Merge branch 'vendor/TCSH'
[dragonfly.git] / lib / libc / locale / xlocale.c
1 /*-
2  * Copyright (c) 2011 The FreeBSD Foundation
3  * All rights reserved.
4  *
5  * This software was developed by David Chisnall under sponsorship from
6  * the FreeBSD Foundation.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  * 1. Redistributions of source code must retain the above copyright
12  *    notice, this list of conditions and the following disclaimer.
13  * 2. Redistributions in binary form must reproduce the above copyright
14  *    notice, this list of conditions and the following disclaimer in the
15  *    documentation and/or other materials provided with the distribution.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27  * SUCH DAMAGE.
28  *
29  * $FreeBSD: head/lib/libc/locale/xlocale.c 303495 2016-07-29 17:18:47Z ed $
30  */
31
32 #include "namespace.h"
33 #include <errno.h>
34 #include <pthread.h>
35 #include <stdio.h>
36 #include <string.h>
37 #include <runetype.h>
38 #include "libc_private.h"
39 #include "xlocale_private.h"
40 #include "un-namespace.h"
41
42 /**
43  * Each locale loader declares a global component.  This is used by setlocale()
44  * and also by xlocale with LC_GLOBAL_LOCALE..
45  */
46 extern struct xlocale_component __xlocale_global_collate;
47 extern struct xlocale_component __xlocale_global_ctype;
48 extern struct xlocale_component __xlocale_global_monetary;
49 extern struct xlocale_component __xlocale_global_numeric;
50 extern struct xlocale_component __xlocale_global_time;
51 extern struct xlocale_component __xlocale_global_messages;
52 /*
53  * And another version for the statically-allocated C locale.  We only have
54  * components for the parts that are expected to be sensible.
55  */
56 extern struct xlocale_component __xlocale_C_collate;
57 extern struct xlocale_component __xlocale_C_ctype;
58
59 #ifndef __NO_TLS
60 /*
61  * The locale for this thread.
62  */
63 __thread locale_t __thread_locale;
64 #endif
65 /*
66  * Flag indicating that one or more per-thread locales exist.
67  */
68 int __has_thread_locale;
69 /*
70  * Private functions in setlocale.c.
71  */
72 const char * __get_locale_env(int category);
73 int __get_locale_str(int category, const char *str, char * const res);
74 int __detect_path_locale(void);
75
76 struct _xlocale __xlocale_global_locale = {
77         .header = {0},
78         .components = {
79                 &__xlocale_global_collate,
80                 &__xlocale_global_ctype,
81                 &__xlocale_global_monetary,
82                 &__xlocale_global_numeric,
83                 &__xlocale_global_time,
84                 &__xlocale_global_messages
85         },
86         .monetary_locale_changed = 1,
87         .using_monetary_locale = 0,
88         .numeric_locale_changed = 1,
89         .using_numeric_locale = 0
90 };
91
92 struct _xlocale __xlocale_C_locale = {
93         .header = {0},
94         .components = {
95                 &__xlocale_C_collate,
96                 &__xlocale_C_ctype,
97                 0, 0, 0, 0
98         },
99         .monetary_locale_changed = 1,
100         .using_monetary_locale = 0,
101         .numeric_locale_changed = 1,
102         .using_numeric_locale = 0
103 };
104
105 static void*(*constructors[])(const char*, locale_t) =
106 {
107         __collate_load,
108         __ctype_load,
109         __monetary_load,
110         __numeric_load,
111         __time_load,
112         __messages_load
113 };
114
115 static pthread_key_t locale_info_key;
116 static int fake_tls;
117 static locale_t thread_local_locale;
118
119 static void init_key(void)
120 {
121
122         _pthread_key_create(&locale_info_key, xlocale_release);
123         _pthread_setspecific(locale_info_key, (void*)42);
124         if (_pthread_getspecific(locale_info_key) == (void*)42) {
125                 _pthread_setspecific(locale_info_key, 0);
126         } else {
127                 fake_tls = 1;
128         }
129         /* At least one per-thread locale has now been set. */
130         __has_thread_locale = 1;
131         __detect_path_locale();
132 }
133
134 static pthread_once_t once_control = PTHREAD_ONCE_INIT;
135
136 static locale_t
137 get_thread_locale(void)
138 {
139
140         _once(&once_control, init_key);
141         
142         return (fake_tls ? thread_local_locale :
143                 _pthread_getspecific(locale_info_key));
144 }
145
146 #ifdef __NO_TLS
147 locale_t
148 __get_locale(void)
149 {
150         locale_t l = get_thread_locale();
151         return (l ? l : &__xlocale_global_locale);
152
153 }
154 #endif
155
156 static void
157 set_thread_locale(locale_t loc)
158 {
159         locale_t l = (loc == LC_GLOBAL_LOCALE) ? 0 : loc;
160
161         _once(&once_control, init_key);
162         
163         if (NULL != l) {
164                 xlocale_retain((struct xlocale_refcounted*)l);
165         }
166         locale_t old = _pthread_getspecific(locale_info_key);
167         if ((NULL != old) && (l != old)) {
168                 xlocale_release((struct xlocale_refcounted*)old);
169         }
170         if (fake_tls) {
171                 thread_local_locale = l;
172         } else {
173                 _pthread_setspecific(locale_info_key, l);
174         }
175 #ifndef __NO_TLS
176         __thread_locale = l;
177         __set_thread_rune_locale(loc);
178 #endif
179 }
180
181 /**
182  * Clean up a locale, once its reference count reaches zero.  This function is
183  * called by xlocale_release(), it should not be called directly.
184  */
185 static void
186 destruct_locale(void *l)
187 {
188         locale_t loc = l;
189
190         for (int type=0 ; type<XLC_LAST ; type++) {
191                 if (loc->components[type]) {
192                         xlocale_release(loc->components[type]);
193                 }
194         }
195         if (loc->csym) {
196                 free(loc->csym);
197         }
198         free(l);
199 }
200
201 /**
202  * Allocates a new, uninitialised, locale.
203  */
204 static locale_t
205 alloc_locale(void)
206 {
207         locale_t new = calloc(sizeof(struct _xlocale), 1);
208
209         new->header.destructor = destruct_locale;
210         new->monetary_locale_changed = 1;
211         new->numeric_locale_changed = 1;
212         return (new);
213 }
214 static void
215 copyflags(locale_t new, locale_t old)
216 {
217         new->using_monetary_locale = old->using_monetary_locale;
218         new->using_numeric_locale = old->using_numeric_locale;
219         new->using_time_locale = old->using_time_locale;
220         new->using_messages_locale = old->using_messages_locale;
221 }
222
223 static int dupcomponent(int type, locale_t base, locale_t new) 
224 {
225         /* Always copy from the global locale, since it has mutable components.
226          */
227         struct xlocale_component *src = base->components[type];
228
229         if (&__xlocale_global_locale == base) {
230                 new->components[type] = constructors[type](src->locale, new);
231                 if (new->components[type]) {
232                         strncpy(new->components[type]->locale, src->locale,
233                             ENCODING_LEN);
234                 }
235         } else if (base->components[type]) {
236                 new->components[type] = xlocale_retain(base->components[type]);
237         } else {
238                 /* If the component was NULL, return success - if base is a
239                  * valid locale then the flag indicating that this isn't
240                  * present should be set.  If it isn't a valid locale, then
241                  * we're stuck anyway. */
242                 return 1;
243         }
244         return (0 != new->components[type]);
245 }
246
247 /*
248  * Public interfaces.  These are the five public functions described by the
249  * xlocale interface.  
250  */
251
252 locale_t newlocale(int mask, const char *locale, locale_t base)
253 {
254         int type;
255         const char *realLocale = locale;
256         const char *np, *cp;
257         char lres[ENCODING_LEN + 1] = "";
258         int len;
259         int useenv = 0;
260         int useslh = 0;
261         int usestr = 0;
262         int success = 1;
263
264         _once(&once_control, init_key);
265
266         locale_t new = alloc_locale();
267         if (NULL == new) {
268                 return (NULL);
269         }
270
271         FIX_LOCALE(base);
272         copyflags(new, base);
273
274         if (NULL == locale) {
275                 realLocale = "C";
276         } else if ('\0' == locale[0]) {
277                 useenv = 1;
278         } else if (strchr(locale, '/') != NULL) {
279                 /*
280                  * Handle system native locale string
281                  * e.g. "C/en_US.UTF-8/C/C/lt_LT/C"
282                  */
283                 useslh = 1;
284                 np = locale;
285         } else if ('L' == locale[0] && strchr(locale, ';') != NULL) {
286                 /*
287                  * We are called from c++ runtime lib with LC_*; string??
288                  */
289                 usestr = 1;
290         }
291
292         for (type=0 ; type<XLC_LAST ; type++) {
293                 if (useslh) {
294                         cp = strchr(np, '/');
295                         if (cp == NULL && type == XLC_LAST - 1) {
296                                 cp = locale + strlen(locale);
297                         } else if (cp == NULL || type == XLC_LAST - 1) {
298                                 errno = EINVAL;
299                                 success = 0;
300                                 break;
301                         }
302                         len = cp - np;
303                         if (len > ENCODING_LEN || len <= 0) {
304                                 errno = EINVAL;
305                                 success = 0;
306                                 break;
307                         }
308                         strncpy(lres, np, len);
309                         lres[len] = '\0';
310                         np = cp + 1;
311                 }
312
313                 if (mask & 1) {
314                         if (useenv) {
315                                 realLocale = __get_locale_env(type + 1);
316                         } else if (useslh) {
317                                 realLocale = lres;
318                         } else if (usestr) {
319                                 __get_locale_str(type + 1, locale, lres);
320                                 realLocale = lres;
321                         }
322                         new->components[type] =
323                              constructors[type](realLocale, new);
324                         if (new->components[type]) {
325                                 strncpy(new->components[type]->locale,
326                                      realLocale, ENCODING_LEN);
327                         } else {
328                                 success = 0;
329                                 break;
330                         }
331                 } else {
332                         if (!dupcomponent(type, base, new)) {
333                                 success = 0;
334                                 break;
335                         }
336                 }
337                 mask >>= 1;
338         }
339         if (0 == success) {
340                 xlocale_release(new);
341                 new = NULL;
342         }
343
344         return (new);
345 }
346
347 locale_t duplocale(locale_t base)
348 {
349         locale_t new = alloc_locale();
350         int type;
351
352         _once(&once_control, init_key);
353
354         if (NULL == new) {
355                 return (NULL);
356         }
357         
358         FIX_LOCALE(base);
359         copyflags(new, base);
360
361         for (type=0 ; type<XLC_LAST ; type++) {
362                 dupcomponent(type, base, new);
363         }
364
365         return (new);
366 }
367
368 /*
369  * Free a locale_t.  This is quite a poorly named function.  It actually
370  * disclaims a reference to a locale_t, rather than freeing it.  
371  */
372 void
373 freelocale(locale_t loc)
374 {
375
376         /*
377          * Fail if we're passed something that isn't a locale. If we're
378          * passed the global locale, pretend that we freed it but don't
379          * actually do anything.
380          */
381         if (loc != NULL && loc != LC_GLOBAL_LOCALE &&
382             loc != &__xlocale_global_locale)
383                 xlocale_release(loc);
384 }
385
386 /*
387  * Returns the name of the locale for a particular component of a locale_t.
388  */
389 const char *querylocale(int mask, locale_t loc)
390 {
391         int type = ffs(mask) - 1;
392         FIX_LOCALE(loc);
393         if (type >= XLC_LAST)
394                 return (NULL);
395         if (loc->components[type])
396                 return (loc->components[type]->locale);
397         return ("C");
398 }
399
400 /*
401  * Installs the specified locale_t as this thread's locale.
402  */
403 locale_t uselocale(locale_t loc)
404 {
405         locale_t old = get_thread_locale();
406         if (NULL != loc) {
407                 set_thread_locale(loc);
408         }
409         return (old ? old : LC_GLOBAL_LOCALE);
410 }
411