Blame localedata/tst-setlocale3.c

Packit 6c4009
/* Regression test for setlocale invalid environment variable handling.
Packit 6c4009
   Copyright (C) 2014-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <locale.h>
Packit 6c4009
#include <stdio.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
Packit 6c4009
/* The result of setlocale may be overwritten by subsequent calls, so
Packit 6c4009
   this wrapper makes a copy.  */
Packit 6c4009
static char *
Packit 6c4009
setlocale_copy (int category, const char *locale)
Packit 6c4009
{
Packit 6c4009
  const char *result = setlocale (category, locale);
Packit 6c4009
  if (result == NULL)
Packit 6c4009
    return NULL;
Packit 6c4009
  return strdup (result);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static char *de_locale;
Packit 6c4009
Packit 6c4009
static void
Packit 6c4009
setlocale_fail (const char *envstring)
Packit 6c4009
{
Packit 6c4009
  setenv ("LC_CTYPE", envstring, 1);
Packit 6c4009
  if (setlocale (LC_CTYPE, "") != NULL)
Packit 6c4009
    {
Packit 6c4009
      printf ("unexpected setlocale success for \"%s\" locale\n", envstring);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
  const char *newloc = setlocale (LC_CTYPE, NULL);
Packit 6c4009
  if (strcmp (newloc, de_locale) != 0)
Packit 6c4009
    {
Packit 6c4009
      printf ("failed setlocale call \"%s\" changed locale to \"%s\"\n",
Packit 6c4009
	      envstring, newloc);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static void
Packit 6c4009
setlocale_success (const char *envstring)
Packit 6c4009
{
Packit 6c4009
  setenv ("LC_CTYPE", envstring, 1);
Packit 6c4009
  char *newloc = setlocale_copy (LC_CTYPE, "");
Packit 6c4009
  if (newloc == NULL)
Packit 6c4009
    {
Packit 6c4009
      printf ("setlocale for \"%s\": %m\n", envstring);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
  if (strcmp (newloc, de_locale) == 0)
Packit 6c4009
    {
Packit 6c4009
      printf ("setlocale with LC_CTYPE=\"%s\" left locale at \"%s\"\n",
Packit 6c4009
	      envstring, de_locale);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
  if (setlocale (LC_CTYPE, de_locale) == NULL)
Packit 6c4009
    {
Packit 6c4009
      printf ("restoring locale \"%s\" with LC_CTYPE=\"%s\": %m\n",
Packit 6c4009
	      de_locale, envstring);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
  char *newloc2 = setlocale_copy (LC_CTYPE, newloc);
Packit 6c4009
  if (newloc2 == NULL)
Packit 6c4009
    {
Packit 6c4009
      printf ("restoring locale \"%s\" following \"%s\": %m\n",
Packit 6c4009
	      newloc, envstring);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
  if (strcmp (newloc, newloc2) != 0)
Packit 6c4009
    {
Packit 6c4009
      printf ("representation of locale \"%s\" changed from \"%s\" to \"%s\"",
Packit 6c4009
	      envstring, newloc, newloc2);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
  free (newloc);
Packit 6c4009
  free (newloc2);
Packit 6c4009
Packit 6c4009
  if (setlocale (LC_CTYPE, de_locale) == NULL)
Packit 6c4009
    {
Packit 6c4009
      printf ("restoring locale \"%s\" with LC_CTYPE=\"%s\": %m\n",
Packit 6c4009
	      de_locale, envstring);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Checks that a known-good locale still works if LC_ALL contains a
Packit 6c4009
   value which should be ignored.  */
Packit 6c4009
static void
Packit 6c4009
setlocale_ignore (const char *to_ignore)
Packit 6c4009
{
Packit 6c4009
  const char *fr_locale = "fr_FR.UTF-8";
Packit 6c4009
  setenv ("LC_CTYPE", fr_locale, 1);
Packit 6c4009
  char *expected_locale = setlocale_copy (LC_CTYPE, "");
Packit 6c4009
  if (expected_locale == NULL)
Packit 6c4009
    {
Packit 6c4009
      printf ("setlocale with LC_CTYPE=\"%s\" failed: %m\n", fr_locale);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
  if (setlocale (LC_CTYPE, de_locale) == NULL)
Packit 6c4009
    {
Packit 6c4009
      printf ("failed to restore locale: %m\n");
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
  unsetenv ("LC_CTYPE");
Packit 6c4009
Packit 6c4009
  setenv ("LC_ALL", to_ignore, 1);
Packit 6c4009
  setenv ("LC_CTYPE", fr_locale, 1);
Packit 6c4009
  const char *actual_locale = setlocale (LC_CTYPE, "");
Packit 6c4009
  if (actual_locale == NULL)
Packit 6c4009
    {
Packit 6c4009
      printf ("setlocale with LC_ALL, LC_CTYPE=\"%s\" failed: %m\n",
Packit 6c4009
	      fr_locale);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
  if (strcmp (actual_locale, expected_locale) != 0)
Packit 6c4009
    {
Packit 6c4009
      printf ("setlocale under LC_ALL failed: got \"%s\", expected \"%s\"\n",
Packit 6c4009
	      actual_locale, expected_locale);
Packit 6c4009
      exit (1);
Packit 6c4009
    }
Packit 6c4009
  unsetenv ("LC_CTYPE");
Packit 6c4009
  setlocale_success (fr_locale);
Packit 6c4009
  unsetenv ("LC_ALL");
Packit 6c4009
  free (expected_locale);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static int
Packit 6c4009
do_test (void)
Packit 6c4009
{
Packit 6c4009
  /* The glibc test harness sets this environment variable
Packit 6c4009
     uncondionally.  */
Packit 6c4009
  unsetenv ("LC_ALL");
Packit 6c4009
Packit 6c4009
  de_locale = setlocale_copy (LC_CTYPE, "de_DE.UTF-8");
Packit 6c4009
  if (de_locale == NULL)
Packit 6c4009
    {
Packit 6c4009
      printf ("setlocale (LC_CTYPE, \"de_DE.UTF-8\"): %m\n");
Packit 6c4009
      return 1;
Packit 6c4009
    }
Packit 6c4009
  setlocale_success ("C");
Packit 6c4009
  setlocale_success ("en_US.UTF-8");
Packit 6c4009
  setlocale_success ("/en_US.UTF-8");
Packit 6c4009
  setlocale_success ("//en_US.UTF-8");
Packit 6c4009
  setlocale_ignore ("");
Packit 6c4009
Packit 6c4009
  setlocale_fail ("does-not-exist");
Packit 6c4009
  setlocale_fail ("/");
Packit 6c4009
  setlocale_fail ("/../localedata/en_US.UTF-8");
Packit 6c4009
  setlocale_fail ("en_US.UTF-8/");
Packit 6c4009
  setlocale_fail ("en_US.UTF-8/..");
Packit 6c4009
  setlocale_fail ("en_US.UTF-8/../en_US.UTF-8");
Packit 6c4009
  setlocale_fail ("../localedata/en_US.UTF-8");
Packit 6c4009
  {
Packit 6c4009
    size_t large_length = 1024;
Packit 6c4009
    char *large_name = malloc (large_length + 1);
Packit 6c4009
    if (large_name == NULL)
Packit 6c4009
      {
Packit 6c4009
	puts ("malloc failure");
Packit 6c4009
	return 1;
Packit 6c4009
      }
Packit 6c4009
    memset (large_name, '/', large_length);
Packit 6c4009
    const char *suffix = "en_US.UTF-8";
Packit 6c4009
    strcpy (large_name + large_length - strlen (suffix), suffix);
Packit 6c4009
    setlocale_fail (large_name);
Packit 6c4009
    free (large_name);
Packit 6c4009
  }
Packit 6c4009
  {
Packit 6c4009
    size_t huge_length = 64 * 1024 * 1024;
Packit 6c4009
    char *huge_name = malloc (huge_length + 1);
Packit 6c4009
    if (huge_name == NULL)
Packit 6c4009
      {
Packit 6c4009
	puts ("malloc failure");
Packit 6c4009
	return 1;
Packit 6c4009
      }
Packit 6c4009
    memset (huge_name, 'X', huge_length);
Packit 6c4009
    huge_name[huge_length] = '\0';
Packit 6c4009
    /* Construct a composite locale specification. */
Packit 6c4009
    const char *prefix = "LC_CTYPE=de_DE.UTF-8;LC_TIME=";
Packit 6c4009
    memcpy (huge_name, prefix, strlen (prefix));
Packit 6c4009
    setlocale_fail (huge_name);
Packit 6c4009
    free (huge_name);
Packit 6c4009
  }
Packit 6c4009
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
#define TEST_FUNCTION do_test ()
Packit 6c4009
#include "../test-skeleton.c"