1*36ac495dSmrg /* Implementation of the bindtextdomain(3) function
2*36ac495dSmrg Copyright (C) 1995-1998, 2000, 2001, 2002 Free Software Foundation, Inc.
3*36ac495dSmrg
4*36ac495dSmrg This program is free software; you can redistribute it and/or modify it
5*36ac495dSmrg under the terms of the GNU Library General Public License as published
6*36ac495dSmrg by the Free Software Foundation; either version 2, or (at your option)
7*36ac495dSmrg any later version.
8*36ac495dSmrg
9*36ac495dSmrg This program is distributed in the hope that it will be useful,
10*36ac495dSmrg but WITHOUT ANY WARRANTY; without even the implied warranty of
11*36ac495dSmrg MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12*36ac495dSmrg Library General Public License for more details.
13*36ac495dSmrg
14*36ac495dSmrg You should have received a copy of the GNU Library General Public
15*36ac495dSmrg License along with this program; if not, write to the Free Software
16*36ac495dSmrg Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301,
17*36ac495dSmrg USA. */
18*36ac495dSmrg
19*36ac495dSmrg #ifdef HAVE_CONFIG_H
20*36ac495dSmrg # include <config.h>
21*36ac495dSmrg #endif
22*36ac495dSmrg
23*36ac495dSmrg #include <stddef.h>
24*36ac495dSmrg #include <stdlib.h>
25*36ac495dSmrg #include <string.h>
26*36ac495dSmrg
27*36ac495dSmrg #ifdef _LIBC
28*36ac495dSmrg # include <libintl.h>
29*36ac495dSmrg #else
30*36ac495dSmrg # include "libgnuintl.h"
31*36ac495dSmrg #endif
32*36ac495dSmrg #include "gettextP.h"
33*36ac495dSmrg
34*36ac495dSmrg #ifdef _LIBC
35*36ac495dSmrg /* We have to handle multi-threaded applications. */
36*36ac495dSmrg # include <bits/libc-lock.h>
37*36ac495dSmrg #else
38*36ac495dSmrg /* Provide dummy implementation if this is outside glibc. */
39*36ac495dSmrg # define __libc_rwlock_define(CLASS, NAME)
40*36ac495dSmrg # define __libc_rwlock_wrlock(NAME)
41*36ac495dSmrg # define __libc_rwlock_unlock(NAME)
42*36ac495dSmrg #endif
43*36ac495dSmrg
44*36ac495dSmrg /* The internal variables in the standalone libintl.a must have different
45*36ac495dSmrg names than the internal variables in GNU libc, otherwise programs
46*36ac495dSmrg using libintl.a cannot be linked statically. */
47*36ac495dSmrg #if !defined _LIBC
48*36ac495dSmrg # define _nl_default_dirname libintl_nl_default_dirname
49*36ac495dSmrg # define _nl_domain_bindings libintl_nl_domain_bindings
50*36ac495dSmrg #endif
51*36ac495dSmrg
52*36ac495dSmrg /* Some compilers, like SunOS4 cc, don't have offsetof in <stddef.h>. */
53*36ac495dSmrg #ifndef offsetof
54*36ac495dSmrg # define offsetof(type,ident) ((size_t)&(((type*)0)->ident))
55*36ac495dSmrg #endif
56*36ac495dSmrg
57*36ac495dSmrg /* @@ end of prolog @@ */
58*36ac495dSmrg
59*36ac495dSmrg /* Contains the default location of the message catalogs. */
60*36ac495dSmrg extern const char _nl_default_dirname[];
61*36ac495dSmrg #ifdef _LIBC
62*36ac495dSmrg extern const char _nl_default_dirname_internal[] attribute_hidden;
63*36ac495dSmrg #else
64*36ac495dSmrg # define INTUSE(name) name
65*36ac495dSmrg #endif
66*36ac495dSmrg
67*36ac495dSmrg /* List with bindings of specific domains. */
68*36ac495dSmrg extern struct binding *_nl_domain_bindings;
69*36ac495dSmrg
70*36ac495dSmrg /* Lock variable to protect the global data in the gettext implementation. */
71*36ac495dSmrg __libc_rwlock_define (extern, _nl_state_lock attribute_hidden)
72*36ac495dSmrg
73*36ac495dSmrg
74*36ac495dSmrg /* Names for the libintl functions are a problem. They must not clash
75*36ac495dSmrg with existing names and they should follow ANSI C. But this source
76*36ac495dSmrg code is also used in GNU C Library where the names have a __
77*36ac495dSmrg prefix. So we have to make a difference here. */
78*36ac495dSmrg #ifdef _LIBC
79*36ac495dSmrg # define BINDTEXTDOMAIN __bindtextdomain
80*36ac495dSmrg # define BIND_TEXTDOMAIN_CODESET __bind_textdomain_codeset
81*36ac495dSmrg # ifndef strdup
82*36ac495dSmrg # define strdup(str) __strdup (str)
83*36ac495dSmrg # endif
84*36ac495dSmrg #else
85*36ac495dSmrg # define BINDTEXTDOMAIN libintl_bindtextdomain
86*36ac495dSmrg # define BIND_TEXTDOMAIN_CODESET libintl_bind_textdomain_codeset
87*36ac495dSmrg #endif
88*36ac495dSmrg
89*36ac495dSmrg /* Prototypes for local functions. */
90*36ac495dSmrg static void set_binding_values PARAMS ((const char *domainname,
91*36ac495dSmrg const char **dirnamep,
92*36ac495dSmrg const char **codesetp));
93*36ac495dSmrg
94*36ac495dSmrg /* Specifies the directory name *DIRNAMEP and the output codeset *CODESETP
95*36ac495dSmrg to be used for the DOMAINNAME message catalog.
96*36ac495dSmrg If *DIRNAMEP or *CODESETP is NULL, the corresponding attribute is not
97*36ac495dSmrg modified, only the current value is returned.
98*36ac495dSmrg If DIRNAMEP or CODESETP is NULL, the corresponding attribute is neither
99*36ac495dSmrg modified nor returned. */
100*36ac495dSmrg static void
set_binding_values(domainname,dirnamep,codesetp)101*36ac495dSmrg set_binding_values (domainname, dirnamep, codesetp)
102*36ac495dSmrg const char *domainname;
103*36ac495dSmrg const char **dirnamep;
104*36ac495dSmrg const char **codesetp;
105*36ac495dSmrg {
106*36ac495dSmrg struct binding *binding;
107*36ac495dSmrg int modified;
108*36ac495dSmrg
109*36ac495dSmrg /* Some sanity checks. */
110*36ac495dSmrg if (domainname == NULL || domainname[0] == '\0')
111*36ac495dSmrg {
112*36ac495dSmrg if (dirnamep)
113*36ac495dSmrg *dirnamep = NULL;
114*36ac495dSmrg if (codesetp)
115*36ac495dSmrg *codesetp = NULL;
116*36ac495dSmrg return;
117*36ac495dSmrg }
118*36ac495dSmrg
119*36ac495dSmrg __libc_rwlock_wrlock (_nl_state_lock);
120*36ac495dSmrg
121*36ac495dSmrg modified = 0;
122*36ac495dSmrg
123*36ac495dSmrg for (binding = _nl_domain_bindings; binding != NULL; binding = binding->next)
124*36ac495dSmrg {
125*36ac495dSmrg int compare = strcmp (domainname, binding->domainname);
126*36ac495dSmrg if (compare == 0)
127*36ac495dSmrg /* We found it! */
128*36ac495dSmrg break;
129*36ac495dSmrg if (compare < 0)
130*36ac495dSmrg {
131*36ac495dSmrg /* It is not in the list. */
132*36ac495dSmrg binding = NULL;
133*36ac495dSmrg break;
134*36ac495dSmrg }
135*36ac495dSmrg }
136*36ac495dSmrg
137*36ac495dSmrg if (binding != NULL)
138*36ac495dSmrg {
139*36ac495dSmrg if (dirnamep)
140*36ac495dSmrg {
141*36ac495dSmrg const char *dirname = *dirnamep;
142*36ac495dSmrg
143*36ac495dSmrg if (dirname == NULL)
144*36ac495dSmrg /* The current binding has be to returned. */
145*36ac495dSmrg *dirnamep = binding->dirname;
146*36ac495dSmrg else
147*36ac495dSmrg {
148*36ac495dSmrg /* The domain is already bound. If the new value and the old
149*36ac495dSmrg one are equal we simply do nothing. Otherwise replace the
150*36ac495dSmrg old binding. */
151*36ac495dSmrg char *result = binding->dirname;
152*36ac495dSmrg if (strcmp (dirname, result) != 0)
153*36ac495dSmrg {
154*36ac495dSmrg if (strcmp (dirname, INTUSE(_nl_default_dirname)) == 0)
155*36ac495dSmrg result = (char *) INTUSE(_nl_default_dirname);
156*36ac495dSmrg else
157*36ac495dSmrg {
158*36ac495dSmrg #if defined _LIBC || defined HAVE_STRDUP
159*36ac495dSmrg result = strdup (dirname);
160*36ac495dSmrg #else
161*36ac495dSmrg size_t len = strlen (dirname) + 1;
162*36ac495dSmrg result = (char *) malloc (len);
163*36ac495dSmrg if (__builtin_expect (result != NULL, 1))
164*36ac495dSmrg memcpy (result, dirname, len);
165*36ac495dSmrg #endif
166*36ac495dSmrg }
167*36ac495dSmrg
168*36ac495dSmrg if (__builtin_expect (result != NULL, 1))
169*36ac495dSmrg {
170*36ac495dSmrg if (binding->dirname != INTUSE(_nl_default_dirname))
171*36ac495dSmrg free (binding->dirname);
172*36ac495dSmrg
173*36ac495dSmrg binding->dirname = result;
174*36ac495dSmrg modified = 1;
175*36ac495dSmrg }
176*36ac495dSmrg }
177*36ac495dSmrg *dirnamep = result;
178*36ac495dSmrg }
179*36ac495dSmrg }
180*36ac495dSmrg
181*36ac495dSmrg if (codesetp)
182*36ac495dSmrg {
183*36ac495dSmrg const char *codeset = *codesetp;
184*36ac495dSmrg
185*36ac495dSmrg if (codeset == NULL)
186*36ac495dSmrg /* The current binding has be to returned. */
187*36ac495dSmrg *codesetp = binding->codeset;
188*36ac495dSmrg else
189*36ac495dSmrg {
190*36ac495dSmrg /* The domain is already bound. If the new value and the old
191*36ac495dSmrg one are equal we simply do nothing. Otherwise replace the
192*36ac495dSmrg old binding. */
193*36ac495dSmrg char *result = binding->codeset;
194*36ac495dSmrg if (result == NULL || strcmp (codeset, result) != 0)
195*36ac495dSmrg {
196*36ac495dSmrg #if defined _LIBC || defined HAVE_STRDUP
197*36ac495dSmrg result = strdup (codeset);
198*36ac495dSmrg #else
199*36ac495dSmrg size_t len = strlen (codeset) + 1;
200*36ac495dSmrg result = (char *) malloc (len);
201*36ac495dSmrg if (__builtin_expect (result != NULL, 1))
202*36ac495dSmrg memcpy (result, codeset, len);
203*36ac495dSmrg #endif
204*36ac495dSmrg
205*36ac495dSmrg if (__builtin_expect (result != NULL, 1))
206*36ac495dSmrg {
207*36ac495dSmrg if (binding->codeset != NULL)
208*36ac495dSmrg free (binding->codeset);
209*36ac495dSmrg
210*36ac495dSmrg binding->codeset = result;
211*36ac495dSmrg binding->codeset_cntr++;
212*36ac495dSmrg modified = 1;
213*36ac495dSmrg }
214*36ac495dSmrg }
215*36ac495dSmrg *codesetp = result;
216*36ac495dSmrg }
217*36ac495dSmrg }
218*36ac495dSmrg }
219*36ac495dSmrg else if ((dirnamep == NULL || *dirnamep == NULL)
220*36ac495dSmrg && (codesetp == NULL || *codesetp == NULL))
221*36ac495dSmrg {
222*36ac495dSmrg /* Simply return the default values. */
223*36ac495dSmrg if (dirnamep)
224*36ac495dSmrg *dirnamep = INTUSE(_nl_default_dirname);
225*36ac495dSmrg if (codesetp)
226*36ac495dSmrg *codesetp = NULL;
227*36ac495dSmrg }
228*36ac495dSmrg else
229*36ac495dSmrg {
230*36ac495dSmrg /* We have to create a new binding. */
231*36ac495dSmrg size_t len = strlen (domainname) + 1;
232*36ac495dSmrg struct binding *new_binding =
233*36ac495dSmrg (struct binding *) malloc (offsetof (struct binding, domainname) + len);
234*36ac495dSmrg
235*36ac495dSmrg if (__builtin_expect (new_binding == NULL, 0))
236*36ac495dSmrg goto failed;
237*36ac495dSmrg
238*36ac495dSmrg memcpy (new_binding->domainname, domainname, len);
239*36ac495dSmrg
240*36ac495dSmrg if (dirnamep)
241*36ac495dSmrg {
242*36ac495dSmrg const char *dirname = *dirnamep;
243*36ac495dSmrg
244*36ac495dSmrg if (dirname == NULL)
245*36ac495dSmrg /* The default value. */
246*36ac495dSmrg dirname = INTUSE(_nl_default_dirname);
247*36ac495dSmrg else
248*36ac495dSmrg {
249*36ac495dSmrg if (strcmp (dirname, INTUSE(_nl_default_dirname)) == 0)
250*36ac495dSmrg dirname = INTUSE(_nl_default_dirname);
251*36ac495dSmrg else
252*36ac495dSmrg {
253*36ac495dSmrg char *result;
254*36ac495dSmrg #if defined _LIBC || defined HAVE_STRDUP
255*36ac495dSmrg result = strdup (dirname);
256*36ac495dSmrg if (__builtin_expect (result == NULL, 0))
257*36ac495dSmrg goto failed_dirname;
258*36ac495dSmrg #else
259*36ac495dSmrg size_t len = strlen (dirname) + 1;
260*36ac495dSmrg result = (char *) malloc (len);
261*36ac495dSmrg if (__builtin_expect (result == NULL, 0))
262*36ac495dSmrg goto failed_dirname;
263*36ac495dSmrg memcpy (result, dirname, len);
264*36ac495dSmrg #endif
265*36ac495dSmrg dirname = result;
266*36ac495dSmrg }
267*36ac495dSmrg }
268*36ac495dSmrg *dirnamep = dirname;
269*36ac495dSmrg new_binding->dirname = (char *) dirname;
270*36ac495dSmrg }
271*36ac495dSmrg else
272*36ac495dSmrg /* The default value. */
273*36ac495dSmrg new_binding->dirname = (char *) INTUSE(_nl_default_dirname);
274*36ac495dSmrg
275*36ac495dSmrg new_binding->codeset_cntr = 0;
276*36ac495dSmrg
277*36ac495dSmrg if (codesetp)
278*36ac495dSmrg {
279*36ac495dSmrg const char *codeset = *codesetp;
280*36ac495dSmrg
281*36ac495dSmrg if (codeset != NULL)
282*36ac495dSmrg {
283*36ac495dSmrg char *result;
284*36ac495dSmrg
285*36ac495dSmrg #if defined _LIBC || defined HAVE_STRDUP
286*36ac495dSmrg result = strdup (codeset);
287*36ac495dSmrg if (__builtin_expect (result == NULL, 0))
288*36ac495dSmrg goto failed_codeset;
289*36ac495dSmrg #else
290*36ac495dSmrg size_t len = strlen (codeset) + 1;
291*36ac495dSmrg result = (char *) malloc (len);
292*36ac495dSmrg if (__builtin_expect (result == NULL, 0))
293*36ac495dSmrg goto failed_codeset;
294*36ac495dSmrg memcpy (result, codeset, len);
295*36ac495dSmrg #endif
296*36ac495dSmrg codeset = result;
297*36ac495dSmrg new_binding->codeset_cntr++;
298*36ac495dSmrg }
299*36ac495dSmrg *codesetp = codeset;
300*36ac495dSmrg new_binding->codeset = (char *) codeset;
301*36ac495dSmrg }
302*36ac495dSmrg else
303*36ac495dSmrg new_binding->codeset = NULL;
304*36ac495dSmrg
305*36ac495dSmrg /* Now enqueue it. */
306*36ac495dSmrg if (_nl_domain_bindings == NULL
307*36ac495dSmrg || strcmp (domainname, _nl_domain_bindings->domainname) < 0)
308*36ac495dSmrg {
309*36ac495dSmrg new_binding->next = _nl_domain_bindings;
310*36ac495dSmrg _nl_domain_bindings = new_binding;
311*36ac495dSmrg }
312*36ac495dSmrg else
313*36ac495dSmrg {
314*36ac495dSmrg binding = _nl_domain_bindings;
315*36ac495dSmrg while (binding->next != NULL
316*36ac495dSmrg && strcmp (domainname, binding->next->domainname) > 0)
317*36ac495dSmrg binding = binding->next;
318*36ac495dSmrg
319*36ac495dSmrg new_binding->next = binding->next;
320*36ac495dSmrg binding->next = new_binding;
321*36ac495dSmrg }
322*36ac495dSmrg
323*36ac495dSmrg modified = 1;
324*36ac495dSmrg
325*36ac495dSmrg /* Here we deal with memory allocation failures. */
326*36ac495dSmrg if (0)
327*36ac495dSmrg {
328*36ac495dSmrg failed_codeset:
329*36ac495dSmrg if (new_binding->dirname != INTUSE(_nl_default_dirname))
330*36ac495dSmrg free (new_binding->dirname);
331*36ac495dSmrg failed_dirname:
332*36ac495dSmrg free (new_binding);
333*36ac495dSmrg failed:
334*36ac495dSmrg if (dirnamep)
335*36ac495dSmrg *dirnamep = NULL;
336*36ac495dSmrg if (codesetp)
337*36ac495dSmrg *codesetp = NULL;
338*36ac495dSmrg }
339*36ac495dSmrg }
340*36ac495dSmrg
341*36ac495dSmrg /* If we modified any binding, we flush the caches. */
342*36ac495dSmrg if (modified)
343*36ac495dSmrg ++_nl_msg_cat_cntr;
344*36ac495dSmrg
345*36ac495dSmrg __libc_rwlock_unlock (_nl_state_lock);
346*36ac495dSmrg }
347*36ac495dSmrg
348*36ac495dSmrg /* Specify that the DOMAINNAME message catalog will be found
349*36ac495dSmrg in DIRNAME rather than in the system locale data base. */
350*36ac495dSmrg char *
BINDTEXTDOMAIN(domainname,dirname)351*36ac495dSmrg BINDTEXTDOMAIN (domainname, dirname)
352*36ac495dSmrg const char *domainname;
353*36ac495dSmrg const char *dirname;
354*36ac495dSmrg {
355*36ac495dSmrg set_binding_values (domainname, &dirname, NULL);
356*36ac495dSmrg return (char *) dirname;
357*36ac495dSmrg }
358*36ac495dSmrg
359*36ac495dSmrg /* Specify the character encoding in which the messages from the
360*36ac495dSmrg DOMAINNAME message catalog will be returned. */
361*36ac495dSmrg char *
BIND_TEXTDOMAIN_CODESET(domainname,codeset)362*36ac495dSmrg BIND_TEXTDOMAIN_CODESET (domainname, codeset)
363*36ac495dSmrg const char *domainname;
364*36ac495dSmrg const char *codeset;
365*36ac495dSmrg {
366*36ac495dSmrg set_binding_values (domainname, NULL, &codeset);
367*36ac495dSmrg return (char *) codeset;
368*36ac495dSmrg }
369*36ac495dSmrg
370*36ac495dSmrg #ifdef _LIBC
371*36ac495dSmrg /* Aliases for function names in GNU C Library. */
372*36ac495dSmrg weak_alias (__bindtextdomain, bindtextdomain);
373*36ac495dSmrg weak_alias (__bind_textdomain_codeset, bind_textdomain_codeset);
374*36ac495dSmrg #endif
375