dcigettext.c
上传用户:tjfeida
上传日期:2013-03-10
资源大小:1917k
文件大小:32k
源码类别:

Ftp客户端

开发平台:

Visual C++

  1. /* Implementation of the internal dcigettext function.
  2.    Copyright (C) 1995-1999, 2000-2002 Free Software Foundation, Inc.
  3.    This program is free software; you can redistribute it and/or modify it
  4.    under the terms of the GNU Library General Public License as published
  5.    by the Free Software Foundation; either version 2, or (at your option)
  6.    any later version.
  7.    This program is distributed in the hope that it will be useful,
  8.    but WITHOUT ANY WARRANTY; without even the implied warranty of
  9.    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  10.    Library General Public License for more details.
  11.    You should have received a copy of the GNU Library General Public
  12.    License along with this program; if not, write to the Free Software
  13.    Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
  14.    USA.  */
  15. /* Tell glibc's <string.h> to provide a prototype for mempcpy().
  16.    This must come before <config.h> because <config.h> may include
  17.    <features.h>, and once <features.h> has been included, it's too late.  */
  18. #ifndef _GNU_SOURCE
  19. # define _GNU_SOURCE 1
  20. #endif
  21. #ifdef HAVE_CONFIG_H
  22. # include <config.h>
  23. #endif
  24. #include <sys/types.h>
  25. #ifdef __GNUC__
  26. # define alloca __builtin_alloca
  27. # define HAVE_ALLOCA 1
  28. #else
  29. # if defined HAVE_ALLOCA_H || defined _LIBC
  30. #  include <alloca.h>
  31. # else
  32. #  ifdef _AIX
  33.  #pragma alloca
  34. #  else
  35. #   ifndef alloca
  36. char *alloca ();
  37. #   endif
  38. #  endif
  39. # endif
  40. #endif
  41. #include <errno.h>
  42. #ifndef errno
  43. extern int errno;
  44. #endif
  45. #ifndef __set_errno
  46. # define __set_errno(val) errno = (val)
  47. #endif
  48. #include <stddef.h>
  49. #include <stdlib.h>
  50. #include <string.h>
  51. #if defined HAVE_UNISTD_H || defined _LIBC
  52. # include <unistd.h>
  53. #endif
  54. #include <locale.h>
  55. #ifdef _LIBC
  56.   /* Guess whether integer division by zero raises signal SIGFPE.
  57.      Set to 1 only if you know for sure.  In case of doubt, set to 0.  */
  58. # if defined __alpha__ || defined __arm__ || defined __i386__ 
  59.      || defined __m68k__ || defined __s390__
  60. #  define INTDIV0_RAISES_SIGFPE 1
  61. # else
  62. #  define INTDIV0_RAISES_SIGFPE 0
  63. # endif
  64. #endif
  65. #if !INTDIV0_RAISES_SIGFPE
  66. # include <signal.h>
  67. #endif
  68. #if defined HAVE_SYS_PARAM_H || defined _LIBC
  69. # include <sys/param.h>
  70. #endif
  71. #include "gettextP.h"
  72. #include "plural-exp.h"
  73. #ifdef _LIBC
  74. # include <libintl.h>
  75. #else
  76. # include "libgnuintl.h"
  77. #endif
  78. #include "hash-string.h"
  79. /* Thread safetyness.  */
  80. #ifdef _LIBC
  81. # include <bits/libc-lock.h>
  82. #else
  83. /* Provide dummy implementation if this is outside glibc.  */
  84. # define __libc_lock_define_initialized(CLASS, NAME)
  85. # define __libc_lock_lock(NAME)
  86. # define __libc_lock_unlock(NAME)
  87. # define __libc_rwlock_define_initialized(CLASS, NAME)
  88. # define __libc_rwlock_rdlock(NAME)
  89. # define __libc_rwlock_unlock(NAME)
  90. #endif
  91. /* Alignment of types.  */
  92. #if defined __GNUC__ && __GNUC__ >= 2
  93. # define alignof(TYPE) __alignof__ (TYPE)
  94. #else
  95. # define alignof(TYPE) 
  96.     ((int) &((struct { char dummy1; TYPE dummy2; } *) 0)->dummy2)
  97. #endif
  98. /* The internal variables in the standalone libintl.a must have different
  99.    names than the internal variables in GNU libc, otherwise programs
  100.    using libintl.a cannot be linked statically.  */
  101. #if !defined _LIBC
  102. # define _nl_default_default_domain libintl_nl_default_default_domain
  103. # define _nl_current_default_domain libintl_nl_current_default_domain
  104. # define _nl_default_dirname libintl_nl_default_dirname
  105. # define _nl_domain_bindings libintl_nl_domain_bindings
  106. #endif
  107. /* Some compilers, like SunOS4 cc, don't have offsetof in <stddef.h>.  */
  108. #ifndef offsetof
  109. # define offsetof(type,ident) ((size_t)&(((type*)0)->ident))
  110. #endif
  111. /* @@ end of prolog @@ */
  112. #ifdef _LIBC
  113. /* Rename the non ANSI C functions.  This is required by the standard
  114.    because some ANSI C functions will require linking with this object
  115.    file and the name space must not be polluted.  */
  116. # define getcwd __getcwd
  117. # ifndef stpcpy
  118. #  define stpcpy __stpcpy
  119. # endif
  120. # define tfind __tfind
  121. #else
  122. # if !defined HAVE_GETCWD
  123. char *getwd ();
  124. #  define getcwd(buf, max) getwd (buf)
  125. # else
  126. char *getcwd ();
  127. # endif
  128. # ifndef HAVE_STPCPY
  129. static char *stpcpy PARAMS ((char *dest, const char *src));
  130. # endif
  131. # ifndef HAVE_MEMPCPY
  132. static void *mempcpy PARAMS ((void *dest, const void *src, size_t n));
  133. # endif
  134. #endif
  135. /* Amount to increase buffer size by in each try.  */
  136. #define PATH_INCR 32
  137. /* The following is from pathmax.h.  */
  138. /* Non-POSIX BSD systems might have gcc's limits.h, which doesn't define
  139.    PATH_MAX but might cause redefinition warnings when sys/param.h is
  140.    later included (as on MORE/BSD 4.3).  */
  141. #if defined _POSIX_VERSION || (defined HAVE_LIMITS_H && !defined __GNUC__)
  142. # include <limits.h>
  143. #endif
  144. #ifndef _POSIX_PATH_MAX
  145. # define _POSIX_PATH_MAX 255
  146. #endif
  147. #if !defined PATH_MAX && defined _PC_PATH_MAX
  148. # define PATH_MAX (pathconf ("/", _PC_PATH_MAX) < 1 ? 1024 : pathconf ("/", _PC_PATH_MAX))
  149. #endif
  150. /* Don't include sys/param.h if it already has been.  */
  151. #if defined HAVE_SYS_PARAM_H && !defined PATH_MAX && !defined MAXPATHLEN
  152. # include <sys/param.h>
  153. #endif
  154. #if !defined PATH_MAX && defined MAXPATHLEN
  155. # define PATH_MAX MAXPATHLEN
  156. #endif
  157. #ifndef PATH_MAX
  158. # define PATH_MAX _POSIX_PATH_MAX
  159. #endif
  160. /* Pathname support.
  161.    ISSLASH(C)           tests whether C is a directory separator character.
  162.    IS_ABSOLUTE_PATH(P)  tests whether P is an absolute path.  If it is not,
  163.                         it may be concatenated to a directory pathname.
  164.    IS_PATH_WITH_DIR(P)  tests whether P contains a directory specification.
  165.  */
  166. #if defined _WIN32 || defined __WIN32__ || defined __EMX__ || defined __DJGPP__
  167.   /* Win32, OS/2, DOS */
  168. # define ISSLASH(C) ((C) == '/' || (C) == '\')
  169. # define HAS_DEVICE(P) 
  170.     ((((P)[0] >= 'A' && (P)[0] <= 'Z') || ((P)[0] >= 'a' && (P)[0] <= 'z')) 
  171.      && (P)[1] == ':')
  172. # define IS_ABSOLUTE_PATH(P) (ISSLASH ((P)[0]) || HAS_DEVICE (P))
  173. # define IS_PATH_WITH_DIR(P) 
  174.     (strchr (P, '/') != NULL || strchr (P, '\') != NULL || HAS_DEVICE (P))
  175. #else
  176.   /* Unix */
  177. # define ISSLASH(C) ((C) == '/')
  178. # define IS_ABSOLUTE_PATH(P) ISSLASH ((P)[0])
  179. # define IS_PATH_WITH_DIR(P) (strchr (P, '/') != NULL)
  180. #endif
  181. /* This is the type used for the search tree where known translations
  182.    are stored.  */
  183. struct known_translation_t
  184. {
  185.   /* Domain in which to search.  */
  186.   char *domainname;
  187.   /* The category.  */
  188.   int category;
  189.   /* State of the catalog counter at the point the string was found.  */
  190.   int counter;
  191.   /* Catalog where the string was found.  */
  192.   struct loaded_l10nfile *domain;
  193.   /* And finally the translation.  */
  194.   const char *translation;
  195.   size_t translation_length;
  196.   /* Pointer to the string in question.  */
  197.   char msgid[ZERO];
  198. };
  199. /* Root of the search tree with known translations.  We can use this
  200.    only if the system provides the `tsearch' function family.  */
  201. #if defined HAVE_TSEARCH || defined _LIBC
  202. # include <search.h>
  203. static void *root;
  204. # ifdef _LIBC
  205. #  define tsearch __tsearch
  206. # endif
  207. /* Function to compare two entries in the table of known translations.  */
  208. static int transcmp PARAMS ((const void *p1, const void *p2));
  209. static int
  210. transcmp (p1, p2)
  211.      const void *p1;
  212.      const void *p2;
  213. {
  214.   const struct known_translation_t *s1;
  215.   const struct known_translation_t *s2;
  216.   int result;
  217.   s1 = (const struct known_translation_t *) p1;
  218.   s2 = (const struct known_translation_t *) p2;
  219.   result = strcmp (s1->msgid, s2->msgid);
  220.   if (result == 0)
  221.     {
  222.       result = strcmp (s1->domainname, s2->domainname);
  223.       if (result == 0)
  224. /* We compare the category last (though this is the cheapest
  225.    operation) since it is hopefully always the same (namely
  226.    LC_MESSAGES).  */
  227. result = s1->category - s2->category;
  228.     }
  229.   return result;
  230. }
  231. #endif
  232. /* Name of the default domain used for gettext(3) prior any call to
  233.    textdomain(3).  The default value for this is "messages".  */
  234. const char _nl_default_default_domain[] attribute_hidden = "messages";
  235. /* Value used as the default domain for gettext(3).  */
  236. const char *_nl_current_default_domain attribute_hidden
  237.      = _nl_default_default_domain;
  238. /* Contains the default location of the message catalogs.  */
  239. #if defined __EMX__
  240. extern const char _nl_default_dirname[];
  241. #else
  242. const char _nl_default_dirname[] = LOCALEDIR;
  243. #endif
  244. /* List with bindings of specific domains created by bindtextdomain()
  245.    calls.  */
  246. struct binding *_nl_domain_bindings;
  247. /* Prototypes for local functions.  */
  248. static char *plural_lookup PARAMS ((struct loaded_l10nfile *domain,
  249.     unsigned long int n,
  250.     const char *translation,
  251.     size_t translation_len))
  252.      internal_function;
  253. static const char *category_to_name PARAMS ((int category)) internal_function;
  254. static const char *guess_category_value PARAMS ((int category,
  255.  const char *categoryname))
  256.      internal_function;
  257. /* For those loosing systems which don't have `alloca' we have to add
  258.    some additional code emulating it.  */
  259. #ifdef HAVE_ALLOCA
  260. /* Nothing has to be done.  */
  261. # define freea(p) /* nothing */
  262. # define ADD_BLOCK(list, address) /* nothing */
  263. # define FREE_BLOCKS(list) /* nothing */
  264. #else
  265. struct block_list
  266. {
  267.   void *address;
  268.   struct block_list *next;
  269. };
  270. # define ADD_BLOCK(list, addr)       
  271.   do {       
  272.     struct block_list *newp = (struct block_list *) malloc (sizeof (*newp));  
  273.     /* If we cannot get a free block we cannot add the new element to       
  274.        the list.  */       
  275.     if (newp != NULL) {       
  276.       newp->address = (addr);       
  277.       newp->next = (list);       
  278.       (list) = newp;       
  279.     }       
  280.   } while (0)
  281. # define FREE_BLOCKS(list)       
  282.   do {       
  283.     while (list != NULL) {       
  284.       struct block_list *old = list;       
  285.       list = list->next;       
  286.       free (old->address);       
  287.       free (old);       
  288.     }       
  289.   } while (0)
  290. # undef alloca
  291. # define alloca(size) (malloc (size))
  292. # define freea(p) free (p)
  293. #endif /* have alloca */
  294. #ifdef _LIBC
  295. /* List of blocks allocated for translations.  */
  296. typedef struct transmem_list
  297. {
  298.   struct transmem_list *next;
  299.   char data[ZERO];
  300. } transmem_block_t;
  301. static struct transmem_list *transmem_list;
  302. #else
  303. typedef unsigned char transmem_block_t;
  304. #endif
  305. /* Names for the libintl functions are a problem.  They must not clash
  306.    with existing names and they should follow ANSI C.  But this source
  307.    code is also used in GNU C Library where the names have a __
  308.    prefix.  So we have to make a difference here.  */
  309. #ifdef _LIBC
  310. # define DCIGETTEXT __dcigettext
  311. #else
  312. # define DCIGETTEXT libintl_dcigettext
  313. #endif
  314. /* Lock variable to protect the global data in the gettext implementation.  */
  315. #ifdef _LIBC
  316. __libc_rwlock_define_initialized (, _nl_state_lock attribute_hidden)
  317. #endif
  318. /* Checking whether the binaries runs SUID must be done and glibc provides
  319.    easier methods therefore we make a difference here.  */
  320. #ifdef _LIBC
  321. # define ENABLE_SECURE __libc_enable_secure
  322. # define DETERMINE_SECURE
  323. #else
  324. # ifndef HAVE_GETUID
  325. #  define getuid() 0
  326. # endif
  327. # ifndef HAVE_GETGID
  328. #  define getgid() 0
  329. # endif
  330. # ifndef HAVE_GETEUID
  331. #  define geteuid() getuid()
  332. # endif
  333. # ifndef HAVE_GETEGID
  334. #  define getegid() getgid()
  335. # endif
  336. static int enable_secure;
  337. # define ENABLE_SECURE (enable_secure == 1)
  338. # define DETERMINE_SECURE 
  339.   if (enable_secure == 0)       
  340.     {       
  341.       if (getuid () != geteuid () || getgid () != getegid ())       
  342. enable_secure = 1;       
  343.       else       
  344. enable_secure = -1;       
  345.     }
  346. #endif
  347. /* Get the function to evaluate the plural expression.  */
  348. #include "eval-plural.h"
  349. /* Look up MSGID in the DOMAINNAME message catalog for the current
  350.    CATEGORY locale and, if PLURAL is nonzero, search over string
  351.    depending on the plural form determined by N.  */
  352. char *
  353. DCIGETTEXT (domainname, msgid1, msgid2, plural, n, category)
  354.      const char *domainname;
  355.      const char *msgid1;
  356.      const char *msgid2;
  357.      int plural;
  358.      unsigned long int n;
  359.      int category;
  360. {
  361. #ifndef HAVE_ALLOCA
  362.   struct block_list *block_list = NULL;
  363. #endif
  364.   struct loaded_l10nfile *domain;
  365.   struct binding *binding;
  366.   const char *categoryname;
  367.   const char *categoryvalue;
  368.   char *dirname, *xdomainname;
  369.   char *single_locale;
  370.   char *retval;
  371.   size_t retlen;
  372.   int saved_errno;
  373. #if defined HAVE_TSEARCH || defined _LIBC
  374.   struct known_translation_t *search;
  375.   struct known_translation_t **foundp = NULL;
  376.   size_t msgid_len;
  377. #endif
  378.   size_t domainname_len;
  379.   /* If no real MSGID is given return NULL.  */
  380.   if (msgid1 == NULL)
  381.     return NULL;
  382.   __libc_rwlock_rdlock (_nl_state_lock);
  383.   /* If DOMAINNAME is NULL, we are interested in the default domain.  If
  384.      CATEGORY is not LC_MESSAGES this might not make much sense but the
  385.      definition left this undefined.  */
  386.   if (domainname == NULL)
  387.     domainname = _nl_current_default_domain;
  388.   /* OS/2 specific: backward compatibility with older libintl versions  */
  389. #ifdef LC_MESSAGES_COMPAT
  390.   if (category == LC_MESSAGES_COMPAT)
  391.     category = LC_MESSAGES;
  392. #endif
  393. #if defined HAVE_TSEARCH || defined _LIBC
  394.   msgid_len = strlen (msgid1) + 1;
  395.   /* Try to find the translation among those which we found at
  396.      some time.  */
  397.   search = (struct known_translation_t *)
  398.    alloca (offsetof (struct known_translation_t, msgid) + msgid_len);
  399.   memcpy (search->msgid, msgid1, msgid_len);
  400.   search->domainname = (char *) domainname;
  401.   search->category = category;
  402.   foundp = (struct known_translation_t **) tfind (search, &root, transcmp);
  403.   freea (search);
  404.   if (foundp != NULL && (*foundp)->counter == _nl_msg_cat_cntr)
  405.     {
  406.       /* Now deal with plural.  */
  407.       if (plural)
  408. retval = plural_lookup ((*foundp)->domain, n, (*foundp)->translation,
  409. (*foundp)->translation_length);
  410.       else
  411. retval = (char *) (*foundp)->translation;
  412.       __libc_rwlock_unlock (_nl_state_lock);
  413.       return retval;
  414.     }
  415. #endif
  416.   /* Preserve the `errno' value.  */
  417.   saved_errno = errno;
  418.   /* See whether this is a SUID binary or not.  */
  419.   DETERMINE_SECURE;
  420.   /* First find matching binding.  */
  421.   for (binding = _nl_domain_bindings; binding != NULL; binding = binding->next)
  422.     {
  423.       int compare = strcmp (domainname, binding->domainname);
  424.       if (compare == 0)
  425. /* We found it!  */
  426. break;
  427.       if (compare < 0)
  428. {
  429.   /* It is not in the list.  */
  430.   binding = NULL;
  431.   break;
  432. }
  433.     }
  434.   if (binding == NULL)
  435.     dirname = (char *) _nl_default_dirname;
  436.   else if (IS_ABSOLUTE_PATH (binding->dirname))
  437.     dirname = binding->dirname;
  438.   else
  439.     {
  440.       /* We have a relative path.  Make it absolute now.  */
  441.       size_t dirname_len = strlen (binding->dirname) + 1;
  442.       size_t path_max;
  443.       char *ret;
  444.       path_max = (unsigned int) PATH_MAX;
  445.       path_max += 2; /* The getcwd docs say to do this.  */
  446.       for (;;)
  447. {
  448.   dirname = (char *) alloca (path_max + dirname_len);
  449.   ADD_BLOCK (block_list, dirname);
  450.   __set_errno (0);
  451.   ret = getcwd (dirname, path_max);
  452.   if (ret != NULL || errno != ERANGE)
  453.     break;
  454.   path_max += path_max / 2;
  455.   path_max += PATH_INCR;
  456. }
  457.       if (ret == NULL)
  458. {
  459.   /* We cannot get the current working directory.  Don't signal an
  460.      error but simply return the default string.  */
  461.   FREE_BLOCKS (block_list);
  462.   __libc_rwlock_unlock (_nl_state_lock);
  463.   __set_errno (saved_errno);
  464.   return (plural == 0
  465.   ? (char *) msgid1
  466.   /* Use the Germanic plural rule.  */
  467.   : n == 1 ? (char *) msgid1 : (char *) msgid2);
  468. }
  469.       stpcpy (stpcpy (strchr (dirname, ''), "/"), binding->dirname);
  470.     }
  471.   /* Now determine the symbolic name of CATEGORY and its value.  */
  472.   categoryname = category_to_name (category);
  473.   categoryvalue = guess_category_value (category, categoryname);
  474.   domainname_len = strlen (domainname);
  475.   xdomainname = (char *) alloca (strlen (categoryname)
  476.  + domainname_len + 5);
  477.   ADD_BLOCK (block_list, xdomainname);
  478.   stpcpy (mempcpy (stpcpy (stpcpy (xdomainname, categoryname), "/"),
  479.   domainname, domainname_len),
  480.   ".mo");
  481.   /* Creating working area.  */
  482.   single_locale = (char *) alloca (strlen (categoryvalue) + 1);
  483.   ADD_BLOCK (block_list, single_locale);
  484.   /* Search for the given string.  This is a loop because we perhaps
  485.      got an ordered list of languages to consider for the translation.  */
  486.   while (1)
  487.     {
  488.       /* Make CATEGORYVALUE point to the next element of the list.  */
  489.       while (categoryvalue[0] != '' && categoryvalue[0] == ':')
  490. ++categoryvalue;
  491.       if (categoryvalue[0] == '')
  492. {
  493.   /* The whole contents of CATEGORYVALUE has been searched but
  494.      no valid entry has been found.  We solve this situation
  495.      by implicitly appending a "C" entry, i.e. no translation
  496.      will take place.  */
  497.   single_locale[0] = 'C';
  498.   single_locale[1] = '';
  499. }
  500.       else
  501. {
  502.   char *cp = single_locale;
  503.   while (categoryvalue[0] != '' && categoryvalue[0] != ':')
  504.     *cp++ = *categoryvalue++;
  505.   *cp = '';
  506.   /* When this is a SUID binary we must not allow accessing files
  507.      outside the dedicated directories.  */
  508.   if (ENABLE_SECURE && IS_PATH_WITH_DIR (single_locale))
  509.     /* Ingore this entry.  */
  510.     continue;
  511. }
  512.       /* If the current locale value is C (or POSIX) we don't load a
  513.  domain.  Return the MSGID.  */
  514.       if (strcmp (single_locale, "C") == 0
  515.   || strcmp (single_locale, "POSIX") == 0)
  516. {
  517.   FREE_BLOCKS (block_list);
  518.   __libc_rwlock_unlock (_nl_state_lock);
  519.   __set_errno (saved_errno);
  520.   return (plural == 0
  521.   ? (char *) msgid1
  522.   /* Use the Germanic plural rule.  */
  523.   : n == 1 ? (char *) msgid1 : (char *) msgid2);
  524. }
  525.       /* Find structure describing the message catalog matching the
  526.  DOMAINNAME and CATEGORY.  */
  527.       domain = _nl_find_domain (dirname, single_locale, xdomainname, binding);
  528.       if (domain != NULL)
  529. {
  530.   retval = _nl_find_msg (domain, binding, msgid1, &retlen);
  531.   if (retval == NULL)
  532.     {
  533.       int cnt;
  534.       for (cnt = 0; domain->successor[cnt] != NULL; ++cnt)
  535. {
  536.   retval = _nl_find_msg (domain->successor[cnt], binding,
  537.  msgid1, &retlen);
  538.   if (retval != NULL)
  539.     {
  540.       domain = domain->successor[cnt];
  541.       break;
  542.     }
  543. }
  544.     }
  545.   if (retval != NULL)
  546.     {
  547.       /* Found the translation of MSGID1 in domain DOMAIN:
  548.  starting at RETVAL, RETLEN bytes.  */
  549.       FREE_BLOCKS (block_list);
  550.       __set_errno (saved_errno);
  551. #if defined HAVE_TSEARCH || defined _LIBC
  552.       if (foundp == NULL)
  553. {
  554.   /* Create a new entry and add it to the search tree.  */
  555.   struct known_translation_t *newp;
  556.   newp = (struct known_translation_t *)
  557.     malloc (offsetof (struct known_translation_t, msgid)
  558.     + msgid_len + domainname_len + 1);
  559.   if (newp != NULL)
  560.     {
  561.       newp->domainname =
  562. mempcpy (newp->msgid, msgid1, msgid_len);
  563.       memcpy (newp->domainname, domainname, domainname_len + 1);
  564.       newp->category = category;
  565.       newp->counter = _nl_msg_cat_cntr;
  566.       newp->domain = domain;
  567.       newp->translation = retval;
  568.       newp->translation_length = retlen;
  569.       /* Insert the entry in the search tree.  */
  570.       foundp = (struct known_translation_t **)
  571. tsearch (newp, &root, transcmp);
  572.       if (foundp == NULL
  573.   || __builtin_expect (*foundp != newp, 0))
  574. /* The insert failed.  */
  575. free (newp);
  576.     }
  577. }
  578.       else
  579. {
  580.   /* We can update the existing entry.  */
  581.   (*foundp)->counter = _nl_msg_cat_cntr;
  582.   (*foundp)->domain = domain;
  583.   (*foundp)->translation = retval;
  584.   (*foundp)->translation_length = retlen;
  585. }
  586. #endif
  587.       /* Now deal with plural.  */
  588.       if (plural)
  589. retval = plural_lookup (domain, n, retval, retlen);
  590.       __libc_rwlock_unlock (_nl_state_lock);
  591.       return retval;
  592.     }
  593. }
  594.     }
  595.   /* NOTREACHED */
  596. }
  597. char *
  598. internal_function
  599. _nl_find_msg (domain_file, domainbinding, msgid, lengthp)
  600.      struct loaded_l10nfile *domain_file;
  601.      struct binding *domainbinding;
  602.      const char *msgid;
  603.      size_t *lengthp;
  604. {
  605.   struct loaded_domain *domain;
  606.   nls_uint32 nstrings;
  607.   size_t act;
  608.   char *result;
  609.   size_t resultlen;
  610.   if (domain_file->decided == 0)
  611.     _nl_load_domain (domain_file, domainbinding);
  612.   if (domain_file->data == NULL)
  613.     return NULL;
  614.   domain = (struct loaded_domain *) domain_file->data;
  615.   nstrings = domain->nstrings;
  616.   /* Locate the MSGID and its translation.  */
  617.   if (domain->hash_tab != NULL)
  618.     {
  619.       /* Use the hashing table.  */
  620.       nls_uint32 len = strlen (msgid);
  621.       nls_uint32 hash_val = hash_string (msgid);
  622.       nls_uint32 idx = hash_val % domain->hash_size;
  623.       nls_uint32 incr = 1 + (hash_val % (domain->hash_size - 2));
  624.       while (1)
  625. {
  626.   nls_uint32 nstr =
  627.     W (domain->must_swap_hash_tab, domain->hash_tab[idx]);
  628.   if (nstr == 0)
  629.     /* Hash table entry is empty.  */
  630.     return NULL;
  631.   nstr--;
  632.   /* Compare msgid with the original string at index nstr.
  633.      We compare the lengths with >=, not ==, because plural entries
  634.      are represented by strings with an embedded NUL.  */
  635.   if (nstr < nstrings
  636.       ? W (domain->must_swap, domain->orig_tab[nstr].length) >= len
  637. && (strcmp (msgid,
  638.     domain->data + W (domain->must_swap,
  639.       domain->orig_tab[nstr].offset))
  640.     == 0)
  641.       : domain->orig_sysdep_tab[nstr - nstrings].length > len
  642. && (strcmp (msgid,
  643.     domain->orig_sysdep_tab[nstr - nstrings].pointer)
  644.     == 0))
  645.     {
  646.       act = nstr;
  647.       goto found;
  648.     }
  649.   if (idx >= domain->hash_size - incr)
  650.     idx -= domain->hash_size - incr;
  651.   else
  652.     idx += incr;
  653. }
  654.       /* NOTREACHED */
  655.     }
  656.   else
  657.     {
  658.       /* Try the default method:  binary search in the sorted array of
  659.  messages.  */
  660.       size_t top, bottom;
  661.       bottom = 0;
  662.       top = nstrings;
  663.       while (bottom < top)
  664. {
  665.   int cmp_val;
  666.   act = (bottom + top) / 2;
  667.   cmp_val = strcmp (msgid, (domain->data
  668.     + W (domain->must_swap,
  669.  domain->orig_tab[act].offset)));
  670.   if (cmp_val < 0)
  671.     top = act;
  672.   else if (cmp_val > 0)
  673.     bottom = act + 1;
  674.   else
  675.     goto found;
  676. }
  677.       /* No translation was found.  */
  678.       return NULL;
  679.     }
  680.  found:
  681.   /* The translation was found at index ACT.  If we have to convert the
  682.      string to use a different character set, this is the time.  */
  683.   if (act < nstrings)
  684.     {
  685.       result = (char *)
  686. (domain->data + W (domain->must_swap, domain->trans_tab[act].offset));
  687.       resultlen = W (domain->must_swap, domain->trans_tab[act].length) + 1;
  688.     }
  689.   else
  690.     {
  691.       result = (char *) domain->trans_sysdep_tab[act - nstrings].pointer;
  692.       resultlen = domain->trans_sysdep_tab[act - nstrings].length;
  693.     }
  694. #if defined _LIBC || HAVE_ICONV
  695.   if (domain->codeset_cntr
  696.       != (domainbinding != NULL ? domainbinding->codeset_cntr : 0))
  697.     {
  698.       /* The domain's codeset has changed through bind_textdomain_codeset()
  699.  since the message catalog was initialized or last accessed.  We
  700.  have to reinitialize the converter.  */
  701.       _nl_free_domain_conv (domain);
  702.       _nl_init_domain_conv (domain_file, domain, domainbinding);
  703.     }
  704.   if (
  705. # ifdef _LIBC
  706.       domain->conv != (__gconv_t) -1
  707. # else
  708. #  if HAVE_ICONV
  709.       domain->conv != (iconv_t) -1
  710. #  endif
  711. # endif
  712.       )
  713.     {
  714.       /* We are supposed to do a conversion.  First allocate an
  715.  appropriate table with the same structure as the table
  716.  of translations in the file, where we can put the pointers
  717.  to the converted strings in.
  718.  There is a slight complication with plural entries.  They
  719.  are represented by consecutive NUL terminated strings.  We
  720.  handle this case by converting RESULTLEN bytes, including
  721.  NULs.  */
  722.       if (domain->conv_tab == NULL
  723.   && ((domain->conv_tab =
  724.  (char **) calloc (nstrings + domain->n_sysdep_strings,
  725.    sizeof (char *)))
  726.       == NULL))
  727. /* Mark that we didn't succeed allocating a table.  */
  728. domain->conv_tab = (char **) -1;
  729.       if (__builtin_expect (domain->conv_tab == (char **) -1, 0))
  730. /* Nothing we can do, no more memory.  */
  731. goto converted;
  732.       if (domain->conv_tab[act] == NULL)
  733. {
  734.   /* We haven't used this string so far, so it is not
  735.      translated yet.  Do this now.  */
  736.   /* We use a bit more efficient memory handling.
  737.      We allocate always larger blocks which get used over
  738.      time.  This is faster than many small allocations.   */
  739.   __libc_lock_define_initialized (static, lock)
  740. # define INITIAL_BLOCK_SIZE 4080
  741.   static unsigned char *freemem;
  742.   static size_t freemem_size;
  743.   const unsigned char *inbuf;
  744.   unsigned char *outbuf;
  745.   int malloc_count;
  746. # ifndef _LIBC
  747.   transmem_block_t *transmem_list = NULL;
  748. # endif
  749.   __libc_lock_lock (lock);
  750.   inbuf = (const unsigned char *) result;
  751.   outbuf = freemem + sizeof (size_t);
  752.   malloc_count = 0;
  753.   while (1)
  754.     {
  755.       transmem_block_t *newmem;
  756. # ifdef _LIBC
  757.       size_t non_reversible;
  758.       int res;
  759.       if (freemem_size < sizeof (size_t))
  760. goto resize_freemem;
  761.       res = __gconv (domain->conv,
  762.      &inbuf, inbuf + resultlen,
  763.      &outbuf,
  764.      outbuf + freemem_size - sizeof (size_t),
  765.      &non_reversible);
  766.       if (res == __GCONV_OK || res == __GCONV_EMPTY_INPUT)
  767. break;
  768.       if (res != __GCONV_FULL_OUTPUT)
  769. {
  770.   __libc_lock_unlock (lock);
  771.   goto converted;
  772. }
  773.       inbuf = result;
  774. # else
  775. #  if HAVE_ICONV
  776.       const char *inptr = (const char *) inbuf;
  777.       size_t inleft = resultlen;
  778.       char *outptr = (char *) outbuf;
  779.       size_t outleft;
  780.       if (freemem_size < sizeof (size_t))
  781. goto resize_freemem;
  782.       outleft = freemem_size - sizeof (size_t);
  783.       if (iconv (domain->conv,
  784.  (ICONV_CONST char **) &inptr, &inleft,
  785.  &outptr, &outleft)
  786.   != (size_t) (-1))
  787. {
  788.   outbuf = (unsigned char *) outptr;
  789.   break;
  790. }
  791.       if (errno != E2BIG)
  792. {
  793.   __libc_lock_unlock (lock);
  794.   goto converted;
  795. }
  796. #  endif
  797. # endif
  798.     resize_freemem:
  799.       /* We must allocate a new buffer or resize the old one.  */
  800.       if (malloc_count > 0)
  801. {
  802.   ++malloc_count;
  803.   freemem_size = malloc_count * INITIAL_BLOCK_SIZE;
  804.   newmem = (transmem_block_t *) realloc (transmem_list,
  805.  freemem_size);
  806. # ifdef _LIBC
  807.   if (newmem != NULL)
  808.     transmem_list = transmem_list->next;
  809.   else
  810.     {
  811.       struct transmem_list *old = transmem_list;
  812.       transmem_list = transmem_list->next;
  813.       free (old);
  814.     }
  815. # endif
  816. }
  817.       else
  818. {
  819.   malloc_count = 1;
  820.   freemem_size = INITIAL_BLOCK_SIZE;
  821.   newmem = (transmem_block_t *) malloc (freemem_size);
  822. }
  823.       if (__builtin_expect (newmem == NULL, 0))
  824. {
  825.   freemem = NULL;
  826.   freemem_size = 0;
  827.   __libc_lock_unlock (lock);
  828.   goto converted;
  829. }
  830. # ifdef _LIBC
  831.       /* Add the block to the list of blocks we have to free
  832.                  at some point.  */
  833.       newmem->next = transmem_list;
  834.       transmem_list = newmem;
  835.       freemem = newmem->data;
  836.       freemem_size -= offsetof (struct transmem_list, data);
  837. # else
  838.       transmem_list = newmem;
  839.       freemem = newmem;
  840. # endif
  841.       outbuf = freemem + sizeof (size_t);
  842.     }
  843.   /* We have now in our buffer a converted string.  Put this
  844.      into the table of conversions.  */
  845.   *(size_t *) freemem = outbuf - freemem - sizeof (size_t);
  846.   domain->conv_tab[act] = (char *) freemem;
  847.   /* Shrink freemem, but keep it aligned.  */
  848.   freemem_size -= outbuf - freemem;
  849.   freemem = outbuf;
  850.   freemem += freemem_size & (alignof (size_t) - 1);
  851.   freemem_size = freemem_size & ~ (alignof (size_t) - 1);
  852.   __libc_lock_unlock (lock);
  853. }
  854.       /* Now domain->conv_tab[act] contains the translation of all
  855.  the plural variants.  */
  856.       result = domain->conv_tab[act] + sizeof (size_t);
  857.       resultlen = *(size_t *) domain->conv_tab[act];
  858.     }
  859.  converted:
  860.   /* The result string is converted.  */
  861. #endif /* _LIBC || HAVE_ICONV */
  862.   *lengthp = resultlen;
  863.   return result;
  864. }
  865. /* Look up a plural variant.  */
  866. static char *
  867. internal_function
  868. plural_lookup (domain, n, translation, translation_len)
  869.      struct loaded_l10nfile *domain;
  870.      unsigned long int n;
  871.      const char *translation;
  872.      size_t translation_len;
  873. {
  874.   struct loaded_domain *domaindata = (struct loaded_domain *) domain->data;
  875.   unsigned long int index;
  876.   const char *p;
  877.   index = plural_eval (domaindata->plural, n);
  878.   if (index >= domaindata->nplurals)
  879.     /* This should never happen.  It means the plural expression and the
  880.        given maximum value do not match.  */
  881.     index = 0;
  882.   /* Skip INDEX strings at TRANSLATION.  */
  883.   p = translation;
  884.   while (index-- > 0)
  885.     {
  886. #ifdef _LIBC
  887.       p = __rawmemchr (p, '');
  888. #else
  889.       p = strchr (p, '');
  890. #endif
  891.       /* And skip over the NUL byte.  */
  892.       p++;
  893.       if (p >= translation + translation_len)
  894. /* This should never happen.  It means the plural expression
  895.    evaluated to a value larger than the number of variants
  896.    available for MSGID1.  */
  897. return (char *) translation;
  898.     }
  899.   return (char *) p;
  900. }
  901. /* Return string representation of locale CATEGORY.  */
  902. static const char *
  903. internal_function
  904. category_to_name (category)
  905.      int category;
  906. {
  907.   const char *retval;
  908.   switch (category)
  909.   {
  910. #ifdef LC_COLLATE
  911.   case LC_COLLATE:
  912.     retval = "LC_COLLATE";
  913.     break;
  914. #endif
  915. #ifdef LC_CTYPE
  916.   case LC_CTYPE:
  917.     retval = "LC_CTYPE";
  918.     break;
  919. #endif
  920. #ifdef LC_MONETARY
  921.   case LC_MONETARY:
  922.     retval = "LC_MONETARY";
  923.     break;
  924. #endif
  925. #ifdef LC_NUMERIC
  926.   case LC_NUMERIC:
  927.     retval = "LC_NUMERIC";
  928.     break;
  929. #endif
  930. #ifdef LC_TIME
  931.   case LC_TIME:
  932.     retval = "LC_TIME";
  933.     break;
  934. #endif
  935. #ifdef LC_MESSAGES
  936.   case LC_MESSAGES:
  937.     retval = "LC_MESSAGES";
  938.     break;
  939. #endif
  940. #ifdef LC_RESPONSE
  941.   case LC_RESPONSE:
  942.     retval = "LC_RESPONSE";
  943.     break;
  944. #endif
  945. #ifdef LC_ALL
  946.   case LC_ALL:
  947.     /* This might not make sense but is perhaps better than any other
  948.        value.  */
  949.     retval = "LC_ALL";
  950.     break;
  951. #endif
  952.   default:
  953.     /* If you have a better idea for a default value let me know.  */
  954.     retval = "LC_XXX";
  955.   }
  956.   return retval;
  957. }
  958. /* Guess value of current locale from value of the environment variables.  */
  959. static const char *
  960. internal_function
  961. guess_category_value (category, categoryname)
  962.      int category;
  963.      const char *categoryname;
  964. {
  965.   const char *language;
  966.   const char *retval;
  967.   /* The highest priority value is the `LANGUAGE' environment
  968.      variable.  But we don't use the value if the currently selected
  969.      locale is the C locale.  This is a GNU extension.  */
  970.   language = getenv ("LANGUAGE");
  971.   if (language != NULL && language[0] == '')
  972.     language = NULL;
  973.   /* We have to proceed with the POSIX methods of looking to `LC_ALL',
  974.      `LC_xxx', and `LANG'.  On some systems this can be done by the
  975.      `setlocale' function itself.  */
  976. #ifdef _LIBC
  977.   retval = setlocale (category, NULL);
  978. #else
  979.   retval = _nl_locale_name (category, categoryname);
  980. #endif
  981.   /* Ignore LANGUAGE if the locale is set to "C" because
  982.      1. "C" locale usually uses the ASCII encoding, and most international
  983. messages use non-ASCII characters. These characters get displayed
  984. as question marks (if using glibc's iconv()) or as invalid 8-bit
  985. characters (because other iconv()s refuse to convert most non-ASCII
  986. characters to ASCII). In any case, the output is ugly.
  987.      2. The precise output of some programs in the "C" locale is specified
  988. by POSIX and should not depend on environment variables like
  989. "LANGUAGE".  We allow such programs to use gettext().  */
  990.   return language != NULL && strcmp (retval, "C") != 0 ? language : retval;
  991. }
  992. /* @@ begin of epilog @@ */
  993. /* We don't want libintl.a to depend on any other library.  So we
  994.    avoid the non-standard function stpcpy.  In GNU C Library this
  995.    function is available, though.  Also allow the symbol HAVE_STPCPY
  996.    to be defined.  */
  997. #if !_LIBC && !HAVE_STPCPY
  998. static char *
  999. stpcpy (dest, src)
  1000.      char *dest;
  1001.      const char *src;
  1002. {
  1003.   while ((*dest++ = *src++) != '')
  1004.     /* Do nothing. */ ;
  1005.   return dest - 1;
  1006. }
  1007. #endif
  1008. #if !_LIBC && !HAVE_MEMPCPY
  1009. static void *
  1010. mempcpy (dest, src, n)
  1011.      void *dest;
  1012.      const void *src;
  1013.      size_t n;
  1014. {
  1015.   return (void *) ((char *) memcpy (dest, src, n) + n);
  1016. }
  1017. #endif
  1018. #ifdef _LIBC
  1019. /* If we want to free all resources we have to do some work at
  1020.    program's end.  */
  1021. static void __attribute__ ((unused))
  1022. free_mem (void)
  1023. {
  1024.   void *old;
  1025.   while (_nl_domain_bindings != NULL)
  1026.     {
  1027.       struct binding *oldp = _nl_domain_bindings;
  1028.       _nl_domain_bindings = _nl_domain_bindings->next;
  1029.       if (oldp->dirname != _nl_default_dirname)
  1030. /* Yes, this is a pointer comparison.  */
  1031. free (oldp->dirname);
  1032.       free (oldp->codeset);
  1033.       free (oldp);
  1034.     }
  1035.   if (_nl_current_default_domain != _nl_default_default_domain)
  1036.     /* Yes, again a pointer comparison.  */
  1037.     free ((char *) _nl_current_default_domain);
  1038.   /* Remove the search tree with the known translations.  */
  1039.   __tdestroy (root, free);
  1040.   root = NULL;
  1041.   while (transmem_list != NULL)
  1042.     {
  1043.       old = transmem_list;
  1044.       transmem_list = transmem_list->next;
  1045.       free (old);
  1046.     }
  1047. }
  1048. text_set_element (__libc_subfreeres, free_mem);
  1049. #endif