strtoul.c
上传用户:ycwykj01
上传日期:2007-01-04
资源大小:1819k
文件大小:4k
源码类别:

网络编程

开发平台:

Unix_Linux

  1. /*
  2.  * Program: String to unsigned long
  3.  *
  4.  * Author: Mark Crispin
  5.  * Networks and Distributed Computing
  6.  * Computing & Communications
  7.  * University of Washington
  8.  * Administration Building, AG-44
  9.  * Seattle, WA  98195
  10.  *
  11.  * Date: 14 February 1995
  12.  * Last Edited: 23 March 1995
  13.  *
  14.  * Copyright 1995 by the University of Washington
  15.  *
  16.  *  Permission to use, copy, modify, and distribute this software and its
  17.  * documentation for any purpose and without fee is hereby granted, provided
  18.  * that the above copyright notice appears in all copies and that both the
  19.  * above copyright notice and this permission notice appear in supporting
  20.  * documentation, and that the name of the University of Washington not be
  21.  * used in advertising or publicity pertaining to distribution of the software
  22.  * without specific, written prior permission.  This software is made
  23.  * available "as is", and
  24.  * THE UNIVERSITY OF WASHINGTON DISCLAIMS ALL WARRANTIES, EXPRESS OR IMPLIED,
  25.  * WITH REGARD TO THIS SOFTWARE, INCLUDING WITHOUT LIMITATION ALL IMPLIED
  26.  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, AND IN
  27.  * NO EVENT SHALL THE UNIVERSITY OF WASHINGTON BE LIABLE FOR ANY SPECIAL,
  28.  * INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
  29.  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, TORT
  30.  * (INCLUDING NEGLIGENCE) OR STRICT LIABILITY, ARISING OUT OF OR IN CONNECTION
  31.  * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  32.  *
  33.  */
  34. /*
  35.  * Turn a string unsigned long into the real thing
  36.  * Accepts: source string
  37.  *     pointer to place to return end pointer
  38.  *     base
  39.  * Returns: parsed unsigned long integer, end pointer is updated
  40.  */
  41. unsigned long strtoul (char *s,char **endp,int base)
  42. {
  43.   unsigned long value = 0; /* the accumulated value */
  44.   /* Yes, you can feed it negative number strings */
  45.   int negative = 0; /* this a negative number? */
  46.   int ok; /* true while valid chars for number */
  47.   char c;
  48.   if (base && (base < 2 || base > 36)) {
  49.     errno = EINVAL; /* insist upon valid base */
  50.     return NIL;
  51.   }
  52.   while (isspace (*s)) s++; /* skip leading whitespace */
  53.   if (base) { /* if have a base */
  54.     switch (*s) { /* check for leading sign char */
  55.     case '-':
  56.       negative = 1; /* yes, negative #.  fall into '+' */
  57.     case '+':
  58.       s++; /* skip the sign character */
  59.     }
  60. /* allow hex prefix "0x" */
  61.     if (base == 16 && *s == '0' && toupper (*(s + 1)) == 'X') s += 2;
  62.     do { /* convert to numeric form if digit*/
  63.       if (isdigit (*s)) c = toint (*s);
  64. /* alphabetic conversion */
  65.       else if (isalpha (*s)) c = *s - (isupper (*s) ? 'A' : 'a') + 10;
  66.       else break; /* else no way it's valid */
  67.       if (c >= base) break; /* digit out of range for base? */
  68.       value = value * base + c; /* accumulate the digit */
  69.     } while (*++s); /* loop until non-numeric character */
  70.   }
  71.   else { /* if base = 0, */
  72.     if (*s == '-') { /* integer constants are allowed a */
  73.       negative = 1; /* leading unary minus, but not */
  74.       s++; /* unary plus. */
  75.     }
  76.     if (*s == '0') { /* if it starts with 0, its either */
  77.       if (toupper (*++s)=='X') {/* 0X, which marks a hex constant, */
  78. s++; /* skip the X */
  79.         base = 16; /* base is hex 16 */
  80.       }
  81.       else base = 8; /* leading 0 means octal number */
  82.     }
  83.     else base = 10; /* otherwise, decimal. */
  84.     do {
  85.       switch (base) { /* char has to be valid for base */
  86.       case 8: /* this digit ok? */
  87. ok = isodigit (*s);
  88. break;
  89.       case 10:
  90. ok = isdigit (*s);
  91. break;
  92.       case 16:
  93. ok = isxdigit (*s);
  94. break;
  95.       default: /* it's good form you know */
  96. return NIL;
  97.       }
  98. /* if valid char, accumulate */
  99.       if (ok) value = value * base + toint (*s++);
  100.     } while (ok);
  101.     if (toupper (*s) == 'L')s++;/* ignore 'l' or 'L' marker */
  102.   }
  103.   if (endp) *endp = s; /* save users endp to after number */
  104. /* negate number if needed */
  105.   return (negative) ? -value : value;
  106. }