diff options
Diffstat (limited to 'gl/setlocale_null-unlocked.c')
-rw-r--r-- | gl/setlocale_null-unlocked.c | 149 |
1 files changed, 149 insertions, 0 deletions
diff --git a/gl/setlocale_null-unlocked.c b/gl/setlocale_null-unlocked.c new file mode 100644 index 00000000..0a86f0df --- /dev/null +++ b/gl/setlocale_null-unlocked.c | |||
@@ -0,0 +1,149 @@ | |||
1 | /* Query the name of the current global locale, without locking. | ||
2 | Copyright (C) 2019-2024 Free Software Foundation, Inc. | ||
3 | |||
4 | This file is free software: you can redistribute it and/or modify | ||
5 | it under the terms of the GNU Lesser General Public License as | ||
6 | published by the Free Software Foundation; either version 2.1 of the | ||
7 | License, or (at your option) any later version. | ||
8 | |||
9 | This file is distributed in the hope that it will be useful, | ||
10 | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
11 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
12 | GNU Lesser General Public License for more details. | ||
13 | |||
14 | You should have received a copy of the GNU Lesser General Public License | ||
15 | along with this program. If not, see <https://www.gnu.org/licenses/>. */ | ||
16 | |||
17 | /* Written by Bruno Haible <bruno@clisp.org>, 2019. */ | ||
18 | |||
19 | #include <config.h> | ||
20 | |||
21 | /* Specification. */ | ||
22 | #include "setlocale_null.h" | ||
23 | |||
24 | #include <errno.h> | ||
25 | #include <locale.h> | ||
26 | #include <string.h> | ||
27 | #if defined _WIN32 && !defined __CYGWIN__ | ||
28 | # include <wchar.h> | ||
29 | #endif | ||
30 | |||
31 | /* Use the system's setlocale() function, not the gnulib override, here. */ | ||
32 | #undef setlocale | ||
33 | |||
34 | const char * | ||
35 | setlocale_null_unlocked (int category) | ||
36 | { | ||
37 | const char *result = setlocale (category, NULL); | ||
38 | |||
39 | #ifdef __ANDROID__ | ||
40 | if (result == NULL) | ||
41 | switch (category) | ||
42 | { | ||
43 | case LC_CTYPE: | ||
44 | case LC_NUMERIC: | ||
45 | case LC_TIME: | ||
46 | case LC_COLLATE: | ||
47 | case LC_MONETARY: | ||
48 | case LC_MESSAGES: | ||
49 | case LC_ALL: | ||
50 | case LC_PAPER: | ||
51 | case LC_NAME: | ||
52 | case LC_ADDRESS: | ||
53 | case LC_TELEPHONE: | ||
54 | case LC_MEASUREMENT: | ||
55 | result = "C"; | ||
56 | break; | ||
57 | default: | ||
58 | break; | ||
59 | } | ||
60 | #endif | ||
61 | |||
62 | return result; | ||
63 | } | ||
64 | |||
65 | int | ||
66 | setlocale_null_r_unlocked (int category, char *buf, size_t bufsize) | ||
67 | { | ||
68 | #if defined _WIN32 && !defined __CYGWIN__ && defined _MSC_VER | ||
69 | /* On native Windows, nowadays, the setlocale() implementation is based | ||
70 | on _wsetlocale() and uses malloc() for the result. We are better off | ||
71 | using _wsetlocale() directly. */ | ||
72 | const wchar_t *result = _wsetlocale (category, NULL); | ||
73 | |||
74 | if (result == NULL) | ||
75 | { | ||
76 | /* CATEGORY is invalid. */ | ||
77 | if (bufsize > 0) | ||
78 | /* Return an empty string in BUF. | ||
79 | This is a convenience for callers that don't want to write explicit | ||
80 | code for handling EINVAL. */ | ||
81 | buf[0] = '\0'; | ||
82 | return EINVAL; | ||
83 | } | ||
84 | else | ||
85 | { | ||
86 | size_t length = wcslen (result); | ||
87 | if (length < bufsize) | ||
88 | { | ||
89 | size_t i; | ||
90 | |||
91 | /* Convert wchar_t[] -> char[], assuming plain ASCII. */ | ||
92 | for (i = 0; i <= length; i++) | ||
93 | buf[i] = result[i]; | ||
94 | |||
95 | return 0; | ||
96 | } | ||
97 | else | ||
98 | { | ||
99 | if (bufsize > 0) | ||
100 | { | ||
101 | /* Return a truncated result in BUF. | ||
102 | This is a convenience for callers that don't want to write | ||
103 | explicit code for handling ERANGE. */ | ||
104 | size_t i; | ||
105 | |||
106 | /* Convert wchar_t[] -> char[], assuming plain ASCII. */ | ||
107 | for (i = 0; i < bufsize; i++) | ||
108 | buf[i] = result[i]; | ||
109 | buf[bufsize - 1] = '\0'; | ||
110 | } | ||
111 | return ERANGE; | ||
112 | } | ||
113 | } | ||
114 | #else | ||
115 | const char *result = setlocale_null_unlocked (category); | ||
116 | |||
117 | if (result == NULL) | ||
118 | { | ||
119 | /* CATEGORY is invalid. */ | ||
120 | if (bufsize > 0) | ||
121 | /* Return an empty string in BUF. | ||
122 | This is a convenience for callers that don't want to write explicit | ||
123 | code for handling EINVAL. */ | ||
124 | buf[0] = '\0'; | ||
125 | return EINVAL; | ||
126 | } | ||
127 | else | ||
128 | { | ||
129 | size_t length = strlen (result); | ||
130 | if (length < bufsize) | ||
131 | { | ||
132 | memcpy (buf, result, length + 1); | ||
133 | return 0; | ||
134 | } | ||
135 | else | ||
136 | { | ||
137 | if (bufsize > 0) | ||
138 | { | ||
139 | /* Return a truncated result in BUF. | ||
140 | This is a convenience for callers that don't want to write | ||
141 | explicit code for handling ERANGE. */ | ||
142 | memcpy (buf, result, bufsize - 1); | ||
143 | buf[bufsize - 1] = '\0'; | ||
144 | } | ||
145 | return ERANGE; | ||
146 | } | ||
147 | } | ||
148 | #endif | ||
149 | } | ||