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:
Petri Lehtinen 2024-03-07 21:02:55 +02:00
parent 9b9b5e81cf
commit 2d1c13224f
7 changed files with 29 additions and 48 deletions

11
CHANGES
View file

@ -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
============ ============

View file

@ -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.

View file

@ -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

View file

@ -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@

View file

@ -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],

View file

@ -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@

View file

@ -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 */