(root)/
m4-1.4.19/
tests/
test-localename.c
       1  /* Test of gl_locale_name function and its variants.
       2     Copyright (C) 2007-2021 Free Software Foundation, Inc.
       3  
       4     This program is free software: you can redistribute it and/or modify
       5     it under the terms of the GNU General Public License as published by
       6     the Free Software Foundation; either version 3 of the License, or
       7     (at your option) any later version.
       8  
       9     This program 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 General Public License for more details.
      13  
      14     You should have received a copy of the GNU 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>, 2007.  */
      18  
      19  #include <config.h>
      20  
      21  #include "localename.h"
      22  
      23  #include <locale.h>
      24  #include <stdlib.h>
      25  #include <string.h>
      26  
      27  #include "macros.h"
      28  
      29  #if HAVE_WORKING_NEWLOCALE && HAVE_WORKING_USELOCALE && !HAVE_FAKE_LOCALES
      30  # define HAVE_GOOD_USELOCALE 1
      31  #endif
      32  
      33  
      34  #if HAVE_GOOD_USELOCALE
      35  
      36  static struct { int cat; int mask; const char *string; } const categories[] =
      37    {
      38        { LC_CTYPE,          LC_CTYPE_MASK,          "LC_CTYPE" },
      39        { LC_NUMERIC,        LC_NUMERIC_MASK,        "LC_NUMERIC" },
      40        { LC_TIME,           LC_TIME_MASK,           "LC_TIME" },
      41        { LC_COLLATE,        LC_COLLATE_MASK,        "LC_COLLATE" },
      42        { LC_MONETARY,       LC_MONETARY_MASK,       "LC_MONETARY" },
      43        { LC_MESSAGES,       LC_MESSAGES_MASK,       "LC_MESSAGES" }
      44  # ifdef LC_PAPER
      45      , { LC_PAPER,          LC_PAPER_MASK,          "LC_PAPER" }
      46  # endif
      47  # ifdef LC_NAME
      48      , { LC_NAME,           LC_NAME_MASK,           "LC_NAME" }
      49  # endif
      50  # ifdef LC_ADDRESS
      51      , { LC_ADDRESS,        LC_ADDRESS_MASK,        "LC_ADDRESS" }
      52  # endif
      53  # ifdef LC_TELEPHONE
      54      , { LC_TELEPHONE,      LC_TELEPHONE_MASK,      "LC_TELEPHONE" }
      55  # endif
      56  # ifdef LC_MEASUREMENT
      57      , { LC_MEASUREMENT,    LC_MEASUREMENT_MASK,    "LC_MEASUREMENT" }
      58  # endif
      59  # ifdef LC_IDENTIFICATION
      60      , { LC_IDENTIFICATION, LC_IDENTIFICATION_MASK, "LC_IDENTIFICATION" }
      61  # endif
      62    };
      63  
      64  #endif
      65  
      66  /* Test the gl_locale_name() function.  */
      67  static void
      68  test_locale_name (void)
      69  {
      70    const char *ret;
      71    const char *name;
      72  
      73    /* Check that gl_locale_name returns non-NULL.  */
      74    ASSERT (gl_locale_name (LC_MESSAGES, "LC_MESSAGES") != NULL);
      75  
      76    /* Get into a defined state,  */
      77    setlocale (LC_ALL, "en_US.UTF-8");
      78  #if HAVE_GOOD_USELOCALE
      79    uselocale (LC_GLOBAL_LOCALE);
      80  #endif
      81  
      82    /* Check that when all environment variables are unset,
      83       gl_locale_name returns the default locale.  */
      84    unsetenv ("LC_ALL");
      85    unsetenv ("LC_CTYPE");
      86    unsetenv ("LC_MESSAGES");
      87    unsetenv ("LC_NUMERIC");
      88    unsetenv ("LANG");
      89    /* Need also to unset all environment variables that specify standard or
      90       non-standard locale categories.  Otherwise, on glibc systems, when some
      91       of these variables are set and reference a nonexistent locale, the
      92       setlocale (LC_ALL, "") call below would fail.  */
      93    unsetenv ("LC_COLLATE");
      94    unsetenv ("LC_MONETARY");
      95    unsetenv ("LC_TIME");
      96    unsetenv ("LC_ADDRESS");
      97    unsetenv ("LC_IDENTIFICATION");
      98    unsetenv ("LC_MEASUREMENT");
      99    unsetenv ("LC_NAME");
     100    unsetenv ("LC_PAPER");
     101    unsetenv ("LC_TELEPHONE");
     102    ret = setlocale (LC_ALL, "");
     103    ASSERT (ret != NULL);
     104    ASSERT (strcmp (gl_locale_name (LC_MESSAGES, "LC_MESSAGES"),
     105                    gl_locale_name_default ()) == 0);
     106    ASSERT (strcmp (gl_locale_name (LC_NUMERIC, "LC_NUMERIC"),
     107                    gl_locale_name_default ()) == 0);
     108  
     109    /* Check that an empty environment variable is treated like an unset
     110       environment variable.  */
     111  
     112    setenv ("LC_ALL", "", 1);
     113    unsetenv ("LC_CTYPE");
     114    unsetenv ("LC_MESSAGES");
     115    unsetenv ("LANG");
     116    setlocale (LC_ALL, "");
     117    ASSERT (strcmp (gl_locale_name (LC_MESSAGES, "LC_MESSAGES"),
     118                    gl_locale_name_default ()) == 0);
     119  
     120    unsetenv ("LC_ALL");
     121    setenv ("LC_CTYPE", "", 1);
     122    unsetenv ("LC_MESSAGES");
     123    unsetenv ("LANG");
     124    setlocale (LC_ALL, "");
     125    ASSERT (strcmp (gl_locale_name (LC_MESSAGES, "LC_MESSAGES"),
     126                    gl_locale_name_default ()) == 0);
     127  
     128    unsetenv ("LC_ALL");
     129    unsetenv ("LC_CTYPE");
     130    setenv ("LC_MESSAGES", "", 1);
     131    unsetenv ("LANG");
     132    setlocale (LC_ALL, "");
     133    ASSERT (strcmp (gl_locale_name (LC_MESSAGES, "LC_MESSAGES"),
     134                    gl_locale_name_default ()) == 0);
     135  
     136    unsetenv ("LC_ALL");
     137    unsetenv ("LC_CTYPE");
     138    unsetenv ("LC_MESSAGES");
     139    setenv ("LANG", "", 1);
     140    setlocale (LC_ALL, "");
     141    ASSERT (strcmp (gl_locale_name (LC_MESSAGES, "LC_MESSAGES"),
     142                    gl_locale_name_default ()) == 0);
     143  
     144    /* Check that LC_ALL overrides the others, and LANG is overridden by the
     145       others.  */
     146  
     147    setenv ("LC_ALL", "C", 1);
     148    unsetenv ("LC_CTYPE");
     149    unsetenv ("LC_MESSAGES");
     150    unsetenv ("LANG");
     151    setlocale (LC_ALL, "");
     152    ASSERT (strcmp (gl_locale_name (LC_MESSAGES, "LC_MESSAGES"), "C") == 0);
     153  
     154    unsetenv ("LC_ALL");
     155    setenv ("LC_CTYPE", "C", 1);
     156    setenv ("LC_MESSAGES", "C", 1);
     157    unsetenv ("LANG");
     158    setlocale (LC_ALL, "");
     159    ASSERT (strcmp (gl_locale_name (LC_MESSAGES, "LC_MESSAGES"), "C") == 0);
     160  
     161    unsetenv ("LC_ALL");
     162    unsetenv ("LC_CTYPE");
     163    unsetenv ("LC_MESSAGES");
     164    setenv ("LANG", "C", 1);
     165    setlocale (LC_ALL, "");
     166    ASSERT (strcmp (gl_locale_name (LC_MESSAGES, "LC_MESSAGES"), "C") == 0);
     167  
     168    /* Check mixed situations.  */
     169  
     170    unsetenv ("LC_ALL");
     171    unsetenv ("LC_CTYPE");
     172    setenv ("LC_MESSAGES", "fr_FR.UTF-8", 1);
     173    setenv ("LANG", "de_DE.UTF-8", 1);
     174    if (setlocale (LC_ALL, "") != NULL)
     175      {
     176        name = gl_locale_name (LC_CTYPE, "LC_CTYPE");
     177  #if defined _WIN32 && !defined __CYGWIN__
     178        /* On native Windows, here,
     179             gl_locale_name_thread (LC_CTYPE, "LC_CTYPE")
     180           returns NULL and
     181             gl_locale_name_posix (LC_CTYPE, "LC_CTYPE")
     182           returns either "de_DE" or "de_DE.UTF-8".  */
     183        ASSERT (strcmp (name, "de_DE") == 0 || strcmp (name, "de_DE.UTF-8") == 0);
     184  #else
     185        ASSERT (strcmp (name, "de_DE.UTF-8") == 0);
     186  #endif
     187        name = gl_locale_name (LC_MESSAGES, "LC_MESSAGES");
     188        ASSERT (strcmp (name, "fr_FR.UTF-8") == 0);
     189      }
     190  
     191    unsetenv ("LC_ALL");
     192    unsetenv ("LC_CTYPE");
     193    setenv ("LC_MESSAGES", "fr_FR.UTF-8", 1);
     194    unsetenv ("LANG");
     195    if (setlocale (LC_ALL, "") != NULL)
     196      {
     197        name = gl_locale_name (LC_CTYPE, "LC_CTYPE");
     198        ASSERT (strcmp (name, gl_locale_name_default ()) == 0);
     199        name = gl_locale_name (LC_MESSAGES, "LC_MESSAGES");
     200        ASSERT (strcmp (name, "fr_FR.UTF-8") == 0);
     201      }
     202  
     203  #if HAVE_GOOD_USELOCALE
     204    /* Check that gl_locale_name considers the thread locale.  */
     205    {
     206      locale_t locale = newlocale (LC_ALL_MASK, "fr_FR.UTF-8", NULL);
     207      if (locale != NULL)
     208        {
     209          uselocale (locale);
     210          name = gl_locale_name (LC_CTYPE, "LC_CTYPE");
     211          ASSERT (strcmp (name, "fr_FR.UTF-8") == 0);
     212          name = gl_locale_name (LC_MESSAGES, "LC_MESSAGES");
     213          ASSERT (strcmp (name, "fr_FR.UTF-8") == 0);
     214          uselocale (LC_GLOBAL_LOCALE);
     215          freelocale (locale);
     216        }
     217    }
     218  
     219    /* Check that gl_locale_name distinguishes different categories of the
     220       thread locale, and that the name is the right one for each.  */
     221    {
     222      unsigned int i;
     223  
     224      for (i = 0; i < SIZEOF (categories); i++)
     225        {
     226          int category_mask = categories[i].mask;
     227          locale_t loc = newlocale (LC_ALL_MASK, "fr_FR.UTF-8", NULL);
     228          if (loc != NULL)
     229            {
     230              locale_t locale = newlocale (category_mask, "de_DE.UTF-8", loc);
     231              if (locale == NULL)
     232                freelocale (loc);
     233              else
     234                {
     235                  unsigned int j;
     236  
     237                  uselocale (locale);
     238                  for (j = 0; j < SIZEOF (categories); j++)
     239                    {
     240                      const char *name_j =
     241                        gl_locale_name (categories[j].cat, categories[j].string);
     242                      if (j == i)
     243                        ASSERT (strcmp (name_j, "de_DE.UTF-8") == 0);
     244                      else
     245                        ASSERT (strcmp (name_j, "fr_FR.UTF-8") == 0);
     246                    }
     247                  uselocale (LC_GLOBAL_LOCALE);
     248                  freelocale (locale);
     249                }
     250            }
     251        }
     252    }
     253  #endif
     254  }
     255  
     256  /* Test the gl_locale_name_thread() function.  */
     257  static void
     258  test_locale_name_thread (void)
     259  {
     260    /* Get into a defined state,  */
     261    setlocale (LC_ALL, "en_US.UTF-8");
     262  
     263  #if HAVE_GOOD_USELOCALE
     264    /* Check that gl_locale_name_thread returns NULL when no thread locale is
     265       set.  */
     266    uselocale (LC_GLOBAL_LOCALE);
     267    ASSERT (gl_locale_name_thread (LC_CTYPE, "LC_CTYPE") == NULL);
     268    ASSERT (gl_locale_name_thread (LC_MESSAGES, "LC_MESSAGES") == NULL);
     269  
     270    /* Check that gl_locale_name_thread considers the thread locale.  */
     271    {
     272      locale_t locale = newlocale (LC_ALL_MASK, "fr_FR.UTF-8", NULL);
     273      if (locale != NULL)
     274        {
     275          const char *name;
     276  
     277          uselocale (locale);
     278          name = gl_locale_name_thread (LC_CTYPE, "LC_CTYPE");
     279          ASSERT (strcmp (name, "fr_FR.UTF-8") == 0);
     280          name = gl_locale_name_thread (LC_MESSAGES, "LC_MESSAGES");
     281          ASSERT (strcmp (name, "fr_FR.UTF-8") == 0);
     282          uselocale (LC_GLOBAL_LOCALE);
     283          freelocale (locale);
     284        }
     285    }
     286  
     287    /* Check that gl_locale_name_thread distinguishes different categories of the
     288       thread locale, and that the name is the right one for each.  */
     289    {
     290      unsigned int i;
     291  
     292      for (i = 0; i < SIZEOF (categories); i++)
     293        {
     294          int category_mask = categories[i].mask;
     295          locale_t loc = newlocale (LC_ALL_MASK, "fr_FR.UTF-8", NULL);
     296          if (loc != NULL)
     297            {
     298              locale_t locale = newlocale (category_mask, "de_DE.UTF-8", loc);
     299              if (locale == NULL)
     300                freelocale (loc);
     301              else
     302                {
     303                  unsigned int j;
     304  
     305                  uselocale (locale);
     306                  for (j = 0; j < SIZEOF (categories); j++)
     307                    {
     308                      const char *name_j =
     309                        gl_locale_name_thread (categories[j].cat,
     310                                               categories[j].string);
     311                      if (j == i)
     312                        ASSERT (strcmp (name_j, "de_DE.UTF-8") == 0);
     313                      else
     314                        ASSERT (strcmp (name_j, "fr_FR.UTF-8") == 0);
     315                    }
     316                  uselocale (LC_GLOBAL_LOCALE);
     317                  freelocale (locale);
     318                }
     319            }
     320        }
     321    }
     322  
     323    /* Check that gl_locale_name_thread returns a string that is allocated with
     324       indefinite extent.  */
     325    {
     326      /* Try many locale names in turn, in order to defeat possible caches.  */
     327      static const char * const choices[] =
     328        {
     329          "C",
     330          "POSIX",
     331          "af_ZA",
     332          "af_ZA.UTF-8",
     333          "am_ET",
     334          "am_ET.UTF-8",
     335          "be_BY",
     336          "be_BY.UTF-8",
     337          "bg_BG",
     338          "bg_BG.UTF-8",
     339          "ca_ES",
     340          "ca_ES.UTF-8",
     341          "cs_CZ",
     342          "cs_CZ.UTF-8",
     343          "da_DK",
     344          "da_DK.UTF-8",
     345          "de_AT",
     346          "de_AT.UTF-8",
     347          "de_CH",
     348          "de_CH.UTF-8",
     349          "de_DE",
     350          "de_DE.UTF-8",
     351          "el_GR",
     352          "el_GR.UTF-8",
     353          "en_AU",
     354          "en_AU.UTF-8",
     355          "en_CA",
     356          "en_CA.UTF-8",
     357          "en_GB",
     358          "en_GB.UTF-8",
     359          "en_IE",
     360          "en_IE.UTF-8",
     361          "en_NZ",
     362          "en_NZ.UTF-8",
     363          "en_US",
     364          "en_US.UTF-8",
     365          "es_ES",
     366          "es_ES.UTF-8",
     367          "et_EE",
     368          "et_EE.UTF-8",
     369          "eu_ES",
     370          "eu_ES.UTF-8",
     371          "fi_FI",
     372          "fi_FI.UTF-8",
     373          "fr_BE",
     374          "fr_BE.UTF-8",
     375          "fr_CA",
     376          "fr_CA.UTF-8",
     377          "fr_CH",
     378          "fr_CH.UTF-8",
     379          "fr_FR",
     380          "fr_FR.UTF-8",
     381          "he_IL",
     382          "he_IL.UTF-8",
     383          "hr_HR",
     384          "hr_HR.UTF-8",
     385          "hu_HU",
     386          "hu_HU.UTF-8",
     387          "hy_AM",
     388          "is_IS",
     389          "is_IS.UTF-8",
     390          "it_CH",
     391          "it_CH.UTF-8",
     392          "it_IT",
     393          "it_IT.UTF-8",
     394          "ja_JP.UTF-8",
     395          "kk_KZ",
     396          "kk_KZ.UTF-8",
     397          "ko_KR.UTF-8",
     398          "lt_LT",
     399          "lt_LT.UTF-8",
     400          "nl_BE",
     401          "nl_BE.UTF-8",
     402          "nl_NL",
     403          "nl_NL.UTF-8",
     404          "no_NO",
     405          "no_NO.UTF-8",
     406          "pl_PL",
     407          "pl_PL.UTF-8",
     408          "pt_BR",
     409          "pt_BR.UTF-8",
     410          "pt_PT",
     411          "pt_PT.UTF-8",
     412          "ro_RO",
     413          "ro_RO.UTF-8",
     414          "ru_RU",
     415          "ru_RU.UTF-8",
     416          "sk_SK",
     417          "sk_SK.UTF-8",
     418          "sl_SI",
     419          "sl_SI.UTF-8",
     420          "sv_SE",
     421          "sv_SE.UTF-8",
     422          "tr_TR",
     423          "tr_TR.UTF-8",
     424          "uk_UA",
     425          "uk_UA.UTF-8",
     426          "zh_CN",
     427          "zh_CN.UTF-8",
     428          "zh_HK",
     429          "zh_HK.UTF-8",
     430          "zh_TW",
     431          "zh_TW.UTF-8"
     432        };
     433      /* Remember which locales are available.  */
     434      unsigned char /* bool */ available[SIZEOF (choices)];
     435      /* Array of remembered results of gl_locale_name_thread.  */
     436      const char *unsaved_names[SIZEOF (choices)][SIZEOF (categories)];
     437      /* Array of remembered results of gl_locale_name_thread, stored in safe
     438         memory.  */
     439      char *saved_names[SIZEOF (choices)][SIZEOF (categories)];
     440      unsigned int j;
     441  
     442      for (j = 0; j < SIZEOF (choices); j++)
     443        {
     444          locale_t locale = newlocale (LC_ALL_MASK, choices[j], NULL);
     445          available[j] = (locale != NULL);
     446          if (locale != NULL)
     447            {
     448              unsigned int i;
     449  
     450              uselocale (locale);
     451              for (i = 0; i < SIZEOF (categories); i++)
     452                {
     453                  unsaved_names[j][i] = gl_locale_name_thread (categories[i].cat, categories[i].string);
     454                  saved_names[j][i] = strdup (unsaved_names[j][i]);
     455                }
     456              uselocale (LC_GLOBAL_LOCALE);
     457              freelocale (locale);
     458            }
     459        }
     460      /* Verify the unsaved_names are still valid.  */
     461      for (j = 0; j < SIZEOF (choices); j++)
     462        if (available[j])
     463          {
     464            unsigned int i;
     465  
     466            for (i = 0; i < SIZEOF (categories); i++)
     467              ASSERT (strcmp (unsaved_names[j][i], saved_names[j][i]) == 0);
     468          }
     469      /* Allocate many locales, without freeing them.  This is an attempt at
     470         overwriting as much of the previously allocated memory as possible.  */
     471      for (j = SIZEOF (choices); j > 0; )
     472        {
     473          j--;
     474          if (available[j])
     475            {
     476              locale_t locale = newlocale (LC_ALL_MASK, choices[j], NULL);
     477              unsigned int i;
     478  
     479              ASSERT (locale != NULL);
     480              uselocale (locale);
     481              for (i = 0; i < SIZEOF (categories); i++)
     482                {
     483                  const char *name = gl_locale_name_thread (categories[i].cat, categories[i].string);
     484                  ASSERT (strcmp (unsaved_names[j][i], name) == 0);
     485                }
     486              uselocale (LC_GLOBAL_LOCALE);
     487              freelocale (locale);
     488            }
     489        }
     490      /* Verify the unsaved_names are still valid.  */
     491      for (j = 0; j < SIZEOF (choices); j++)
     492        if (available[j])
     493          {
     494            unsigned int i;
     495  
     496            for (i = 0; i < SIZEOF (categories); i++)
     497              {
     498                ASSERT (strcmp (unsaved_names[j][i], saved_names[j][i]) == 0);
     499                free (saved_names[j][i]);
     500              }
     501          }
     502    }
     503  #else
     504    /* Check that gl_locale_name_thread always returns NULL.  */
     505    ASSERT (gl_locale_name_thread (LC_CTYPE, "LC_CTYPE") == NULL);
     506    ASSERT (gl_locale_name_thread (LC_MESSAGES, "LC_MESSAGES") == NULL);
     507  #endif
     508  }
     509  
     510  /* Test the gl_locale_name_posix() function.  */
     511  static void
     512  test_locale_name_posix (void)
     513  {
     514    const char *ret;
     515    const char *name;
     516  
     517    /* Get into a defined state,  */
     518    setlocale (LC_ALL, "en_US.UTF-8");
     519  #if HAVE_GOOD_USELOCALE
     520    uselocale (LC_GLOBAL_LOCALE);
     521  #endif
     522  
     523    /* Check that when all environment variables are unset,
     524       gl_locale_name_posix returns either NULL or the default locale.  */
     525    unsetenv ("LC_ALL");
     526    unsetenv ("LC_CTYPE");
     527    unsetenv ("LC_MESSAGES");
     528    unsetenv ("LC_NUMERIC");
     529    unsetenv ("LANG");
     530    /* Need also to unset all environment variables that specify standard or
     531       non-standard locale categories.  Otherwise, on glibc systems, when some
     532       of these variables are set and reference a nonexistent locale, the
     533       setlocale (LC_ALL, "") call below would fail.  */
     534    unsetenv ("LC_COLLATE");
     535    unsetenv ("LC_MONETARY");
     536    unsetenv ("LC_TIME");
     537    unsetenv ("LC_ADDRESS");
     538    unsetenv ("LC_IDENTIFICATION");
     539    unsetenv ("LC_MEASUREMENT");
     540    unsetenv ("LC_NAME");
     541    unsetenv ("LC_PAPER");
     542    unsetenv ("LC_TELEPHONE");
     543    ret = setlocale (LC_ALL, "");
     544    ASSERT (ret != NULL);
     545    name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     546    ASSERT (name == NULL || strcmp (name, gl_locale_name_default ()) == 0);
     547    name = gl_locale_name_posix (LC_NUMERIC, "LC_NUMERIC");
     548    ASSERT (name == NULL || strcmp (name, gl_locale_name_default ()) == 0);
     549  
     550    /* Check that an empty environment variable is treated like an unset
     551       environment variable.  */
     552  
     553    setenv ("LC_ALL", "", 1);
     554    unsetenv ("LC_CTYPE");
     555    unsetenv ("LC_MESSAGES");
     556    unsetenv ("LANG");
     557    setlocale (LC_ALL, "");
     558    name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     559    ASSERT (name == NULL || strcmp (name, gl_locale_name_default ()) == 0);
     560  
     561    unsetenv ("LC_ALL");
     562    setenv ("LC_CTYPE", "", 1);
     563    unsetenv ("LC_MESSAGES");
     564    unsetenv ("LANG");
     565    setlocale (LC_ALL, "");
     566    name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     567    ASSERT (name == NULL || strcmp (name, gl_locale_name_default ()) == 0);
     568  
     569    unsetenv ("LC_ALL");
     570    unsetenv ("LC_CTYPE");
     571    setenv ("LC_MESSAGES", "", 1);
     572    unsetenv ("LANG");
     573    setlocale (LC_ALL, "");
     574    name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     575    ASSERT (name == NULL || strcmp (name, gl_locale_name_default ()) == 0);
     576  
     577    unsetenv ("LC_ALL");
     578    unsetenv ("LC_CTYPE");
     579    unsetenv ("LC_MESSAGES");
     580    setenv ("LANG", "", 1);
     581    setlocale (LC_ALL, "");
     582    name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     583    ASSERT (name == NULL || strcmp (name, gl_locale_name_default ()) == 0);
     584  
     585    /* Check that LC_ALL overrides the others, and LANG is overridden by the
     586       others.  */
     587  
     588    setenv ("LC_ALL", "C", 1);
     589    unsetenv ("LC_CTYPE");
     590    unsetenv ("LC_MESSAGES");
     591    unsetenv ("LANG");
     592    setlocale (LC_ALL, "");
     593    name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     594    ASSERT (strcmp (name, "C") == 0);
     595  
     596    unsetenv ("LC_ALL");
     597    setenv ("LC_CTYPE", "C", 1);
     598    setenv ("LC_MESSAGES", "C", 1);
     599    unsetenv ("LANG");
     600    setlocale (LC_ALL, "");
     601    name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     602    ASSERT (strcmp (name, "C") == 0);
     603  
     604    unsetenv ("LC_ALL");
     605    unsetenv ("LC_CTYPE");
     606    unsetenv ("LC_MESSAGES");
     607    setenv ("LANG", "C", 1);
     608    setlocale (LC_ALL, "");
     609    name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     610    ASSERT (strcmp (name, "C") == 0);
     611  
     612    /* Check mixed situations.  */
     613  
     614    unsetenv ("LC_ALL");
     615    unsetenv ("LC_CTYPE");
     616    setenv ("LC_MESSAGES", "fr_FR.UTF-8", 1);
     617    setenv ("LANG", "de_DE.UTF-8", 1);
     618    if (setlocale (LC_ALL, "") != NULL)
     619      {
     620        name = gl_locale_name_posix (LC_CTYPE, "LC_CTYPE");
     621  #if defined _WIN32 && !defined __CYGWIN__
     622        ASSERT (strcmp (name, "de_DE") == 0 || strcmp (name, "de_DE.UTF-8") == 0);
     623  #else
     624        ASSERT (strcmp (name, "de_DE.UTF-8") == 0);
     625  #endif
     626        name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     627        ASSERT (strcmp (name, "fr_FR.UTF-8") == 0);
     628      }
     629  
     630    unsetenv ("LC_ALL");
     631    unsetenv ("LC_CTYPE");
     632    setenv ("LC_MESSAGES", "fr_FR.UTF-8", 1);
     633    unsetenv ("LANG");
     634    if (setlocale (LC_ALL, "") != NULL)
     635      {
     636        name = gl_locale_name_posix (LC_CTYPE, "LC_CTYPE");
     637        ASSERT (name == NULL || strcmp (name, gl_locale_name_default ()) == 0);
     638        name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     639        ASSERT (strcmp (name, "fr_FR.UTF-8") == 0);
     640      }
     641  
     642  #if HAVE_GOOD_USELOCALE
     643    /* Check that gl_locale_name_posix ignores the thread locale.  */
     644    {
     645      locale_t locale = newlocale (LC_ALL_MASK, "fr_FR.UTF-8", NULL);
     646      if (locale != NULL)
     647        {
     648          unsetenv ("LC_ALL");
     649          unsetenv ("LC_CTYPE");
     650          unsetenv ("LC_MESSAGES");
     651          setenv ("LANG", "C", 1);
     652          setlocale (LC_ALL, "");
     653          uselocale (locale);
     654          name = gl_locale_name_posix (LC_MESSAGES, "LC_MESSAGES");
     655          ASSERT (strcmp (name, "C") == 0);
     656          uselocale (LC_GLOBAL_LOCALE);
     657          freelocale (locale);
     658        }
     659    }
     660  #endif
     661  }
     662  
     663  /* Test the gl_locale_name_environ() function.  */
     664  static void
     665  test_locale_name_environ (void)
     666  {
     667    const char *name;
     668  
     669    /* Get into a defined state,  */
     670    setlocale (LC_ALL, "en_US.UTF-8");
     671  #if HAVE_GOOD_USELOCALE
     672    uselocale (LC_GLOBAL_LOCALE);
     673  #endif
     674  
     675    /* Check that when all environment variables are unset,
     676       gl_locale_name_environ returns NULL.  */
     677    unsetenv ("LC_ALL");
     678    unsetenv ("LC_CTYPE");
     679    unsetenv ("LC_MESSAGES");
     680    unsetenv ("LC_NUMERIC");
     681    unsetenv ("LANG");
     682    ASSERT (gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES") == NULL);
     683    ASSERT (gl_locale_name_environ (LC_NUMERIC, "LC_NUMERIC") == NULL);
     684  
     685    /* Check that an empty environment variable is treated like an unset
     686       environment variable.  */
     687  
     688    setenv ("LC_ALL", "", 1);
     689    unsetenv ("LC_CTYPE");
     690    unsetenv ("LC_MESSAGES");
     691    unsetenv ("LANG");
     692    ASSERT (gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES") == NULL);
     693  
     694    unsetenv ("LC_ALL");
     695    setenv ("LC_CTYPE", "", 1);
     696    unsetenv ("LC_MESSAGES");
     697    unsetenv ("LANG");
     698    ASSERT (gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES") == NULL);
     699  
     700    unsetenv ("LC_ALL");
     701    unsetenv ("LC_CTYPE");
     702    setenv ("LC_MESSAGES", "", 1);
     703    unsetenv ("LANG");
     704    ASSERT (gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES") == NULL);
     705  
     706    unsetenv ("LC_ALL");
     707    unsetenv ("LC_CTYPE");
     708    unsetenv ("LC_MESSAGES");
     709    setenv ("LANG", "", 1);
     710    ASSERT (gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES") == NULL);
     711  
     712    /* Check that LC_ALL overrides the others, and LANG is overridden by the
     713       others.  */
     714  
     715    setenv ("LC_ALL", "C", 1);
     716    unsetenv ("LC_CTYPE");
     717    unsetenv ("LC_MESSAGES");
     718    unsetenv ("LANG");
     719    name = gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES");
     720    ASSERT (strcmp (name, "C") == 0);
     721  
     722    unsetenv ("LC_ALL");
     723    setenv ("LC_CTYPE", "C", 1);
     724    setenv ("LC_MESSAGES", "C", 1);
     725    unsetenv ("LANG");
     726    name = gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES");
     727    ASSERT (strcmp (name, "C") == 0);
     728  
     729    unsetenv ("LC_ALL");
     730    unsetenv ("LC_CTYPE");
     731    unsetenv ("LC_MESSAGES");
     732    setenv ("LANG", "C", 1);
     733    name = gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES");
     734    ASSERT (strcmp (name, "C") == 0);
     735  
     736    /* Check mixed situations.  */
     737  
     738    unsetenv ("LC_ALL");
     739    unsetenv ("LC_CTYPE");
     740    setenv ("LC_MESSAGES", "fr_FR.UTF-8", 1);
     741    setenv ("LANG", "de_DE.UTF-8", 1);
     742    name = gl_locale_name_environ (LC_CTYPE, "LC_CTYPE");
     743    ASSERT (strcmp (name, "de_DE.UTF-8") == 0);
     744    name = gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES");
     745    ASSERT (strcmp (name, "fr_FR.UTF-8") == 0);
     746  
     747    unsetenv ("LC_ALL");
     748    unsetenv ("LC_CTYPE");
     749    setenv ("LC_MESSAGES", "fr_FR.UTF-8", 1);
     750    unsetenv ("LANG");
     751    name = gl_locale_name_environ (LC_CTYPE, "LC_CTYPE");
     752    ASSERT (name == NULL);
     753    name = gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES");
     754    ASSERT (strcmp (name, "fr_FR.UTF-8") == 0);
     755  
     756  #if HAVE_GOOD_USELOCALE
     757    /* Check that gl_locale_name_environ ignores the thread locale.  */
     758    {
     759      locale_t locale = newlocale (LC_ALL_MASK, "fr_FR.UTF-8", NULL);
     760      if (locale != NULL)
     761        {
     762          unsetenv ("LC_ALL");
     763          unsetenv ("LC_CTYPE");
     764          unsetenv ("LC_MESSAGES");
     765          setenv ("LANG", "C", 1);
     766          setlocale (LC_ALL, "");
     767          uselocale (locale);
     768          name = gl_locale_name_environ (LC_MESSAGES, "LC_MESSAGES");
     769          ASSERT (strcmp (name, "C") == 0);
     770          uselocale (LC_GLOBAL_LOCALE);
     771          freelocale (locale);
     772        }
     773    }
     774  #endif
     775  }
     776  
     777  /* Test the gl_locale_name_default() function.  */
     778  static void
     779  test_locale_name_default (void)
     780  {
     781    const char *name = gl_locale_name_default ();
     782  
     783    ASSERT (name != NULL);
     784  
     785    /* Only Mac OS X and Windows have a facility for the user to set the default
     786       locale.  */
     787  #if !((defined __APPLE__ && defined __MACH__) || (defined _WIN32 || defined __CYGWIN__))
     788    ASSERT (strcmp (name, "C") == 0);
     789  #endif
     790  
     791  #if HAVE_GOOD_USELOCALE
     792    /* Check that gl_locale_name_default ignores the thread locale.  */
     793    {
     794      locale_t locale = newlocale (LC_ALL_MASK, "fr_FR.UTF-8", NULL);
     795      if (locale != NULL)
     796        {
     797          uselocale (locale);
     798          ASSERT (strcmp (gl_locale_name_default (), name) == 0);
     799          uselocale (LC_GLOBAL_LOCALE);
     800          freelocale (locale);
     801        }
     802    }
     803  #endif
     804  }
     805  
     806  int
     807  main ()
     808  {
     809    test_locale_name ();
     810    test_locale_name_thread ();
     811    test_locale_name_posix ();
     812    test_locale_name_environ ();
     813    test_locale_name_default ();
     814  
     815    return 0;
     816  }