mirror of
https://github.com/akheron/jansson.git
synced 2025-04-04 13:05:02 +00:00
Use sprintf() to determine locale's decimal point
This should fix thread safety of encoding and decoding, since localeconv() is not tread safe after all.
This commit is contained in:
parent
9b9b5e81cf
commit
2d1c13224f
7 changed files with 29 additions and 48 deletions
11
CHANGES
11
CHANGES
|
@ -1,3 +1,14 @@
|
||||||
|
Version 2.14.1
|
||||||
|
==============
|
||||||
|
|
||||||
|
Work in progress
|
||||||
|
|
||||||
|
* Fixes:
|
||||||
|
|
||||||
|
- Fix thread safety of encoding and decoding when `uselocale` or `newlocale`
|
||||||
|
is used to switch locales inside the threads (#674, #675, #677. Thanks to
|
||||||
|
Bruno Haible the report and help with fixing.)
|
||||||
|
|
||||||
Version 2.14
|
Version 2.14
|
||||||
============
|
============
|
||||||
|
|
||||||
|
|
|
@ -215,18 +215,7 @@ if (NOT DEFINED JSON_INT_T)
|
||||||
endif ()
|
endif ()
|
||||||
endif ()
|
endif ()
|
||||||
|
|
||||||
|
|
||||||
# If locale.h and localeconv() are available, define to 1, otherwise to 0.
|
|
||||||
check_include_files (locale.h HAVE_LOCALE_H)
|
check_include_files (locale.h HAVE_LOCALE_H)
|
||||||
check_function_exists (localeconv HAVE_LOCALECONV)
|
|
||||||
|
|
||||||
if (HAVE_LOCALECONV AND HAVE_LOCALE_H)
|
|
||||||
set (JSON_HAVE_LOCALECONV 1)
|
|
||||||
else ()
|
|
||||||
set (JSON_HAVE_LOCALECONV 0)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# check if we have setlocale
|
|
||||||
check_function_exists(setlocale HAVE_SETLOCALE)
|
check_function_exists(setlocale HAVE_SETLOCALE)
|
||||||
|
|
||||||
# Check what the inline keyword is.
|
# Check what the inline keyword is.
|
||||||
|
|
|
@ -32,10 +32,6 @@
|
||||||
otherwise to 0. */
|
otherwise to 0. */
|
||||||
#define JSON_INTEGER_IS_LONG_LONG 1
|
#define JSON_INTEGER_IS_LONG_LONG 1
|
||||||
|
|
||||||
/* If locale.h and localeconv() are available, define to 1,
|
|
||||||
otherwise to 0. */
|
|
||||||
#define JSON_HAVE_LOCALECONV 0
|
|
||||||
|
|
||||||
/* Maximum recursion depth for parsing JSON input.
|
/* Maximum recursion depth for parsing JSON input.
|
||||||
This limits the depth of e.g. array-within-array constructions. */
|
This limits the depth of e.g. array-within-array constructions. */
|
||||||
#define JSON_PARSER_MAX_DEPTH 2048
|
#define JSON_PARSER_MAX_DEPTH 2048
|
||||||
|
|
|
@ -56,9 +56,6 @@
|
||||||
#define JSON_INTEGER_FORMAT @JSON_INTEGER_FORMAT@
|
#define JSON_INTEGER_FORMAT @JSON_INTEGER_FORMAT@
|
||||||
|
|
||||||
|
|
||||||
/* If locale.h and localeconv() are available, define to 1, otherwise to 0. */
|
|
||||||
#define JSON_HAVE_LOCALECONV @JSON_HAVE_LOCALECONV@
|
|
||||||
|
|
||||||
/* If __atomic builtins are available they will be used to manage
|
/* If __atomic builtins are available they will be used to manage
|
||||||
reference counts of json_t. */
|
reference counts of json_t. */
|
||||||
#define JSON_HAVE_ATOMIC_BUILTINS @JSON_HAVE_ATOMIC_BUILTINS@
|
#define JSON_HAVE_ATOMIC_BUILTINS @JSON_HAVE_ATOMIC_BUILTINS@
|
||||||
|
|
|
@ -34,7 +34,7 @@ esac
|
||||||
AC_SUBST([json_inline])
|
AC_SUBST([json_inline])
|
||||||
|
|
||||||
# Checks for library functions.
|
# Checks for library functions.
|
||||||
AC_CHECK_FUNCS([close getpid gettimeofday localeconv open read sched_yield strtoll])
|
AC_CHECK_FUNCS([close getpid gettimeofday open read setlocale sched_yield strtoll])
|
||||||
|
|
||||||
AC_MSG_CHECKING([for gcc __sync builtins])
|
AC_MSG_CHECKING([for gcc __sync builtins])
|
||||||
have_sync_builtins=no
|
have_sync_builtins=no
|
||||||
|
@ -74,12 +74,6 @@ case "$ac_cv_type_long_long_int$ac_cv_func_strtoll" in
|
||||||
esac
|
esac
|
||||||
AC_SUBST([json_have_long_long])
|
AC_SUBST([json_have_long_long])
|
||||||
|
|
||||||
case "$ac_cv_header_locale_h$ac_cv_func_localeconv" in
|
|
||||||
yesyes) json_have_localeconv=1;;
|
|
||||||
*) json_have_localeconv=0;;
|
|
||||||
esac
|
|
||||||
AC_SUBST([json_have_localeconv])
|
|
||||||
|
|
||||||
# Features
|
# Features
|
||||||
AC_ARG_ENABLE([urandom],
|
AC_ARG_ENABLE([urandom],
|
||||||
[AS_HELP_STRING([--disable-urandom],
|
[AS_HELP_STRING([--disable-urandom],
|
||||||
|
|
|
@ -32,10 +32,6 @@
|
||||||
otherwise to 0. */
|
otherwise to 0. */
|
||||||
#define JSON_INTEGER_IS_LONG_LONG @json_have_long_long@
|
#define JSON_INTEGER_IS_LONG_LONG @json_have_long_long@
|
||||||
|
|
||||||
/* If locale.h and localeconv() are available, define to 1,
|
|
||||||
otherwise to 0. */
|
|
||||||
#define JSON_HAVE_LOCALECONV @json_have_localeconv@
|
|
||||||
|
|
||||||
/* If __atomic builtins are available they will be used to manage
|
/* If __atomic builtins are available they will be used to manage
|
||||||
reference counts of json_t. */
|
reference counts of json_t. */
|
||||||
#define JSON_HAVE_ATOMIC_BUILTINS @json_have_atomic_builtins@
|
#define JSON_HAVE_ATOMIC_BUILTINS @json_have_atomic_builtins@
|
||||||
|
|
|
@ -11,57 +11,57 @@
|
||||||
#include <jansson_private_config.h>
|
#include <jansson_private_config.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if JSON_HAVE_LOCALECONV
|
|
||||||
#include <locale.h>
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
- This code assumes that the decimal separator is exactly one
|
- This code assumes that the decimal separator is exactly one
|
||||||
character.
|
character.
|
||||||
|
|
||||||
- If setlocale() is called by another thread between the call to
|
- If setlocale() is called by another thread between the call to
|
||||||
localeconv() and the call to sprintf() or strtod(), the result may
|
get_decimal_point() and the call to sprintf() or strtod(), the
|
||||||
be wrong. setlocale() is not thread-safe and should not be used
|
result may be wrong. setlocale() is not thread-safe and should
|
||||||
this way. Multi-threaded programs should use uselocale() instead.
|
not be used this way. Multi-threaded programs should use
|
||||||
|
uselocale() instead.
|
||||||
*/
|
*/
|
||||||
|
static char get_decimal_point() {
|
||||||
|
char buf[3];
|
||||||
|
sprintf(buf, "%#.0f", 1.0); // "1." in the current locale
|
||||||
|
return buf[1];
|
||||||
|
}
|
||||||
|
|
||||||
static void to_locale(strbuffer_t *strbuffer) {
|
static void to_locale(strbuffer_t *strbuffer) {
|
||||||
const char *point;
|
char point;
|
||||||
char *pos;
|
char *pos;
|
||||||
|
|
||||||
point = localeconv()->decimal_point;
|
point = get_decimal_point();
|
||||||
if (*point == '.') {
|
if (point == '.') {
|
||||||
/* No conversion needed */
|
/* No conversion needed */
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
pos = strchr(strbuffer->value, '.');
|
pos = strchr(strbuffer->value, '.');
|
||||||
if (pos)
|
if (pos)
|
||||||
*pos = *point;
|
*pos = point;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void from_locale(char *buffer) {
|
static void from_locale(char *buffer) {
|
||||||
const char *point;
|
char point;
|
||||||
char *pos;
|
char *pos;
|
||||||
|
|
||||||
point = localeconv()->decimal_point;
|
point = get_decimal_point();
|
||||||
if (*point == '.') {
|
if (point == '.') {
|
||||||
/* No conversion needed */
|
/* No conversion needed */
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
pos = strchr(buffer, *point);
|
pos = strchr(buffer, point);
|
||||||
if (pos)
|
if (pos)
|
||||||
*pos = '.';
|
*pos = '.';
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
int jsonp_strtod(strbuffer_t *strbuffer, double *out) {
|
int jsonp_strtod(strbuffer_t *strbuffer, double *out) {
|
||||||
double value;
|
double value;
|
||||||
char *end;
|
char *end;
|
||||||
|
|
||||||
#if JSON_HAVE_LOCALECONV
|
|
||||||
to_locale(strbuffer);
|
to_locale(strbuffer);
|
||||||
#endif
|
|
||||||
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
value = strtod(strbuffer->value, &end);
|
value = strtod(strbuffer->value, &end);
|
||||||
|
@ -92,9 +92,7 @@ int jsonp_dtostr(char *buffer, size_t size, double value, int precision) {
|
||||||
if (length >= size)
|
if (length >= size)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
#if JSON_HAVE_LOCALECONV
|
|
||||||
from_locale(buffer);
|
from_locale(buffer);
|
||||||
#endif
|
|
||||||
|
|
||||||
/* Make sure there's a dot or 'e' in the output. Otherwise
|
/* Make sure there's a dot or 'e' in the output. Otherwise
|
||||||
a real is converted to an integer when decoding */
|
a real is converted to an integer when decoding */
|
||||||
|
|
Loading…
Add table
Reference in a new issue