Blame time/alt_digit.c

Packit 6c4009
/* Helper functions used by strftime/strptime to handle alternate digits.
Packit 6c4009
   Copyright (C) 1995-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/localeinfo.h"
Packit 6c4009
#include <libc-lock.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <wchar.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <stdint.h>
Packit 6c4009
Packit 6c4009
/* Some of the functions here must not be used while setlocale is called.  */
Packit 6c4009
__libc_rwlock_define (extern, __libc_setlocale_lock attribute_hidden)
Packit 6c4009
Packit 6c4009
#define CURRENT(item) (current->values[_NL_ITEM_INDEX (item)].string)
Packit 6c4009
#define CURRENT_WSTR(item) \
Packit 6c4009
  ((wchar_t *) current->values[_NL_ITEM_INDEX (item)].wstr)
Packit 6c4009
Packit 6c4009
static void
Packit 6c4009
_nl_init_alt_digit (struct __locale_data *current)
Packit 6c4009
{
Packit 6c4009
  struct lc_time_data *data;
Packit 6c4009
Packit 6c4009
  if (current->private.time == NULL)
Packit 6c4009
    {
Packit 6c4009
      current->private.time = malloc (sizeof *current->private.time);
Packit 6c4009
      if (current->private.time == NULL)
Packit 6c4009
	return;
Packit 6c4009
      memset (current->private.time, 0, sizeof *current->private.time);
Packit 6c4009
      current->private.cleanup = &_nl_cleanup_time;
Packit 6c4009
    }
Packit 6c4009
  data = current->private.time;
Packit 6c4009
Packit 6c4009
  if (! data->alt_digits_initialized)
Packit 6c4009
    {
Packit 6c4009
      const char *ptr = CURRENT (ALT_DIGITS);
Packit 6c4009
      size_t cnt;
Packit 6c4009
Packit 6c4009
      data->alt_digits_initialized = 1;
Packit 6c4009
Packit 6c4009
      if (ptr != NULL)
Packit 6c4009
	{
Packit 6c4009
	  data->alt_digits = malloc (100 * sizeof (const char *));
Packit 6c4009
	  if (data->alt_digits != NULL)
Packit 6c4009
	    for (cnt = 0; cnt < 100; ++cnt)
Packit 6c4009
	      {
Packit 6c4009
		data->alt_digits[cnt] = ptr;
Packit 6c4009
Packit 6c4009
		/* Skip digit format. */
Packit 6c4009
		ptr = strchr (ptr, '\0') + 1;
Packit 6c4009
	      }
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
const char *
Packit 6c4009
_nl_get_alt_digit (unsigned int number, struct __locale_data *current)
Packit 6c4009
{
Packit 6c4009
  const char *result;
Packit 6c4009
Packit 6c4009
  if (number >= 100 || CURRENT (ALT_DIGITS)[0] == '\0')
Packit 6c4009
    return NULL;
Packit 6c4009
Packit 6c4009
  __libc_rwlock_wrlock (__libc_setlocale_lock);
Packit 6c4009
Packit 6c4009
  if (current->private.time == NULL
Packit 6c4009
      || ! current->private.time->alt_digits_initialized)
Packit 6c4009
    _nl_init_alt_digit (current);
Packit 6c4009
Packit 6c4009
  result = ((current->private.time != NULL
Packit 6c4009
	     && current->private.time->alt_digits != NULL)
Packit 6c4009
	    ? current->private.time->alt_digits[number]
Packit 6c4009
	    : NULL);
Packit 6c4009
Packit 6c4009
  __libc_rwlock_unlock (__libc_setlocale_lock);
Packit 6c4009
Packit 6c4009
  return result;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
const wchar_t *
Packit 6c4009
_nl_get_walt_digit (unsigned int number, struct __locale_data *current)
Packit 6c4009
{
Packit 6c4009
  const wchar_t *result = NULL;
Packit 6c4009
  struct lc_time_data *data;
Packit 6c4009
Packit 6c4009
  if (number >= 100 || CURRENT_WSTR (_NL_WALT_DIGITS)[0] == L'\0')
Packit 6c4009
    return NULL;
Packit 6c4009
Packit 6c4009
  __libc_rwlock_wrlock (__libc_setlocale_lock);
Packit 6c4009
Packit 6c4009
  if (current->private.time == NULL)
Packit 6c4009
    {
Packit 6c4009
      current->private.time = malloc (sizeof *current->private.time);
Packit 6c4009
      if (current->private.time == NULL)
Packit 6c4009
	goto out;
Packit 6c4009
      memset (current->private.time, 0, sizeof *current->private.time);
Packit 6c4009
      current->private.cleanup = &_nl_cleanup_time;
Packit 6c4009
    }
Packit 6c4009
  data = current->private.time;
Packit 6c4009
Packit 6c4009
  if (! data->walt_digits_initialized)
Packit 6c4009
    {
Packit 6c4009
      const wchar_t *ptr = CURRENT_WSTR (_NL_WALT_DIGITS);
Packit 6c4009
      size_t cnt;
Packit 6c4009
Packit 6c4009
      data->walt_digits_initialized = 1;
Packit 6c4009
Packit 6c4009
      if (ptr != NULL)
Packit 6c4009
	{
Packit 6c4009
	  data->walt_digits = malloc (100 * sizeof (const uint32_t *));
Packit 6c4009
	  if (data->walt_digits != NULL)
Packit 6c4009
	    for (cnt = 0; cnt < 100; ++cnt)
Packit 6c4009
	      {
Packit 6c4009
		data->walt_digits[cnt] = ptr;
Packit 6c4009
Packit 6c4009
		/* Skip digit format. */
Packit 6c4009
		ptr = __wcschr (ptr, L'\0') + 1;
Packit 6c4009
	      }
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (data->walt_digits != NULL)
Packit 6c4009
    result = data->walt_digits[number];
Packit 6c4009
Packit 6c4009
 out:
Packit 6c4009
  __libc_rwlock_unlock (__libc_setlocale_lock);
Packit 6c4009
Packit 6c4009
  return (wchar_t *) result;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
_nl_parse_alt_digit (const char **strp, struct __locale_data *current)
Packit 6c4009
{
Packit 6c4009
  const char *str = *strp;
Packit 6c4009
  int result = -1;
Packit 6c4009
  size_t cnt;
Packit 6c4009
  size_t maxlen = 0;
Packit 6c4009
Packit 6c4009
  if (CURRENT_WSTR (_NL_WALT_DIGITS)[0] == L'\0')
Packit 6c4009
    return result;
Packit 6c4009
Packit 6c4009
  __libc_rwlock_wrlock (__libc_setlocale_lock);
Packit 6c4009
Packit 6c4009
  if (current->private.time == NULL
Packit 6c4009
      || ! current->private.time->alt_digits_initialized)
Packit 6c4009
    _nl_init_alt_digit (current);
Packit 6c4009
Packit 6c4009
  if (current->private.time != NULL &&
Packit 6c4009
      current->private.time->alt_digits != NULL)
Packit 6c4009
    /* Matching is not unambiguous.  The alternative digits could be like
Packit 6c4009
       I, II, III, ... and the first one is a substring of the second
Packit 6c4009
       and third.  Therefore we must keep on searching until we found
Packit 6c4009
       the longest possible match.  Note that this is not specified in
Packit 6c4009
       the standard.  */
Packit 6c4009
    for (cnt = 0; cnt < 100; ++cnt)
Packit 6c4009
      {
Packit 6c4009
	const char *const dig = current->private.time->alt_digits[cnt];
Packit 6c4009
	size_t len = strlen (dig);
Packit 6c4009
Packit 6c4009
	if (len > maxlen && strncmp (dig, str, len) == 0)
Packit 6c4009
	  {
Packit 6c4009
	    maxlen = len;
Packit 6c4009
	    result = (int) cnt;
Packit 6c4009
	  }
Packit 6c4009
      }
Packit 6c4009
Packit 6c4009
  __libc_rwlock_unlock (__libc_setlocale_lock);
Packit 6c4009
Packit 6c4009
  if (result != -1)
Packit 6c4009
    *strp += maxlen;
Packit 6c4009
Packit 6c4009
  return result;
Packit 6c4009
}