xref: /freebsd/contrib/sendmail/libsm/strto.c (revision e2c0e292e8a7ca00ba99bcfccc9e637f45c3e8b1)
140266059SGregory Neil Shapiro /*
25dd76dd0SGregory Neil Shapiro  * Copyright (c) 2000-2001 Proofpoint, Inc. and its suppliers.
340266059SGregory Neil Shapiro  *      All rights reserved.
440266059SGregory Neil Shapiro  * Copyright (c) 1992
540266059SGregory Neil Shapiro  *	The Regents of the University of California.  All rights reserved.
640266059SGregory Neil Shapiro  *
740266059SGregory Neil Shapiro  * By using this file, you agree to the terms and conditions set
840266059SGregory Neil Shapiro  * forth in the LICENSE file which can be found at the top level of
940266059SGregory Neil Shapiro  * the sendmail distribution.
1040266059SGregory Neil Shapiro  */
1140266059SGregory Neil Shapiro 
1240266059SGregory Neil Shapiro #include <sm/gen.h>
134313cc83SGregory Neil Shapiro SM_IDSTR(id, "@(#)$Id: strto.c,v 1.19 2013-11-22 20:51:43 ca Exp $")
1440266059SGregory Neil Shapiro 
1540266059SGregory Neil Shapiro #include <sys/param.h>
1640266059SGregory Neil Shapiro #include <sys/types.h>
1740266059SGregory Neil Shapiro #include <stdlib.h>
1840266059SGregory Neil Shapiro #include <ctype.h>
1940266059SGregory Neil Shapiro #include <errno.h>
2040266059SGregory Neil Shapiro #include <sm/limits.h>
2140266059SGregory Neil Shapiro #include <sm/conf.h>
2240266059SGregory Neil Shapiro #include <sm/string.h>
2340266059SGregory Neil Shapiro 
2440266059SGregory Neil Shapiro /*
2540266059SGregory Neil Shapiro **  SM_STRTOLL --  Convert a string to a (signed) long long integer.
2640266059SGregory Neil Shapiro **
2740266059SGregory Neil Shapiro **  Ignores `locale' stuff.  Assumes that the upper and lower case
2840266059SGregory Neil Shapiro **  alphabets and digits are each contiguous.
2940266059SGregory Neil Shapiro **
3040266059SGregory Neil Shapiro **	Parameters:
3140266059SGregory Neil Shapiro **		nptr -- string containing number
3240266059SGregory Neil Shapiro **		endptr -- location of first invalid character
3340266059SGregory Neil Shapiro **		base -- numeric base that 'nptr' number is based in
3440266059SGregory Neil Shapiro **
3540266059SGregory Neil Shapiro **	Returns:
3640266059SGregory Neil Shapiro **		Failure: on underflow LLONG_MIN is returned; on overflow
3740266059SGregory Neil Shapiro **			LLONG_MAX is returned and errno is set.
3840266059SGregory Neil Shapiro **			When 'endptr' == '\0' then the entire string 'nptr'
3940266059SGregory Neil Shapiro **			was valid.
4040266059SGregory Neil Shapiro **		Success: returns the converted number
4140266059SGregory Neil Shapiro */
4240266059SGregory Neil Shapiro 
4340266059SGregory Neil Shapiro LONGLONG_T
4440266059SGregory Neil Shapiro sm_strtoll(nptr, endptr, base)
4540266059SGregory Neil Shapiro 	const char *nptr;
4640266059SGregory Neil Shapiro 	char **endptr;
4740266059SGregory Neil Shapiro 	register int base;
4840266059SGregory Neil Shapiro {
4940266059SGregory Neil Shapiro 	register bool neg;
5040266059SGregory Neil Shapiro 	register const char *s;
5140266059SGregory Neil Shapiro 	register LONGLONG_T acc, cutoff;
5240266059SGregory Neil Shapiro 	register int c;
5340266059SGregory Neil Shapiro 	register int any, cutlim;
5440266059SGregory Neil Shapiro 
5540266059SGregory Neil Shapiro 	/*
5640266059SGregory Neil Shapiro 	**  Skip white space and pick up leading +/- sign if any.
5740266059SGregory Neil Shapiro 	**  If base is 0, allow 0x for hex and 0 for octal, else
5840266059SGregory Neil Shapiro 	**  assume decimal; if base is already 16, allow 0x.
5940266059SGregory Neil Shapiro 	*/
6040266059SGregory Neil Shapiro 
6140266059SGregory Neil Shapiro 	s = nptr;
6240266059SGregory Neil Shapiro 	do
6340266059SGregory Neil Shapiro 	{
6440266059SGregory Neil Shapiro 		c = (unsigned char) *s++;
6540266059SGregory Neil Shapiro 	} while (isascii(c) && isspace(c));
6640266059SGregory Neil Shapiro 	if (c == '-')
6740266059SGregory Neil Shapiro 	{
6840266059SGregory Neil Shapiro 		neg = true;
6940266059SGregory Neil Shapiro 		c = *s++;
7040266059SGregory Neil Shapiro 	}
7140266059SGregory Neil Shapiro 	else
7240266059SGregory Neil Shapiro 	{
7340266059SGregory Neil Shapiro 		neg = false;
7440266059SGregory Neil Shapiro 		if (c == '+')
7540266059SGregory Neil Shapiro 			c = *s++;
7640266059SGregory Neil Shapiro 	}
7740266059SGregory Neil Shapiro 	if ((base == 0 || base == 16) &&
7840266059SGregory Neil Shapiro 	    c == '0' && (*s == 'x' || *s == 'X'))
7940266059SGregory Neil Shapiro 	{
8040266059SGregory Neil Shapiro 		c = s[1];
8140266059SGregory Neil Shapiro 		s += 2;
8240266059SGregory Neil Shapiro 		base = 16;
8340266059SGregory Neil Shapiro 	}
8440266059SGregory Neil Shapiro 	if (base == 0)
8540266059SGregory Neil Shapiro 		base = c == '0' ? 8 : 10;
8640266059SGregory Neil Shapiro 
8740266059SGregory Neil Shapiro 	/*
8840266059SGregory Neil Shapiro 	**  Compute the cutoff value between legal numbers and illegal
8940266059SGregory Neil Shapiro 	**  numbers.  That is the largest legal value, divided by the
9040266059SGregory Neil Shapiro 	**  base.  An input number that is greater than this value, if
9140266059SGregory Neil Shapiro 	**  followed by a legal input character, is too big.  One that
9240266059SGregory Neil Shapiro 	**  is equal to this value may be valid or not; the limit
9340266059SGregory Neil Shapiro 	**  between valid and invalid numbers is then based on the last
9440266059SGregory Neil Shapiro 	**  digit.  For instance, if the range for long-long's is
9540266059SGregory Neil Shapiro 	**  [-9223372036854775808..9223372036854775807] and the input base
9640266059SGregory Neil Shapiro 	**  is 10, cutoff will be set to 922337203685477580 and cutlim to
9740266059SGregory Neil Shapiro 	**  either 7 (!neg) or 8 (neg), meaning that if we have
9840266059SGregory Neil Shapiro 	**  accumulated a value > 922337203685477580, or equal but the
9940266059SGregory Neil Shapiro 	**  next digit is > 7 (or 8), the number is too big, and we will
10040266059SGregory Neil Shapiro 	**  return a range error.
10140266059SGregory Neil Shapiro 	**
10240266059SGregory Neil Shapiro 	**  Set any if any `digits' consumed; make it negative to indicate
10340266059SGregory Neil Shapiro 	**  overflow.
10440266059SGregory Neil Shapiro 	*/
10540266059SGregory Neil Shapiro 
10640266059SGregory Neil Shapiro 	cutoff = neg ? LLONG_MIN : LLONG_MAX;
10740266059SGregory Neil Shapiro 	cutlim = cutoff % base;
10840266059SGregory Neil Shapiro 	cutoff /= base;
10940266059SGregory Neil Shapiro 	if (neg)
11040266059SGregory Neil Shapiro 	{
11140266059SGregory Neil Shapiro 		if (cutlim > 0)
11240266059SGregory Neil Shapiro 		{
11340266059SGregory Neil Shapiro 			cutlim -= base;
11440266059SGregory Neil Shapiro 			cutoff += 1;
11540266059SGregory Neil Shapiro 		}
11640266059SGregory Neil Shapiro 		cutlim = -cutlim;
11740266059SGregory Neil Shapiro 	}
11840266059SGregory Neil Shapiro 	for (acc = 0, any = 0;; c = (unsigned char) *s++)
11940266059SGregory Neil Shapiro 	{
12040266059SGregory Neil Shapiro 		if (isascii(c) && isdigit(c))
12140266059SGregory Neil Shapiro 			c -= '0';
12240266059SGregory Neil Shapiro 		else if (isascii(c) && isalpha(c))
12340266059SGregory Neil Shapiro 			c -= isupper(c) ? 'A' - 10 : 'a' - 10;
12440266059SGregory Neil Shapiro 		else
12540266059SGregory Neil Shapiro 			break;
12640266059SGregory Neil Shapiro 		if (c >= base)
12740266059SGregory Neil Shapiro 			break;
12840266059SGregory Neil Shapiro 		if (any < 0)
12940266059SGregory Neil Shapiro 			continue;
13040266059SGregory Neil Shapiro 		if (neg)
13140266059SGregory Neil Shapiro 		{
13240266059SGregory Neil Shapiro 			if (acc < cutoff || (acc == cutoff && c > cutlim))
13340266059SGregory Neil Shapiro 			{
13440266059SGregory Neil Shapiro 				any = -1;
13540266059SGregory Neil Shapiro 				acc = LLONG_MIN;
13640266059SGregory Neil Shapiro 				errno = ERANGE;
13740266059SGregory Neil Shapiro 			}
13840266059SGregory Neil Shapiro 			else
13940266059SGregory Neil Shapiro 			{
14040266059SGregory Neil Shapiro 				any = 1;
14140266059SGregory Neil Shapiro 				acc *= base;
14240266059SGregory Neil Shapiro 				acc -= c;
14340266059SGregory Neil Shapiro 			}
14440266059SGregory Neil Shapiro 		}
14540266059SGregory Neil Shapiro 		else
14640266059SGregory Neil Shapiro 		{
14740266059SGregory Neil Shapiro 			if (acc > cutoff || (acc == cutoff && c > cutlim))
14840266059SGregory Neil Shapiro 			{
14940266059SGregory Neil Shapiro 				any = -1;
15040266059SGregory Neil Shapiro 				acc = LLONG_MAX;
15140266059SGregory Neil Shapiro 				errno = ERANGE;
15240266059SGregory Neil Shapiro 			}
15340266059SGregory Neil Shapiro 			else
15440266059SGregory Neil Shapiro 			{
15540266059SGregory Neil Shapiro 				any = 1;
15640266059SGregory Neil Shapiro 				acc *= base;
15740266059SGregory Neil Shapiro 				acc += c;
15840266059SGregory Neil Shapiro 			}
15940266059SGregory Neil Shapiro 		}
16040266059SGregory Neil Shapiro 	}
161*5b0945b5SGregory Neil Shapiro 	if (endptr != NULL)
16240266059SGregory Neil Shapiro 		*endptr = (char *) (any ? s - 1 : nptr);
16340266059SGregory Neil Shapiro 	return acc;
16440266059SGregory Neil Shapiro }
16540266059SGregory Neil Shapiro 
16640266059SGregory Neil Shapiro /*
16740266059SGregory Neil Shapiro **  SM_STRTOULL --  Convert a string to an unsigned long long integer.
16840266059SGregory Neil Shapiro **
16940266059SGregory Neil Shapiro **  Ignores `locale' stuff.  Assumes that the upper and lower case
17040266059SGregory Neil Shapiro **  alphabets and digits are each contiguous.
17140266059SGregory Neil Shapiro **
17240266059SGregory Neil Shapiro **	Parameters:
17340266059SGregory Neil Shapiro **		nptr -- string containing (unsigned) number
17440266059SGregory Neil Shapiro **		endptr -- location of first invalid character
17540266059SGregory Neil Shapiro **		base -- numeric base that 'nptr' number is based in
17640266059SGregory Neil Shapiro **
17740266059SGregory Neil Shapiro **	Returns:
17840266059SGregory Neil Shapiro **		Failure: on overflow ULLONG_MAX is returned and errno is set.
17940266059SGregory Neil Shapiro **			When 'endptr' == '\0' then the entire string 'nptr'
18040266059SGregory Neil Shapiro **			was valid.
18140266059SGregory Neil Shapiro **		Success: returns the converted number
18240266059SGregory Neil Shapiro */
18340266059SGregory Neil Shapiro 
18440266059SGregory Neil Shapiro ULONGLONG_T
sm_strtoull(nptr,endptr,base)18540266059SGregory Neil Shapiro sm_strtoull(nptr, endptr, base)
18640266059SGregory Neil Shapiro 	const char *nptr;
18740266059SGregory Neil Shapiro 	char **endptr;
18840266059SGregory Neil Shapiro 	register int base;
18940266059SGregory Neil Shapiro {
19040266059SGregory Neil Shapiro 	register const char *s;
19140266059SGregory Neil Shapiro 	register ULONGLONG_T acc, cutoff;
19240266059SGregory Neil Shapiro 	register int c;
19340266059SGregory Neil Shapiro 	register bool neg;
19440266059SGregory Neil Shapiro 	register int any, cutlim;
19540266059SGregory Neil Shapiro 
19640266059SGregory Neil Shapiro 	/* See sm_strtoll for comments as to the logic used. */
19740266059SGregory Neil Shapiro 	s = nptr;
19840266059SGregory Neil Shapiro 	do
19940266059SGregory Neil Shapiro 	{
20040266059SGregory Neil Shapiro 		c = (unsigned char) *s++;
20140266059SGregory Neil Shapiro 	} while (isascii(c) && isspace(c));
20240266059SGregory Neil Shapiro 	neg = (c == '-');
20340266059SGregory Neil Shapiro 	if (neg)
20440266059SGregory Neil Shapiro 	{
20540266059SGregory Neil Shapiro 		c = *s++;
20640266059SGregory Neil Shapiro 	}
20740266059SGregory Neil Shapiro 	else
20840266059SGregory Neil Shapiro 	{
20940266059SGregory Neil Shapiro 		if (c == '+')
21040266059SGregory Neil Shapiro 			c = *s++;
21140266059SGregory Neil Shapiro 	}
21240266059SGregory Neil Shapiro 	if ((base == 0 || base == 16) &&
21340266059SGregory Neil Shapiro 	    c == '0' && (*s == 'x' || *s == 'X'))
21440266059SGregory Neil Shapiro 	{
21540266059SGregory Neil Shapiro 		c = s[1];
21640266059SGregory Neil Shapiro 		s += 2;
21740266059SGregory Neil Shapiro 		base = 16;
21840266059SGregory Neil Shapiro 	}
21940266059SGregory Neil Shapiro 	if (base == 0)
22040266059SGregory Neil Shapiro 		base = c == '0' ? 8 : 10;
22140266059SGregory Neil Shapiro 
22240266059SGregory Neil Shapiro 	cutoff = ULLONG_MAX / (ULONGLONG_T)base;
22340266059SGregory Neil Shapiro 	cutlim = ULLONG_MAX % (ULONGLONG_T)base;
22440266059SGregory Neil Shapiro 	for (acc = 0, any = 0;; c = (unsigned char) *s++)
22540266059SGregory Neil Shapiro 	{
22640266059SGregory Neil Shapiro 		if (isascii(c) && isdigit(c))
22740266059SGregory Neil Shapiro 			c -= '0';
22840266059SGregory Neil Shapiro 		else if (isascii(c) && isalpha(c))
22940266059SGregory Neil Shapiro 			c -= isupper(c) ? 'A' - 10 : 'a' - 10;
23040266059SGregory Neil Shapiro 		else
23140266059SGregory Neil Shapiro 			break;
23240266059SGregory Neil Shapiro 		if (c >= base)
23340266059SGregory Neil Shapiro 			break;
23440266059SGregory Neil Shapiro 		if (any < 0)
23540266059SGregory Neil Shapiro 			continue;
23640266059SGregory Neil Shapiro 		if (acc > cutoff || (acc == cutoff && c > cutlim))
23740266059SGregory Neil Shapiro 		{
23840266059SGregory Neil Shapiro 			any = -1;
23940266059SGregory Neil Shapiro 			acc = ULLONG_MAX;
24040266059SGregory Neil Shapiro 			errno = ERANGE;
24140266059SGregory Neil Shapiro 		}
24240266059SGregory Neil Shapiro 		else
24340266059SGregory Neil Shapiro 		{
24440266059SGregory Neil Shapiro 			any = 1;
24540266059SGregory Neil Shapiro 			acc *= (ULONGLONG_T)base;
24640266059SGregory Neil Shapiro 			acc += c;
24740266059SGregory Neil Shapiro 		}
24840266059SGregory Neil Shapiro 	}
24940266059SGregory Neil Shapiro 	if (neg && any > 0)
25040266059SGregory Neil Shapiro 		acc = -((LONGLONG_T) acc);
251*5b0945b5SGregory Neil Shapiro 	if (endptr != NULL)
25240266059SGregory Neil Shapiro 		*endptr = (char *) (any ? s - 1 : nptr);
25340266059SGregory Neil Shapiro 	return acc;
25440266059SGregory Neil Shapiro }
255