X-Git-Url: https://sigrok.org/gitweb/?a=blobdiff_plain;f=src%2Fstrutil.c;h=30e992b815468e5a86c1210652b3a19aece2720b;hb=8f484ca78ef8603f854ad80df02f251631dc1330;hp=02afc034c67249acdcd5078518c3793b5d114780;hpb=d2391b5453cd54b4bb087bdfbf2964202ba29098;p=libsigrok.git diff --git a/src/strutil.c b/src/strutil.c index 02afc034..30e992b8 100644 --- a/src/strutil.c +++ b/src/strutil.c @@ -23,6 +23,7 @@ #include #include #include +#include #include #include "libsigrok-internal.h" @@ -168,6 +169,38 @@ SR_PRIV int sr_atof(const char *str, float *ret) return SR_OK; } +/** + * @private + * + * Convert a string representation of a numeric value to a double. The + * conversion is strict and will fail if the complete string does not represent + * a valid double. The function sets errno according to the details of the + * failure. This version ignores the locale. + * + * @param str The string representation to convert. + * @param ret Pointer to double where the result of the conversion will be stored. + * + * @retval SR_OK Conversion successful. + * @retval SR_ERR Failure. + */ +SR_PRIV int sr_atod_ascii(const char *str, double *ret) +{ + double tmp; + char *endptr = NULL; + + errno = 0; + tmp = g_ascii_strtod(str, &endptr); + + if (!endptr || *endptr || errno) { + if (!errno) + errno = EINVAL; + return SR_ERR; + } + + *ret = tmp; + return SR_OK; +} + /** * @private * @@ -233,13 +266,19 @@ SR_API int sr_parse_rational(const char *str, struct sr_rational *ret) int64_t denominator = 1; int32_t fractional_len = 0; int32_t exponent = 0; + bool is_negative = false; errno = 0; integral = g_ascii_strtoll(str, &endptr, 10); - if (errno) + if (str == endptr && (str[0] == '-' || str[0] == '+') && str[1] == '.') + endptr += 1; + else if (errno) return SR_ERR; + if (integral < 0 || str[0] == '-') + is_negative = true; + if (*endptr == '.') { const char* start = endptr + 1; fractional = g_ascii_strtoll(start, &endptr, 10); @@ -261,7 +300,7 @@ SR_API int sr_parse_rational(const char *str, struct sr_rational *ret) integral *= 10; exponent -= fractional_len; - if (integral >= 0) + if (!is_negative) integral += fractional; else integral -= fractional; @@ -412,11 +451,11 @@ SR_API char *sr_period_string(uint64_t v_p, uint64_t v_q) SR_API char *sr_voltage_string(uint64_t v_p, uint64_t v_q) { if (v_q == 1000) - return g_strdup_printf("%" PRIu64 "mV", v_p); + return g_strdup_printf("%" PRIu64 " mV", v_p); else if (v_q == 1) - return g_strdup_printf("%" PRIu64 "V", v_p); + return g_strdup_printf("%" PRIu64 " V", v_p); else - return g_strdup_printf("%gV", (float)v_p / (float)v_q); + return g_strdup_printf("%g V", (float)v_p / (float)v_q); } /**