summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--parse-xml.c54
1 files changed, 32 insertions, 22 deletions
diff --git a/parse-xml.c b/parse-xml.c
index ddb3ba347..e23ebc0a9 100644
--- a/parse-xml.c
+++ b/parse-xml.c
@@ -195,28 +195,32 @@ static void divedatetime(char *buffer, void *_when)
}
}
-union int_or_float {
- double fp;
-};
-
enum number_type {
NEITHER,
FLOAT
};
-static enum number_type integer_or_float(char *buffer, union int_or_float *res)
+static enum number_type parse_float(char *buffer, double *res, char **endp)
{
- char *end;
- double fp;
+ double val;
errno = 0;
- fp = g_ascii_strtod(buffer, &end);
- if (!errno && end != buffer) {
- res->fp = fp;
- return FLOAT;
- }
+ val = g_ascii_strtod(buffer, endp);
+ if (errno || *endp == buffer)
+ return NEITHER;
+
+ *res = val;
+ return FLOAT;
+}
+
+union int_or_float {
+ double fp;
+};
- return NEITHER;
+static enum number_type integer_or_float(char *buffer, union int_or_float *res)
+{
+ char *end;
+ return parse_float(buffer, &res->fp, &end);
}
static void pressure(char *buffer, void *_press)
@@ -361,18 +365,24 @@ static void duration(char *buffer, void *_time)
static void percent(char *buffer, void *_fraction)
{
fraction_t *fraction = _fraction;
- union int_or_float val;
+ double val;
+ char *end;
- switch (integer_or_float(buffer, &val)) {
+ switch (parse_float(buffer, &val, &end)) {
case FLOAT:
- /* Turn fractions into percent.. */
- if (val.fp <= 1.0)
- val.fp *= 100;
- /* Then turn percent into our integer permille format */
- if (val.fp <= 100.0)
- fraction->permille = val.fp * 10 + 0.5;
- break;
+ /* Turn fractions into percent unless explicit.. */
+ if (val <= 1.0) {
+ while (isspace(*end))
+ end++;
+ if (*end != '%')
+ val *= 100;
+ }
+ /* Then turn percent into our integer permille format */
+ if (val >= 0 && val <= 100.0) {
+ fraction->permille = val * 10 + 0.5;
+ break;
+ }
default:
printf("Strange percentage reading %s\n", buffer);
break;