Fix gcc10 compiler warnings
[legacy-proxmark3.git] / client / jansson / strconv.c
blob8075481e3b76d365fd26b87c300a07032d0a448d
1 #include <assert.h>
2 #include <errno.h>
3 #include <stdio.h>
4 #include <string.h>
5 #include <math.h>
6 #ifdef __MINGW32__
7 #undef __NO_ISOCEXT /* ensure stdlib.h will declare prototypes for mingw own 'strtod' replacement, called '__strtod' */
8 #endif
9 #include "jansson_private.h"
10 #include "strbuffer.h"
12 /* need jansson_private_config.h to get the correct snprintf */
13 #ifdef HAVE_CONFIG_H
14 #include <jansson_private_config.h>
15 #endif
17 #ifdef __MINGW32__
18 #define strtod __strtod
19 #endif
21 #if JSON_HAVE_LOCALECONV
22 #include <locale.h>
25 - This code assumes that the decimal separator is exactly one
26 character.
28 - If setlocale() is called by another thread between the call to
29 localeconv() and the call to sprintf() or strtod(), the result may
30 be wrong. setlocale() is not thread-safe and should not be used
31 this way. Multi-threaded programs should use uselocale() instead.
34 static void to_locale(strbuffer_t *strbuffer)
36 const char *point;
37 char *pos;
39 point = localeconv()->decimal_point;
40 if(*point == '.') {
41 /* No conversion needed */
42 return;
45 pos = strchr(strbuffer->value, '.');
46 if(pos)
47 *pos = *point;
50 static void from_locale(char *buffer)
52 const char *point;
53 char *pos;
55 point = localeconv()->decimal_point;
56 if(*point == '.') {
57 /* No conversion needed */
58 return;
61 pos = strchr(buffer, *point);
62 if(pos)
63 *pos = '.';
65 #endif
67 int jsonp_strtod(strbuffer_t *strbuffer, double *out)
69 double value;
70 char *end;
72 #if JSON_HAVE_LOCALECONV
73 to_locale(strbuffer);
74 #endif
76 errno = 0;
77 value = strtod(strbuffer->value, &end);
78 assert(end == strbuffer->value + strbuffer->length);
80 if((value == HUGE_VAL || value == -HUGE_VAL) && errno == ERANGE) {
81 /* Overflow */
82 return -1;
85 *out = value;
86 return 0;
89 int jsonp_dtostr(char *buffer, size_t size, double value, int precision)
91 int ret;
92 char *start, *end;
93 size_t length;
95 if (precision == 0)
96 precision = 17;
98 ret = snprintf(buffer, size, "%.*g", precision, value);
99 if(ret < 0)
100 return -1;
102 length = (size_t)ret;
103 if(length >= size)
104 return -1;
106 #if JSON_HAVE_LOCALECONV
107 from_locale(buffer);
108 #endif
110 /* Make sure there's a dot or 'e' in the output. Otherwise
111 a real is converted to an integer when decoding */
112 if(strchr(buffer, '.') == NULL &&
113 strchr(buffer, 'e') == NULL)
115 if(length + 3 >= size) {
116 /* No space to append ".0" */
117 return -1;
119 buffer[length] = '.';
120 buffer[length + 1] = '0';
121 buffer[length + 2] = '\0';
122 length += 2;
125 /* Remove leading '+' from positive exponent. Also remove leading
126 zeros from exponents (added by some printf() implementations) */
127 start = strchr(buffer, 'e');
128 if(start) {
129 start++;
130 end = start + 1;
132 if(*start == '-')
133 start++;
135 while(*end == '0')
136 end++;
138 if(end != start) {
139 memmove(start, end, length - (size_t)(end - buffer));
140 length -= (size_t)(end - start);
144 return (int)length;