4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
23 * Copyright 2011 Nexenta Systems, Inc. All rights reserved.
27 * Copyright 2009 Sun Microsystems, Inc. All rights reserved.
28 * Use is subject to license terms.
31 /* Copyright (c) 1988 AT&T */
32 /* All Rights Reserved */
34 #if defined(_KERNEL) && !defined(_BOOT)
35 #include <sys/errno.h>
36 #else /* _KERNEL && !_BOOT */
37 #if !defined(_BOOT) && !defined(_KMDB)
39 #endif /* !_BOOT && !_KMDB */
44 #endif /* _KERNEL && !_BOOT */
45 #include "strtolctype.h"
46 #include <sys/types.h>
48 #if defined(_KERNEL) && !defined(_BOOT)
50 ddi_strtol(const char *str
, char **nptr
, int base
, long *result
)
51 #else /* _KERNEL && !_BOOT */
53 strtol(const char *str
, char **nptr
, int base
)
54 #endif /* _KERNEL && !_BOOT */
62 const char **ptr
= (const char **)nptr
;
63 const unsigned char *ustr
= (const unsigned char *)str
;
65 if (ptr
!= (const char **)0)
66 *ptr
= (char *)ustr
; /* in case no number is formed */
67 if (base
< 0 || base
> MBASE
|| base
== 1) {
68 /* base is invalid -- should be a fatal error */
69 #if defined(_KERNEL) && !defined(_BOOT)
71 #else /* _KERNEL && !_BOOT */
74 #endif /* _KERNEL && !_BOOT */
76 if (!isalnum(c
= *ustr
)) {
90 else if (ustr
[1] == 'x' || ustr
[1] == 'X')
95 * for any base > 10, the digits incrementally following
96 * 9 are assumed to be "abc...z" or "ABC...Z"
98 if (!lisalnum(c
) || (xx
= DIGIT(c
)) >= base
) {
99 /* no number formed */
100 #if defined(_KERNEL) && !defined(_BOOT)
102 #else /* _KERNEL && !_BOOT */
105 #endif /* _KERNEL && !_BOOT */
107 if (base
== 16 && c
== '0' && (ustr
[1] == 'x' || ustr
[1] == 'X') &&
109 c
= *(ustr
+= 2); /* skip over leading "0x" or "0X" */
111 /* this code assumes that abs(LONG_MIN) >= abs(LONG_MAX) */
116 multmin
= limit
/ (long)base
;
118 for (c
= *++ustr
; lisalnum(c
) && (xx
= DIGIT(c
)) < base
; ) {
119 /* accumulate neg avoids surprises near LONG_MAX */
123 if (val
< limit
+ xx
)
128 if (ptr
!= (const char **)0)
130 #if defined(_KERNEL) && !defined(_BOOT)
131 *result
= neg
? val
: -val
;
133 #else /* _KERNEL && !_BOOT */
134 return (neg
? val
: -val
);
135 #endif /* _KERNEL && !_BOOT */
138 for (c
= *++ustr
; lisalnum(c
) && (xx
= DIGIT(c
)) < base
; (c
= *++ustr
))
140 if (ptr
!= (const char **)0)
142 #if defined(_KERNEL) && !defined(_BOOT)
144 #else /* _KERNEL && !_BOOT */
146 return (neg
? LONG_MIN
: LONG_MAX
);
147 #endif /* _KERNEL && !_BOOT */