mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 10:44:55 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			151 lines
		
	
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			151 lines
		
	
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
 | |
| #include "Python.h"
 | |
| 
 | |
| #if defined(__sgi) && defined(WITH_THREAD) && !defined(_SGI_MP_SOURCE)
 | |
| #define _SGI_MP_SOURCE
 | |
| #endif
 | |
| 
 | |
| /* Convert a possibly signed character to a nonnegative int */
 | |
| /* XXX This assumes characters are 8 bits wide */
 | |
| #ifdef __CHAR_UNSIGNED__
 | |
| #define Py_CHARMASK(c)		(c)
 | |
| #else
 | |
| #define Py_CHARMASK(c)		((c) & 0xff)
 | |
| #endif
 | |
| 
 | |
| /* strtol and strtoul, renamed to avoid conflicts */
 | |
| 
 | |
| /*
 | |
| **	strtoul
 | |
| **		This is a general purpose routine for converting
 | |
| **		an ascii string to an integer in an arbitrary base.
 | |
| **		Leading white space is ignored.  If 'base' is zero
 | |
| **		it looks for a leading 0, 0x or 0X to tell which
 | |
| **		base.  If these are absent it defaults to 10.
 | |
| **		Base must be 0 or between 2 and 36 (inclusive).
 | |
| **		If 'ptr' is non-NULL it will contain a pointer to
 | |
| **		the end of the scan.
 | |
| **		Errors due to bad pointers will probably result in
 | |
| **		exceptions - we don't check for them.
 | |
| */
 | |
| 
 | |
| #include <ctype.h>
 | |
| #ifndef DONT_HAVE_ERRNO_H
 | |
| #include <errno.h>
 | |
| #endif
 | |
| 
 | |
| unsigned long
 | |
| PyOS_strtoul(register char *str, char **ptr, int base)
 | |
| {
 | |
|     register unsigned long	result;	/* return value of the function */
 | |
|     register int		c;	/* current input character */
 | |
|     register unsigned long	temp;	/* used in overflow testing */
 | |
|     int				ovf;	/* true if overflow occurred */
 | |
| 
 | |
|     result = 0;
 | |
|     ovf = 0;
 | |
| 
 | |
| /* catch silly bases */
 | |
|     if (base != 0 && (base < 2 || base > 36))
 | |
|     {
 | |
| 	if (ptr)
 | |
| 	    *ptr = str;
 | |
| 	return 0;
 | |
|     }
 | |
| 
 | |
| /* skip leading white space */
 | |
|     while (*str && isspace(Py_CHARMASK(*str)))
 | |
| 	str++;
 | |
| 
 | |
| /* check for leading 0 or 0x for auto-base or base 16 */
 | |
|     switch (base)
 | |
|     {
 | |
|     case 0:		/* look for leading 0, 0x or 0X */
 | |
| 	if (*str == '0')
 | |
| 	{
 | |
| 	    str++;
 | |
| 	    if (*str == 'x' || *str == 'X')
 | |
| 	    {
 | |
| 		str++;
 | |
| 		base = 16;
 | |
| 	    }
 | |
| 	    else
 | |
| 		base = 8;
 | |
| 	}
 | |
| 	else
 | |
| 	    base = 10;
 | |
| 	break;
 | |
| 
 | |
|     case 16:	/* skip leading 0x or 0X */
 | |
| 	if (*str == '0' && (*(str+1) == 'x' || *(str+1) == 'X'))
 | |
| 	    str += 2;
 | |
| 	break;
 | |
|     }
 | |
| 
 | |
| /* do the conversion */
 | |
|     while ((c = Py_CHARMASK(*str)) != '\0')
 | |
|     {
 | |
| 	if (isdigit(c) && c - '0' < base)
 | |
| 	    c -= '0';
 | |
| 	else
 | |
| 	{
 | |
| 	    if (isupper(c))
 | |
| 		c = tolower(c);
 | |
| 	    if (c >= 'a' && c <= 'z')
 | |
| 		c -= 'a' - 10;
 | |
| 	    else	/* non-"digit" character */
 | |
| 		break;
 | |
| 	    if (c >= base)	/* non-"digit" character */
 | |
| 		break;
 | |
| 	}
 | |
| 	temp = result;
 | |
| 	result = result * base + c;
 | |
| 	if(base == 10) {
 | |
| 		if(((long)(result - c) / base != (long)temp))	/* overflow */
 | |
| 			ovf = 1;
 | |
| 	}
 | |
| 	else {
 | |
| 		if ((result - c) / base != temp)	/* overflow */
 | |
| 			ovf = 1;
 | |
| 	}
 | |
| 	str++;
 | |
|     }
 | |
| 
 | |
| /* set pointer to point to the last character scanned */
 | |
|     if (ptr)
 | |
| 	*ptr = str;
 | |
|     if (ovf)
 | |
|     {
 | |
| 	result = (unsigned long) ~0L;
 | |
| 	errno = ERANGE;
 | |
|     }
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| long
 | |
| PyOS_strtol(char *str, char **ptr, int base)
 | |
| {
 | |
| 	long result;
 | |
| 	char sign;
 | |
| 	
 | |
| 	while (*str && isspace(Py_CHARMASK(*str)))
 | |
| 		str++;
 | |
| 	
 | |
| 	sign = *str;
 | |
| 	if (sign == '+' || sign == '-')
 | |
| 		str++;
 | |
| 	
 | |
| 	result = (long) PyOS_strtoul(str, ptr, base);
 | |
| 	
 | |
| 	/* Signal overflow if the result appears negative,
 | |
| 	   except for the largest negative integer */
 | |
| 	if (result < 0 && !(sign == '-' && result == -result)) {
 | |
| 		errno = ERANGE;
 | |
| 		result = 0x7fffffff;
 | |
| 	}
 | |
| 	
 | |
| 	if (sign == '-')
 | |
| 		result = -result;
 | |
| 	
 | |
| 	return result;
 | |
| }
 | 
