glibc/localedata/tst-langinfo.c

279 lines
5.2 KiB
C
Raw Normal View History

/* Test driver for nl_langinfo[_l] functions.
Copyright (C) 2000-2019 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>.
The GNU C Library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
The GNU C Library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with the GNU C Library; if not, see
<http://www.gnu.org/licenses/>. */
#include <langinfo.h>
#include <locale.h>
#include <stdio.h>
#include <string.h>
struct map
{
const char *str;
int val;
} map[] =
{
#define VAL(name) { #name, name }
VAL (ABDAY_1),
VAL (ABDAY_2),
VAL (ABDAY_3),
VAL (ABDAY_4),
VAL (ABDAY_5),
VAL (ABDAY_6),
VAL (ABDAY_7),
VAL (ABMON_1),
VAL (ABMON_10),
VAL (ABMON_11),
VAL (ABMON_12),
VAL (ABMON_2),
VAL (ABMON_3),
VAL (ABMON_4),
VAL (ABMON_5),
VAL (ABMON_6),
VAL (ABMON_7),
VAL (ABMON_8),
VAL (ABMON_9),
VAL (ALT_DIGITS),
Implement alternative month names (bug 10871). Some languages (Slavic, Baltic, etc.) require a genitive case of the month name when formatting a full date (with the day number) while they require a nominative case when referring to the month standalone. This requirement cannot be fulfilled without providing two forms for each month name. From now it is specified that nl_langinfo(MON_1) series (up to MON_12) and strftime("%B") generate the month names in the grammatical form used when the month is a part of a complete date. If the grammatical form used when the month is named by itself is needed, the new values nl_langinfo(ALTMON_1) (up to ALTMON_12) and strftime("%OB") are supported. This new feature is optional so the languages which do not need it or do not yet provide the updated locales simply do not use it and their behaviour is unchanged. [BZ #10871] * locale/C-time.c (_nl_C_LC_TIME): Add alternative month names, define them as the same as primary full month names explicitly. * locale/categories.def (LC_TIME): Add alt_mon and wide-alt_mon. * locale/langinfo.h (__ALTMON_1, __ALTMON_2, __ALTMON_3, __ALTMON_4, __ALTMON_5, __ALTMON_6, __ALTMON_7, __ALTMON_8, __ALTMON_9, __ALTMON_10, __ALTMON_11, __ALTMON_12, _NL_WALTMON_1, _NL_WALTMON_2, _NL_WALTMON_3, _NL_WALTMON_4, _NL_WALTMON_5, _NL_WALTMON_6, _NL_WALTMON_7, _NL_WALTMON_8, _NL_WALTMON_9, _NL_WALTMON_10, _NL_WALTMON_11, _NL_WALTMON_12): New enum constants. [__USE_GNU] (ALTMON_1, ALTMON_2, ALTMON_3, ALTMON_4, ALTMON_5, ALTMON_6, ALTMON_7, ALTMON_8, ALTMON_9, ALTMON_10, ALTMON_11, ALTMON_12): New macros. * locale/programs/ld-time.c (struct locale_time_t): Add alt_mon, walt_mon, and alt_mon_defined members. (time_output): Output alt_mon and walt_mon members. (time_read): Read them, initialize them as copies of mon and wmon respectively if they are missing, initialize alt_mon_defined. * locale/programs/locfile-kw.gperf (alt_mon): Define. * locale/programs/locfile-kw.h: Regenerate. * locale/programs/locfile-token.h (tok_alt_mon): New enum constant. * localedata/tst-langinfo.c (map): Add tests for the new constants ALTMON_1 .. ALTMON_12. * time/Makefile [$(run-built-tests) = yes] (LOCALES): Add fr_FR.UTF-8 and pl_PL.UTF-8. * time/strftime_l.c (f_altmonth): New macro. (__strftime_internal): Handle %OB format. * time/strptime_l.c [_LIBC] (alt_month_name): New macro. (__strptime_internal): Handle %OB format. * time/tst-strptime.c (day_tests): Add tests to parse different forms of month names including the new %OB format specifier. Reviewed-by: Carlos O'Donell <carlos@redhat.com>
2017-11-14 01:19:35 +00:00
VAL (ALTMON_1),
VAL (ALTMON_10),
VAL (ALTMON_11),
VAL (ALTMON_12),
VAL (ALTMON_2),
VAL (ALTMON_3),
VAL (ALTMON_4),
VAL (ALTMON_5),
VAL (ALTMON_6),
VAL (ALTMON_7),
VAL (ALTMON_8),
VAL (ALTMON_9),
VAL (AM_STR),
VAL (CRNCYSTR),
VAL (CURRENCY_SYMBOL),
VAL (DAY_1),
VAL (DAY_2),
VAL (DAY_3),
VAL (DAY_4),
VAL (DAY_5),
VAL (DAY_6),
VAL (DAY_7),
VAL (DECIMAL_POINT),
VAL (D_FMT),
VAL (D_T_FMT),
VAL (ERA),
VAL (ERA_D_FMT),
VAL (ERA_D_T_FMT),
VAL (ERA_T_FMT),
VAL (ERA_YEAR),
VAL (FRAC_DIGITS),
VAL (GROUPING),
VAL (INT_CURR_SYMBOL),
VAL (INT_FRAC_DIGITS),
VAL (MON_1),
VAL (MON_10),
VAL (MON_11),
VAL (MON_12),
VAL (MON_2),
VAL (MON_3),
VAL (MON_4),
VAL (MON_5),
VAL (MON_6),
VAL (MON_7),
VAL (MON_8),
VAL (MON_9),
VAL (MON_DECIMAL_POINT),
VAL (MON_GROUPING),
VAL (MON_THOUSANDS_SEP),
VAL (NEGATIVE_SIGN),
VAL (NOEXPR),
VAL (NOSTR),
VAL (N_CS_PRECEDES),
VAL (N_SEP_BY_SPACE),
VAL (N_SIGN_POSN),
VAL (PM_STR),
VAL (POSITIVE_SIGN),
VAL (P_CS_PRECEDES),
VAL (P_SEP_BY_SPACE),
VAL (P_SIGN_POSN),
VAL (RADIXCHAR),
VAL (THOUSANDS_SEP),
VAL (THOUSEP),
VAL (T_FMT),
VAL (T_FMT_AMPM),
VAL (YESEXPR),
VAL (YESSTR)
};
static int
map_paramstr (const char *str)
{
int low = 0;
int high = sizeof (map) / sizeof (map[0]);
while (low < high)
{
int med = (low + high) / 2;
int cmpres;
cmpres = strcmp (str, map[med].str);
if (cmpres == 0)
return map[med].val;
else if (cmpres > 0)
low = med + 1;
else
high = med;
}
return -1;
}
#ifdef DEBUG
# define REASON(str) printf ("\"%s\" ignored: %s\n", buf, str)
#else
# define REASON(str)
#endif
static int
do_test (void)
{
int result = 0;
while (! feof (stdin))
{
char buf[1000];
char *rp;
char *locale;
char *paramstr;
char *expected;
int param;
if (fgets (buf, sizeof (buf), stdin) == NULL)
break;
/* Split the fields. There are three is them:
1. locale
2. langinfo() parameter
3. expected result; this can be a string with white space etc.
*/
rp = buf;
while (*rp == ' ' || *rp == '\t')
++rp;
if (*rp == '#')
{
/* It's a comment line. Ignore it. */
REASON ("comment");
continue;
}
locale = rp;
while (*rp != '\0' && *rp != ' ' && *rp != '\t' && *rp != '\n')
++rp;
if (*rp == '\0' || *rp == '\n')
{
/* Incomplete line. */
REASON ("incomplete line");
continue;
}
*rp++ = '\0';
while (*rp == ' ' || *rp == '\t')
++rp;
paramstr = rp;
while (*rp != '\0' && *rp != ' ' && *rp != '\t' && *rp != '\n')
++rp;
if (*rp == '\0' || *rp == '\n')
{
/* Incomplete line. */
REASON ("incomplete line");
continue;
}
*rp++ = '\0';
while (*rp == ' ' || *rp == '\t')
++rp;
if (*rp == '"')
{
char *wp;
expected = wp = ++rp;
while (*rp != '"' && *rp != '\n' && *rp != '\0')
{
if (*rp == '\\')
{
++rp;
if (*rp == '\0')
break;
if (*rp >= '0' && *rp <= '9')
{
int val = *rp - '0';
if (rp[1] >= '0' && rp[1] <= '9')
{
++rp;
val *= 10;
val += *rp - '0';
if (rp[1] >= '0' && rp[1] <= '9')
{
++rp;
val *= 10;
val += *rp - '0';
}
}
*rp = val;
}
}
*wp++ = *rp++;
}
if (*rp != '"')
{
REASON ("missing '\"'");
continue;
}
*wp = '\0';
}
else
{
expected = rp;
while (*rp != '\0' && *rp != '\n')
++rp;
*rp = '\0';
}
param = map_paramstr (paramstr);
if (param == -1)
{
/* Invalid parameter. */
REASON ("invalid parameter");
continue;
}
result = test_locale (locale, paramstr, param, expected);
}
return result;
}
#define TEST_FUNCTION do_test ()
#include "../test-skeleton.c"