2011-08-28 16:58:26 -07:00
# include <stdio.h>
2011-08-29 17:51:54 -07:00
# include <ctype.h>
# include <string.h>
2011-08-30 16:23:47 -07:00
# include <stdlib.h>
# include <errno.h>
2011-11-05 13:51:37 -07:00
# include <unistd.h>
2013-01-31 11:08:36 +11:00
# include <assert.h>
2011-10-02 21:59:54 -07:00
# define __USE_XOPEN
2011-08-30 15:22:48 -07:00
# include <time.h>
2011-08-28 16:58:26 -07:00
# include <libxml/parser.h>
2013-03-15 19:02:14 +02:00
# include <libxml/parserInternals.h>
2011-08-28 16:58:26 -07:00
# include <libxml/tree.h>
2011-11-05 12:39:17 +02:00
# include <libxslt/transform.h>
2013-10-06 08:55:58 -07:00
# include "gettext.h"
2013-03-05 07:10:39 +02:00
# include <sqlite3.h>
2011-08-30 18:23:59 -07:00
# include "dive.h"
Assemble the actual Suunto serial number
It turns out that the serial number returned by libdivecomputer isn't
really the serial number as interpreted by the vendor. Those tend to be
strings, but libdivecomputer gives us a 32bit number.
Some experimenting showed that for the Suunto devies tested the serial
number is encoded in that 32bit number:
It so happens that the Suunto serial number strings are strings that have
all numbers, but they aren't *one* number. They are four bytes
representing two numbers each, and the "23500027" string is actually the
four bytes 23 50 00 27 (0x17 0x32 0x00 0x1b). And libdivecomputer has
incorrectly parsed those four bytes as one number, not as the encoded
serial number string it is. So the value 389152795 is actually hex
0x1732001b, which is 0x17 0x32 0x00 0x1b, which is - 23 50 00 27.
This should be done by libdivecomputer, but hey, in the meantime this at
least shows the concept. And helps test the XML save/restore code.
It depends on the two patches that create the whole "device.c"
infrastructure, of course. With this, my dive file ends up having the
settings section look like this:
<divecomputerid model='Suunto Vyper Air' deviceid='d4629110'
serial='01201094' firmware='1.1.22'/>
<divecomputerid model='Suunto HelO2' deviceid='995dd566'
serial='23500027' firmware='1.0.4'/>
where the format of the firmware version is something I guessed at,
but it was the obvious choice (again, it's byte-based, I'm ignoring
the high byte that is zero for both of my Suuntos).
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-01-09 16:14:21 -08:00
# include "device.h"
2011-08-30 16:42:05 -07:00
2013-11-02 12:00:16 -07:00
int verbose , quit ;
2011-08-30 13:48:05 -07:00
2013-10-16 22:05:19 +03:00
static xmlDoc * test_xslt_transforms ( xmlDoc * doc , const char * * params , char * * error ) ;
2013-10-04 13:21:31 -07:00
char * xslt_path ;
2013-02-22 11:20:46 -08:00
2013-01-31 11:08:36 +11:00
/* the dive table holds the overall dive list; target table points at
* the table we are currently filling */
2011-08-30 18:40:25 -07:00
struct dive_table dive_table ;
2013-01-31 11:08:36 +11:00
struct dive_table * target_table = NULL ;
2013-02-22 11:20:46 -08:00
2013-05-21 23:13:45 -07:00
static void parser_error ( char * * error , const char * fmt , . . . )
2013-02-22 11:20:46 -08:00
{
va_list args ;
2013-05-21 23:13:45 -07:00
char * tmp ;
2013-02-22 11:20:46 -08:00
if ( ! error )
return ;
2013-05-21 23:13:45 -07:00
tmp = malloc ( 1024 ) ;
2013-02-22 11:20:46 -08:00
va_start ( args , fmt ) ;
2013-05-21 23:13:45 -07:00
vsnprintf ( tmp , 1024 , fmt , args ) ;
2013-02-22 11:20:46 -08:00
va_end ( args ) ;
2013-05-21 23:13:45 -07:00
if ( * error ) {
int len = strlen ( * error ) + strlen ( tmp ) + 1 ;
* error = realloc ( * error , len ) ;
strncat ( * error , tmp , strlen ( tmp ) ) ;
free ( tmp ) ;
} else {
* error = tmp ;
}
2013-02-22 11:20:46 -08:00
}
2011-08-30 18:40:25 -07:00
/*
* Add a dive into the dive_table array
*/
2013-01-31 11:08:36 +11:00
static void record_dive_to_table ( struct dive * dive , struct dive_table * table )
2011-08-30 17:18:33 -07:00
{
2013-01-31 11:08:36 +11:00
assert ( table ! = NULL ) ;
int nr = table - > nr , allocated = table - > allocated ;
struct dive * * dives = table - > dives ;
2011-08-30 17:18:33 -07:00
2011-08-30 18:40:25 -07:00
if ( nr > = allocated ) {
allocated = ( nr + 32 ) * 3 / 2 ;
dives = realloc ( dives , allocated * sizeof ( struct dive * ) ) ;
if ( ! dives )
exit ( 1 ) ;
2013-01-31 11:08:36 +11:00
table - > dives = dives ;
table - > allocated = allocated ;
2011-08-30 16:42:05 -07:00
}
2011-09-03 13:19:26 -07:00
dives [ nr ] = fixup_dive ( dive ) ;
2013-01-31 11:08:36 +11:00
table - > nr = nr + 1 ;
2011-08-30 17:18:33 -07:00
}
2013-01-31 11:08:36 +11:00
void record_dive ( struct dive * dive )
2012-04-02 19:19:01 -07:00
{
2013-01-31 11:08:36 +11:00
record_dive_to_table ( dive , & dive_table ) ;
2012-04-02 19:19:01 -07:00
}
2011-09-01 11:22:05 -07:00
static void start_match ( const char * type , const char * name , char * buffer )
2011-08-30 13:58:19 -07:00
{
2011-09-01 11:22:05 -07:00
if ( verbose > 2 )
printf ( " Matching %s '%s' (%s) \n " ,
type , name , buffer ) ;
2011-08-30 13:58:19 -07:00
}
2011-09-01 11:22:05 -07:00
static void nonmatch ( const char * type , const char * name , char * buffer )
2011-08-30 13:58:19 -07:00
{
2011-09-01 11:22:05 -07:00
if ( verbose > 1 )
printf ( " Unable to match %s '%s' (%s) \n " ,
type , name , buffer ) ;
2011-08-30 13:58:19 -07:00
}
2011-08-30 15:22:48 -07:00
typedef void ( * matchfn_t ) ( char * buffer , void * ) ;
2011-08-30 13:48:05 -07:00
2011-09-01 11:22:05 -07:00
static int match ( const char * pattern , int plen ,
2013-11-02 12:00:16 -07:00
const char * name ,
2011-09-01 11:22:05 -07:00
matchfn_t fn , char * buf , void * data )
2011-08-30 13:48:05 -07:00
{
2013-11-02 12:00:16 -07:00
switch ( name [ plen ] ) {
case ' \0 ' : case ' . ' :
break ;
default :
2011-09-01 11:22:05 -07:00
return 0 ;
2013-11-02 12:00:16 -07:00
}
if ( memcmp ( pattern , name , plen ) )
2011-08-30 15:22:48 -07:00
return 0 ;
fn ( buf , data ) ;
return 1 ;
2011-08-30 13:48:05 -07:00
}
2011-09-06 19:07:17 -07:00
2013-01-10 17:26:10 -08:00
struct units xml_parsing_units ;
2012-12-10 09:20:57 -08:00
const struct units SI_units = SI_UNITS ;
const struct units IMPERIAL_units = IMPERIAL_UNITS ;
2011-09-06 19:07:17 -07:00
2011-08-30 13:48:05 -07:00
/*
* Dive info as it is being built up . .
*/
2012-11-24 16:50:21 -10:00
static struct divecomputer * cur_dc ;
2012-09-19 23:42:11 -04:00
static struct dive * cur_dive ;
static dive_trip_t * cur_trip = NULL ;
2012-01-05 08:16:08 -08:00
static struct sample * cur_sample ;
2011-09-22 18:02:54 -07:00
static struct {
int active ;
duration_t time ;
int type , flags , value ;
const char * name ;
2012-01-05 08:16:08 -08:00
} cur_event ;
2012-12-26 13:47:54 -08:00
static struct {
struct {
const char * model ;
uint32_t deviceid ;
Assemble the actual Suunto serial number
It turns out that the serial number returned by libdivecomputer isn't
really the serial number as interpreted by the vendor. Those tend to be
strings, but libdivecomputer gives us a 32bit number.
Some experimenting showed that for the Suunto devies tested the serial
number is encoded in that 32bit number:
It so happens that the Suunto serial number strings are strings that have
all numbers, but they aren't *one* number. They are four bytes
representing two numbers each, and the "23500027" string is actually the
four bytes 23 50 00 27 (0x17 0x32 0x00 0x1b). And libdivecomputer has
incorrectly parsed those four bytes as one number, not as the encoded
serial number string it is. So the value 389152795 is actually hex
0x1732001b, which is 0x17 0x32 0x00 0x1b, which is - 23 50 00 27.
This should be done by libdivecomputer, but hey, in the meantime this at
least shows the concept. And helps test the XML save/restore code.
It depends on the two patches that create the whole "device.c"
infrastructure, of course. With this, my dive file ends up having the
settings section look like this:
<divecomputerid model='Suunto Vyper Air' deviceid='d4629110'
serial='01201094' firmware='1.1.22'/>
<divecomputerid model='Suunto HelO2' deviceid='995dd566'
serial='23500027' firmware='1.0.4'/>
where the format of the firmware version is something I guessed at,
but it was the obvious choice (again, it's byte-based, I'm ignoring
the high byte that is zero for both of my Suuntos).
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-01-09 16:14:21 -08:00
const char * nickname , * serial_nr , * firmware ;
2012-12-26 13:47:54 -08:00
} dc ;
} cur_settings ;
2013-10-05 00:29:09 -07:00
static bool in_settings = FALSE ;
2012-01-05 08:16:08 -08:00
static struct tm cur_tm ;
2012-03-23 21:07:53 -07:00
static int cur_cylinder_index , cur_ws_index ;
2012-12-30 18:11:01 -08:00
static int lastndl , laststoptime , laststopdepth , lastcns , lastpo2 , lastindeco ;
First step in cleaning up cylinder pressure sensor logic
This clarifies/changes the meaning of our "cylinderindex" entry in our
samples. It has been rather confused, because different dive computers
have done things differently, and the naming really hasn't helped.
There are two totally different - and independent - cylinder "indexes":
- the pressure sensor index, which indicates which cylinder the sensor
data is from.
- the "active cylinder" index, which indicates which cylinder we actually
breathe from.
These two values really are totally independent, and have nothing
what-so-ever to do with each other. The sensor index may well be fixed:
many dive computers only support a single pressure sensor (whether
wireless or wired), and the sensor index is thus always zero.
Other dive computers may support multiple pressure sensors, and the gas
switch event may - or may not - indicate that the sensor changed too. A
dive computer might give the sensor data for *all* cylinders it can read,
regardless of which one is the one we're actively breathing. In fact, some
dive computers might give sensor data for not just *your* cylinder, but
your buddies.
This patch renames "cylinderindex" in the samples as "sensor", making it
quite clear that it's about which sensor index the pressure data in the
sample is about.
The way we figure out which is the currently active gas is with an
explicit has change event. If a computer (like the Uemis Zurich) joins the
two concepts together, then a sensor change should also create a gas
switch event. This patch also changes the Uemis importer to do that.
Finally, it should be noted that the plot info works totally separately
from the sample data, and is about what we actually *display*, not about
the sample pressures etc. In the plot info, the "cylinderindex" does in
fact mean the currently active cylinder, and while it is initially set to
match the sensor information from the samples, we then walk the gas change
events and fix it up - and if the active cylinder differs from the sensor
cylinder, we clear the sensor data.
[Dirk Hohndel: this conflicted with some of my recent changes - I think
I merged things correctly...]
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-12-30 20:00:51 -08:00
static int lastcylinderindex , lastsensor ;
2011-08-30 15:22:48 -07:00
2013-02-22 08:52:35 -08:00
/*
* If we don ' t have an explicit dive computer ,
* we use the implicit one that every dive has . .
*/
static struct divecomputer * get_dc ( void )
{
return cur_dc ? : & cur_dive - > dc ;
}
2011-09-05 13:45:14 -07:00
static enum import_source {
UNKNOWN ,
LIBDIVECOMPUTER ,
2011-09-05 14:29:08 -07:00
DIVINGLOG ,
2011-09-06 17:01:28 -07:00
UDDF ,
2011-09-05 13:45:14 -07:00
} import_source ;
2011-08-30 15:22:48 -07:00
static void divedate ( char * buffer , void * _when )
{
int d , m , y ;
2013-01-28 21:07:52 -08:00
int hh , mm , ss ;
2012-09-19 17:35:52 -07:00
timestamp_t * when = _when ;
2013-01-28 21:07:52 -08:00
hh = 0 ; mm = 0 ; ss = 0 ;
if ( sscanf ( buffer , " %d.%d.%d %d:%d:%d " , & d , & m , & y , & hh , & mm , & ss ) > = 3 ) {
/* This is ok, and we got at least the date */
} else if ( sscanf ( buffer , " %d-%d-%d %d:%d:%d " , & y , & m , & d , & hh , & mm , & ss ) > = 3 ) {
/* This is also ok */
2011-09-01 17:13:39 -07:00
} else {
fprintf ( stderr , " Unable to parse date '%s' \n " , buffer ) ;
2013-01-28 21:07:52 -08:00
return ;
2011-08-30 15:22:48 -07:00
}
2013-01-28 21:07:52 -08:00
cur_tm . tm_year = y ;
cur_tm . tm_mon = m - 1 ;
cur_tm . tm_mday = d ;
cur_tm . tm_hour = hh ;
cur_tm . tm_min = mm ;
cur_tm . tm_sec = ss ;
* when = utc_mktime ( & cur_tm ) ;
2011-08-30 15:22:48 -07:00
}
static void divetime ( char * buffer , void * _when )
{
int h , m , s = 0 ;
2012-09-19 17:35:52 -07:00
timestamp_t * when = _when ;
2011-08-30 15:22:48 -07:00
if ( sscanf ( buffer , " %d:%d:%d " , & h , & m , & s ) > = 2 ) {
2012-01-05 08:16:08 -08:00
cur_tm . tm_hour = h ;
cur_tm . tm_min = m ;
cur_tm . tm_sec = s ;
2013-01-28 21:07:52 -08:00
* when = utc_mktime ( & cur_tm ) ;
2011-08-30 15:22:48 -07:00
}
}
2011-08-30 16:59:03 -07:00
/* Libdivecomputer: "2011-03-20 10:22:38" */
static void divedatetime ( char * buffer , void * _when )
{
int y , m , d ;
int hr , min , sec ;
2012-09-19 17:35:52 -07:00
timestamp_t * when = _when ;
2011-08-30 16:59:03 -07:00
if ( sscanf ( buffer , " %d-%d-%d %d:%d:%d " ,
& y , & m , & d , & hr , & min , & sec ) = = 6 ) {
2012-01-05 08:16:08 -08:00
cur_tm . tm_year = y ;
cur_tm . tm_mon = m - 1 ;
cur_tm . tm_mday = d ;
cur_tm . tm_hour = hr ;
cur_tm . tm_min = min ;
cur_tm . tm_sec = sec ;
* when = utc_mktime ( & cur_tm ) ;
2011-08-30 16:59:03 -07:00
}
}
2013-11-02 02:12:42 +01:00
enum ParseState { FINDSTART , FINDEND } ;
2013-04-09 13:06:30 -07:00
static void divetags ( char * buffer , void * _tags )
{
2013-11-02 02:12:42 +01:00
struct tag_entry * tags = _tags ;
char tag [ 128 ] ;
int i = 0 , start = 0 , end = 0 ;
enum ParseState state = FINDEND ;
i = 0 ;
while ( i < strlen ( buffer ) ) {
if ( buffer [ i ] = = ' , ' ) {
if ( state = = FINDSTART ) {
/* Detect empty tags */
} else if ( state = = FINDEND ) {
/* Found end of tag */
if ( i > 1 ) {
if ( buffer [ i - 1 ] ! = ' \\ ' ) {
strncpy ( tag , buffer + start , end - start + 1 ) ;
tag [ end - start + 1 ] = ' \0 ' ;
state = FINDSTART ;
taglist_add_tag ( tags , tag ) ;
}
} else {
state = FINDSTART ;
2013-04-09 13:06:30 -07:00
}
}
2013-11-02 02:12:42 +01:00
} else if ( buffer [ i ] = = ' ' ) {
/* Handled */
} else {
/* Found start of tag */
if ( state = = FINDSTART ) {
state = FINDEND ;
start = i ;
} else if ( state = = FINDEND ) {
end = i ;
}
2013-04-09 13:06:30 -07:00
}
2013-11-02 02:12:42 +01:00
i + + ;
}
if ( state = = FINDEND ) {
if ( end < start )
end = strlen ( buffer ) - 1 ;
if ( strlen ( buffer ) > 0 ) {
strncpy ( tag , buffer + start , end - start + 1 ) ;
tag [ end - start + 1 ] = ' \0 ' ;
taglist_add_tag ( tags , tag ) ;
}
}
2013-04-09 13:06:30 -07:00
}
2011-08-30 16:23:47 -07:00
enum number_type {
NEITHER ,
FLOAT
} ;
2013-10-05 00:29:09 -07:00
double ascii_strtod ( char * str , char * * ptr )
{
2013-10-07 08:57:27 -07:00
char * p = str , c , * ep ;
double val = 0.0 ;
double decimal ;
int sign = 0 , esign = 0 ;
int numbers = 0 , dot = 0 ;
/* skip spaces */
while ( isspace ( c = * p + + ) )
/* */ ;
/* optional sign */
switch ( c ) {
case ' - ' :
sign = 1 ;
/* fallthrough */
case ' + ' :
c = * p + + ;
}
/* Mantissa */
for ( ; ; c = * p + + ) {
if ( c = = ' . ' ) {
if ( dot )
goto done ;
dot = 1 ;
decimal = 1.0 ;
continue ;
2013-10-05 00:29:09 -07:00
}
2013-10-07 08:57:27 -07:00
if ( c > = ' 0 ' & & c < = ' 9 ' ) {
numbers + + ;
if ( dot ) {
decimal / = 10 ;
val + = ( c - ' 0 ' ) * decimal ;
} else {
val = ( val * 10 ) + ( c - ' 0 ' ) ;
}
continue ;
2013-10-05 00:29:09 -07:00
}
2013-10-07 08:57:27 -07:00
if ( c ! = ' e ' & & c ! = ' E ' )
goto done ;
break ;
2013-10-05 00:29:09 -07:00
}
2013-10-07 08:57:27 -07:00
if ( ! numbers )
goto done ;
/* Exponent */
ep = p ;
c = * ep + + ;
switch ( c ) {
case ' - ' :
esign = 1 ;
/* fallthrough */
case ' + ' :
c = * ep + + ;
2013-10-05 00:29:09 -07:00
}
2013-10-07 08:57:27 -07:00
if ( c > = ' 0 ' & & c < = ' 9 ' ) {
p = ep ;
int exponent = c - ' 0 ' ;
for ( ; ; ) {
c = * p + + ;
if ( c < ' 0 ' | | c > ' 9 ' )
break ;
exponent * = 10 ;
exponent + = c - ' 0 ' ;
2013-10-05 00:29:09 -07:00
}
2013-10-07 08:57:27 -07:00
/* We're not going to bother playing games */
if ( exponent > 308 )
exponent = 308 ;
while ( exponent - - > 0 ) {
if ( esign )
val / = 10 ;
else
val * = 10 ;
2013-10-05 00:29:09 -07:00
}
}
2013-10-07 08:57:27 -07:00
done :
if ( ! numbers )
goto no_conversion ;
2013-10-07 10:59:50 -07:00
if ( ptr )
* ptr = p - 1 ;
2013-10-07 08:57:27 -07:00
return sign ? - val : val ;
no_conversion :
2013-10-07 10:59:50 -07:00
if ( ptr )
* ptr = str ;
2013-10-05 00:29:09 -07:00
return 0.0 ;
}
2013-02-22 16:18:39 -08:00
static enum number_type parse_float ( char * buffer , double * res , char * * endp )
2011-08-30 16:23:47 -07:00
{
2013-02-22 16:18:39 -08:00
double val ;
2013-10-05 00:29:09 -07:00
static bool first_time = TRUE ;
2011-08-30 16:23:47 -07:00
2013-01-23 12:09:29 -08:00
errno = 0 ;
2013-10-05 00:29:09 -07:00
val = ascii_strtod ( buffer , endp ) ;
2013-02-22 16:18:39 -08:00
if ( errno | | * endp = = buffer )
return NEITHER ;
2013-03-07 11:43:51 -08:00
if ( * * endp = = ' , ' ) {
if ( val = = rint ( val ) ) {
/* we really want to send an error if this is a Subsurface native file
* as this is likely indication of a bug - but right now we don ' t have
* that information available */
if ( first_time ) {
fprintf ( stderr , " Floating point value with decimal comma (%s)? \n " , buffer ) ;
first_time = FALSE ;
}
/* Try again */
* * endp = ' . ' ;
2013-10-05 00:29:09 -07:00
val = ascii_strtod ( buffer , endp ) ;
2013-03-07 11:43:51 -08:00
}
}
2013-02-22 16:18:39 -08:00
* res = val ;
return FLOAT ;
}
union int_or_float {
double fp ;
} ;
2011-08-30 16:23:47 -07:00
2013-02-22 16:18:39 -08:00
static enum number_type integer_or_float ( char * buffer , union int_or_float * res )
{
char * end ;
return parse_float ( buffer , & res - > fp , & end ) ;
2011-08-30 16:23:47 -07:00
}
static void pressure ( char * buffer , void * _press )
{
2011-09-02 13:59:39 -07:00
double mbar ;
2011-08-30 16:23:47 -07:00
pressure_t * pressure = _press ;
union int_or_float val ;
switch ( integer_or_float ( buffer , & val ) ) {
case FLOAT :
2011-09-02 14:06:26 -07:00
/* Just ignore zero values */
if ( ! val . fp )
break ;
2013-01-10 17:26:10 -08:00
switch ( xml_parsing_units . pressure ) {
2011-09-06 17:01:28 -07:00
case PASCAL :
mbar = val . fp / 100 ;
break ;
2011-09-02 13:59:39 -07:00
case BAR :
/* Assume mbar, but if it's really small, it's bar */
mbar = val . fp ;
if ( mbar < 5000 )
mbar = mbar * 1000 ;
break ;
case PSI :
mbar = val . fp * 68.95 ;
2011-08-30 16:23:47 -07:00
break ;
}
2011-09-02 13:59:39 -07:00
if ( mbar > 5 & & mbar < 500000 ) {
pressure - > mbar = mbar + 0.5 ;
2011-08-30 16:23:47 -07:00
break ;
}
2011-09-02 13:59:39 -07:00
/* fallthrough */
2011-08-30 16:23:47 -07:00
default :
printf ( " Strange pressure reading %s \n " , buffer ) ;
}
}
2012-11-12 20:57:49 +01:00
static void salinity ( char * buffer , void * _salinity )
{
int * salinity = _salinity ;
union int_or_float val ;
switch ( integer_or_float ( buffer , & val ) ) {
case FLOAT :
* salinity = val . fp * 10.0 + 0.5 ;
break ;
default :
printf ( " Strange salinity reading %s \n " , buffer ) ;
}
}
2011-08-30 16:23:47 -07:00
static void depth ( char * buffer , void * _depth )
{
depth_t * depth = _depth ;
union int_or_float val ;
switch ( integer_or_float ( buffer , & val ) ) {
case FLOAT :
2013-01-10 17:26:10 -08:00
switch ( xml_parsing_units . length ) {
2011-09-02 13:59:39 -07:00
case METERS :
depth - > mm = val . fp * 1000 + 0.5 ;
break ;
case FEET :
depth - > mm = val . fp * 304.8 + 0.5 ;
break ;
}
2011-08-30 16:23:47 -07:00
break ;
default :
printf ( " Strange depth reading %s \n " , buffer ) ;
}
}
2011-12-23 19:41:16 -08:00
static void weight ( char * buffer , void * _weight )
{
weight_t * weight = _weight ;
union int_or_float val ;
switch ( integer_or_float ( buffer , & val ) ) {
case FLOAT :
2013-01-10 17:26:10 -08:00
switch ( xml_parsing_units . weight ) {
2011-12-23 19:41:16 -08:00
case KG :
weight - > grams = val . fp * 1000 + 0.5 ;
break ;
case LBS :
weight - > grams = val . fp * 453.6 + 0.5 ;
break ;
}
break ;
default :
2012-11-24 12:12:16 +02:00
printf ( " Strange weight reading %s \n " , buffer ) ;
2011-12-23 19:41:16 -08:00
}
}
2011-08-30 16:23:47 -07:00
static void temperature ( char * buffer , void * _temperature )
{
temperature_t * temperature = _temperature ;
union int_or_float val ;
switch ( integer_or_float ( buffer , & val ) ) {
case FLOAT :
2013-01-10 17:26:10 -08:00
switch ( xml_parsing_units . temperature ) {
2011-09-06 17:01:28 -07:00
case KELVIN :
temperature - > mkelvin = val . fp * 1000 ;
break ;
2011-09-02 13:59:39 -07:00
case CELSIUS :
2013-01-24 23:09:53 +01:00
temperature - > mkelvin = val . fp * 1000 + ZERO_C_IN_MKELVIN + 0.5 ;
2011-08-30 16:23:47 -07:00
break ;
2011-09-02 13:59:39 -07:00
case FAHRENHEIT :
2011-08-30 16:23:47 -07:00
temperature - > mkelvin = ( val . fp + 459.67 ) * 5000 / 9 ;
break ;
}
break ;
default :
printf ( " Strange temperature reading %s \n " , buffer ) ;
}
2013-03-08 11:52:10 -08:00
/* temperatures outside -40C .. +70C should be ignored */
if ( temperature - > mkelvin < ZERO_C_IN_MKELVIN - 40000 | |
temperature - > mkelvin > ZERO_C_IN_MKELVIN + 70000 )
temperature - > mkelvin = 0 ;
2011-08-30 16:23:47 -07:00
}
static void sampletime ( char * buffer , void * _time )
{
2011-08-30 17:45:03 -07:00
int i ;
int min , sec ;
2011-08-30 16:23:47 -07:00
duration_t * time = _time ;
2011-08-30 17:45:03 -07:00
i = sscanf ( buffer , " %d:%d " , & min , & sec ) ;
switch ( i ) {
case 1 :
sec = min ;
min = 0 ;
/* fallthrough */
case 2 :
time - > seconds = sec + min * 60 ;
2011-08-30 16:23:47 -07:00
break ;
default :
printf ( " Strange sample time reading %s \n " , buffer ) ;
}
}
2011-08-30 17:45:03 -07:00
static void duration ( char * buffer , void * _time )
{
2013-02-25 14:19:16 -08:00
/* DivingLog 5.08 (and maybe other versions) appear to sometimes
* store the dive time as 44.00 instead of 44 : 00 ;
* This attempts to parse this in a fairly robust way */
if ( ! strchr ( buffer , ' : ' ) & & strchr ( buffer , ' . ' ) ) {
char * mybuffer = strdup ( buffer ) ;
char * dot = strchr ( mybuffer , ' . ' ) ;
* dot = ' : ' ;
sampletime ( mybuffer , _time ) ;
} else {
sampletime ( buffer , _time ) ;
}
2011-08-30 17:45:03 -07:00
}
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
static void percent ( char * buffer , void * _fraction )
{
fraction_t * fraction = _fraction ;
2013-02-22 16:18:39 -08:00
double val ;
char * end ;
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
2013-02-22 16:18:39 -08:00
switch ( parse_float ( buffer , & val , & end ) ) {
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
case FLOAT :
2013-02-22 16:18:39 -08:00
/* Turn fractions into percent unless explicit.. */
if ( val < = 1.0 ) {
2013-10-05 00:29:09 -07:00
while ( isspace ( * end ) )
2013-02-22 16:18:39 -08:00
end + + ;
if ( * end ! = ' % ' )
val * = 100 ;
}
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
2013-02-22 16:18:39 -08:00
/* Then turn percent into our integer permille format */
if ( val > = 0 & & val < = 100.0 ) {
fraction - > permille = val * 10 + 0.5 ;
break ;
}
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
default :
2013-10-09 22:48:35 -07:00
printf ( translate ( " gettextFromC " , " Strange percentage reading %s \n " ) , buffer ) ;
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
break ;
}
}
static void gasmix ( char * buffer , void * _fraction )
{
2011-09-01 13:46:24 -07:00
/* libdivecomputer does negative percentages. */
if ( * buffer = = ' - ' )
return ;
2012-01-05 08:16:08 -08:00
if ( cur_cylinder_index < MAX_CYLINDERS )
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
percent ( buffer , _fraction ) ;
}
2011-09-01 16:26:11 -07:00
static void gasmix_nitrogen ( char * buffer , void * _gasmix )
{
/* Ignore n2 percentages. There's no value in them. */
}
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
2011-09-03 20:31:18 -07:00
static void cylindersize ( char * buffer , void * _volume )
{
volume_t * volume = _volume ;
union int_or_float val ;
switch ( integer_or_float ( buffer , & val ) ) {
case FLOAT :
volume - > mliter = val . fp * 1000 + 0.5 ;
break ;
default :
printf ( " Strange volume reading %s \n " , buffer ) ;
break ;
}
}
2011-09-01 19:56:04 -07:00
static void utf8_string ( char * buffer , void * _res )
{
2012-12-28 18:18:23 +02:00
int size ;
char * res ;
2013-10-05 00:29:09 -07:00
while ( isspace ( * buffer ) )
2012-12-28 18:18:23 +02:00
buffer + + ;
size = strlen ( buffer ) ;
2013-10-05 00:29:09 -07:00
while ( size & & isspace ( buffer [ size - 1 ] ) )
2012-12-28 18:18:23 +02:00
size - - ;
if ( ! size )
return ;
res = malloc ( size + 1 ) ;
memcpy ( res , buffer , size ) ;
res [ size ] = 0 ;
* ( char * * ) _res = res ;
2011-09-01 19:56:04 -07:00
}
2011-09-01 11:22:05 -07:00
# define MATCH(pattern, fn, dest) \
2013-11-02 12:00:16 -07:00
match ( pattern , strlen ( pattern ) , name , fn , buf , dest )
2011-09-01 11:22:05 -07:00
2011-09-02 15:01:53 -07:00
static void get_index ( char * buffer , void * _i )
{
int * i = _i ;
* i = atoi ( buffer ) ;
}
2013-01-29 22:30:02 +01:00
static void get_rating ( char * buffer , void * _i )
{
int * i = _i ;
int j = atoi ( buffer ) ;
if ( j > = 0 & & j < = 5 ) {
* i = j ;
}
}
2012-12-07 20:08:29 -08:00
static void double_to_permil ( char * buffer , void * _i )
{
int * i = _i ;
2013-10-05 00:29:09 -07:00
* i = ascii_strtod ( buffer , NULL ) * 1000.0 + 0.5 ;
2012-12-07 20:08:29 -08:00
}
2012-11-25 11:44:27 -08:00
static void hex_value ( char * buffer , void * _i )
{
uint32_t * i = _i ;
* i = strtol ( buffer , NULL , 16 ) ;
}
2012-08-21 22:04:24 -07:00
static void get_tripflag ( char * buffer , void * _tf )
{
tripflag_t * tf = _tf ;
2012-11-26 14:52:07 -08:00
* tf = strcmp ( buffer , " NOTRIP " ) ? TF_NONE : NO_TRIP ;
2012-08-21 22:04:24 -07:00
}
2011-09-05 14:29:08 -07:00
/*
* Divinglog is crazy . The temperatures are in celsius . EXCEPT
* for the sample temperatures , that are in Fahrenheit .
* WTF ?
2011-09-11 13:16:23 -07:00
*
* Oh , and I think Diving Log * internally * probably kept them
* in celsius , because I ' m seeing entries like
*
* < Temp > 32.0 < / Temp >
*
* in there . Which is freezing , aka 0 degC . I bet the " 0 " is
* what Diving Log uses for " no temperature " .
*
* So throw away crap like that .
2012-10-10 18:14:55 +09:00
*
* It gets worse . Sometimes the sample temperatures are in
* Celsius , which apparently happens if you are in a SI
* locale . So we now do :
*
* - temperatures < 32.0 = = Celsius
* - temperature = = 32.0 - > garbage , it ' s a missing temperature ( zero converted from C to F )
* - temperatures > 32.0 = = Fahrenheit
2011-09-05 14:29:08 -07:00
*/
static void fahrenheit ( char * buffer , void * _temperature )
{
temperature_t * temperature = _temperature ;
union int_or_float val ;
switch ( integer_or_float ( buffer , & val ) ) {
case FLOAT :
2011-09-11 13:16:23 -07:00
/* Floating point equality is evil, but works for small integers */
if ( val . fp = = 32.0 )
break ;
2012-10-10 18:14:55 +09:00
if ( val . fp < 32.0 )
temperature - > mkelvin = C_to_mkelvin ( val . fp ) ;
else
temperature - > mkelvin = F_to_mkelvin ( val . fp ) ;
2011-09-05 14:29:08 -07:00
break ;
default :
fprintf ( stderr , " Crazy Diving Log temperature reading %s \n " , buffer ) ;
}
}
2011-09-11 12:24:57 -07:00
/*
* Did I mention how bat - shit crazy divinglog is ? The sample
* pressures are in PSI . But the tank working pressure is in
* bar . WTF ^ 2 ?
*
2011-09-15 09:43:14 -07:00
* Crazy stuff like this is why subsurface has everything in
2011-09-11 12:24:57 -07:00
* these inconvenient typed structures , and you have to say
* " pressure->mbar " to get the actual value . Exactly so that
* you can never have unit confusion .
2012-10-10 18:14:55 +09:00
*
* It gets worse : sometimes apparently the pressures are in
* bar , sometimes in psi . Dirk suspects that this may be a
* DivingLog Uemis importer bug , and that they are always
* supposed to be in bar , but that the importer got the
* sample importing wrong .
*
* Sadly , there ' s no way to really tell . So I think we just
* have to have some arbitrary cut - off point where we assume
* that smaller values mean bar . . Not good .
2011-09-11 12:24:57 -07:00
*/
2012-10-10 18:14:55 +09:00
static void psi_or_bar ( char * buffer , void * _pressure )
2011-09-11 12:24:57 -07:00
{
pressure_t * pressure = _pressure ;
union int_or_float val ;
switch ( integer_or_float ( buffer , & val ) ) {
case FLOAT :
2012-10-10 18:14:55 +09:00
if ( val . fp > 400 )
pressure - > mbar = psi_to_mbar ( val . fp ) ;
else
pressure - > mbar = val . fp * 1000 + 0.5 ;
2011-09-11 12:24:57 -07:00
break ;
default :
fprintf ( stderr , " Crazy Diving Log PSI reading %s \n " , buffer ) ;
}
}
2013-11-02 12:00:16 -07:00
static int divinglog_fill_sample ( struct sample * sample , const char * name , char * buf )
2011-09-05 14:29:08 -07:00
{
2013-11-02 12:00:16 -07:00
return MATCH ( " time.p " , sampletime , & sample - > time ) | |
MATCH ( " depth.p " , depth , & sample - > depth ) | |
MATCH ( " temp.p " , fahrenheit , & sample - > temperature ) | |
MATCH ( " press1.p " , psi_or_bar , & sample - > cylinderpressure ) | |
2011-09-05 14:29:08 -07:00
0 ;
}
2013-02-22 08:52:35 -08:00
static void uddf_gasswitch ( char * buffer , void * _sample )
{
struct sample * sample = _sample ;
int idx = atoi ( buffer ) ;
int seconds = sample - > time . seconds ;
struct dive * dive = cur_dive ;
struct divecomputer * dc = get_dc ( ) ;
add_gas_switch_event ( dive , dc , seconds , idx ) ;
}
2013-11-02 12:00:16 -07:00
static int uddf_fill_sample ( struct sample * sample , const char * name , char * buf )
2011-09-06 17:33:52 -07:00
{
2013-11-02 12:00:16 -07:00
return MATCH ( " divetime " , sampletime , & sample - > time ) | |
MATCH ( " depth " , depth , & sample - > depth ) | |
MATCH ( " temperature " , temperature , & sample - > temperature ) | |
MATCH ( " tankpressure " , pressure , & sample - > cylinderpressure ) | |
MATCH ( " ref.switchmix " , uddf_gasswitch , sample ) | |
2011-09-06 17:33:52 -07:00
0 ;
}
2011-09-22 18:02:54 -07:00
static void eventtime ( char * buffer , void * _duration )
{
duration_t * duration = _duration ;
sampletime ( buffer , duration ) ;
2012-01-05 08:16:08 -08:00
if ( cur_sample )
duration - > seconds + = cur_sample - > time . seconds ;
2011-09-22 18:02:54 -07:00
}
2013-01-01 17:29:38 -08:00
static void try_to_match_autogroup ( const char * name , char * buf )
{
int autogroupvalue ;
start_match ( " autogroup " , name , buf ) ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " state.autogroup " , get_index , & autogroupvalue ) ) {
2013-01-01 17:29:38 -08:00
set_autogroup ( autogroupvalue ) ;
return ;
}
nonmatch ( " autogroup " , name , buf ) ;
}
2012-12-26 13:47:54 -08:00
static void try_to_fill_dc_settings ( const char * name , char * buf )
{
start_match ( " divecomputerid " , name , buf ) ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " model.divecomputerid " , utf8_string , & cur_settings . dc . model ) )
2012-12-26 13:47:54 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " deviceid.divecomputerid " , hex_value , & cur_settings . dc . deviceid ) )
2012-12-26 13:47:54 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " nickname.divecomputerid " , utf8_string , & cur_settings . dc . nickname ) )
2012-12-26 13:47:54 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " serial.divecomputerid " , utf8_string , & cur_settings . dc . serial_nr ) )
Assemble the actual Suunto serial number
It turns out that the serial number returned by libdivecomputer isn't
really the serial number as interpreted by the vendor. Those tend to be
strings, but libdivecomputer gives us a 32bit number.
Some experimenting showed that for the Suunto devies tested the serial
number is encoded in that 32bit number:
It so happens that the Suunto serial number strings are strings that have
all numbers, but they aren't *one* number. They are four bytes
representing two numbers each, and the "23500027" string is actually the
four bytes 23 50 00 27 (0x17 0x32 0x00 0x1b). And libdivecomputer has
incorrectly parsed those four bytes as one number, not as the encoded
serial number string it is. So the value 389152795 is actually hex
0x1732001b, which is 0x17 0x32 0x00 0x1b, which is - 23 50 00 27.
This should be done by libdivecomputer, but hey, in the meantime this at
least shows the concept. And helps test the XML save/restore code.
It depends on the two patches that create the whole "device.c"
infrastructure, of course. With this, my dive file ends up having the
settings section look like this:
<divecomputerid model='Suunto Vyper Air' deviceid='d4629110'
serial='01201094' firmware='1.1.22'/>
<divecomputerid model='Suunto HelO2' deviceid='995dd566'
serial='23500027' firmware='1.0.4'/>
where the format of the firmware version is something I guessed at,
but it was the obvious choice (again, it's byte-based, I'm ignoring
the high byte that is zero for both of my Suuntos).
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-01-09 16:14:21 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " firmware.divecomputerid " , utf8_string , & cur_settings . dc . firmware ) )
Assemble the actual Suunto serial number
It turns out that the serial number returned by libdivecomputer isn't
really the serial number as interpreted by the vendor. Those tend to be
strings, but libdivecomputer gives us a 32bit number.
Some experimenting showed that for the Suunto devies tested the serial
number is encoded in that 32bit number:
It so happens that the Suunto serial number strings are strings that have
all numbers, but they aren't *one* number. They are four bytes
representing two numbers each, and the "23500027" string is actually the
four bytes 23 50 00 27 (0x17 0x32 0x00 0x1b). And libdivecomputer has
incorrectly parsed those four bytes as one number, not as the encoded
serial number string it is. So the value 389152795 is actually hex
0x1732001b, which is 0x17 0x32 0x00 0x1b, which is - 23 50 00 27.
This should be done by libdivecomputer, but hey, in the meantime this at
least shows the concept. And helps test the XML save/restore code.
It depends on the two patches that create the whole "device.c"
infrastructure, of course. With this, my dive file ends up having the
settings section look like this:
<divecomputerid model='Suunto Vyper Air' deviceid='d4629110'
serial='01201094' firmware='1.1.22'/>
<divecomputerid model='Suunto HelO2' deviceid='995dd566'
serial='23500027' firmware='1.0.4'/>
where the format of the firmware version is something I guessed at,
but it was the obvious choice (again, it's byte-based, I'm ignoring
the high byte that is zero for both of my Suuntos).
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-01-09 16:14:21 -08:00
return ;
2012-12-26 13:47:54 -08:00
nonmatch ( " divecomputerid " , name , buf ) ;
}
2011-09-22 18:02:54 -07:00
static void try_to_fill_event ( const char * name , char * buf )
{
start_match ( " event " , name , buf ) ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " event " , utf8_string , & cur_event . name ) )
2011-09-22 18:02:54 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " name " , utf8_string , & cur_event . name ) )
2011-09-22 18:02:54 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " time " , eventtime , & cur_event . time ) )
2011-09-22 18:02:54 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " type " , get_index , & cur_event . type ) )
2011-09-22 18:02:54 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " flags " , get_index , & cur_event . flags ) )
2011-09-22 18:02:54 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " value " , get_index , & cur_event . value ) )
2011-09-30 21:55:51 -07:00
return ;
2011-09-22 18:02:54 -07:00
nonmatch ( " event " , name , buf ) ;
}
2013-11-02 12:00:16 -07:00
static int match_dc_data_fields ( struct divecomputer * dc , const char * name , char * buf )
2013-01-23 10:25:31 -08:00
{
2013-11-02 12:00:16 -07:00
if ( MATCH ( " maxdepth " , depth , & dc - > maxdepth ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " meandepth " , depth , & dc - > meandepth ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " max.depth " , depth , & dc - > maxdepth ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " mean.depth " , depth , & dc - > meandepth ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " duration " , duration , & dc - > duration ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " divetime " , duration , & dc - > duration ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " divetimesec " , duration , & dc - > duration ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " surfacetime " , duration , & dc - > surfacetime ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " airtemp " , temperature , & dc - > airtemp ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " watertemp " , temperature , & dc - > watertemp ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " air.temperature " , temperature , & dc - > airtemp ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " water.temperature " , temperature , & dc - > watertemp ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " pressure.surface " , pressure , & dc - > surface_pressure ) )
2013-01-23 10:25:31 -08:00
return 1 ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " salinity.water " , salinity , & dc - > salinity ) )
2013-01-23 10:25:31 -08:00
return 1 ;
return 0 ;
}
2012-11-24 16:50:21 -10:00
/* We're in the top-level dive xml. Try to convert whatever value to a dive value */
static void try_to_fill_dc ( struct divecomputer * dc , const char * name , char * buf )
{
start_match ( " divecomputer " , name , buf ) ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " date " , divedate , & dc - > when ) )
2012-11-24 16:50:21 -10:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " time " , divetime , & dc - > when ) )
2012-11-24 16:50:21 -10:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " model " , utf8_string , & dc - > model ) )
2012-11-25 11:44:27 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " deviceid " , hex_value , & dc - > deviceid ) )
2012-11-24 16:50:21 -10:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " diveid " , hex_value , & dc - > diveid ) )
2012-11-24 16:50:21 -10:00
return ;
2013-11-02 12:00:16 -07:00
if ( match_dc_data_fields ( dc , name , buf ) )
2013-01-23 10:25:31 -08:00
return ;
2012-11-24 16:50:21 -10:00
nonmatch ( " divecomputer " , name , buf ) ;
}
First step in cleaning up cylinder pressure sensor logic
This clarifies/changes the meaning of our "cylinderindex" entry in our
samples. It has been rather confused, because different dive computers
have done things differently, and the naming really hasn't helped.
There are two totally different - and independent - cylinder "indexes":
- the pressure sensor index, which indicates which cylinder the sensor
data is from.
- the "active cylinder" index, which indicates which cylinder we actually
breathe from.
These two values really are totally independent, and have nothing
what-so-ever to do with each other. The sensor index may well be fixed:
many dive computers only support a single pressure sensor (whether
wireless or wired), and the sensor index is thus always zero.
Other dive computers may support multiple pressure sensors, and the gas
switch event may - or may not - indicate that the sensor changed too. A
dive computer might give the sensor data for *all* cylinders it can read,
regardless of which one is the one we're actively breathing. In fact, some
dive computers might give sensor data for not just *your* cylinder, but
your buddies.
This patch renames "cylinderindex" in the samples as "sensor", making it
quite clear that it's about which sensor index the pressure data in the
sample is about.
The way we figure out which is the currently active gas is with an
explicit has change event. If a computer (like the Uemis Zurich) joins the
two concepts together, then a sensor change should also create a gas
switch event. This patch also changes the Uemis importer to do that.
Finally, it should be noted that the plot info works totally separately
from the sample data, and is about what we actually *display*, not about
the sample pressures etc. In the plot info, the "cylinderindex" does in
fact mean the currently active cylinder, and while it is initially set to
match the sensor information from the samples, we then walk the gas change
events and fix it up - and if the active cylinder differs from the sensor
cylinder, we clear the sensor data.
[Dirk Hohndel: this conflicted with some of my recent changes - I think
I merged things correctly...]
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-12-30 20:00:51 -08:00
void add_gas_switch_event ( struct dive * dive , struct divecomputer * dc , int seconds , int idx )
{
/* The gas switch event format is insane. It will be fixed, I think */
int o2 = dive - > cylinder [ idx ] . gasmix . o2 . permille ;
int he = dive - > cylinder [ idx ] . gasmix . he . permille ;
int value ;
if ( ! o2 )
2013-01-14 23:53:38 +01:00
o2 = O2_IN_AIR ;
First step in cleaning up cylinder pressure sensor logic
This clarifies/changes the meaning of our "cylinderindex" entry in our
samples. It has been rather confused, because different dive computers
have done things differently, and the naming really hasn't helped.
There are two totally different - and independent - cylinder "indexes":
- the pressure sensor index, which indicates which cylinder the sensor
data is from.
- the "active cylinder" index, which indicates which cylinder we actually
breathe from.
These two values really are totally independent, and have nothing
what-so-ever to do with each other. The sensor index may well be fixed:
many dive computers only support a single pressure sensor (whether
wireless or wired), and the sensor index is thus always zero.
Other dive computers may support multiple pressure sensors, and the gas
switch event may - or may not - indicate that the sensor changed too. A
dive computer might give the sensor data for *all* cylinders it can read,
regardless of which one is the one we're actively breathing. In fact, some
dive computers might give sensor data for not just *your* cylinder, but
your buddies.
This patch renames "cylinderindex" in the samples as "sensor", making it
quite clear that it's about which sensor index the pressure data in the
sample is about.
The way we figure out which is the currently active gas is with an
explicit has change event. If a computer (like the Uemis Zurich) joins the
two concepts together, then a sensor change should also create a gas
switch event. This patch also changes the Uemis importer to do that.
Finally, it should be noted that the plot info works totally separately
from the sample data, and is about what we actually *display*, not about
the sample pressures etc. In the plot info, the "cylinderindex" does in
fact mean the currently active cylinder, and while it is initially set to
match the sensor information from the samples, we then walk the gas change
events and fix it up - and if the active cylinder differs from the sensor
cylinder, we clear the sensor data.
[Dirk Hohndel: this conflicted with some of my recent changes - I think
I merged things correctly...]
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-12-30 20:00:51 -08:00
o2 = ( o2 + 5 ) / 10 ;
he = ( he + 5 ) / 10 ;
value = o2 + ( he < < 16 ) ;
2013-09-18 09:06:37 +02:00
add_event ( dc , seconds , 25 , 0 , value , " gaschange " ) ; /* SAMPLE_EVENT_GASCHANGE2 */
First step in cleaning up cylinder pressure sensor logic
This clarifies/changes the meaning of our "cylinderindex" entry in our
samples. It has been rather confused, because different dive computers
have done things differently, and the naming really hasn't helped.
There are two totally different - and independent - cylinder "indexes":
- the pressure sensor index, which indicates which cylinder the sensor
data is from.
- the "active cylinder" index, which indicates which cylinder we actually
breathe from.
These two values really are totally independent, and have nothing
what-so-ever to do with each other. The sensor index may well be fixed:
many dive computers only support a single pressure sensor (whether
wireless or wired), and the sensor index is thus always zero.
Other dive computers may support multiple pressure sensors, and the gas
switch event may - or may not - indicate that the sensor changed too. A
dive computer might give the sensor data for *all* cylinders it can read,
regardless of which one is the one we're actively breathing. In fact, some
dive computers might give sensor data for not just *your* cylinder, but
your buddies.
This patch renames "cylinderindex" in the samples as "sensor", making it
quite clear that it's about which sensor index the pressure data in the
sample is about.
The way we figure out which is the currently active gas is with an
explicit has change event. If a computer (like the Uemis Zurich) joins the
two concepts together, then a sensor change should also create a gas
switch event. This patch also changes the Uemis importer to do that.
Finally, it should be noted that the plot info works totally separately
from the sample data, and is about what we actually *display*, not about
the sample pressures etc. In the plot info, the "cylinderindex" does in
fact mean the currently active cylinder, and while it is initially set to
match the sensor information from the samples, we then walk the gas change
events and fix it up - and if the active cylinder differs from the sensor
cylinder, we clear the sensor data.
[Dirk Hohndel: this conflicted with some of my recent changes - I think
I merged things correctly...]
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-12-30 20:00:51 -08:00
}
static void get_cylinderindex ( char * buffer , void * _i )
{
int * i = _i ;
* i = atoi ( buffer ) ;
if ( lastcylinderindex ! = * i ) {
2013-01-18 10:39:25 -08:00
add_gas_switch_event ( cur_dive , get_dc ( ) , cur_sample - > time . seconds , * i ) ;
First step in cleaning up cylinder pressure sensor logic
This clarifies/changes the meaning of our "cylinderindex" entry in our
samples. It has been rather confused, because different dive computers
have done things differently, and the naming really hasn't helped.
There are two totally different - and independent - cylinder "indexes":
- the pressure sensor index, which indicates which cylinder the sensor
data is from.
- the "active cylinder" index, which indicates which cylinder we actually
breathe from.
These two values really are totally independent, and have nothing
what-so-ever to do with each other. The sensor index may well be fixed:
many dive computers only support a single pressure sensor (whether
wireless or wired), and the sensor index is thus always zero.
Other dive computers may support multiple pressure sensors, and the gas
switch event may - or may not - indicate that the sensor changed too. A
dive computer might give the sensor data for *all* cylinders it can read,
regardless of which one is the one we're actively breathing. In fact, some
dive computers might give sensor data for not just *your* cylinder, but
your buddies.
This patch renames "cylinderindex" in the samples as "sensor", making it
quite clear that it's about which sensor index the pressure data in the
sample is about.
The way we figure out which is the currently active gas is with an
explicit has change event. If a computer (like the Uemis Zurich) joins the
two concepts together, then a sensor change should also create a gas
switch event. This patch also changes the Uemis importer to do that.
Finally, it should be noted that the plot info works totally separately
from the sample data, and is about what we actually *display*, not about
the sample pressures etc. In the plot info, the "cylinderindex" does in
fact mean the currently active cylinder, and while it is initially set to
match the sensor information from the samples, we then walk the gas change
events and fix it up - and if the active cylinder differs from the sensor
cylinder, we clear the sensor data.
[Dirk Hohndel: this conflicted with some of my recent changes - I think
I merged things correctly...]
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-12-30 20:00:51 -08:00
lastcylinderindex = * i ;
}
}
static void get_sensor ( char * buffer , void * _i )
{
int * i = _i ;
* i = atoi ( buffer ) ;
lastsensor = * i ;
}
2011-08-30 15:22:48 -07:00
/* We're in samples - try to convert the random xml value to something useful */
static void try_to_fill_sample ( struct sample * sample , const char * name , char * buf )
{
2012-12-30 18:11:01 -08:00
int in_deco ;
2011-08-30 16:23:47 -07:00
2011-09-01 11:22:05 -07:00
start_match ( " sample " , name , buf ) ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " pressure.sample " , pressure , & sample - > cylinderpressure ) )
2011-08-30 16:23:47 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " cylpress.sample " , pressure , & sample - > cylinderpressure ) )
2011-08-30 16:23:47 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " cylinderindex.sample " , get_cylinderindex , & sample - > sensor ) )
First step in cleaning up cylinder pressure sensor logic
This clarifies/changes the meaning of our "cylinderindex" entry in our
samples. It has been rather confused, because different dive computers
have done things differently, and the naming really hasn't helped.
There are two totally different - and independent - cylinder "indexes":
- the pressure sensor index, which indicates which cylinder the sensor
data is from.
- the "active cylinder" index, which indicates which cylinder we actually
breathe from.
These two values really are totally independent, and have nothing
what-so-ever to do with each other. The sensor index may well be fixed:
many dive computers only support a single pressure sensor (whether
wireless or wired), and the sensor index is thus always zero.
Other dive computers may support multiple pressure sensors, and the gas
switch event may - or may not - indicate that the sensor changed too. A
dive computer might give the sensor data for *all* cylinders it can read,
regardless of which one is the one we're actively breathing. In fact, some
dive computers might give sensor data for not just *your* cylinder, but
your buddies.
This patch renames "cylinderindex" in the samples as "sensor", making it
quite clear that it's about which sensor index the pressure data in the
sample is about.
The way we figure out which is the currently active gas is with an
explicit has change event. If a computer (like the Uemis Zurich) joins the
two concepts together, then a sensor change should also create a gas
switch event. This patch also changes the Uemis importer to do that.
Finally, it should be noted that the plot info works totally separately
from the sample data, and is about what we actually *display*, not about
the sample pressures etc. In the plot info, the "cylinderindex" does in
fact mean the currently active cylinder, and while it is initially set to
match the sensor information from the samples, we then walk the gas change
events and fix it up - and if the active cylinder differs from the sensor
cylinder, we clear the sensor data.
[Dirk Hohndel: this conflicted with some of my recent changes - I think
I merged things correctly...]
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-12-30 20:00:51 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " sensor.sample " , get_sensor , & sample - > sensor ) )
2011-10-19 10:06:11 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " depth.sample " , depth , & sample - > depth ) )
2011-08-30 16:23:47 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " temp.sample " , temperature , & sample - > temperature ) )
2011-09-01 16:41:10 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " temperature.sample " , temperature , & sample - > temperature ) )
2011-08-30 16:23:47 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " sampletime.sample " , sampletime , & sample - > time ) )
2011-08-30 16:23:47 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " time.sample " , sampletime , & sample - > time ) )
2011-08-30 16:23:47 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " ndl.sample " , sampletime , & sample - > ndl ) )
2012-12-01 13:02:30 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " in_deco.sample " , get_index , & in_deco ) ) {
2012-12-30 18:11:01 -08:00
sample - > in_deco = ( in_deco = = 1 ) ;
return ;
}
2013-11-02 12:00:16 -07:00
if ( MATCH ( " stoptime.sample " , sampletime , & sample - > stoptime ) )
2012-12-01 13:02:30 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " stopdepth.sample " , depth , & sample - > stopdepth ) )
2012-12-01 13:02:30 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " cns.sample " , get_index , & sample - > cns ) )
2012-12-11 13:40:07 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " po2.sample " , double_to_permil , & sample - > po2 ) )
2012-12-11 13:40:07 -08:00
return ;
2011-08-30 16:23:47 -07:00
2011-09-05 13:45:14 -07:00
switch ( import_source ) {
2011-09-05 14:29:08 -07:00
case DIVINGLOG :
2013-11-02 12:00:16 -07:00
if ( divinglog_fill_sample ( sample , name , buf ) )
2011-09-05 14:29:08 -07:00
return ;
break ;
2011-09-06 17:33:52 -07:00
case UDDF :
2013-11-02 12:00:16 -07:00
if ( uddf_fill_sample ( sample , name , buf ) )
2011-09-06 17:33:52 -07:00
return ;
break ;
2011-09-05 13:45:14 -07:00
default :
break ;
2011-09-02 11:32:48 -07:00
}
2011-09-01 11:22:05 -07:00
nonmatch ( " sample " , name , buf ) ;
2011-08-30 15:22:48 -07:00
}
2011-09-05 14:29:08 -07:00
static const char * country , * city ;
static void divinglog_place ( char * place , void * _location )
{
char * * location = _location ;
2013-03-03 17:53:43 -08:00
char buffer [ 1024 ] , * p ;
2011-09-05 14:29:08 -07:00
int len ;
len = snprintf ( buffer , sizeof ( buffer ) ,
" %s%s%s%s%s " ,
place ,
city ? " , " : " " ,
city ? city : " " ,
country ? " , " : " " ,
country ? country : " " ) ;
p = malloc ( len + 1 ) ;
memcpy ( p , buffer , len + 1 ) ;
* location = p ;
city = NULL ;
country = NULL ;
}
2013-11-02 12:00:16 -07:00
static int divinglog_dive_match ( struct dive * dive , const char * name , char * buf )
{
return MATCH ( " divedate " , divedate , & dive - > when ) | |
MATCH ( " entrytime " , divetime , & dive - > when ) | |
MATCH ( " divetime " , duration , & dive - > dc . duration ) | |
MATCH ( " depth " , depth , & dive - > dc . maxdepth ) | |
MATCH ( " depthavg " , depth , & dive - > dc . meandepth ) | |
MATCH ( " tanktype " , utf8_string , & dive - > cylinder [ 0 ] . type . description ) | |
MATCH ( " tanksize " , cylindersize , & dive - > cylinder [ 0 ] . type . size ) | |
MATCH ( " presw " , pressure , & dive - > cylinder [ 0 ] . type . workingpressure ) | |
MATCH ( " press " , pressure , & dive - > cylinder [ 0 ] . start ) | |
MATCH ( " prese " , pressure , & dive - > cylinder [ 0 ] . end ) | |
MATCH ( " comments " , utf8_string , & dive - > notes ) | |
MATCH ( " names.buddy " , utf8_string , & dive - > buddy ) | |
MATCH ( " name.country " , utf8_string , & country ) | |
MATCH ( " name.city " , utf8_string , & city ) | |
MATCH ( " name.place " , divinglog_place , & dive - > location ) | |
2011-09-05 14:29:08 -07:00
0 ;
}
2011-09-06 17:33:52 -07:00
/*
* Uddf specifies ISO 8601 time format .
*
* There are many variations on that . This handles the useful cases .
*/
static void uddf_datetime ( char * buffer , void * _when )
{
char c ;
int y , m , d , hh , mm , ss ;
2012-09-19 17:35:52 -07:00
timestamp_t * when = _when ;
2011-09-06 17:33:52 -07:00
struct tm tm = { 0 } ;
int i ;
i = sscanf ( buffer , " %d-%d-%d%c%d:%d:%d " , & y , & m , & d , & c , & hh , & mm , & ss ) ;
if ( i = = 7 )
goto success ;
ss = 0 ;
if ( i = = 6 )
goto success ;
i = sscanf ( buffer , " %04d%02d%02d%c%02d%02d%02d " , & y , & m , & d , & c , & hh , & mm , & ss ) ;
if ( i = = 7 )
goto success ;
ss = 0 ;
if ( i = = 6 )
goto success ;
bad_date :
printf ( " Bad date time %s \n " , buffer ) ;
return ;
success :
if ( c ! = ' T ' & & c ! = ' ' )
goto bad_date ;
tm . tm_year = y ;
tm . tm_mon = m - 1 ;
tm . tm_mday = d ;
tm . tm_hour = hh ;
tm . tm_min = mm ;
tm . tm_sec = ss ;
* when = utc_mktime ( & tm ) ;
}
2013-02-22 08:52:35 -08:00
# define uddf_datedata(name, offset) \
static void uddf_ # # name ( char * buffer , void * _when ) \
{ timestamp_t * when = _when ; \
cur_tm . tm_ # # name = atoi ( buffer ) + offset ; \
* when = utc_mktime ( & cur_tm ) ; }
uddf_datedata ( year , 0 )
uddf_datedata ( mon , - 1 )
uddf_datedata ( mday , 0 )
uddf_datedata ( hour , 0 )
uddf_datedata ( min , 0 )
2013-11-02 12:00:16 -07:00
static int uddf_dive_match ( struct dive * dive , const char * name , char * buf )
2011-09-06 17:33:52 -07:00
{
2013-11-02 12:00:16 -07:00
return MATCH ( " datetime " , uddf_datetime , & dive - > when ) | |
MATCH ( " diveduration " , duration , & dive - > dc . duration ) | |
MATCH ( " greatestdepth " , depth , & dive - > dc . maxdepth ) | |
MATCH ( " year.date " , uddf_year , & dive - > when ) | |
MATCH ( " month.date " , uddf_mon , & dive - > when ) | |
MATCH ( " day.date " , uddf_mday , & dive - > when ) | |
MATCH ( " hour.time " , uddf_hour , & dive - > when ) | |
MATCH ( " minute.time " , uddf_min , & dive - > when ) | |
2011-09-06 17:33:52 -07:00
0 ;
}
2012-12-05 09:59:52 -08:00
/*
* This parses " floating point " into micro - degrees .
* We don ' t do exponentials etc , if somebody does
* gps locations in that format , they are insane .
*/
static degrees_t parse_degrees ( char * buf , char * * end )
{
int sign = 1 , decimals = 6 , value = 0 ;
degrees_t ret ;
2013-10-05 00:29:09 -07:00
while ( isspace ( * buf ) )
2012-12-05 09:59:52 -08:00
buf + + ;
switch ( * buf ) {
case ' - ' :
sign = - 1 ;
/* fallthrough */
case ' + ' :
buf + + ;
}
while ( isdigit ( * buf ) ) {
value = 10 * value + * buf - ' 0 ' ;
buf + + ;
}
/* Get the first six decimals if they exist */
if ( * buf = = ' . ' )
buf + + ;
do {
value * = 10 ;
if ( isdigit ( * buf ) ) {
value + = * buf - ' 0 ' ;
buf + + ;
}
} while ( - - decimals ) ;
/* Rounding */
switch ( * buf ) {
case ' 5 ' . . . ' 9 ' :
value + + ;
}
while ( isdigit ( * buf ) )
buf + + ;
* end = buf ;
ret . udeg = value * sign ;
return ret ;
}
2013-01-23 11:53:42 -08:00
static void gps_lat ( char * buffer , void * _dive )
{
char * end ;
struct dive * dive = _dive ;
dive - > latitude = parse_degrees ( buffer , & end ) ;
}
static void gps_long ( char * buffer , void * _dive )
{
char * end ;
struct dive * dive = _dive ;
dive - > longitude = parse_degrees ( buffer , & end ) ;
}
2011-09-15 18:16:07 -07:00
static void gps_location ( char * buffer , void * _dive )
{
2012-11-18 07:55:41 -10:00
char * end ;
2011-09-15 18:16:07 -07:00
struct dive * dive = _dive ;
2012-12-05 09:59:52 -08:00
dive - > latitude = parse_degrees ( buffer , & end ) ;
dive - > longitude = parse_degrees ( end , & end ) ;
2011-09-15 18:16:07 -07:00
}
2011-08-30 15:22:48 -07:00
/* We're in the top-level dive xml. Try to convert whatever value to a dive value */
2012-11-23 16:05:38 -10:00
static void try_to_fill_dive ( struct dive * dive , const char * name , char * buf )
2011-08-30 15:22:48 -07:00
{
2011-09-01 11:22:05 -07:00
start_match ( " dive " , name , buf ) ;
2011-09-05 14:29:08 -07:00
switch ( import_source ) {
case DIVINGLOG :
2013-11-02 12:00:16 -07:00
if ( divinglog_dive_match ( dive , name , buf ) )
2011-09-05 14:29:08 -07:00
return ;
break ;
2011-09-06 17:33:52 -07:00
case UDDF :
2013-11-02 12:00:16 -07:00
if ( uddf_dive_match ( dive , name , buf ) )
2011-09-06 17:33:52 -07:00
return ;
break ;
2011-09-05 14:29:08 -07:00
default :
break ;
}
2013-11-02 12:00:16 -07:00
if ( MATCH ( " number " , get_index , & dive - > number ) )
2011-09-11 11:36:33 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " tags " , divetags , dive - > tag_list ) )
2013-04-09 17:54:36 +02:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " tripflag " , get_tripflag , & dive - > tripflag ) )
2012-08-21 22:04:24 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " date " , divedate , & dive - > when ) )
2011-08-30 15:22:48 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " time " , divetime , & dive - > when ) )
2011-08-30 15:22:48 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " datetime " , divedatetime , & dive - > when ) )
2011-08-30 16:59:03 -07:00
return ;
2013-01-23 10:25:31 -08:00
/*
* Legacy format note : per - dive depths and duration get saved
* in the first dive computer entry
*/
2013-11-02 12:00:16 -07:00
if ( match_dc_data_fields ( & dive - > dc , name , buf ) )
2012-11-12 20:57:49 +01:00
return ;
2013-01-23 10:25:31 -08:00
2013-11-02 12:00:16 -07:00
if ( MATCH ( " cylinderstartpressure " , pressure , & dive - > cylinder [ 0 ] . start ) )
2011-08-30 17:45:03 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " cylinderendpressure " , pressure , & dive - > cylinder [ 0 ] . end ) )
2011-08-30 17:45:03 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " gps " , gps_location , dive ) )
2011-09-15 18:16:07 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " Place " , gps_location , dive ) )
2013-02-25 09:20:58 +02:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " latitude " , gps_lat , dive ) )
2013-01-23 11:53:42 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " sitelat " , gps_lat , dive ) )
2013-01-28 21:11:01 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " lat " , gps_lat , dive ) )
2013-02-25 09:20:58 +02:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " longitude " , gps_long , dive ) )
2013-01-23 11:53:42 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " sitelon " , gps_long , dive ) )
2013-01-28 21:11:01 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " lon " , gps_long , dive ) )
2013-02-25 09:20:58 +02:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " location " , utf8_string , & dive - > location ) )
2011-09-01 19:56:04 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " name.dive " , utf8_string , & dive - > location ) )
2013-01-23 11:53:42 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " suit " , utf8_string , & dive - > suit ) )
2012-08-14 16:07:25 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " divesuit " , utf8_string , & dive - > suit ) )
2012-08-17 20:22:37 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " notes " , utf8_string , & dive - > notes ) )
2011-09-01 19:56:04 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " divemaster " , utf8_string , & dive - > divemaster ) )
2011-09-13 14:58:06 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " buddy " , utf8_string , & dive - > buddy ) )
2011-09-13 14:58:06 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " rating.dive " , get_rating , & dive - > rating ) )
2011-12-07 11:58:16 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " visibility.dive " , get_rating , & dive - > visibility ) )
2012-10-28 15:49:02 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " size.cylinder " , cylindersize , & dive - > cylinder [ cur_cylinder_index ] . type . size ) )
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " workpressure.cylinder " , pressure , & dive - > cylinder [ cur_cylinder_index ] . type . workingpressure ) )
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " description.cylinder " , utf8_string , & dive - > cylinder [ cur_cylinder_index ] . type . description ) )
2011-09-04 15:07:47 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " start.cylinder " , pressure , & dive - > cylinder [ cur_cylinder_index ] . start ) )
2011-09-05 09:12:54 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " end.cylinder " , pressure , & dive - > cylinder [ cur_cylinder_index ] . end ) )
2011-09-05 09:12:54 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " description.weightsystem " , utf8_string , & dive - > weightsystem [ cur_ws_index ] . description ) )
2011-12-23 19:41:16 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " weight.weightsystem " , weight , & dive - > weightsystem [ cur_ws_index ] . weight ) )
2011-12-23 19:41:16 -08:00
return ;
2012-03-23 21:07:53 -07:00
if ( MATCH ( " weight " , weight , & dive - > weightsystem [ cur_ws_index ] . weight ) )
2011-12-23 19:41:16 -08:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " o2 " , gasmix , & dive - > cylinder [ cur_cylinder_index ] . gasmix . o2 ) )
2011-09-03 20:31:18 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " o2percent " , gasmix , & dive - > cylinder [ cur_cylinder_index ] . gasmix . o2 ) )
2013-01-31 22:54:14 +01:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " n2 " , gasmix_nitrogen , & dive - > cylinder [ cur_cylinder_index ] . gasmix ) )
2011-09-03 20:31:18 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " he " , gasmix , & dive - > cylinder [ cur_cylinder_index ] . gasmix . he ) )
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " air.divetemperature " , temperature , & dive - > airtemp ) )
2013-02-14 09:44:18 -08:00
return ;
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
2011-09-01 11:22:05 -07:00
nonmatch ( " dive " , name , buf ) ;
2011-08-30 15:22:48 -07:00
}
2011-08-30 13:48:05 -07:00
2012-08-21 22:04:24 -07:00
/* We're in the top-level trip xml. Try to convert whatever value to a trip value */
2012-09-19 23:42:11 -04:00
static void try_to_fill_trip ( dive_trip_t * * dive_trip_p , const char * name , char * buf )
2012-08-21 22:04:24 -07:00
{
start_match ( " trip " , name , buf ) ;
2012-09-19 23:42:11 -04:00
dive_trip_t * dive_trip = * dive_trip_p ;
2012-08-21 22:04:24 -07:00
2013-11-02 12:00:16 -07:00
if ( MATCH ( " date " , divedate , & dive_trip - > when ) )
2012-08-29 17:24:15 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " time " , divetime , & dive_trip - > when ) )
2012-08-21 22:04:24 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " location " , utf8_string , & dive_trip - > location ) )
2012-08-21 22:04:24 -07:00
return ;
2013-11-02 12:00:16 -07:00
if ( MATCH ( " notes " , utf8_string , & dive_trip - > notes ) )
2012-08-21 22:04:24 -07:00
return ;
nonmatch ( " trip " , name , buf ) ;
}
2011-08-29 21:32:27 -07:00
/*
2012-09-15 18:32:15 -07:00
* While in some formats file boundaries are dive boundaries , in many
* others ( as for example in our native format ) there are
2011-08-29 21:32:27 -07:00
* multiple dives per file , so there can be other events too that
* trigger a " new dive " marker and you may get some nesting due
* to that . Just ignore nesting levels .
2012-09-15 18:32:15 -07:00
* On the flipside it is possible that we start an XML file that ends
* up having no dives in it at all - don ' t create a bogus empty dive
* for those . It ' s not entirely clear what is the minimum set of data
* to make a dive valid , but if it has no location , no date and no
* samples I ' m pretty sure it ' s useless .
2011-08-29 21:32:27 -07:00
*/
2013-10-05 00:29:09 -07:00
static bool is_dive ( void )
2012-09-15 18:32:15 -07:00
{
return ( cur_dive & &
2012-11-23 16:51:27 -10:00
( cur_dive - > location | | cur_dive - > when | | cur_dive - > dc . samples ) ) ;
2012-09-15 18:32:15 -07:00
}
2012-12-19 12:36:56 -08:00
static void reset_dc_info ( struct divecomputer * dc )
{
2012-12-30 18:11:01 -08:00
lastcns = lastpo2 = lastndl = laststoptime = laststopdepth = lastindeco = 0 ;
First step in cleaning up cylinder pressure sensor logic
This clarifies/changes the meaning of our "cylinderindex" entry in our
samples. It has been rather confused, because different dive computers
have done things differently, and the naming really hasn't helped.
There are two totally different - and independent - cylinder "indexes":
- the pressure sensor index, which indicates which cylinder the sensor
data is from.
- the "active cylinder" index, which indicates which cylinder we actually
breathe from.
These two values really are totally independent, and have nothing
what-so-ever to do with each other. The sensor index may well be fixed:
many dive computers only support a single pressure sensor (whether
wireless or wired), and the sensor index is thus always zero.
Other dive computers may support multiple pressure sensors, and the gas
switch event may - or may not - indicate that the sensor changed too. A
dive computer might give the sensor data for *all* cylinders it can read,
regardless of which one is the one we're actively breathing. In fact, some
dive computers might give sensor data for not just *your* cylinder, but
your buddies.
This patch renames "cylinderindex" in the samples as "sensor", making it
quite clear that it's about which sensor index the pressure data in the
sample is about.
The way we figure out which is the currently active gas is with an
explicit has change event. If a computer (like the Uemis Zurich) joins the
two concepts together, then a sensor change should also create a gas
switch event. This patch also changes the Uemis importer to do that.
Finally, it should be noted that the plot info works totally separately
from the sample data, and is about what we actually *display*, not about
the sample pressures etc. In the plot info, the "cylinderindex" does in
fact mean the currently active cylinder, and while it is initially set to
match the sensor information from the samples, we then walk the gas change
events and fix it up - and if the active cylinder differs from the sensor
cylinder, we clear the sensor data.
[Dirk Hohndel: this conflicted with some of my recent changes - I think
I merged things correctly...]
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-12-30 20:00:51 -08:00
lastsensor = lastcylinderindex = 0 ;
2012-12-19 12:36:56 -08:00
}
2012-12-26 13:47:54 -08:00
static void reset_dc_settings ( void )
{
free ( ( void * ) cur_settings . dc . model ) ;
free ( ( void * ) cur_settings . dc . nickname ) ;
Assemble the actual Suunto serial number
It turns out that the serial number returned by libdivecomputer isn't
really the serial number as interpreted by the vendor. Those tend to be
strings, but libdivecomputer gives us a 32bit number.
Some experimenting showed that for the Suunto devies tested the serial
number is encoded in that 32bit number:
It so happens that the Suunto serial number strings are strings that have
all numbers, but they aren't *one* number. They are four bytes
representing two numbers each, and the "23500027" string is actually the
four bytes 23 50 00 27 (0x17 0x32 0x00 0x1b). And libdivecomputer has
incorrectly parsed those four bytes as one number, not as the encoded
serial number string it is. So the value 389152795 is actually hex
0x1732001b, which is 0x17 0x32 0x00 0x1b, which is - 23 50 00 27.
This should be done by libdivecomputer, but hey, in the meantime this at
least shows the concept. And helps test the XML save/restore code.
It depends on the two patches that create the whole "device.c"
infrastructure, of course. With this, my dive file ends up having the
settings section look like this:
<divecomputerid model='Suunto Vyper Air' deviceid='d4629110'
serial='01201094' firmware='1.1.22'/>
<divecomputerid model='Suunto HelO2' deviceid='995dd566'
serial='23500027' firmware='1.0.4'/>
where the format of the firmware version is something I guessed at,
but it was the obvious choice (again, it's byte-based, I'm ignoring
the high byte that is zero for both of my Suuntos).
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-01-09 16:14:21 -08:00
free ( ( void * ) cur_settings . dc . serial_nr ) ;
free ( ( void * ) cur_settings . dc . firmware ) ;
2012-12-26 13:47:54 -08:00
cur_settings . dc . model = NULL ;
cur_settings . dc . nickname = NULL ;
Assemble the actual Suunto serial number
It turns out that the serial number returned by libdivecomputer isn't
really the serial number as interpreted by the vendor. Those tend to be
strings, but libdivecomputer gives us a 32bit number.
Some experimenting showed that for the Suunto devies tested the serial
number is encoded in that 32bit number:
It so happens that the Suunto serial number strings are strings that have
all numbers, but they aren't *one* number. They are four bytes
representing two numbers each, and the "23500027" string is actually the
four bytes 23 50 00 27 (0x17 0x32 0x00 0x1b). And libdivecomputer has
incorrectly parsed those four bytes as one number, not as the encoded
serial number string it is. So the value 389152795 is actually hex
0x1732001b, which is 0x17 0x32 0x00 0x1b, which is - 23 50 00 27.
This should be done by libdivecomputer, but hey, in the meantime this at
least shows the concept. And helps test the XML save/restore code.
It depends on the two patches that create the whole "device.c"
infrastructure, of course. With this, my dive file ends up having the
settings section look like this:
<divecomputerid model='Suunto Vyper Air' deviceid='d4629110'
serial='01201094' firmware='1.1.22'/>
<divecomputerid model='Suunto HelO2' deviceid='995dd566'
serial='23500027' firmware='1.0.4'/>
where the format of the firmware version is something I guessed at,
but it was the obvious choice (again, it's byte-based, I'm ignoring
the high byte that is zero for both of my Suuntos).
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-01-09 16:14:21 -08:00
cur_settings . dc . serial_nr = NULL ;
cur_settings . dc . firmware = NULL ;
2012-12-26 13:47:54 -08:00
cur_settings . dc . deviceid = 0 ;
}
2013-01-01 17:29:38 -08:00
static void settings_start ( void )
2012-12-26 13:47:54 -08:00
{
in_settings = TRUE ;
2013-01-01 17:29:38 -08:00
}
static void settings_end ( void )
{
in_settings = FALSE ;
}
static void dc_settings_start ( void )
{
2012-12-26 13:47:54 -08:00
reset_dc_settings ( ) ;
}
static void dc_settings_end ( void )
{
2013-06-17 15:58:26 -07:00
create_device_node ( cur_settings . dc . model , cur_settings . dc . deviceid , cur_settings . dc . serial_nr ,
cur_settings . dc . firmware , cur_settings . dc . nickname ) ;
2012-12-26 13:47:54 -08:00
reset_dc_settings ( ) ;
}
2011-08-29 21:32:27 -07:00
static void dive_start ( void )
{
2012-01-05 08:16:08 -08:00
if ( cur_dive )
2011-08-30 15:22:48 -07:00
return ;
2012-01-05 08:16:08 -08:00
cur_dive = alloc_dive ( ) ;
2012-12-19 12:36:56 -08:00
reset_dc_info ( & cur_dive - > dc ) ;
2012-01-05 08:16:08 -08:00
memset ( & cur_tm , 0 , sizeof ( cur_tm ) ) ;
New XML format for saving dives
This patch makes the trips nest, and it also fixes the fact that you never
saved the trip notes (you could edit it, but saving would throw it away).
I did *not* change the indentation of the dives, so the trip stuff shows
up the the beginning of the line, at the same level as the <dive> and
<dives> thing. I think it's fairly readable xml, though, and we haven't
really had proper "indentation shows nesting" anyway, since the top-level
"<dives>" thing also didn't indent stuff inside of it.
Anyway, the way I wrote it, it still parses your old "INTRIP" stuff etc,
so as far as I know, it should happily read the old-style XML too. At
least it seemed to work with your xml file that already had the old-style
one (I haven't committed my divetrips, exactly because I didn't like the
new format).
It always saves in the new style, though.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-09-30 12:36:18 -07:00
if ( cur_trip ) {
2012-11-10 19:51:03 +01:00
add_dive_to_trip ( cur_dive , cur_trip ) ;
New XML format for saving dives
This patch makes the trips nest, and it also fixes the fact that you never
saved the trip notes (you could edit it, but saving would throw it away).
I did *not* change the indentation of the dives, so the trip stuff shows
up the the beginning of the line, at the same level as the <dive> and
<dives> thing. I think it's fairly readable xml, though, and we haven't
really had proper "indentation shows nesting" anyway, since the top-level
"<dives>" thing also didn't indent stuff inside of it.
Anyway, the way I wrote it, it still parses your old "INTRIP" stuff etc,
so as far as I know, it should happily read the old-style XML too. At
least it seemed to work with your xml file that already had the old-style
one (I haven't committed my divetrips, exactly because I didn't like the
new format).
It always saves in the new style, though.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-09-30 12:36:18 -07:00
cur_dive - > tripflag = IN_TRIP ;
}
2011-08-29 21:32:27 -07:00
}
static void dive_end ( void )
{
New XML format for saving dives
This patch makes the trips nest, and it also fixes the fact that you never
saved the trip notes (you could edit it, but saving would throw it away).
I did *not* change the indentation of the dives, so the trip stuff shows
up the the beginning of the line, at the same level as the <dive> and
<dives> thing. I think it's fairly readable xml, though, and we haven't
really had proper "indentation shows nesting" anyway, since the top-level
"<dives>" thing also didn't indent stuff inside of it.
Anyway, the way I wrote it, it still parses your old "INTRIP" stuff etc,
so as far as I know, it should happily read the old-style XML too. At
least it seemed to work with your xml file that already had the old-style
one (I haven't committed my divetrips, exactly because I didn't like the
new format).
It always saves in the new style, though.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-09-30 12:36:18 -07:00
if ( ! cur_dive )
2011-08-30 13:48:05 -07:00
return ;
New XML format for saving dives
This patch makes the trips nest, and it also fixes the fact that you never
saved the trip notes (you could edit it, but saving would throw it away).
I did *not* change the indentation of the dives, so the trip stuff shows
up the the beginning of the line, at the same level as the <dive> and
<dives> thing. I think it's fairly readable xml, though, and we haven't
really had proper "indentation shows nesting" anyway, since the top-level
"<dives>" thing also didn't indent stuff inside of it.
Anyway, the way I wrote it, it still parses your old "INTRIP" stuff etc,
so as far as I know, it should happily read the old-style XML too. At
least it seemed to work with your xml file that already had the old-style
one (I haven't committed my divetrips, exactly because I didn't like the
new format).
It always saves in the new style, though.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-09-30 12:36:18 -07:00
if ( ! is_dive ( ) )
free ( cur_dive ) ;
else
2013-01-31 11:08:36 +11:00
record_dive_to_table ( cur_dive , target_table ) ;
2012-01-05 08:16:08 -08:00
cur_dive = NULL ;
2012-11-24 16:50:21 -10:00
cur_dc = NULL ;
2012-01-05 08:16:08 -08:00
cur_cylinder_index = 0 ;
2012-03-23 21:07:53 -07:00
cur_ws_index = 0 ;
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
}
2012-08-21 22:04:24 -07:00
static void trip_start ( void )
{
if ( cur_trip )
return ;
New XML format for saving dives
This patch makes the trips nest, and it also fixes the fact that you never
saved the trip notes (you could edit it, but saving would throw it away).
I did *not* change the indentation of the dives, so the trip stuff shows
up the the beginning of the line, at the same level as the <dive> and
<dives> thing. I think it's fairly readable xml, though, and we haven't
really had proper "indentation shows nesting" anyway, since the top-level
"<dives>" thing also didn't indent stuff inside of it.
Anyway, the way I wrote it, it still parses your old "INTRIP" stuff etc,
so as far as I know, it should happily read the old-style XML too. At
least it seemed to work with your xml file that already had the old-style
one (I haven't committed my divetrips, exactly because I didn't like the
new format).
It always saves in the new style, though.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-09-30 12:36:18 -07:00
dive_end ( ) ;
2012-09-19 23:42:11 -04:00
cur_trip = calloc ( sizeof ( dive_trip_t ) , 1 ) ;
2012-08-21 22:04:24 -07:00
memset ( & cur_tm , 0 , sizeof ( cur_tm ) ) ;
}
static void trip_end ( void )
{
if ( ! cur_trip )
return ;
2012-09-05 13:54:22 -07:00
insert_trip ( & cur_trip ) ;
2012-08-21 22:04:24 -07:00
cur_trip = NULL ;
}
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
static void event_start ( void )
{
2012-01-05 08:16:08 -08:00
memset ( & cur_event , 0 , sizeof ( cur_event ) ) ;
cur_event . active = 1 ;
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
}
static void event_end ( void )
{
2012-11-24 16:50:21 -10:00
struct divecomputer * dc = get_dc ( ) ;
2012-12-24 03:51:39 +02:00
if ( cur_event . name ) {
if ( strcmp ( cur_event . name , " surface " ) ! = 0 )
add_event ( dc , cur_event . time . seconds ,
cur_event . type , cur_event . flags ,
cur_event . value , cur_event . name ) ;
free ( ( void * ) cur_event . name ) ;
}
2012-01-05 08:16:08 -08:00
cur_event . active = 0 ;
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
}
2011-09-03 20:31:18 -07:00
static void cylinder_start ( void )
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
{
}
2011-09-03 20:31:18 -07:00
static void cylinder_end ( void )
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
{
2012-01-05 08:16:08 -08:00
cur_cylinder_index + + ;
2011-08-29 21:32:27 -07:00
}
2011-12-23 19:41:16 -08:00
static void ws_start ( void )
{
}
static void ws_end ( void )
{
2012-03-23 21:07:53 -07:00
cur_ws_index + + ;
2011-12-23 19:41:16 -08:00
}
2011-08-29 21:32:27 -07:00
static void sample_start ( void )
{
2012-11-24 16:50:21 -10:00
cur_sample = prepare_sample ( get_dc ( ) ) ;
2012-12-01 13:02:30 -08:00
cur_sample - > ndl . seconds = lastndl ;
2012-12-30 18:11:01 -08:00
cur_sample - > in_deco = lastindeco ;
2012-12-01 13:02:30 -08:00
cur_sample - > stoptime . seconds = laststoptime ;
cur_sample - > stopdepth . mm = laststopdepth ;
2012-12-07 20:08:29 -08:00
cur_sample - > cns = lastcns ;
cur_sample - > po2 = lastpo2 ;
First step in cleaning up cylinder pressure sensor logic
This clarifies/changes the meaning of our "cylinderindex" entry in our
samples. It has been rather confused, because different dive computers
have done things differently, and the naming really hasn't helped.
There are two totally different - and independent - cylinder "indexes":
- the pressure sensor index, which indicates which cylinder the sensor
data is from.
- the "active cylinder" index, which indicates which cylinder we actually
breathe from.
These two values really are totally independent, and have nothing
what-so-ever to do with each other. The sensor index may well be fixed:
many dive computers only support a single pressure sensor (whether
wireless or wired), and the sensor index is thus always zero.
Other dive computers may support multiple pressure sensors, and the gas
switch event may - or may not - indicate that the sensor changed too. A
dive computer might give the sensor data for *all* cylinders it can read,
regardless of which one is the one we're actively breathing. In fact, some
dive computers might give sensor data for not just *your* cylinder, but
your buddies.
This patch renames "cylinderindex" in the samples as "sensor", making it
quite clear that it's about which sensor index the pressure data in the
sample is about.
The way we figure out which is the currently active gas is with an
explicit has change event. If a computer (like the Uemis Zurich) joins the
two concepts together, then a sensor change should also create a gas
switch event. This patch also changes the Uemis importer to do that.
Finally, it should be noted that the plot info works totally separately
from the sample data, and is about what we actually *display*, not about
the sample pressures etc. In the plot info, the "cylinderindex" does in
fact mean the currently active cylinder, and while it is initially set to
match the sensor information from the samples, we then walk the gas change
events and fix it up - and if the active cylinder differs from the sensor
cylinder, we clear the sensor data.
[Dirk Hohndel: this conflicted with some of my recent changes - I think
I merged things correctly...]
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-12-30 20:00:51 -08:00
cur_sample - > sensor = lastsensor ;
2011-08-29 21:32:27 -07:00
}
static void sample_end ( void )
{
2012-01-05 08:16:08 -08:00
if ( ! cur_dive )
2011-08-30 13:48:05 -07:00
return ;
2011-08-31 14:36:53 -07:00
2012-11-24 16:50:21 -10:00
finish_sample ( get_dc ( ) ) ;
2012-12-01 13:02:30 -08:00
lastndl = cur_sample - > ndl . seconds ;
2012-12-30 18:11:01 -08:00
lastindeco = cur_sample - > in_deco ;
2012-12-01 13:02:30 -08:00
laststoptime = cur_sample - > stoptime . seconds ;
laststopdepth = cur_sample - > stopdepth . mm ;
2012-12-07 20:08:29 -08:00
lastcns = cur_sample - > cns ;
lastpo2 = cur_sample - > po2 ;
2012-01-05 08:16:08 -08:00
cur_sample = NULL ;
2011-08-29 21:32:27 -07:00
}
2012-11-24 16:50:21 -10:00
static void divecomputer_start ( void )
{
struct divecomputer * dc ;
/* Start from the previous dive computer */
dc = & cur_dive - > dc ;
while ( dc - > next )
dc = dc - > next ;
/* Did we already fill that in? */
2012-11-25 11:44:27 -08:00
if ( dc - > samples | | dc - > model | | dc - > when ) {
2012-11-24 16:50:21 -10:00
struct divecomputer * newdc = calloc ( 1 , sizeof ( * newdc ) ) ;
if ( newdc ) {
dc - > next = newdc ;
dc = newdc ;
}
}
/* .. this is the one we'll use */
cur_dc = dc ;
2012-12-19 12:36:56 -08:00
reset_dc_info ( dc ) ;
2012-11-24 16:50:21 -10:00
}
static void divecomputer_end ( void )
{
if ( ! cur_dc - > when )
cur_dc - > when = cur_dive - > when ;
cur_dc = NULL ;
}
2012-12-28 18:18:23 +02:00
static void entry ( const char * name , char * buf )
2011-08-29 21:32:27 -07:00
{
2012-12-26 13:47:54 -08:00
if ( in_settings ) {
try_to_fill_dc_settings ( name , buf ) ;
2013-01-01 17:29:38 -08:00
try_to_match_autogroup ( name , buf ) ;
2012-12-26 13:47:54 -08:00
return ;
}
2012-01-05 08:16:08 -08:00
if ( cur_event . active ) {
2011-09-22 18:02:54 -07:00
try_to_fill_event ( name , buf ) ;
return ;
}
2012-01-05 08:16:08 -08:00
if ( cur_sample ) {
try_to_fill_sample ( cur_sample , name , buf ) ;
2011-08-30 13:48:05 -07:00
return ;
}
2012-11-24 16:50:21 -10:00
if ( cur_dc ) {
try_to_fill_dc ( cur_dc , name , buf ) ;
return ;
}
2012-01-05 08:16:08 -08:00
if ( cur_dive ) {
2012-11-23 16:05:38 -10:00
try_to_fill_dive ( cur_dive , name , buf ) ;
2011-08-30 13:48:05 -07:00
return ;
}
New XML format for saving dives
This patch makes the trips nest, and it also fixes the fact that you never
saved the trip notes (you could edit it, but saving would throw it away).
I did *not* change the indentation of the dives, so the trip stuff shows
up the the beginning of the line, at the same level as the <dive> and
<dives> thing. I think it's fairly readable xml, though, and we haven't
really had proper "indentation shows nesting" anyway, since the top-level
"<dives>" thing also didn't indent stuff inside of it.
Anyway, the way I wrote it, it still parses your old "INTRIP" stuff etc,
so as far as I know, it should happily read the old-style XML too. At
least it seemed to work with your xml file that already had the old-style
one (I haven't committed my divetrips, exactly because I didn't like the
new format).
It always saves in the new style, though.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2012-09-30 12:36:18 -07:00
if ( cur_trip ) {
try_to_fill_trip ( & cur_trip , name , buf ) ;
return ;
}
2011-08-29 21:32:27 -07:00
}
2011-08-29 17:51:54 -07:00
static const char * nodename ( xmlNode * node , char * buf , int len )
2011-08-28 16:58:26 -07:00
{
2013-11-02 12:00:16 -07:00
int levels = 2 ;
char * p = buf ;
2011-08-30 14:38:39 -07:00
if ( ! node | | ! node - > name )
return " root " ;
2011-08-29 17:51:54 -07:00
2013-11-02 12:00:16 -07:00
if ( node - > parent & & ! strcmp ( node - > name , " text " ) )
node = node - > parent ;
/* Make sure it's always NUL-terminated */
p [ - - len ] = 0 ;
2011-08-29 17:51:54 -07:00
for ( ; ; ) {
const char * name = node - > name ;
2013-11-02 12:00:16 -07:00
char c ;
while ( ( c = * name + + ) ! = 0 ) {
/* Cheaper 'tolower()' for ASCII */
c = ( c > = ' A ' & & c < = ' Z ' ) ? c - ' A ' + ' a ' : c ;
* p + + = c ;
2011-08-29 17:51:54 -07:00
if ( ! - - len )
return buf ;
}
2013-11-02 12:00:16 -07:00
* p = 0 ;
2011-08-29 17:51:54 -07:00
node = node - > parent ;
if ( ! node | | ! node - > name )
return buf ;
2013-11-02 12:00:16 -07:00
* p + + = ' . ' ;
2011-08-29 17:51:54 -07:00
if ( ! - - len )
return buf ;
2013-11-02 12:00:16 -07:00
if ( ! - - levels )
return buf ;
2011-08-29 17:51:54 -07:00
}
}
2013-11-02 12:00:16 -07:00
# define MAXNAME 32
2011-08-29 17:51:54 -07:00
2011-08-29 21:32:27 -07:00
static void visit_one_node ( xmlNode * node )
2011-08-29 17:51:54 -07:00
{
2012-12-28 18:18:23 +02:00
char * content ;
2011-08-29 17:51:54 -07:00
char buffer [ MAXNAME ] ;
const char * name ;
content = node - > content ;
2012-12-28 18:18:23 +02:00
if ( ! content | | xmlIsBlankNode ( node ) )
2011-08-29 17:51:54 -07:00
return ;
name = nodename ( node , buffer , sizeof ( buffer ) ) ;
2011-08-28 16:58:26 -07:00
2012-12-28 18:18:23 +02:00
entry ( name , content ) ;
2011-08-28 16:58:26 -07:00
}
2011-09-01 11:22:05 -07:00
static void traverse ( xmlNode * root ) ;
static void traverse_properties ( xmlNode * node )
{
xmlAttr * p ;
for ( p = node - > properties ; p ; p = p - > next )
traverse ( p - > children ) ;
}
static void visit ( xmlNode * n )
{
visit_one_node ( n ) ;
traverse_properties ( n ) ;
traverse ( n - > children ) ;
}
2011-09-05 14:29:08 -07:00
static void DivingLog_importer ( void )
{
import_source = DIVINGLOG ;
/*
* Diving Log units are really strange .
*
* Temperatures are in C , except in samples ,
* when they are in Fahrenheit . Depths are in
2011-09-11 12:24:57 -07:00
* meters , an dpressure is in PSI in the samples ,
* but in bar when it comes to working pressure .
*
* Crazy f * % ^ morons .
2011-09-05 14:29:08 -07:00
*/
2013-01-10 17:26:10 -08:00
xml_parsing_units = SI_units ;
2011-09-05 14:29:08 -07:00
}
2011-09-06 17:01:28 -07:00
static void uddf_importer ( void )
{
import_source = UDDF ;
2013-01-10 17:26:10 -08:00
xml_parsing_units = SI_units ;
xml_parsing_units . pressure = PASCAL ;
xml_parsing_units . temperature = KELVIN ;
2011-09-06 17:01:28 -07:00
}
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
/*
* I ' m sure this could be done as some fancy DTD rules .
* It ' s just not worth the headache .
*/
static struct nesting {
const char * name ;
void ( * start ) ( void ) , ( * end ) ( void ) ;
} nesting [ ] = {
2012-12-26 13:47:54 -08:00
{ " divecomputerid " , dc_settings_start , dc_settings_end } ,
2013-01-01 17:29:38 -08:00
{ " settings " , settings_start , settings_end } ,
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
{ " dive " , dive_start , dive_end } ,
2011-09-05 14:29:08 -07:00
{ " Dive " , dive_start , dive_end } ,
2012-08-21 22:04:24 -07:00
{ " trip " , trip_start , trip_end } ,
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
{ " sample " , sample_start , sample_end } ,
2011-09-06 17:01:28 -07:00
{ " waypoint " , sample_start , sample_end } ,
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
{ " SAMPLE " , sample_start , sample_end } ,
2011-09-02 11:32:48 -07:00
{ " reading " , sample_start , sample_end } ,
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
{ " event " , event_start , event_end } ,
2013-02-22 08:52:35 -08:00
{ " mix " , cylinder_start , cylinder_end } ,
2011-09-03 20:31:18 -07:00
{ " gasmix " , cylinder_start , cylinder_end } ,
{ " cylinder " , cylinder_start , cylinder_end } ,
2011-12-23 19:41:16 -08:00
{ " weightsystem " , ws_start , ws_end } ,
2012-11-24 16:50:21 -10:00
{ " divecomputer " , divecomputer_start , divecomputer_end } ,
2011-09-05 14:29:08 -07:00
{ " P " , sample_start , sample_end } ,
/* Import type recognition */
{ " Divinglog " , DivingLog_importer } ,
2011-09-06 17:01:28 -07:00
{ " uddf " , uddf_importer } ,
2011-09-05 14:29:08 -07:00
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
{ NULL , }
} ;
2011-09-01 11:22:05 -07:00
static void traverse ( xmlNode * root )
2011-08-28 16:58:26 -07:00
{
xmlNode * n ;
2011-09-01 11:22:05 -07:00
for ( n = root ; n ; n = n - > next ) {
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
struct nesting * rule = nesting ;
2011-08-29 21:32:27 -07:00
parse-xml: allow XML nodes with empty tag names
They happen for CDATA content, where libxml2 turns the CDATA fields into
a child of the parent entry, but without a name.
Now, of course, any sane person would just want to use the CDATA as the
string value of the parent itself, but libxml2 probably does this
insanity for a reason. And the reason is probably that some misguided
people want to *write* XML using libxml2, and then the stupid child node
actually acts as a "now I want you to write this data as CDATA".
Whatever the reason, let's just ignore it. We will just traverse such a
nameless child and be happy, and we'll give the nameless child the name
of the parent. Our XML node matching logic will then never see this
insane nameless child at all, and doesn't have to care.
Our whole XML parsing rule-of-thumb is to take the whole "be strict in
what you output, but generous in what you accept" to its logical
conclusion. Because we will literally accept almost anything, in any
format. You can mix tags or attributes wildly, and youc an use CDATA or
not as you see fit. We just don't care.
We're the honeybadger of the divelog world.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-01-28 21:12:32 -08:00
if ( ! n - > name ) {
visit ( n ) ;
continue ;
}
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
do {
if ( ! strcmp ( rule - > name , n - > name ) )
break ;
rule + + ;
} while ( rule - > name ) ;
2011-08-29 21:32:27 -07:00
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
if ( rule - > start )
rule - > start ( ) ;
2011-09-01 11:22:05 -07:00
visit ( n ) ;
Start parsing gas mixes
The suunto xml is just completely crazy. What's the helium percentage
companion to "o2pct"? Would it be "hepct"? No. It's "hepct_0".
Ok, so they didn't number the first o2pct, which could be seen as sane:
that's the only mix value that should always exist. And they clearly
started their indexing with 0. So with multiple mixes, you'd then
expect "o2pct_1" and "hepct_1", right?
Wrong! Because XML people are crazy, the second O2 mix percentage is
obviously "o2pct_2". So the O2 percentages are one-based, with an
implicit one. But the He percentages are zero-based with an explicit
zero. So the second mix is "o2pct_2" and "hepct_1".
I'd like to ask what drugs Suunto people are on, but hey, it's a Finnish
company. No need to ask. Vodka explains everything. LOTS AND LOTS OF
VODKA.
In comparison, the libdivecomputer output is nice and sane, and uses a
'gasmix' node. Of course, now we have so many different XML nesting
nodes to check that I just made it an array of different noces. That
also allows me to mark the suunto case, so that we only do the "check
for crazy alcoholic xml entries" when it's a suunto file.
The "type of file" thing is probably a good idea for deciding on default
units too. Some day.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-01 13:32:52 -07:00
if ( rule - > end )
rule - > end ( ) ;
2011-08-28 16:58:26 -07:00
}
}
2011-09-02 11:32:48 -07:00
/* Per-file reset */
static void reset_all ( void )
{
/*
* We reset the units for each file . You ' d think it was
* a per - dive property , but I ' m not going to trust people
* to do per - dive setup . If the xml does have per - dive
* data within one file , we might have to reset it per
* dive for that format .
*/
2013-01-10 17:26:10 -08:00
xml_parsing_units = SI_units ;
2011-09-05 13:45:14 -07:00
import_source = UNKNOWN ;
2011-09-02 11:32:48 -07:00
}
2013-03-15 19:02:14 +02:00
/* divelog.de sends us xml files that claim to be iso-8859-1
* but once we decode the HTML encoded characters they turn
* into UTF - 8 instead . So skip the incorrect encoding
* declaration and decode the HTML encoded characters */
const char * preprocess_divelog_de ( const char * buffer )
{
char * ret = strstr ( buffer , " <DIVELOGSDATA> " ) ;
if ( ret ) {
xmlParserCtxtPtr ctx ;
char buf [ ] = " " ;
2013-03-26 21:59:22 +02:00
int i ;
for ( i = 0 ; i < strlen ( ret ) ; + + i )
if ( ! isascii ( ret [ i ] ) )
return buffer ;
2013-03-15 19:02:14 +02:00
ctx = xmlCreateMemoryParserCtxt ( buf , sizeof ( buf ) ) ;
ret = xmlStringLenDecodeEntities ( ctx , ret , strlen ( ret ) , XML_SUBSTITUTE_REF , 0 , 0 , 0 ) ;
return ret ;
}
return buffer ;
}
2013-01-31 11:08:36 +11:00
void parse_xml_buffer ( const char * url , const char * buffer , int size ,
2013-10-16 22:05:19 +03:00
struct dive_table * table , const char * * params , char * * error )
2011-08-28 16:58:26 -07:00
{
xmlDoc * doc ;
2013-03-15 19:02:14 +02:00
const char * res = preprocess_divelog_de ( buffer ) ;
2011-08-28 16:58:26 -07:00
2013-01-31 11:08:36 +11:00
target_table = table ;
2013-03-15 19:02:14 +02:00
doc = xmlReadMemory ( res , strlen ( res ) , url , NULL , 0 ) ;
2013-03-17 07:12:23 +02:00
if ( res ! = buffer )
free ( ( char * ) res ) ;
2011-08-28 16:58:26 -07:00
if ( ! doc ) {
2013-10-09 22:48:35 -07:00
fprintf ( stderr , translate ( " gettextFromC " , " Failed to parse '%s'. \n " ) , url ) ;
parser_error ( error , translate ( " gettextFromC " , " Failed to parse '%s' " ) , url ) ;
2011-08-28 16:58:26 -07:00
return ;
}
2011-09-02 11:32:48 -07:00
reset_all ( ) ;
2011-08-29 21:32:27 -07:00
dive_start ( ) ;
2013-10-16 22:05:19 +03:00
doc = test_xslt_transforms ( doc , params , error ) ;
2011-08-29 21:32:27 -07:00
traverse ( xmlDocGetRootElement ( doc ) ) ;
dive_end ( ) ;
2011-08-28 16:58:26 -07:00
xmlFreeDoc ( doc ) ;
}
2013-03-05 07:10:39 +02:00
extern int dm4_events ( void * handle , int columns , char * * data , char * * column )
{
event_start ( ) ;
if ( data [ 1 ] )
cur_event . time . seconds = atoi ( data [ 1 ] ) ;
if ( data [ 2 ] ) {
switch ( atoi ( data [ 2 ] ) ) {
case 1 :
/* 1 Mandatory Safety Stop */
cur_event . name = strdup ( " safety stop (mandatory) " ) ;
break ;
case 3 :
/* 3 Deco */
/* What is Subsurface's term for going to
* deco ? */
cur_event . name = strdup ( " deco " ) ;
break ;
case 4 :
/* 4 Ascent warning */
cur_event . name = strdup ( " ascent " ) ;
break ;
case 5 :
/* 5 Ceiling broken */
cur_event . name = strdup ( " violation " ) ;
break ;
2013-03-07 07:23:40 +02:00
case 6 :
/* 6 Mandatory safety stop ceiling error */
cur_event . name = strdup ( " violation " ) ;
break ;
2013-07-26 20:51:27 +02:00
case 7 :
/* 7 Below deco floor */
cur_event . name = strdup ( " below floor " ) ;
break ;
2013-03-07 07:23:40 +02:00
case 8 :
/* 8 Dive time alarm */
cur_event . name = strdup ( " divetime " ) ;
break ;
case 9 :
/* 9 Depth alarm */
cur_event . name = strdup ( " maxdepth " ) ;
break ;
2013-03-05 07:10:39 +02:00
case 10 :
/* 10 OLF 80% */
case 11 :
/* 11 OLF 100% */
cur_event . name = strdup ( " OLF " ) ;
break ;
case 12 :
/* 12 High ppO2 */
cur_event . name = strdup ( " PO2 " ) ;
break ;
2013-07-26 20:51:27 +02:00
case 13 :
/* 13 Air time */
cur_event . name = strdup ( " airtime " ) ;
break ;
2013-03-05 07:10:39 +02:00
case 18 :
/* 18 Ceiling error */
cur_event . name = strdup ( " ceiling " ) ;
break ;
case 19 :
/* 19 Surfaced */
cur_event . name = strdup ( " surface " ) ;
break ;
2013-03-07 07:23:40 +02:00
case 257 :
/* 257 Dive active */
/* This seems to be given after surface
* when descending again . Ignoring it . */
break ;
2013-03-05 07:10:39 +02:00
case 258 :
/* 258 Bookmark */
2013-03-18 18:23:10 +02:00
if ( data [ 3 ] ) {
cur_event . name = strdup ( " heading " ) ;
2013-03-07 07:23:40 +02:00
cur_event . value = atoi ( data [ 3 ] ) ;
2013-03-18 18:23:10 +02:00
} else {
cur_event . name = strdup ( " bookmark " ) ;
}
2013-03-05 07:10:39 +02:00
break ;
default :
cur_event . name = strdup ( " unknown " ) ;
2013-03-07 07:23:40 +02:00
cur_event . value = atoi ( data [ 2 ] ) ;
2013-03-05 07:10:39 +02:00
break ;
}
}
event_end ( ) ;
return 0 ;
}
extern int dm4_dive ( void * param , int columns , char * * data , char * * column )
{
int i , interval , retval = 0 ;
sqlite3 * handle = ( sqlite3 * ) param ;
float * profileBlob ;
unsigned char * tempBlob ;
int * pressureBlob ;
char * err = NULL ;
char get_events_template [ ] = " select * from Mark where DiveId = %d " ;
char get_events [ 64 ] ;
dive_start ( ) ;
cur_dive - > number = atoi ( data [ 0 ] ) ;
2013-05-15 08:07:41 +03:00
cur_dive - > when = ( time_t ) ( atol ( data [ 1 ] ) ) ;
2013-03-05 07:10:39 +02:00
if ( data [ 2 ] )
utf8_string ( data [ 2 ] , & cur_dive - > notes ) ;
/*
* DM4 stores Duration and DiveTime . It looks like DiveTime is
* 10 to 60 seconds shorter than Duration . However , I have no
* idea what is the difference and which one should be used .
* Duration = data [ 3 ]
* DiveTime = data [ 15 ]
*/
2013-05-14 21:34:54 -04:00
if ( data [ 3 ] )
cur_dive - > duration . seconds = atoi ( data [ 3 ] ) ;
2013-03-05 07:10:39 +02:00
if ( data [ 15 ] )
2013-05-14 21:34:54 -04:00
cur_dive - > dc . duration . seconds = atoi ( data [ 15 ] ) ;
2013-03-05 07:10:39 +02:00
/*
* TODO : the deviceid hash should be calculated here .
*/
settings_start ( ) ;
dc_settings_start ( ) ;
if ( data [ 4 ] )
utf8_string ( data [ 4 ] , & cur_settings . dc . serial_nr ) ;
if ( data [ 5 ] )
utf8_string ( data [ 5 ] , & cur_settings . dc . model ) ;
cur_settings . dc . deviceid = 0xffffffff ;
dc_settings_end ( ) ;
settings_end ( ) ;
if ( data [ 6 ] )
2013-05-14 21:34:54 -04:00
cur_dive - > dc . maxdepth . mm = atof ( data [ 6 ] ) * 1000 ;
2013-03-05 07:10:39 +02:00
if ( data [ 8 ] )
2013-05-14 21:34:54 -04:00
cur_dive - > dc . airtemp . mkelvin = ( atoi ( data [ 8 ] ) + 273.15 ) * 1000 ;
2013-03-05 07:10:39 +02:00
if ( data [ 9 ] )
2013-05-14 21:34:54 -04:00
cur_dive - > dc . watertemp . mkelvin = ( atoi ( data [ 9 ] ) + 273.15 ) * 1000 ;
2013-03-05 07:10:39 +02:00
/*
* TODO : handle multiple cylinders
*/
cylinder_start ( ) ;
2013-03-08 21:44:10 +02:00
if ( data [ 22 ] & & atoi ( data [ 22 ] ) > 0 )
cur_dive - > cylinder [ cur_cylinder_index ] . start . mbar = atoi ( data [ 22 ] ) ;
else if ( data [ 10 ] & & atoi ( data [ 10 ] ) > 0 )
cur_dive - > cylinder [ cur_cylinder_index ] . start . mbar = atoi ( data [ 10 ] ) ;
if ( data [ 23 ] & & atoi ( data [ 23 ] ) > 0 )
cur_dive - > cylinder [ cur_cylinder_index ] . end . mbar = ( atoi ( data [ 23 ] ) ) ;
if ( data [ 11 ] & & atoi ( data [ 11 ] ) > 0 )
2013-03-05 07:10:39 +02:00
cur_dive - > cylinder [ cur_cylinder_index ] . end . mbar = ( atoi ( data [ 11 ] ) ) ;
if ( data [ 12 ] )
cur_dive - > cylinder [ cur_cylinder_index ] . type . size . mliter = ( atof ( data [ 12 ] ) ) * 1000 ;
if ( data [ 13 ] )
cur_dive - > cylinder [ cur_cylinder_index ] . type . workingpressure . mbar = ( atoi ( data [ 13 ] ) ) ;
if ( data [ 20 ] )
cur_dive - > cylinder [ cur_cylinder_index ] . gasmix . o2 . permille = atoi ( data [ 20 ] ) * 10 ;
if ( data [ 21 ] )
cur_dive - > cylinder [ cur_cylinder_index ] . gasmix . he . permille = atoi ( data [ 21 ] ) * 10 ;
cylinder_end ( ) ;
if ( data [ 14 ] )
2013-05-14 21:34:54 -04:00
cur_dive - > dc . surface_pressure . mbar = ( atoi ( data [ 14 ] ) * 1000 ) ;
2013-03-05 07:10:39 +02:00
interval = data [ 16 ] ? atoi ( data [ 16 ] ) : 0 ;
profileBlob = ( float * ) data [ 17 ] ;
tempBlob = ( unsigned char * ) data [ 18 ] ;
pressureBlob = ( int * ) data [ 19 ] ;
for ( i = 0 ; interval & & i * interval < cur_dive - > duration . seconds ; i + + ) {
sample_start ( ) ;
cur_sample - > time . seconds = i * interval ;
if ( profileBlob )
cur_sample - > depth . mm = profileBlob [ i ] * 1000 ;
else
2013-05-14 21:34:54 -04:00
cur_sample - > depth . mm = cur_dive - > dc . maxdepth . mm ;
2013-03-05 07:10:39 +02:00
2013-03-07 06:40:28 +02:00
if ( tempBlob )
2013-03-05 07:10:39 +02:00
cur_sample - > temperature . mkelvin = ( tempBlob [ i ] + 273.15 ) * 1000 ;
2013-03-07 06:40:28 +02:00
if ( data [ 19 ] & & data [ 19 ] [ 0 ] )
2013-03-05 07:10:39 +02:00
cur_sample - > cylinderpressure . mbar = pressureBlob [ i ] ;
sample_end ( ) ;
}
snprintf ( get_events , sizeof ( get_events ) - 1 , get_events_template , cur_dive - > number ) ;
retval = sqlite3_exec ( handle , get_events , & dm4_events , 0 , & err ) ;
if ( retval ! = SQLITE_OK ) {
2013-10-20 11:56:38 +03:00
fprintf ( stderr , " %s " , translate ( " gettextFromC " , " Database query get_events failed. \n " ) ) ;
2013-03-05 07:10:39 +02:00
return 1 ;
}
dive_end ( ) ;
/*
for ( i = 0 ; i < columns ; + + i ) {
fprintf ( stderr , " %s \t " , column [ i ] ) ;
}
fprintf ( stderr , " \n " ) ;
for ( i = 0 ; i < columns ; + + i ) {
fprintf ( stderr , " %s \t " , data [ i ] ) ;
}
fprintf ( stderr , " \n " ) ;
//exit(0);
*/
return SQLITE_OK ;
}
int parse_dm4_buffer ( const char * url , const char * buffer , int size ,
2013-05-21 23:13:45 -07:00
struct dive_table * table , char * * error )
2013-03-05 07:10:39 +02:00
{
int retval ;
char * err = NULL ;
sqlite3 * handle ;
target_table = table ;
2013-05-09 20:57:39 +03:00
/* StartTime is converted from Suunto's nano seconds to standard
* time . We also need epoch , not seconds since year 1. */
2013-05-15 08:07:41 +03:00
char get_dives [ ] = " select D.DiveId,StartTime/10000000-62135596800,Note,Duration,SourceSerialNumber,Source,MaxDepth,SampleInterval,StartTemperature,BottomTemperature,D.StartPressure,D.EndPressure,Size,CylinderWorkPressure,SurfacePressure,DiveTime,SampleInterval,ProfileBlob,TemperatureBlob,PressureBlob,Oxygen,Helium,MIX.StartPressure,MIX.EndPressure FROM Dive AS D JOIN DiveMixture AS MIX ON D.DiveId=MIX.DiveId " ;
2013-03-05 07:10:39 +02:00
retval = sqlite3_open ( url , & handle ) ;
if ( retval ) {
2013-10-09 22:48:35 -07:00
fprintf ( stderr , translate ( " gettextFromC " , " Database connection failed '%s'. \n " ) , url ) ;
2013-03-05 07:10:39 +02:00
return 1 ;
}
retval = sqlite3_exec ( handle , get_dives , & dm4_dive , handle , & err ) ;
if ( retval ! = SQLITE_OK ) {
2013-10-09 22:48:35 -07:00
fprintf ( stderr , translate ( " gettextFromC " , " Database query failed '%s'. \n " ) , url ) ;
2013-03-05 07:10:39 +02:00
return 1 ;
}
sqlite3_close ( handle ) ;
return 0 ;
}
2011-08-30 18:40:25 -07:00
void parse_xml_init ( void )
2011-08-30 16:28:59 -07:00
{
2011-08-28 16:58:26 -07:00
LIBXML_TEST_VERSION
}
2011-11-05 12:39:17 +02:00
2012-09-18 18:33:55 +03:00
void parse_xml_exit ( void )
{
xmlCleanupParser ( ) ;
}
2011-11-05 13:51:37 -07:00
static xsltStylesheetPtr try_get_stylesheet ( const char * path , int len , const char * name )
{
xsltStylesheetPtr ret ;
int namelen = strlen ( name ) ;
char * filename = malloc ( len + 1 + namelen + 1 ) ;
if ( ! filename )
return NULL ;
memcpy ( filename , path , len ) ;
2013-10-05 00:29:09 -07:00
# ifdef WIN32
filename [ len ] = ' \\ ' ;
# else
filename [ len ] = ' / ' ;
# endif
2011-11-05 13:51:37 -07:00
memcpy ( filename + len + 1 , name , namelen + 1 ) ;
ret = NULL ;
if ( ! access ( filename , R_OK ) )
ret = xsltParseStylesheetFile ( filename ) ;
free ( filename ) ;
return ret ;
}
2013-03-10 10:24:49 +02:00
xsltStylesheetPtr get_stylesheet ( const char * name )
2011-11-05 13:51:37 -07:00
{
2012-05-12 12:53:41 -07:00
const char * path , * next ;
path = getenv ( " SUBSURFACE_XSLT_PATH " ) ;
if ( ! path )
path = xslt_path ;
2011-11-05 13:51:37 -07:00
do {
int len ;
xsltStylesheetPtr ret ;
next = strchr ( path , ' : ' ) ;
len = strlen ( path ) ;
if ( next ) {
len = next - path ;
next + + ;
}
ret = try_get_stylesheet ( path , len , name ) ;
if ( ret )
return ret ;
} while ( ( path = next ) ! = NULL ) ;
return NULL ;
}
2011-11-16 07:12:43 +02:00
static struct xslt_files {
const char * root ;
const char * file ;
} xslt_files [ ] = {
2011-11-16 07:12:44 +02:00
{ " SUUNTO " , " SuuntoSDM.xslt " } ,
2013-04-22 07:02:48 +03:00
{ " Dive " , " SuuntoDM4.xslt " } ,
2011-11-16 07:12:43 +02:00
{ " JDiveLog " , " jdivelog2subsurface.xslt " } ,
2013-02-06 21:31:40 +02:00
{ " dives " , " MacDive.xslt " } ,
2013-02-20 22:15:42 +05:30
{ " DIVELOGSDATA " , " divelogs.xslt " } ,
2013-02-24 18:13:26 +02:00
{ " uddf " , " uddf.xslt " } ,
2013-06-07 17:45:56 +03:00
{ " UDDF " , " uddf.xslt " } ,
2013-02-26 20:13:28 +02:00
{ " profile " , " udcf.xslt " } ,
Test dives
On Tue, Feb 26, 2013 at 6:19 PM, Dirk Hohndel <dirk@hohndel.org> wrote:
> Miika Turkia <miika.turkia@gmail.com> writes:
>
>> On Tue, Feb 26, 2013 at 5:54 PM, Dirk Hohndel <dirk@hohndel.org> wrote:
>>> Miika Turkia <miika.turkia@gmail.com> writes:
>>>
>>>> On Tue, Feb 26, 2013 at 6:45 AM, Dirk Hohndel <dirk@hohndel.org> wrote:
>>>>> I added a few test dives exported from other software (Dm3, DiveLog 5.08,
>>>>> JDiveLog 10.2 from Mac - last one doesn't parse because of encoding... Also
>>>>> a composed XML zip file from DiveLog isn't supported, yet)
>>>>> Find them under dives
>>>>
>>>> Now this gets interesting. Is there ANY logic in the units in the
>>>> dives/TestDiveDiveLog5.08.xml? It seems that some of the temperatures
>>>> are in C (Airtemp and Watertemp) and some in F (samples). Otherwise I
>>>> would guess we are talking metric here, but I do not see any specs
>>>> (another log I have seen from DivingLog had all the units in metric,
>>>> as far as I could guess). BTW is the Weight in kg or lb?
>>>
>>> I have purchased DivingLog and should be able to create any combination
>>> of data for the test file that we could possibly want. I'll do a set in
>>> a moment that describe what SHOULD be there in their notes, maybe that
>>> will clear things up.
>>>
>>> That said, I really want to release 3.0.1 in the next couple of hours,
>>> so this may have to wait for 3.0.2 (if we end up needing that) or 3.1.
>>>
>>>> The divelogs.de UDCF format looks like it shouldn't take long to write
>>>> support for. I'll look into it this evening.
>>>
>>> It is evening for you, right? No pressure, just making sure I understand
>>> what may be coming in in patches in the next hour or two
>>
>> I currently have one version of the DivingLog XSLT. So a bit of
>> verification and that could possibly be used as is. However, this
>> could use a bit more testing than a new support to make sure things
>> are not going to be any worse than they currently are.
>>
>> I have not started with the UDCF yet, but that could be reasonably
>> fast to implement. However, no guarantees. (And yes, it is evening for
>> me)
>
> I can hold off 3.0.1 a couple hours longer if that is a realistic thing
> to do. I don't see the UDCF as that important since we have a different
> format from them that we support. So I think the best possible DivingLog
> support would be my preference.
>
> I'll add a few more exported dives from DivingLog next (and fix the
> naming of the existing ones).
In that case, here is the DivingLog XSLT if anyone can give it a test.
miika
From 4a62058f4f6fd4780f04bce6e1fe45e20abcf33f Mon Sep 17 00:00:00 2001
From: Miika Turkia <miika.turkia@gmail.com>
Date: Tue, 26 Feb 2013 17:46:53 +0200
Subject: [PATCH] XSLT for DivingLog
Signed-off-by: Miika Turkia <miika.turkia@gmail.com>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-02-26 18:31:22 +02:00
{ " Divinglog " , " DivingLog.xslt " } ,
2013-09-29 15:44:38 +03:00
{ " csv " , " csv2xml.xslt " } ,
2011-11-16 07:12:43 +02:00
{ NULL , }
} ;
2013-10-16 22:05:19 +03:00
static xmlDoc * test_xslt_transforms ( xmlDoc * doc , const char * * params , char * * error )
2011-11-05 12:39:17 +02:00
{
2011-11-16 07:12:43 +02:00
struct xslt_files * info = xslt_files ;
2011-11-05 12:39:17 +02:00
xmlDoc * transformed ;
xsltStylesheetPtr xslt = NULL ;
xmlNode * root_element = xmlDocGetRootElement ( doc ) ;
2013-02-07 21:20:02 +02:00
char * attribute ;
2011-11-16 07:12:43 +02:00
while ( ( info - > root ) & & ( strcasecmp ( root_element - > name , info - > root ) ! = 0 ) ) {
info + + ;
}
if ( info - > root ) {
2013-02-07 21:20:02 +02:00
attribute = xmlGetProp ( xmlFirstElementChild ( root_element ) , " name " ) ;
2013-02-21 03:07:22 +02:00
if ( attribute ) {
if ( strcasecmp ( attribute , " subsurface " ) = = 0 ) {
free ( ( void * ) attribute ) ;
return doc ;
}
free ( ( void * ) attribute ) ;
}
2013-02-07 21:20:02 +02:00
2011-11-05 12:39:17 +02:00
xmlSubstituteEntitiesDefault ( 1 ) ;
2011-11-16 07:12:43 +02:00
xslt = get_stylesheet ( info - > file ) ;
2013-02-22 11:20:46 -08:00
if ( xslt = = NULL ) {
2013-10-09 22:48:35 -07:00
parser_error ( error , translate ( " gettextFromC " , " Can't open stylesheet (%s)/%s " ) , xslt_path , info - > file ) ;
2011-11-05 12:39:17 +02:00
return doc ;
2013-02-22 11:20:46 -08:00
}
2013-10-16 22:05:19 +03:00
transformed = xsltApplyStylesheet ( xslt , doc , params ) ;
2011-11-05 12:39:17 +02:00
xmlFreeDoc ( doc ) ;
xsltFreeStylesheet ( xslt ) ;
2013-10-16 22:05:19 +03:00
2011-11-05 12:39:17 +02:00
return transformed ;
}
return doc ;
}