diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2014-03-09 14:45:20 -0700 |
---|---|---|
committer | Dirk Hohndel <dirk@hohndel.org> | 2014-03-09 19:36:38 -0700 |
commit | 6ef38967ffa0d8a78193698d29e86ffbab1bb864 (patch) | |
tree | 542ac500cdf3e56e73ea6a44e1ff7ae5617a25b8 | |
parent | 820cbe96da886b72b190911e12573d511d3db4b5 (diff) | |
download | subsurface-6ef38967ffa0d8a78193698d29e86ffbab1bb864.tar.gz |
parse cylinder and weightsystem information
This makes the sample parsing helper function for key-value pair parsing
more generic, and uses it for parsing cylinders and weightsystems too.
Events still to go, and then we have the "setting" section (for dive
computer nicknames and firmware information) that we don't actually save
yet in the git format.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
-rw-r--r-- | load-git.c | 168 |
1 files changed, 135 insertions, 33 deletions
diff --git a/load-git.c b/load-git.c index cd54b0f71..a8304a0af 100644 --- a/load-git.c +++ b/load-git.c @@ -58,6 +58,20 @@ static depth_t get_depth(const char *line) return d; } +static volume_t get_volume(const char *line) +{ + volume_t v; + v.mliter = rint(1000*ascii_strtod(line, NULL)); + return v; +} + +static weight_t get_weight(const char *line) +{ + weight_t w; + w.grams = rint(1000*ascii_strtod(line, NULL)); + return w; +} + static pressure_t get_pressure(const char *line) { pressure_t p; @@ -65,6 +79,13 @@ static pressure_t get_pressure(const char *line) return p; } +static fraction_t get_fraction(const char *line) +{ + fraction_t f; + f.permille = rint(10*ascii_strtod(line, NULL)); + return f; +} + static void update_date(timestamp_t *when, const char *line) { unsigned yyyy, mm, dd; @@ -167,9 +188,114 @@ static void parse_dive_visibility(char *line, struct membuffer *str, void *_dive static void parse_dive_notrip(char *line, struct membuffer *str, void *_dive) { struct dive *dive = _dive; dive->tripflag = NO_TRIP; } -/* FIXME! Cylinders and weigthsystems not parsed */ -static void parse_dive_cylinder(char *line, struct membuffer *str, void *_dive) { } -static void parse_dive_weightsystem(char *line, struct membuffer *str, void *_dive) { } +/* Parse key=val parts of samples and cylinders etc */ +static char *parse_keyvalue_entry(void (*fn)(void *, const char *, const char *), void *fndata, char *line) +{ + char *key = line, *val, c; + + while ((c = *line) != 0) { + if (isspace(c) || c == '=') + break; + line++; + } + + if (c == '=') + *line++ = 0; + val = line; + + while ((c = *line) != 0) { + if (isspace(c)) + break; + line++; + } + if (c) + *line++ = 0; + + fn(fndata, key, val); + return line; +} + +static int cylinder_index, weightsystem_index; + +static void parse_cylinder_keyvalue(void *_cylinder, const char *key, const char *value) +{ + cylinder_t *cylinder = _cylinder; + if (!strcmp(key, "vol")) { + cylinder->type.size = get_volume(value); + return; + } + if (!strcmp(key, "workpressure")) { + cylinder->type.workingpressure = get_pressure(value); + return; + } + /* This is handled by the "get_utf8()" */ + if (!strcmp(key, "description")) + return; + if (!strcmp(key, "o2")) { + cylinder->gasmix.o2 = get_fraction(value); + return; + } + if (!strcmp(key, "he")) { + cylinder->gasmix.he = get_fraction(value); + return; + } + if (!strcmp(key, "start")) { + cylinder->start = get_pressure(value); + return; + } + if (!strcmp(key, "end")) { + cylinder->end = get_pressure(value); + return; + } + report_error("Unknown cylinder key/value pair (%s/%s)", key, value); +} + +static void parse_dive_cylinder(char *line, struct membuffer *str, void *_dive) +{ + struct dive *dive = _dive; + cylinder_t *cylinder = dive->cylinder + cylinder_index; + + cylinder_index++; + cylinder->type.description = get_utf8(str); + for (;;) { + char c; + while (isspace(c = *line)) + line++; + if (!c) + break; + line = parse_keyvalue_entry(parse_cylinder_keyvalue, cylinder, line); + } +} + +static void parse_weightsystem_keyvalue(void *_ws, const char *key, const char *value) +{ + weightsystem_t *ws = _ws; + if (!strcmp(key, "weight")) { + ws->weight = get_weight(value); + return; + } + /* This is handled by the "get_utf8()" */ + if (!strcmp(key, "description")) + return; + report_error("Unknown weightsystem key/value pair (%s/%s)", key, value); +} + +static void parse_dive_weightsystem(char *line, struct membuffer *str, void *_dive) +{ + struct dive *dive = _dive; + weightsystem_t *ws = dive->weightsystem + weightsystem_index; + + weightsystem_index++; + ws->description = get_utf8(str); + for (;;) { + char c; + while (isspace(c = *line)) + line++; + if (!c) + break; + line = parse_keyvalue_entry(parse_weightsystem_keyvalue, ws, line); + } +} static int match_action(char *line, struct membuffer *str, void *data, struct keyword_action *action, unsigned nr_action) @@ -212,8 +338,10 @@ report_error("Unmatched action '%s'", line); } /* FIXME! We should do the array thing here too. */ -static void parse_sample_keyvalue(struct sample *sample, const char *key, const char *value) +static void parse_sample_keyvalue(void *_sample, const char *key, const char *value) { + struct sample *sample = _sample; + if (!strcmp(key, "sensor")) { sample->sensor = atoi(value); return; @@ -252,34 +380,7 @@ static void parse_sample_keyvalue(struct sample *sample, const char *key, const sample->bearing = atoi(value); return; } - report_error("Unexpected sample key/value pair: '%s'='%s'\n", key, value); -} - -/* Parse key=val parts of the samples */ -static char *parse_sample_entry(struct sample *sample, char *line) -{ - char *key = line, *val, c; - - while ((c = *line) != 0) { - if (isspace(c) || c == '=') - break; - line++; - } - - if (c == '=') - *line++ = 0; - val = line; - - while ((c = *line) != 0) { - if (isspace(c)) - break; - line++; - } - if (c) - *line++ = 0; - - parse_sample_keyvalue(sample, key, val); - return line; + report_error("Unexpected sample key/value pair (%s/%s)", key, value); } static char *parse_sample_unit(struct sample *sample, double val, char *unit) @@ -351,7 +452,7 @@ static void sample_parser(char *line, struct divecomputer *dc) break; /* Less common sample entries have a name */ if (c >= 'a' && c <= 'z') { - line = parse_sample_entry(sample, line); + line = parse_keyvalue_entry(parse_sample_keyvalue, sample, line); } else { const char *end; double val = ascii_strtod(line, &end); @@ -871,6 +972,7 @@ static int parse_dive_entry(git_repository *repo, const git_tree_entry *entry, c return report_error("Unable to read dive file"); if (*suffix) dive->number = atoi(suffix+1); + cylinder_index = weightsystem_index = 0; for_each_line(blob, dive_parser, active_dive); git_blob_free(blob); return 0; |