10Sstevel@tonic-gate #if defined(LIBC_SCCS) && !defined(lint) 20Sstevel@tonic-gate static const char sccsid[] = "@(#)strtoul.c 8.1 (Berkeley) 6/4/93"; 3*11038SRao.Shoaib@Sun.COM static const char rcsid[] = "$Id: strtoul.c,v 1.4 2008/02/18 03:49:08 marka Exp $"; 40Sstevel@tonic-gate #endif /* LIBC_SCCS and not lint */ 50Sstevel@tonic-gate 60Sstevel@tonic-gate /* 70Sstevel@tonic-gate * Copyright (c) 1990, 1993 80Sstevel@tonic-gate * The Regents of the University of California. All rights reserved. 90Sstevel@tonic-gate * 100Sstevel@tonic-gate * Redistribution and use in source and binary forms, with or without 110Sstevel@tonic-gate * modification, are permitted provided that the following conditions 120Sstevel@tonic-gate * are met: 130Sstevel@tonic-gate * 1. Redistributions of source code must retain the above copyright 140Sstevel@tonic-gate * notice, this list of conditions and the following disclaimer. 150Sstevel@tonic-gate * 2. Redistributions in binary form must reproduce the above copyright 160Sstevel@tonic-gate * notice, this list of conditions and the following disclaimer in the 170Sstevel@tonic-gate * documentation and/or other materials provided with the distribution. 180Sstevel@tonic-gate * 3. All advertising materials mentioning features or use of this software 190Sstevel@tonic-gate * must display the following acknowledgement: 200Sstevel@tonic-gate * This product includes software developed by the University of 210Sstevel@tonic-gate * California, Berkeley and its contributors. 220Sstevel@tonic-gate * 4. Neither the name of the University nor the names of its contributors 230Sstevel@tonic-gate * may be used to endorse or promote products derived from this software 240Sstevel@tonic-gate * without specific prior written permission. 250Sstevel@tonic-gate * 260Sstevel@tonic-gate * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 270Sstevel@tonic-gate * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 280Sstevel@tonic-gate * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 290Sstevel@tonic-gate * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 300Sstevel@tonic-gate * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 310Sstevel@tonic-gate * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 320Sstevel@tonic-gate * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 330Sstevel@tonic-gate * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 340Sstevel@tonic-gate * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 350Sstevel@tonic-gate * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 360Sstevel@tonic-gate * SUCH DAMAGE. 370Sstevel@tonic-gate */ 380Sstevel@tonic-gate 390Sstevel@tonic-gate #include "port_before.h" 400Sstevel@tonic-gate 410Sstevel@tonic-gate #include <sys/types.h> 420Sstevel@tonic-gate #include <sys/param.h> 430Sstevel@tonic-gate 440Sstevel@tonic-gate #include <ctype.h> 450Sstevel@tonic-gate #include <errno.h> 460Sstevel@tonic-gate #include <limits.h> 470Sstevel@tonic-gate #include <stdlib.h> 480Sstevel@tonic-gate 490Sstevel@tonic-gate #include "port_after.h" 500Sstevel@tonic-gate 510Sstevel@tonic-gate #ifndef NEED_STRTOUL 520Sstevel@tonic-gate int __strtoul_unneeded__; 530Sstevel@tonic-gate #else 540Sstevel@tonic-gate 55*11038SRao.Shoaib@Sun.COM /*% 560Sstevel@tonic-gate * Convert a string to an unsigned long integer. 570Sstevel@tonic-gate * 580Sstevel@tonic-gate * Ignores `locale' stuff. Assumes that the upper and lower case 590Sstevel@tonic-gate * alphabets and digits are each contiguous. 600Sstevel@tonic-gate */ 610Sstevel@tonic-gate u_long 620Sstevel@tonic-gate strtoul(const char *nptr, char **endptr, int base) { 630Sstevel@tonic-gate const char *s = nptr; 640Sstevel@tonic-gate u_long acc, cutoff; 650Sstevel@tonic-gate int neg, c, any, cutlim; 660Sstevel@tonic-gate 670Sstevel@tonic-gate neg = 0; 680Sstevel@tonic-gate 690Sstevel@tonic-gate /* 700Sstevel@tonic-gate * See strtol for comments as to the logic used. 710Sstevel@tonic-gate */ 720Sstevel@tonic-gate do { 73*11038SRao.Shoaib@Sun.COM c = *(const unsigned char *)s++; 740Sstevel@tonic-gate } while (isspace(c)); 750Sstevel@tonic-gate if (c == '-') { 760Sstevel@tonic-gate neg = 1; 770Sstevel@tonic-gate c = *s++; 780Sstevel@tonic-gate } else if (c == '+') 790Sstevel@tonic-gate c = *s++; 800Sstevel@tonic-gate if ((base == 0 || base == 16) && 810Sstevel@tonic-gate c == '0' && (*s == 'x' || *s == 'X')) { 820Sstevel@tonic-gate c = s[1]; 830Sstevel@tonic-gate s += 2; 840Sstevel@tonic-gate base = 16; 850Sstevel@tonic-gate } 860Sstevel@tonic-gate if (base == 0) 870Sstevel@tonic-gate base = c == '0' ? 8 : 10; 880Sstevel@tonic-gate cutoff = (u_long)ULONG_MAX / (u_long)base; 890Sstevel@tonic-gate cutlim = (u_long)ULONG_MAX % (u_long)base; 90*11038SRao.Shoaib@Sun.COM for (acc = 0, any = 0;; c = *(const unsigned char*)s++) { 910Sstevel@tonic-gate if (isdigit(c)) 920Sstevel@tonic-gate c -= '0'; 930Sstevel@tonic-gate else if (isalpha(c)) 940Sstevel@tonic-gate c -= isupper(c) ? 'A' - 10 : 'a' - 10; 950Sstevel@tonic-gate else 960Sstevel@tonic-gate break; 970Sstevel@tonic-gate if (c >= base) 980Sstevel@tonic-gate break; 99*11038SRao.Shoaib@Sun.COM if (any < 0 || acc > cutoff || (acc == cutoff && c > cutlim)) 1000Sstevel@tonic-gate any = -1; 1010Sstevel@tonic-gate else { 1020Sstevel@tonic-gate any = 1; 1030Sstevel@tonic-gate acc *= base; 1040Sstevel@tonic-gate acc += c; 1050Sstevel@tonic-gate } 1060Sstevel@tonic-gate } 1070Sstevel@tonic-gate if (any < 0) { 1080Sstevel@tonic-gate acc = ULONG_MAX; 1090Sstevel@tonic-gate errno = ERANGE; 1100Sstevel@tonic-gate } else if (neg) 1110Sstevel@tonic-gate acc = -acc; 1120Sstevel@tonic-gate if (endptr != 0) 113*11038SRao.Shoaib@Sun.COM DE_CONST((any ? s - 1 : nptr), *endptr); 1140Sstevel@tonic-gate return (acc); 1150Sstevel@tonic-gate } 1160Sstevel@tonic-gate 1170Sstevel@tonic-gate #endif /*NEED_STRTOUL*/ 118*11038SRao.Shoaib@Sun.COM 119*11038SRao.Shoaib@Sun.COM /*! \file */ 120