]> git.tdb.fi Git - ext/subsurface.git/commitdiff
Start parsing gas mixes
authorLinus Torvalds <torvalds@linux-foundation.org>
Thu, 1 Sep 2011 20:32:52 +0000 (13:32 -0700)
committerLinus Torvalds <torvalds@linux-foundation.org>
Thu, 1 Sep 2011 20:32:52 +0000 (13:32 -0700)
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>
dive.h
parse-xml.c

diff --git a/dive.h b/dive.h
index 23e2f544c0a959158e161e53953fa6893c029d8a..0bac6b2b9eeb580727c2c7cddbf6efc71f19e89f 100644 (file)
--- a/dive.h
+++ b/dive.h
@@ -67,7 +67,7 @@ typedef struct {
 typedef struct {
        fraction_t o2;
        fraction_t n2;
-       fraction_t he2;
+       fraction_t he;
 } gasmix_t;
 
 typedef struct {
@@ -100,6 +100,8 @@ struct sample {
        int tankindex;
 };
 
+#define MAX_MIXES (4)
+
 struct dive {
        const char *name;
        time_t when;
@@ -108,6 +110,7 @@ struct dive {
        depth_t visibility;
        temperature_t airtemp, watertemp;
        pressure_t beginning_pressure, end_pressure;
+       gasmix_t gasmix[MAX_MIXES];
        int samples;
        struct sample sample[];
 };
index 95625d12f550cf34854c9949ca2496af08b39ea9..892e0fb2014fe8abe99b5e7bbc5fd3e7565dc076 100644 (file)
@@ -69,6 +69,8 @@ static int alloc_samples;
 static struct dive *dive;
 static struct sample *sample;
 static struct tm tm;
+static int suunto;
+static int event_index, gasmix_index;
 
 static time_t utc_mktime(struct tm *tm)
 {
@@ -300,6 +302,35 @@ static void duration(char *buffer, void *_time)
        sampletime(buffer, _time);
 }
 
+static void percent(char *buffer, void *_fraction)
+{
+       fraction_t *fraction = _fraction;
+       union int_or_float val;
+
+       switch (integer_or_float(buffer, &val)) {
+       /* C or F? Who knows? Let's default to Celsius */
+       case INTEGER:
+               val.fp = val.i;
+               /* Fallthrough */
+       case FLOAT:
+               if (val.fp <= 100.0)
+                       fraction->permille = val.fp * 10 + 0.5;
+               break;
+
+       default:
+               printf("Strange percentage reading %s\n", buffer);
+               break;
+       }
+       free(buffer);
+}
+
+static void gasmix(char *buffer, void *_fraction)
+{
+       if (gasmix_index < MAX_MIXES)
+               percent(buffer, _fraction);
+}
+
+
 #define MATCH(pattern, fn, dest) \
        match(pattern, strlen(pattern), name, len, fn, buf, dest)
 
@@ -325,6 +356,21 @@ static void try_to_fill_sample(struct sample *sample, const char *name, char *bu
        nonmatch("sample", name, buf);
 }
 
+/*
+ * Crazy suunto xml. Look at how those o2/he things match up.
+ */
+static int suunto_dive_match(struct dive *dive, const char *name, int len, char *buf)
+{
+       return  MATCH(".o2pct", percent, &dive->gasmix[0].o2) ||
+               MATCH(".hepct_0", percent, &dive->gasmix[0].he) ||
+               MATCH(".o2pct_2", percent, &dive->gasmix[1].o2) ||
+               MATCH(".hepct_1", percent, &dive->gasmix[1].he) ||
+               MATCH(".o2pct_3", percent, &dive->gasmix[2].o2) ||
+               MATCH(".hepct_2", percent, &dive->gasmix[2].he) ||
+               MATCH(".o2pct_4", percent, &dive->gasmix[3].o2) ||
+               MATCH(".hepct_3", percent, &dive->gasmix[3].he);
+}
+
 /* We're in the top-level dive xml. Try to convert whatever value to a dive value */
 static void try_to_fill_dive(struct dive *dive, const char *name, char *buf)
 {
@@ -355,6 +401,18 @@ static void try_to_fill_dive(struct dive *dive, const char *name, char *buf)
                return;
        if (MATCH(".cylinderendpressure", pressure, &dive->end_pressure))
                return;
+
+       if (MATCH(".o2", gasmix, &dive->gasmix[gasmix_index].o2))
+               return;
+       if (MATCH(".n2", gasmix, &dive->gasmix[gasmix_index].n2))
+               return;
+       if (MATCH(".he", gasmix, &dive->gasmix[gasmix_index].he))
+               return;
+
+       /* Suunto XML files are some crazy sh*t. */
+       if (suunto && suunto_dive_match(dive, name, len, buf))
+               return;
+
        nonmatch("dive", name, buf);
 }
 
@@ -415,6 +473,35 @@ static void dive_end(void)
                dive->name = generate_name(dive);
        record_dive(dive);
        dive = NULL;
+       gasmix_index = 0;
+}
+
+static void suunto_start(void)
+{
+       suunto++;
+}
+
+static void suunto_end(void)
+{
+       suunto--;
+}
+
+static void event_start(void)
+{
+}
+
+static void event_end(void)
+{
+       event_index++;
+}
+
+static void gasmix_start(void)
+{
+}
+
+static void gasmix_end(void)
+{
+       gasmix_index++;
 }
 
 static void sample_start(void)
@@ -435,6 +522,7 @@ static void sample_start(void)
        }
        sample = dive->sample + nr;
        memset(sample, 0, sizeof(*sample));
+       event_index = 0;
 }
 
 static void sample_end(void)
@@ -550,34 +638,41 @@ static void visit(xmlNode *n)
        traverse(n->children);
 }
 
+/*
+ * 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[] = {
+       { "dive", dive_start, dive_end },
+       { "SUUNTO", suunto_start, suunto_end },
+       { "sample", sample_start, sample_end },
+       { "SAMPLE", sample_start, sample_end },
+       { "event", event_start, event_end },
+       { "gasmix", gasmix_start, gasmix_end },
+       { NULL, }
+};
+
 static void traverse(xmlNode *root)
 {
        xmlNode *n;
 
        for (n = root; n; n = n->next) {
-               /* XML from libdivecomputer: 'dive' per new dive */
-               if (!strcmp(n->name, "dive")) {
-                       dive_start();
-                       visit(n);
-                       dive_end();
-                       continue;
-               }
+               struct nesting *rule = nesting;
 
-               /*
-                * At least both libdivecomputer and Suunto
-                * agree on "sample".
-                *
-                * Well - almost. Ignore case.
-                */
-               if (!strcasecmp(n->name, "sample")) {
-                       sample_start();
-                       visit(n);
-                       sample_end();
-                       continue;
-               }
+               do {
+                       if (!strcmp(rule->name, n->name))
+                               break;
+                       rule++;
+               } while (rule->name);
 
-               /* Anything else - just visit it and recurse */
+               if (rule->start)
+                       rule->start();
                visit(n);
+               if (rule->end)
+                       rule->end();
        }
 }