Blame gnulib-tests/test-wcrtomb-w32.c

Packit 33f14e
/* Test of conversion of wide character to multibyte character.
Packit 33f14e
   Copyright (C) 2008-2017 Free Software Foundation, Inc.
Packit 33f14e
Packit 33f14e
   This program is free software: you can redistribute it and/or modify
Packit 33f14e
   it under the terms of the GNU General Public License as published by
Packit 33f14e
   the Free Software Foundation; either version 3 of the License, or
Packit 33f14e
   (at your option) any later version.
Packit 33f14e
Packit 33f14e
   This program is distributed in the hope that it will be useful,
Packit 33f14e
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 33f14e
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 33f14e
   GNU General Public License for more details.
Packit 33f14e
Packit 33f14e
   You should have received a copy of the GNU General Public License
Packit 33f14e
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 33f14e
Packit 33f14e
#include <config.h>
Packit 33f14e
Packit 33f14e
#include <wchar.h>
Packit 33f14e
Packit 33f14e
#include <locale.h>
Packit 33f14e
#include <stdio.h>
Packit 33f14e
#include <stdlib.h>
Packit 33f14e
#include <string.h>
Packit 33f14e
Packit 33f14e
#include "macros.h"
Packit 33f14e
Packit 33f14e
#if (defined _WIN32 || defined __WIN32__) && !defined __CYGWIN__
Packit 33f14e
Packit 33f14e
static int
Packit 33f14e
test_one_locale (const char *name, int codepage)
Packit 33f14e
{
Packit 33f14e
  char buf[64];
Packit 33f14e
  size_t ret;
Packit 33f14e
Packit 33f14e
# if 1
Packit 33f14e
  /* Portable code to set the locale.  */
Packit 33f14e
  {
Packit 33f14e
    char name_with_codepage[1024];
Packit 33f14e
Packit 33f14e
    sprintf (name_with_codepage, "%s.%d", name, codepage);
Packit 33f14e
Packit 33f14e
    /* Set the locale.  */
Packit 33f14e
    if (setlocale (LC_ALL, name_with_codepage) == NULL)
Packit 33f14e
      return 77;
Packit 33f14e
  }
Packit 33f14e
# else
Packit 33f14e
  /* Hacky way to set a locale.codepage combination that setlocale() refuses
Packit 33f14e
     to set.  */
Packit 33f14e
  {
Packit 33f14e
    /* Codepage of the current locale, set with setlocale().
Packit 33f14e
       Not necessarily the same as GetACP().  */
Packit 33f14e
    extern __declspec(dllimport) unsigned int __lc_codepage;
Packit 33f14e
Packit 33f14e
    /* Set the locale.  */
Packit 33f14e
    if (setlocale (LC_ALL, name) == NULL)
Packit 33f14e
      return 77;
Packit 33f14e
Packit 33f14e
    /* Clobber the codepage and MB_CUR_MAX, both set by setlocale().  */
Packit 33f14e
    __lc_codepage = codepage;
Packit 33f14e
    switch (codepage)
Packit 33f14e
      {
Packit 33f14e
      case 1252:
Packit 33f14e
      case 1256:
Packit 33f14e
        MB_CUR_MAX = 1;
Packit 33f14e
        break;
Packit 33f14e
      case 932:
Packit 33f14e
      case 950:
Packit 33f14e
      case 936:
Packit 33f14e
        MB_CUR_MAX = 2;
Packit 33f14e
        break;
Packit 33f14e
      case 54936:
Packit 33f14e
      case 65001:
Packit 33f14e
        MB_CUR_MAX = 4;
Packit 33f14e
        break;
Packit 33f14e
      }
Packit 33f14e
Packit 33f14e
    /* Test whether the codepage is really available.  */
Packit 33f14e
    {
Packit 33f14e
      mbstate_t state;
Packit 33f14e
      wchar_t wc;
Packit 33f14e
Packit 33f14e
      memset (&state, '\0', sizeof (mbstate_t));
Packit 33f14e
      if (mbrtowc (&wc, " ", 1, &state) == (size_t)(-1))
Packit 33f14e
        return 77;
Packit 33f14e
    }
Packit 33f14e
  }
Packit 33f14e
# endif
Packit 33f14e
Packit 33f14e
  /* Test NUL character.  */
Packit 33f14e
  {
Packit 33f14e
    buf[0] = 'x';
Packit 33f14e
    ret = wcrtomb (buf, 0, NULL);
Packit 33f14e
    ASSERT (ret == 1);
Packit 33f14e
    ASSERT (buf[0] == '\0');
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  /* Test single bytes.  */
Packit 33f14e
  {
Packit 33f14e
    int c;
Packit 33f14e
Packit 33f14e
    for (c = 0; c < 0x100; c++)
Packit 33f14e
      switch (c)
Packit 33f14e
        {
Packit 33f14e
        case '\t': case '\v': case '\f':
Packit 33f14e
        case ' ': case '!': case '"': case '#': case '%':
Packit 33f14e
        case '&': case '\'': case '(': case ')': case '*':
Packit 33f14e
        case '+': case ',': case '-': case '.': case '/':
Packit 33f14e
        case '0': case '1': case '2': case '3': case '4':
Packit 33f14e
        case '5': case '6': case '7': case '8': case '9':
Packit 33f14e
        case ':': case ';': case '<': case '=': case '>':
Packit 33f14e
        case '?':
Packit 33f14e
        case 'A': case 'B': case 'C': case 'D': case 'E':
Packit 33f14e
        case 'F': case 'G': case 'H': case 'I': case 'J':
Packit 33f14e
        case 'K': case 'L': case 'M': case 'N': case 'O':
Packit 33f14e
        case 'P': case 'Q': case 'R': case 'S': case 'T':
Packit 33f14e
        case 'U': case 'V': case 'W': case 'X': case 'Y':
Packit 33f14e
        case 'Z':
Packit 33f14e
        case '[': case '\\': case ']': case '^': case '_':
Packit 33f14e
        case 'a': case 'b': case 'c': case 'd': case 'e':
Packit 33f14e
        case 'f': case 'g': case 'h': case 'i': case 'j':
Packit 33f14e
        case 'k': case 'l': case 'm': case 'n': case 'o':
Packit 33f14e
        case 'p': case 'q': case 'r': case 's': case 't':
Packit 33f14e
        case 'u': case 'v': case 'w': case 'x': case 'y':
Packit 33f14e
        case 'z': case '{': case '|': case '}': case '~':
Packit 33f14e
          /* c is in the ISO C "basic character set".  */
Packit 33f14e
          ret = wcrtomb (buf, btowc (c), NULL);
Packit 33f14e
          ASSERT (ret == 1);
Packit 33f14e
          ASSERT (buf[0] == (char) c);
Packit 33f14e
          break;
Packit 33f14e
        }
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  /* Test special calling convention, passing a NULL pointer.  */
Packit 33f14e
  {
Packit 33f14e
    ret = wcrtomb (NULL, '\0', NULL);
Packit 33f14e
    ASSERT (ret == 1);
Packit 33f14e
    ret = wcrtomb (NULL, btowc ('x'), NULL);
Packit 33f14e
    ASSERT (ret == 1);
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  switch (codepage)
Packit 33f14e
    {
Packit 33f14e
    case 1252:
Packit 33f14e
      /* Locale encoding is CP1252, an extension of ISO-8859-1.  */
Packit 33f14e
      {
Packit 33f14e
        /* Convert "B\374\337er": "Büßer" */
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x00FC, NULL);
Packit 33f14e
        ASSERT (ret == 1);
Packit 33f14e
        ASSERT (memcmp (buf, "\374", 1) == 0);
Packit 33f14e
        ASSERT (buf[1] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x00DF, NULL);
Packit 33f14e
        ASSERT (ret == 1);
Packit 33f14e
        ASSERT (memcmp (buf, "\337", 1) == 0);
Packit 33f14e
        ASSERT (buf[1] == 'x');
Packit 33f14e
      }
Packit 33f14e
      return 0;
Packit 33f14e
Packit 33f14e
    case 1256:
Packit 33f14e
      /* Locale encoding is CP1256, not the same as ISO-8859-6.  */
Packit 33f14e
      {
Packit 33f14e
        /* Convert "x\302\341\346y": "xآلوy" */
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x0622, NULL);
Packit 33f14e
        ASSERT (ret == 1);
Packit 33f14e
        ASSERT (memcmp (buf, "\302", 1) == 0);
Packit 33f14e
        ASSERT (buf[1] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x0644, NULL);
Packit 33f14e
        ASSERT (ret == 1);
Packit 33f14e
        ASSERT (memcmp (buf, "\341", 1) == 0);
Packit 33f14e
        ASSERT (buf[1] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x0648, NULL);
Packit 33f14e
        ASSERT (ret == 1);
Packit 33f14e
        ASSERT (memcmp (buf, "\346", 1) == 0);
Packit 33f14e
        ASSERT (buf[1] == 'x');
Packit 33f14e
      }
Packit 33f14e
      return 0;
Packit 33f14e
Packit 33f14e
    case 932:
Packit 33f14e
      /* Locale encoding is CP932, similar to Shift_JIS.  */
Packit 33f14e
      {
Packit 33f14e
        /* Convert "<\223\372\226\173\214\352>": "<日本語>" */
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x65E5, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\223\372", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x672C, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\226\173", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x8A9E, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\214\352", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
      }
Packit 33f14e
      return 0;
Packit 33f14e
Packit 33f14e
    case 950:
Packit 33f14e
      /* Locale encoding is CP950, similar to Big5.  */
Packit 33f14e
      {
Packit 33f14e
        /* Convert "<\244\351\245\273\273\171>": "<日本語>" */
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x65E5, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\244\351", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x672C, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\245\273", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x8A9E, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\273\171", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
      }
Packit 33f14e
      return 0;
Packit 33f14e
Packit 33f14e
    case 936:
Packit 33f14e
      /* Locale encoding is CP936 = GBK, an extension of GB2312.  */
Packit 33f14e
      {
Packit 33f14e
        /* Convert "<\310\325\261\276\325\132>": "<日本語>" */
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x65E5, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\310\325", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x672C, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\261\276", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x8A9E, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\325\132", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
      }
Packit 33f14e
      return 0;
Packit 33f14e
Packit 33f14e
    case 54936:
Packit 33f14e
      /* Locale encoding is CP54936 = GB18030.  */
Packit 33f14e
      {
Packit 33f14e
        /* Convert "B\250\271\201\060\211\070er": "Büßer" */
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x00FC, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\250\271", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x00DF, NULL);
Packit 33f14e
        ASSERT (ret == 4);
Packit 33f14e
        ASSERT (memcmp (buf, "\201\060\211\070", 4) == 0);
Packit 33f14e
        ASSERT (buf[4] == 'x');
Packit 33f14e
      }
Packit 33f14e
      return 0;
Packit 33f14e
Packit 33f14e
    case 65001:
Packit 33f14e
      /* Locale encoding is CP65001 = UTF-8.  */
Packit 33f14e
      {
Packit 33f14e
        /* Convert "B\303\274\303\237er": "Büßer" */
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x00FC, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\303\274", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
Packit 33f14e
        memset (buf, 'x', 8);
Packit 33f14e
        ret = wcrtomb (buf, 0x00DF, NULL);
Packit 33f14e
        ASSERT (ret == 2);
Packit 33f14e
        ASSERT (memcmp (buf, "\303\237", 2) == 0);
Packit 33f14e
        ASSERT (buf[2] == 'x');
Packit 33f14e
      }
Packit 33f14e
      return 0;
Packit 33f14e
Packit 33f14e
    default:
Packit 33f14e
      return 1;
Packit 33f14e
    }
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
int
Packit 33f14e
main (int argc, char *argv[])
Packit 33f14e
{
Packit 33f14e
  int codepage = atoi (argv[argc - 1]);
Packit 33f14e
  int result;
Packit 33f14e
  int i;
Packit 33f14e
Packit 33f14e
  result = 77;
Packit 33f14e
  for (i = 1; i < argc - 1; i++)
Packit 33f14e
    {
Packit 33f14e
      int ret = test_one_locale (argv[i], codepage);
Packit 33f14e
Packit 33f14e
      if (ret != 77)
Packit 33f14e
        result = ret;
Packit 33f14e
    }
Packit 33f14e
Packit 33f14e
  if (result == 77)
Packit 33f14e
    {
Packit 33f14e
      fprintf (stderr, "Skipping test: found no locale with codepage %d\n",
Packit 33f14e
               codepage);
Packit 33f14e
    }
Packit 33f14e
  return result;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
#else
Packit 33f14e
Packit 33f14e
int
Packit 33f14e
main (int argc, char *argv[])
Packit 33f14e
{
Packit 33f14e
  fputs ("Skipping test: not a native Windows system\n", stderr);
Packit 33f14e
  return 77;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
#endif