xref: /netbsd-src/external/gpl3/gcc.old/dist/intl/bindtextdom.c (revision 36ac495d2b3ea2b9d96377b2143ebfedac224b92)
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