| /* Copyright (C) 1991, 1992, 1994, 1995 Free Software Foundation, Inc. |
| |
| This file is part of the GNU C Library. |
| |
| The GNU C Library is free software; you can redistribute it and/or |
| modify it under the terms of the GNU Library General Public License as |
| published by the Free Software Foundation; either version 2 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 |
| Library General Public License for more details. |
| |
| You should have received a copy of the GNU Library General Public |
| License along with the GNU C Library; see the file COPYING.LIB. If |
| not, write to the Free Software Foundation, Inc., 675 Mass Ave, |
| Cambridge, MA 02139, USA. */ |
| |
| /* Convert NPTR to an `unsigned long int' or `long int' in base BASE. |
| If BASE is 0 the base is determined by the presence of a leading |
| zero, indicating octal or a leading "0x" or "0X", indicating hexadecimal. |
| If BASE is < 2 or > 36, it is reset to 10. |
| If ENDPTR is not NULL, a pointer to the character after the last |
| one converted is stored in *ENDPTR. */ |
| |
| #include "ctype.h" |
| |
| int strtol (nptr, endptr, base) |
| const char *nptr; |
| char **endptr; |
| int base; |
| { |
| int negative; |
| register unsigned int cutoff; |
| register unsigned int cutlim; |
| register unsigned int i; |
| register const char *s; |
| register unsigned char c; |
| const char *save, *end; |
| int overflow; |
| |
| if (base < 0 || base == 1 || base > 36) |
| base = 10; |
| |
| save = s = nptr; |
| |
| /* Skip white space. */ |
| while (((unsigned char) *s) <= 32 && *s) |
| ++s; |
| if (*s == '\0') |
| goto noconv; |
| |
| /* Check for a sign. */ |
| if (*s == '-') { |
| negative = 1; |
| ++s; |
| } else if (*s == '+') { |
| negative = 0; |
| ++s; |
| } else |
| negative = 0; |
| |
| if ((base == 16 && s[0] == '0' && (s[1] == 'X')) || (s[1] == 'x')) |
| s += 2; |
| |
| /* If BASE is zero, figure it out ourselves. */ |
| if (base == 0) { |
| if (*s == '0') { |
| if (s[1] == 'X' || s[1] == 'x') { |
| s += 2; |
| base = 16; |
| } else |
| base = 8; |
| } else |
| base = 10; |
| } |
| |
| /* Save the pointer so we can check later if anything happened. */ |
| save = s; |
| |
| end = 0; |
| |
| cutoff = 0x7FFFFFFF / (unsigned int) base; |
| cutlim = 0x7FFFFFFF % (unsigned int) base; |
| |
| overflow = 0; |
| i = 0; |
| for (c = *s; c != '\0'; c = *++s) { |
| if (s == end) |
| break; |
| if (c >= '0' && c <= '9') |
| c -= '0'; |
| else if (c >= 'A' && c <= 'Z') |
| c = c - 'A' + 10; |
| else if (c >= 'a' && c <= 'z') |
| c = c - 'a' + 10; |
| else |
| break; |
| if (c >= base) |
| break; |
| /* Check for overflow. */ |
| if (i > cutoff || (i == cutoff && c > cutlim)) |
| overflow = 1; |
| else { |
| i *= (unsigned int) base; |
| i += c; |
| } |
| } |
| |
| /* Check if anything actually happened. */ |
| if (s == save) |
| goto noconv; |
| |
| /* Store in ENDPTR the address of one character |
| past the last character we converted. */ |
| if (endptr) |
| *endptr = (char *) s; |
| |
| if (overflow) |
| return negative ? (int) 0x80000000 : (int) 0x7FFFFFFF; |
| |
| /* Return the result of the appropriate sign. */ |
| return (negative ? -i : i); |
| |
| noconv: |
| /* We must handle a special case here: the base is 0 or 16 and the |
| first two characters and '0' and 'x', but the rest are no |
| hexadecimal digits. This is no error case. We return 0 and |
| ENDPTR points to the `x`. */ |
| if (endptr) { |
| if (save - nptr >= 2 && tolower (save[-1]) == 'x' && save[-2] == '0') |
| *endptr = (char *) &save[-1]; |
| else |
| /* There was no number to convert. */ |
| *endptr = (char *) nptr; |
| } |
| |
| return 0L; |
| } |