be2474d3500844d3a478d89647b59e08a1fba8a5
[karo-tx-redboot.git] / packages / language / c / libc / stdlib / v2_0 / src / strtoul.cxx
1 //===========================================================================
2 //
3 //      strtoul.cxx
4 //
5 //      ISO standard string to long int conversion function defined in
6 //      section 7.10.1.6 of the standard
7 //
8 //===========================================================================
9 //####ECOSGPLCOPYRIGHTBEGIN####
10 // -------------------------------------------
11 // This file is part of eCos, the Embedded Configurable Operating System.
12 // Copyright (C) 1998, 1999, 2000, 2001, 2002 Red Hat, Inc.
13 //
14 // eCos is free software; you can redistribute it and/or modify it under
15 // the terms of the GNU General Public License as published by the Free
16 // Software Foundation; either version 2 or (at your option) any later version.
17 //
18 // eCos is distributed in the hope that it will be useful, but WITHOUT ANY
19 // WARRANTY; without even the implied warranty of MERCHANTABILITY or
20 // FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
21 // for more details.
22 //
23 // You should have received a copy of the GNU General Public License along
24 // with eCos; if not, write to the Free Software Foundation, Inc.,
25 // 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
26 //
27 // As a special exception, if other files instantiate templates or use macros
28 // or inline functions from this file, or you compile this file and link it
29 // with other works to produce a work based on this file, this file does not
30 // by itself cause the resulting work to be covered by the GNU General Public
31 // License. However the source code for this file must still be made available
32 // in accordance with section (3) of the GNU General Public License.
33 //
34 // This exception does not invalidate any other reasons why a work based on
35 // this file might be covered by the GNU General Public License.
36 //
37 // Alternative licenses for eCos may be arranged by contacting Red Hat, Inc.
38 // at http://sources.redhat.com/ecos/ecos-license/
39 // -------------------------------------------
40 //####ECOSGPLCOPYRIGHTEND####
41 //===========================================================================
42 //#####DESCRIPTIONBEGIN####
43 //
44 // Author(s):    jlarmour
45 // Contributors: 
46 // Date:         2000-04-30
47 // Purpose:     
48 // Description: 
49 // Usage:       
50 //
51 //####DESCRIPTIONEND####
52 //
53 //===========================================================================
54 //
55 // This code is based on original code with the following copyright:
56 //
57 /*
58  * Copyright (c) 1990 Regents of the University of California.
59  * All rights reserved.
60  *
61  * Redistribution and use in source and binary forms, with or without
62  * modification, are permitted provided that the following conditions
63  * are met:
64  * 1. Redistributions of source code must retain the above copyright
65  *    notice, this list of conditions and the following disclaimer.
66  * 2. Redistributions in binary form must reproduce the above copyright
67  *    notice, this list of conditions and the following disclaimer in the
68  *    documentation and/or other materials provided with the distribution.
69  * 3. Neither the name of the University nor the names of its contributors
70  *    may be used to endorse or promote products derived from this software
71  *    without specific prior written permission.
72  *
73  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
74  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
75  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
76  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
77  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
78  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
79  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
80  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
81  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
82  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
83  * SUCH DAMAGE.
84  */
85
86
87 // CONFIGURATION
88
89 #include <pkgconf/libc_stdlib.h>   // Configuration header
90
91 // INCLUDES
92
93 #include <cyg/infra/cyg_type.h>     // Common type definitions and support
94 #include <cyg/infra/cyg_trac.h>     // Tracing support
95 #include <cyg/infra/cyg_ass.h>      // Assertion support
96 #include <limits.h>                 // Definition of ULONG_MAX
97 #include <ctype.h>                  // Definition of many ctype functions
98 #include <errno.h>                  // Error code definitions
99 #include <stdlib.h>                 // Header for all stdlib functions
100                                     // (like this one)
101
102
103 // FUNCTIONS
104
105 //
106 // Convert a string to an unsigned long integer.
107 //
108 // Ignores `locale' stuff.  Assumes that the upper and lower case
109 // alphabets and digits are each contiguous.
110 //
111
112 unsigned long
113 strtoul( const char *nptr, char **endptr, int base )
114 {
115     const char *s = nptr;
116     unsigned long acc;
117     int c;
118     unsigned long cutoff;
119     int neg = 0, any, cutlim;
120     
121     CYG_REPORT_FUNCNAMETYPE( "strtoul", "returning long %d" );
122     CYG_REPORT_FUNCARG3( "nptr=%08x, endptr=%08x, base=%d",
123                          nptr, endptr, base );
124     CYG_CHECK_DATA_PTR( nptr, "nptr is not a valid pointer!" );
125
126     if (endptr != NULL)
127         CYG_CHECK_DATA_PTR( endptr, "endptr is not a valid pointer!" );
128     //
129     // See strtol for comments as to the logic used.
130     //
131     do {
132         c = *s++;
133     } while (isspace(c));
134     if (c == '-') {
135         neg = 1;
136         c = *s++;
137     } else if (c == '+')
138         c = *s++;
139     if ((base == 0 || base == 16) &&
140         c == '0' && (*s == 'x' || *s == 'X')) {
141         c = s[1];
142         s += 2;
143         base = 16;
144     }
145     if (base == 0)
146         base = c == '0' ? 8 : 10;
147     cutoff = (unsigned long)ULONG_MAX / (unsigned long)base;
148     cutlim = (unsigned long)ULONG_MAX % (unsigned long)base;
149     for (acc = 0, any = 0;; c = *s++) {
150         if (isdigit(c))
151             c -= '0';
152         else if (isalpha(c))
153             c -= isupper(c) ? 'A' - 10 : 'a' - 10;
154         else
155             break;
156         if (c >= base)
157             break;
158         if (any < 0 || acc > cutoff || acc == cutoff && c > cutlim)
159             any = -1;
160         else {
161             any = 1;
162             acc *= base;
163             acc += c;
164         }
165     }
166     if (any < 0) {
167         acc = ULONG_MAX;
168         errno = ERANGE;
169     } else if (neg)
170         acc = -acc;
171     if (endptr != 0)
172         *endptr = (char *) (any ? s - 1 : nptr);
173     
174     CYG_REPORT_RETVAL( acc );
175
176     return acc;
177 } // strtoul()
178
179 // EOF strtoul.cxx