_strtoi.h revision 1.5 1 /* $NetBSD: _strtoi.h,v 1.5 2024/07/24 09:11:27 kre Exp $ */
2
3 /*-
4 * Copyright (c) 1990, 1993
5 * The Regents of the University of California. All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3. Neither the name of the University nor the names of its contributors
16 * may be used to endorse or promote products derived from this software
17 * without specific prior written permission.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
23 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29 * SUCH DAMAGE.
30 *
31 * Original version ID:
32 * NetBSD: src/lib/libc/locale/_wcstoul.h,v 1.2 2003/08/07 16:43:03 agc Exp
33 *
34 * Created by Kamil Rytarowski, based on ID:
35 * NetBSD: src/common/lib/libc/stdlib/_strtoul.h,v 1.7 2013/05/17 12:55:56 joerg Exp
36 */
37
38 /*
39 * function template for strtoi and strtou
40 *
41 * parameters:
42 * _FUNCNAME : function name
43 * __TYPE : return and range limits type
44 * __WRAPPED : wrapped function, strtoimax or strtoumax
45 */
46
47 #define __WRAPPED_L_(x) x ## _l
48 #define __WRAPPED_L__(x) __WRAPPED_L_(x)
49 #define __WRAPPED_L __WRAPPED_L__(__WRAPPED)
50
51 #if defined(_KERNEL) || defined(_STANDALONE) || \
52 defined(HAVE_NBTOOL_CONFIG_H) || defined(BCS_ONLY)
53 __TYPE
54 _FUNCNAME(const char * __restrict nptr, char ** __restrict endptr, int base,
55 __TYPE lo, __TYPE hi, int * rstatus)
56 #else
57 #include <locale.h>
58 #include "setlocale_local.h"
59 #define INT_FUNCNAME_(pre, name, post) pre ## name ## post
60 #define INT_FUNCNAME(pre, name, post) INT_FUNCNAME_(pre, name, post)
61
62 static __TYPE
63 INT_FUNCNAME(_int_, _FUNCNAME, _l)(const char * __restrict nptr,
64 char ** __restrict endptr, int base,
65 __TYPE lo, __TYPE hi, int * rstatus, locale_t loc)
66 #endif
67 {
68 #if !defined(_KERNEL) && !defined(_STANDALONE)
69 int serrno;
70 #endif
71 __TYPE im;
72 char *ep;
73 int rep;
74
75 _DIAGASSERT(hi >= lo);
76
77 _DIAGASSERT(nptr != NULL);
78 /* endptr may be NULL */
79
80 if (endptr == NULL)
81 endptr = &ep;
82
83 if (rstatus == NULL)
84 rstatus = &rep;
85
86 *rstatus = 0; /* assume there will be no errors */
87
88 if (base != 0 && (base < 2 || base > 36)) {
89 #if !defined(_KERNEL) && !defined(_STANDALONE)
90 *rstatus = EINVAL;
91 if (endptr != NULL)
92 /* LINTED interface specification */
93 *endptr = __UNCONST(nptr);
94 return 0;
95 #else
96 panic("%s: invalid base %d", __func__, base);
97 #endif
98 }
99
100 #if !defined(_KERNEL) && !defined(_STANDALONE)
101 serrno = errno;
102 errno = 0;
103 #endif
104
105 #if defined(_KERNEL) || defined(_STANDALONE) || \
106 defined(HAVE_NBTOOL_CONFIG_H) || defined(BCS_ONLY)
107 im = __WRAPPED(nptr, endptr, base);
108 #else
109 im = __WRAPPED_L(nptr, endptr, base, loc);
110 #endif
111
112 #if !defined(_KERNEL) && !defined(_STANDALONE)
113 /* EINVAL here can only mean "nothing converted" */
114 if (errno != EINVAL)
115 *rstatus = errno;
116 errno = serrno;
117 #endif
118
119 /* No digits were found */
120 if (*rstatus == 0 && nptr == *endptr)
121 *rstatus = ECANCELED;
122
123 if (im < lo) {
124 if (*rstatus == 0)
125 *rstatus = ERANGE;
126 return lo;
127 }
128
129 if (im > hi) {
130 if (*rstatus == 0)
131 *rstatus = ERANGE;
132 return hi;
133 }
134
135 /* There are further characters after number */
136 if (*rstatus == 0 && **endptr != '\0')
137 *rstatus = ENOTSUP;
138
139 return im;
140 }
141
142 #if !defined(_KERNEL) && !defined(_STANDALONE) && \
143 !defined(HAVE_NBTOOL_CONFIG_H) && !defined(BCS_ONLY)
144 __TYPE
145 _FUNCNAME(const char * __restrict nptr, char ** __restrict endptr, int base,
146 __TYPE lo, __TYPE hi, int * rstatus)
147 {
148 return INT_FUNCNAME(_int_, _FUNCNAME, _l)(nptr, endptr, base, lo, hi,
149 rstatus, _current_locale());
150 }
151
152 __TYPE
153 INT_FUNCNAME(, _FUNCNAME, _l)(const char * __restrict nptr,
154 char ** __restrict endptr, int base,
155 __TYPE lo, __TYPE hi, int * rstatus, locale_t loc)
156 {
157 return INT_FUNCNAME(_int_, _FUNCNAME, _l)(nptr, endptr, base, lo, hi,
158 rstatus, loc);
159 }
160 #endif
161