Blame stdlib/strtol_l.c

Packit 6c4009
/* Convert string representing a number to integer value, using given locale.
Packit 6c4009
   Copyright (C) 1997-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
   Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
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
Packit 6c4009
#if HAVE_CONFIG_H
Packit 6c4009
# include <config.h>
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#ifdef _LIBC
Packit 6c4009
# define USE_NUMBER_GROUPING
Packit 6c4009
# define HAVE_LIMITS_H
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#include <ctype.h>
Packit 6c4009
#include <errno.h>
Packit 6c4009
#ifndef __set_errno
Packit 6c4009
# define __set_errno(Val) errno = (Val)
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#ifdef HAVE_LIMITS_H
Packit 6c4009
# include <limits.h>
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#include <stddef.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <locale.h>
Packit 6c4009
#include <stdint.h>
Packit 6c4009
#include <bits/wordsize.h>
Packit 6c4009
Packit 6c4009
#ifdef USE_NUMBER_GROUPING
Packit 6c4009
# include "../locale/localeinfo.h"
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
/* Nonzero if we are defining `strtoul' or `strtoull', operating on
Packit 6c4009
   unsigned integers.  */
Packit 6c4009
#ifndef UNSIGNED
Packit 6c4009
# define UNSIGNED 0
Packit 6c4009
# define INT LONG int
Packit 6c4009
#else
Packit 6c4009
# define INT unsigned LONG int
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
/* Determine the name.  */
Packit 6c4009
#if UNSIGNED
Packit 6c4009
# ifdef USE_WIDE_CHAR
Packit 6c4009
#  ifdef QUAD
Packit 6c4009
#   define strtol_l wcstoull_l
Packit 6c4009
#  else
Packit 6c4009
#   define strtol_l wcstoul_l
Packit 6c4009
#  endif
Packit 6c4009
# else
Packit 6c4009
#  ifdef QUAD
Packit 6c4009
#   define strtol_l strtoull_l
Packit 6c4009
#  else
Packit 6c4009
#   define strtol_l strtoul_l
Packit 6c4009
#  endif
Packit 6c4009
# endif
Packit 6c4009
#else
Packit 6c4009
# ifdef USE_WIDE_CHAR
Packit 6c4009
#  ifdef QUAD
Packit 6c4009
#   define strtol_l wcstoll_l
Packit 6c4009
#  else
Packit 6c4009
#   define strtol_l wcstol_l
Packit 6c4009
#  endif
Packit 6c4009
# else
Packit 6c4009
#  ifdef QUAD
Packit 6c4009
#   define strtol_l strtoll_l
Packit 6c4009
#  else
Packit 6c4009
#   define strtol_l strtol_l
Packit 6c4009
#  endif
Packit 6c4009
# endif
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#define __strtol_l __strtol_l2(strtol_l)
Packit 6c4009
#define __strtol_l2(name) __strtol_l3(name)
Packit 6c4009
#define __strtol_l3(name) __##name
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* If QUAD is defined, we are defining `strtoll' or `strtoull',
Packit 6c4009
   operating on `long long int's.  */
Packit 6c4009
#ifdef QUAD
Packit 6c4009
# define LONG long long
Packit 6c4009
# define STRTOL_LONG_MIN LONG_LONG_MIN
Packit 6c4009
# define STRTOL_LONG_MAX LONG_LONG_MAX
Packit 6c4009
# define STRTOL_ULONG_MAX ULONG_LONG_MAX
Packit 6c4009
#else
Packit 6c4009
# define LONG long
Packit 6c4009
Packit 6c4009
# ifndef ULONG_MAX
Packit 6c4009
#  define ULONG_MAX ((unsigned long int) ~(unsigned long int) 0)
Packit 6c4009
# endif
Packit 6c4009
# ifndef LONG_MAX
Packit 6c4009
#  define LONG_MAX ((long int) (ULONG_MAX >> 1))
Packit 6c4009
# endif
Packit 6c4009
# define STRTOL_LONG_MIN LONG_MIN
Packit 6c4009
# define STRTOL_LONG_MAX LONG_MAX
Packit 6c4009
# define STRTOL_ULONG_MAX ULONG_MAX
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* We use this code for the extended locale handling where the
Packit 6c4009
   function gets as an additional argument the locale which has to be
Packit 6c4009
   used.  To access the values we have to redefine the _NL_CURRENT and
Packit 6c4009
   _NL_CURRENT_WORD macros.  */
Packit 6c4009
#undef _NL_CURRENT
Packit 6c4009
#define _NL_CURRENT(category, item) \
Packit 6c4009
  (current->values[_NL_ITEM_INDEX (item)].string)
Packit 6c4009
#undef _NL_CURRENT_WORD
Packit 6c4009
#define _NL_CURRENT_WORD(category, item) \
Packit 6c4009
  ((uint32_t) current->values[_NL_ITEM_INDEX (item)].word)
Packit 6c4009
Packit 6c4009
#if defined _LIBC || defined HAVE_WCHAR_H
Packit 6c4009
# include <wchar.h>
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#ifdef USE_WIDE_CHAR
Packit 6c4009
# include <wctype.h>
Packit 6c4009
# define L_(Ch) L##Ch
Packit 6c4009
# define UCHAR_TYPE wint_t
Packit 6c4009
# define STRING_TYPE wchar_t
Packit 6c4009
# define ISSPACE(Ch) __iswspace_l ((Ch), loc)
Packit 6c4009
# define ISALPHA(Ch) __iswalpha_l ((Ch), _nl_C_locobj_ptr)
Packit 6c4009
# define TOUPPER(Ch) __towupper_l ((Ch), _nl_C_locobj_ptr)
Packit 6c4009
#else
Packit 6c4009
# if defined _LIBC \
Packit 6c4009
   || defined STDC_HEADERS || (!defined isascii && !defined HAVE_ISASCII)
Packit 6c4009
#  define IN_CTYPE_DOMAIN(c) 1
Packit 6c4009
# else
Packit 6c4009
#  define IN_CTYPE_DOMAIN(c) isascii(c)
Packit 6c4009
# endif
Packit 6c4009
# define L_(Ch) Ch
Packit 6c4009
# define UCHAR_TYPE unsigned char
Packit 6c4009
# define STRING_TYPE char
Packit 6c4009
# define ISSPACE(Ch) __isspace_l ((Ch), loc)
Packit 6c4009
# define ISALPHA(Ch) __isalpha_l ((Ch), _nl_C_locobj_ptr)
Packit 6c4009
# define TOUPPER(Ch) __toupper_l ((Ch), _nl_C_locobj_ptr)
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#define INTERNAL(X) INTERNAL1(X)
Packit 6c4009
#define INTERNAL1(X) __##X##_internal
Packit 6c4009
#define WEAKNAME(X) WEAKNAME1(X)
Packit 6c4009
Packit 6c4009
#ifdef USE_NUMBER_GROUPING
Packit 6c4009
/* This file defines a function to check for correct grouping.  */
Packit 6c4009
# include "grouping.h"
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* Define tables of maximum values and remainders in order to detect
Packit 6c4009
   overflow.  Do this at compile-time in order to avoid the runtime
Packit 6c4009
   overhead of the division.  */
Packit 6c4009
extern const unsigned long __strtol_ul_max_tab[] attribute_hidden;
Packit 6c4009
extern const unsigned char __strtol_ul_rem_tab[] attribute_hidden;
Packit 6c4009
#if defined(QUAD) && __WORDSIZE == 32
Packit 6c4009
extern const unsigned long long __strtol_ull_max_tab[] attribute_hidden;
Packit 6c4009
extern const unsigned char __strtol_ull_rem_tab[] attribute_hidden;
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#define DEF(TYPE, NAME)							   \
Packit 6c4009
  const TYPE NAME[] attribute_hidden =					   \
Packit 6c4009
  {									   \
Packit 6c4009
    F(2), F(3), F(4), F(5), F(6), F(7), F(8), F(9), F(10), 		   \
Packit 6c4009
    F(11), F(12), F(13), F(14), F(15), F(16), F(17), F(18), F(19), F(20),  \
Packit 6c4009
    F(21), F(22), F(23), F(24), F(25), F(26), F(27), F(28), F(29), F(30),  \
Packit 6c4009
    F(31), F(32), F(33), F(34), F(35), F(36)				   \
Packit 6c4009
  }
Packit 6c4009
Packit 6c4009
#if !UNSIGNED && !defined (USE_WIDE_CHAR) && !defined (QUAD)
Packit 6c4009
# define F(X)	ULONG_MAX / X
Packit 6c4009
  DEF (unsigned long, __strtol_ul_max_tab);
Packit 6c4009
# undef F
Packit 6c4009
# define F(X)	ULONG_MAX % X
Packit 6c4009
  DEF (unsigned char, __strtol_ul_rem_tab);
Packit 6c4009
# undef F
Packit 6c4009
#endif
Packit 6c4009
#if !UNSIGNED && !defined (USE_WIDE_CHAR) && defined (QUAD) \
Packit 6c4009
    && __WORDSIZE == 32
Packit 6c4009
# define F(X)	ULONG_LONG_MAX / X
Packit 6c4009
  DEF (unsigned long long, __strtol_ull_max_tab);
Packit 6c4009
# undef F
Packit 6c4009
# define F(X)	ULONG_LONG_MAX % X
Packit 6c4009
  DEF (unsigned char, __strtol_ull_rem_tab);
Packit 6c4009
# undef F
Packit 6c4009
#endif
Packit 6c4009
#undef DEF
Packit 6c4009
Packit 6c4009
/* Define some more readable aliases for these arrays which correspond
Packit 6c4009
   to how they'll be used in the function below.  */
Packit 6c4009
#define jmax_tab	__strtol_ul_max_tab
Packit 6c4009
#if defined(QUAD) && __WORDSIZE == 32
Packit 6c4009
# define cutoff_tab	__strtol_ull_max_tab
Packit 6c4009
# define cutlim_tab	__strtol_ull_rem_tab
Packit 6c4009
#else
Packit 6c4009
# define cutoff_tab	__strtol_ul_max_tab
Packit 6c4009
# define cutlim_tab	__strtol_ul_rem_tab
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* Convert NPTR to an `unsigned long int' or `long int' in base BASE.
Packit 6c4009
   If BASE is 0 the base is determined by the presence of a leading
Packit 6c4009
   zero, indicating octal or a leading "0x" or "0X", indicating hexadecimal.
Packit 6c4009
   If BASE is < 2 or > 36, it is reset to 10.
Packit 6c4009
   If ENDPTR is not NULL, a pointer to the character after the last
Packit 6c4009
   one converted is stored in *ENDPTR.  */
Packit 6c4009
Packit 6c4009
INT
Packit 6c4009
INTERNAL (__strtol_l) (const STRING_TYPE *nptr, STRING_TYPE **endptr,
Packit 6c4009
		       int base, int group, locale_t loc)
Packit 6c4009
{
Packit 6c4009
  int negative;
Packit 6c4009
  unsigned LONG int cutoff;
Packit 6c4009
  unsigned int cutlim;
Packit 6c4009
  unsigned LONG int i;
Packit 6c4009
  const STRING_TYPE *s;
Packit 6c4009
  UCHAR_TYPE c;
Packit 6c4009
  const STRING_TYPE *save, *end;
Packit 6c4009
  int overflow;
Packit 6c4009
#ifndef USE_WIDE_CHAR
Packit 6c4009
  size_t cnt;
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#ifdef USE_NUMBER_GROUPING
Packit 6c4009
  struct __locale_data *current = loc->__locales[LC_NUMERIC];
Packit 6c4009
  /* The thousands character of the current locale.  */
Packit 6c4009
# ifdef USE_WIDE_CHAR
Packit 6c4009
  wchar_t thousands = L'\0';
Packit 6c4009
# else
Packit 6c4009
  const char *thousands = NULL;
Packit 6c4009
  size_t thousands_len = 0;
Packit 6c4009
# endif
Packit 6c4009
  /* The numeric grouping specification of the current locale,
Packit 6c4009
     in the format described in <locale.h>.  */
Packit 6c4009
  const char *grouping;
Packit 6c4009
Packit 6c4009
  if (__glibc_unlikely (group))
Packit 6c4009
    {
Packit 6c4009
      grouping = _NL_CURRENT (LC_NUMERIC, GROUPING);
Packit 6c4009
      if (*grouping <= 0 || *grouping == CHAR_MAX)
Packit 6c4009
	grouping = NULL;
Packit 6c4009
      else
Packit 6c4009
	{
Packit 6c4009
	  /* Figure out the thousands separator character.  */
Packit 6c4009
# ifdef USE_WIDE_CHAR
Packit 6c4009
#  ifdef _LIBC
Packit 6c4009
	  thousands = _NL_CURRENT_WORD (LC_NUMERIC,
Packit 6c4009
					_NL_NUMERIC_THOUSANDS_SEP_WC);
Packit 6c4009
#  endif
Packit 6c4009
	  if (thousands == L'\0')
Packit 6c4009
	    grouping = NULL;
Packit 6c4009
# else
Packit 6c4009
#  ifdef _LIBC
Packit 6c4009
	  thousands = _NL_CURRENT (LC_NUMERIC, THOUSANDS_SEP);
Packit 6c4009
#  endif
Packit 6c4009
	  if (*thousands == '\0')
Packit 6c4009
	    {
Packit 6c4009
	      thousands = NULL;
Packit 6c4009
	      grouping = NULL;
Packit 6c4009
	    }
Packit 6c4009
# endif
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    grouping = NULL;
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
  if (base < 0 || base == 1 || base > 36)
Packit 6c4009
    {
Packit 6c4009
      __set_errno (EINVAL);
Packit 6c4009
      return 0;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  save = s = nptr;
Packit 6c4009
Packit 6c4009
  /* Skip white space.  */
Packit 6c4009
  while (ISSPACE (*s))
Packit 6c4009
    ++s;
Packit 6c4009
  if (__glibc_unlikely (*s == L_('\0')))
Packit 6c4009
    goto noconv;
Packit 6c4009
Packit 6c4009
  /* Check for a sign.  */
Packit 6c4009
  negative = 0;
Packit 6c4009
  if (*s == L_('-'))
Packit 6c4009
    {
Packit 6c4009
      negative = 1;
Packit 6c4009
      ++s;
Packit 6c4009
    }
Packit 6c4009
  else if (*s == L_('+'))
Packit 6c4009
    ++s;
Packit 6c4009
Packit 6c4009
  /* Recognize number prefix and if BASE is zero, figure it out ourselves.  */
Packit 6c4009
  if (*s == L_('0'))
Packit 6c4009
    {
Packit 6c4009
      if ((base == 0 || base == 16) && TOUPPER (s[1]) == L_('X'))
Packit 6c4009
	{
Packit 6c4009
	  s += 2;
Packit 6c4009
	  base = 16;
Packit 6c4009
	}
Packit 6c4009
      else if (base == 0)
Packit 6c4009
	base = 8;
Packit 6c4009
    }
Packit 6c4009
  else if (base == 0)
Packit 6c4009
    base = 10;
Packit 6c4009
Packit 6c4009
  /* Save the pointer so we can check later if anything happened.  */
Packit 6c4009
  save = s;
Packit 6c4009
Packit 6c4009
#ifdef USE_NUMBER_GROUPING
Packit 6c4009
  if (base != 10)
Packit 6c4009
    grouping = NULL;
Packit 6c4009
Packit 6c4009
  if (__glibc_unlikely (grouping != NULL))
Packit 6c4009
    {
Packit 6c4009
# ifndef USE_WIDE_CHAR
Packit 6c4009
      thousands_len = strlen (thousands);
Packit 6c4009
# endif
Packit 6c4009
Packit 6c4009
      /* Find the end of the digit string and check its grouping.  */
Packit 6c4009
      end = s;
Packit 6c4009
      if (
Packit 6c4009
# ifdef USE_WIDE_CHAR
Packit 6c4009
	  *s != thousands
Packit 6c4009
# else
Packit 6c4009
	  ({ for (cnt = 0; cnt < thousands_len; ++cnt)
Packit 6c4009
	       if (thousands[cnt] != end[cnt])
Packit 6c4009
		 break;
Packit 6c4009
	     cnt < thousands_len; })
Packit 6c4009
# endif
Packit 6c4009
	  )
Packit 6c4009
	{
Packit 6c4009
	  for (c = *end; c != L_('\0'); c = *++end)
Packit 6c4009
	    if (((STRING_TYPE) c < L_('0') || (STRING_TYPE) c > L_('9'))
Packit 6c4009
# ifdef USE_WIDE_CHAR
Packit 6c4009
		&& (wchar_t) c != thousands
Packit 6c4009
# else
Packit 6c4009
		&& ({ for (cnt = 0; cnt < thousands_len; ++cnt)
Packit 6c4009
			if (thousands[cnt] != end[cnt])
Packit 6c4009
			  break;
Packit 6c4009
		      cnt < thousands_len; })
Packit 6c4009
# endif
Packit 6c4009
		&& (!ISALPHA (c)
Packit 6c4009
		    || (int) (TOUPPER (c) - L_('A') + 10) >= base))
Packit 6c4009
	      break;
Packit 6c4009
Packit 6c4009
# ifdef USE_WIDE_CHAR
Packit 6c4009
	  end = __correctly_grouped_prefixwc (s, end, thousands, grouping);
Packit 6c4009
# else
Packit 6c4009
	  end = __correctly_grouped_prefixmb (s, end, thousands, grouping);
Packit 6c4009
# endif
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
#endif
Packit 6c4009
    end = NULL;
Packit 6c4009
Packit 6c4009
  /* Avoid runtime division; lookup cutoff and limit.  */
Packit 6c4009
  cutoff = cutoff_tab[base - 2];
Packit 6c4009
  cutlim = cutlim_tab[base - 2];
Packit 6c4009
Packit 6c4009
  overflow = 0;
Packit 6c4009
  i = 0;
Packit 6c4009
  c = *s;
Packit 6c4009
  if (sizeof (long int) != sizeof (LONG int))
Packit 6c4009
    {
Packit 6c4009
      unsigned long int j = 0;
Packit 6c4009
      unsigned long int jmax = jmax_tab[base - 2];
Packit 6c4009
Packit 6c4009
      for (;c != L_('\0'); c = *++s)
Packit 6c4009
	{
Packit 6c4009
	  if (s == end)
Packit 6c4009
	    break;
Packit 6c4009
	  if (c >= L_('0') && c <= L_('9'))
Packit 6c4009
	    c -= L_('0');
Packit 6c4009
#ifdef USE_NUMBER_GROUPING
Packit 6c4009
# ifdef USE_WIDE_CHAR
Packit 6c4009
	  else if (grouping && (wchar_t) c == thousands)
Packit 6c4009
	    continue;
Packit 6c4009
# else
Packit 6c4009
	  else if (thousands_len)
Packit 6c4009
	    {
Packit 6c4009
	      for (cnt = 0; cnt < thousands_len; ++cnt)
Packit 6c4009
		if (thousands[cnt] != s[cnt])
Packit 6c4009
		  break;
Packit 6c4009
	      if (cnt == thousands_len)
Packit 6c4009
		{
Packit 6c4009
		  s += thousands_len - 1;
Packit 6c4009
		  continue;
Packit 6c4009
		}
Packit 6c4009
	      if (ISALPHA (c))
Packit 6c4009
		c = TOUPPER (c) - L_('A') + 10;
Packit 6c4009
	      else
Packit 6c4009
		break;
Packit 6c4009
	    }
Packit 6c4009
# endif
Packit 6c4009
#endif
Packit 6c4009
	  else if (ISALPHA (c))
Packit 6c4009
	    c = TOUPPER (c) - L_('A') + 10;
Packit 6c4009
	  else
Packit 6c4009
	    break;
Packit 6c4009
	  if ((int) c >= base)
Packit 6c4009
	    break;
Packit 6c4009
	  /* Note that we never can have an overflow.  */
Packit 6c4009
	  else if (j >= jmax)
Packit 6c4009
	    {
Packit 6c4009
	      /* We have an overflow.  Now use the long representation.  */
Packit 6c4009
	      i = (unsigned LONG int) j;
Packit 6c4009
	      goto use_long;
Packit 6c4009
	    }
Packit 6c4009
	  else
Packit 6c4009
	    j = j * (unsigned long int) base + c;
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      i = (unsigned LONG int) j;
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    for (;c != L_('\0'); c = *++s)
Packit 6c4009
      {
Packit 6c4009
	if (s == end)
Packit 6c4009
	  break;
Packit 6c4009
	if (c >= L_('0') && c <= L_('9'))
Packit 6c4009
	  c -= L_('0');
Packit 6c4009
#ifdef USE_NUMBER_GROUPING
Packit 6c4009
# ifdef USE_WIDE_CHAR
Packit 6c4009
	else if (grouping && (wchar_t) c == thousands)
Packit 6c4009
	  continue;
Packit 6c4009
# else
Packit 6c4009
	else if (thousands_len)
Packit 6c4009
	  {
Packit 6c4009
	    for (cnt = 0; cnt < thousands_len; ++cnt)
Packit 6c4009
	      if (thousands[cnt] != s[cnt])
Packit 6c4009
		break;
Packit 6c4009
	    if (cnt == thousands_len)
Packit 6c4009
	      {
Packit 6c4009
		s += thousands_len - 1;
Packit 6c4009
		continue;
Packit 6c4009
	      }
Packit 6c4009
	    if (ISALPHA (c))
Packit 6c4009
	      c = TOUPPER (c) - L_('A') + 10;
Packit 6c4009
	    else
Packit 6c4009
	      break;
Packit 6c4009
	  }
Packit 6c4009
# endif
Packit 6c4009
#endif
Packit 6c4009
	else if (ISALPHA (c))
Packit 6c4009
	  c = TOUPPER (c) - L_('A') + 10;
Packit 6c4009
	else
Packit 6c4009
	  break;
Packit 6c4009
	if ((int) c >= base)
Packit 6c4009
	  break;
Packit 6c4009
	/* Check for overflow.  */
Packit 6c4009
	if (i > cutoff || (i == cutoff && c > cutlim))
Packit 6c4009
	  overflow = 1;
Packit 6c4009
	else
Packit 6c4009
	  {
Packit 6c4009
	  use_long:
Packit 6c4009
	    i *= (unsigned LONG int) base;
Packit 6c4009
	    i += c;
Packit 6c4009
	  }
Packit 6c4009
      }
Packit 6c4009
Packit 6c4009
  /* Check if anything actually happened.  */
Packit 6c4009
  if (s == save)
Packit 6c4009
    goto noconv;
Packit 6c4009
Packit 6c4009
  /* Store in ENDPTR the address of one character
Packit 6c4009
     past the last character we converted.  */
Packit 6c4009
  if (endptr != NULL)
Packit 6c4009
    *endptr = (STRING_TYPE *) s;
Packit 6c4009
Packit 6c4009
#if !UNSIGNED
Packit 6c4009
  /* Check for a value that is within the range of
Packit 6c4009
     `unsigned LONG int', but outside the range of `LONG int'.  */
Packit 6c4009
  if (overflow == 0
Packit 6c4009
      && i > (negative
Packit 6c4009
	      ? -((unsigned LONG int) (STRTOL_LONG_MIN + 1)) + 1
Packit 6c4009
	      : (unsigned LONG int) STRTOL_LONG_MAX))
Packit 6c4009
    overflow = 1;
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
  if (__glibc_unlikely (overflow))
Packit 6c4009
    {
Packit 6c4009
      __set_errno (ERANGE);
Packit 6c4009
#if UNSIGNED
Packit 6c4009
      return STRTOL_ULONG_MAX;
Packit 6c4009
#else
Packit 6c4009
      return negative ? STRTOL_LONG_MIN : STRTOL_LONG_MAX;
Packit 6c4009
#endif
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  /* Return the result of the appropriate sign.  */
Packit 6c4009
  return negative ? -i : i;
Packit 6c4009
Packit 6c4009
noconv:
Packit 6c4009
  /* We must handle a special case here: the base is 0 or 16 and the
Packit 6c4009
     first two characters are '0' and 'x', but the rest are no
Packit 6c4009
     hexadecimal digits.  This is no error case.  We return 0 and
Packit 6c4009
     ENDPTR points to the `x`.  */
Packit 6c4009
  if (endptr != NULL)
Packit 6c4009
    {
Packit 6c4009
      if (save - nptr >= 2 && TOUPPER (save[-1]) == L_('X')
Packit 6c4009
	  && save[-2] == L_('0'))
Packit 6c4009
	*endptr = (STRING_TYPE *) &save[-1];
Packit 6c4009
      else
Packit 6c4009
	/*  There was no number to convert.  */
Packit 6c4009
	*endptr = (STRING_TYPE *) nptr;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return 0L;
Packit 6c4009
}
Packit 6c4009
#if defined _LIBC && !defined USE_WIDE_CHAR
Packit 6c4009
libc_hidden_def (INTERNAL (__strtol_l))
Packit 6c4009
#endif
Packit 6c4009

Packit 6c4009
/* External user entry point.  */
Packit 6c4009
Packit 6c4009
#if _LIBC - 0 == 0
Packit 6c4009
Packit 6c4009
/* Prototype.  */
Packit 6c4009
extern INT __strtol_l (const STRING_TYPE *nptr, STRING_TYPE **endptr,
Packit 6c4009
		       int base);
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
Packit 6c4009
INT
Packit 6c4009
#ifdef weak_function
Packit 6c4009
weak_function
Packit 6c4009
#endif
Packit 6c4009
__strtol_l (const STRING_TYPE *nptr, STRING_TYPE **endptr,
Packit 6c4009
	    int base, locale_t loc)
Packit 6c4009
{
Packit 6c4009
  return INTERNAL (__strtol_l) (nptr, endptr, base, 0, loc);
Packit 6c4009
}
Packit 6c4009
libc_hidden_def (__strtol_l)
Packit 6c4009
weak_alias (__strtol_l, strtol_l)