Reunite uspace string-to-int conversion functions
[helenos.git] / uspace / lib / c / generic / strtol.c
bloba687f0a3dd2dcf0b83b8f1d4c27ad6256cbce286
1 /*
2 * Copyright (c) 2005 Martin Decky
3 * Copyright (c) 2008 Jiri Svoboda
4 * Copyright (c) 2011 Martin Sucha
5 * Copyright (c) 2011 Oleg Romanenko
6 * Copyright (c) 2011 Jiri Zarevucky
7 * All rights reserved.
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions
11 * are met:
13 * - Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * - Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
18 * - The name of the author may not be used to endorse or promote products
19 * derived from this software without specific prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
22 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
23 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
24 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
25 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
26 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
27 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
28 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
29 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
30 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
33 /** @addtogroup libc
34 * @{
36 /** @file
39 #include <assert.h>
40 #include <ctype.h>
41 #include <errno.h>
42 #include <inttypes.h>
43 #include <limits.h>
44 #include <stdbool.h>
45 #include <stdlib.h>
46 #include <str.h>
48 // FIXME: The original HelenOS functions return EOVERFLOW instead
49 // of ERANGE. It's a pointless distinction from standard functions,
50 // so we should change that. Beware the callers though.
52 // TODO: more unit tests
54 static inline int _digit_value(int c)
56 if (isdigit(c)) {
57 return c - '0';
58 } else if (islower(c)) {
59 return c - 'a' + 10;
60 } else if (isupper(c)) {
61 return c - 'A' + 10;
63 return INT_MAX;
67 * FIXME: workaround for GCC "optimizing" the overflow check
68 * into soft-emulated 128b multiplication using `__multi3`,
69 * which we don't currently implement.
71 __attribute__((noinline)) static uintmax_t _max_value(int base)
73 return UINTMAX_MAX / base;
76 static inline int _prefixbase(const char *restrict *nptrptr, bool nonstd)
78 const char *nptr = *nptrptr;
80 if (nptr[0] != '0')
81 return 10;
83 if (nptr[1] == 'x' || nptr[1] == 'X') {
84 if (_digit_value(nptr[2]) < 16) {
85 *nptrptr += 2;
86 return 16;
90 if (nonstd) {
91 switch (nptr[1]) {
92 case 'b':
93 case 'B':
94 if (_digit_value(nptr[2]) < 2) {
95 *nptrptr += 2;
96 return 2;
98 break;
99 case 'o':
100 case 'O':
101 if (_digit_value(nptr[2]) < 8) {
102 *nptrptr += 2;
103 return 8;
105 break;
106 case 'd':
107 case 'D':
108 case 't':
109 case 'T':
110 if (_digit_value(nptr[2]) < 10) {
111 *nptrptr += 2;
112 return 10;
114 break;
118 return 8;
121 static inline uintmax_t _strtoumax(
122 const char *restrict nptr, char **restrict endptr, int base,
123 bool *restrict sgn, errno_t *err, bool nonstd)
125 assert(nptr != NULL);
126 assert(sgn != NULL);
128 const char *first = nptr;
130 /* Skip leading whitespace. */
132 while (isspace(*nptr)) {
133 nptr++;
136 /* Parse sign, if any. */
138 switch (*nptr) {
139 case '-':
140 *sgn = true;
141 nptr++;
142 break;
143 case '+':
144 nptr++;
145 break;
148 /* Figure out the base. */
150 if (base == 0)
151 base = _prefixbase(&nptr, nonstd);
153 if (base == 16 && !nonstd) {
155 * Standard strto* functions allow hexadecimal prefix to be
156 * present when base is explicitly set to 16.
157 * Our nonstandard str_* functions don't allow it.
158 * I don't know if that is intended, just matching the original
159 * functionality here.
162 if (nptr[0] == '0' && (nptr[1] == 'x' || nptr[1] == 'X') &&
163 _digit_value(nptr[2]) < base)
164 nptr += 2;
167 if (base < 2 || base > 36) {
168 *err = EINVAL;
169 return 0;
172 /* Must be at least one digit. */
174 if (_digit_value(*nptr) >= base) {
175 /* No digits on input. */
176 if (endptr != NULL)
177 *endptr = (char *) first;
178 return 0;
181 /* Read the value. */
183 uintmax_t result = 0;
184 uintmax_t max = _max_value(base);
185 int digit;
187 while (digit = _digit_value(*nptr), digit < base) {
188 if (result > max ||
189 __builtin_add_overflow(result * base, digit, &result)) {
191 *err = nonstd ? EOVERFLOW : ERANGE;
192 result = UINTMAX_MAX;
193 break;
196 nptr++;
199 /* Set endptr. */
201 if (endptr != NULL) {
203 * Move the pointer to the end of the number,
204 * in case it isn't there already.
205 * This can happen when the number has legal formatting,
206 * but is out of range of the target type.
208 while (_digit_value(*nptr) < base) {
209 nptr++;
212 *endptr = (char *) nptr;
215 return result;
218 static inline intmax_t _strtosigned(const char *nptr, char **endptr, int base,
219 intmax_t min, intmax_t max, errno_t *err, bool nonstd)
221 bool sgn = false;
222 uintmax_t number = _strtoumax(nptr, endptr, base, &sgn, err, nonstd);
224 if (number > (uintmax_t) max) {
225 if (sgn && (number - 1 == (uintmax_t) max)) {
226 return min;
229 *err = nonstd ? EOVERFLOW : ERANGE;
230 return (sgn ? min : max);
233 return (sgn ? -number : number);
236 static inline uintmax_t _strtounsigned(const char *nptr, char **endptr, int base,
237 uintmax_t max, errno_t *err, bool nonstd)
239 bool sgn = false;
240 uintmax_t number = _strtoumax(nptr, endptr, base, &sgn, err, nonstd);
242 if (nonstd && sgn) {
243 /* Do not allow negative values */
244 *err = EINVAL;
245 return 0;
248 if (number > max) {
249 *err = nonstd ? EOVERFLOW : ERANGE;
250 return max;
253 return (sgn ? -number : number);
256 /** Convert initial part of string to long int according to given base.
257 * The number may begin with an arbitrary number of whitespaces followed by
258 * optional sign (`+' or `-'). If the base is 0 or 16, the prefix `0x' may be
259 * inserted and the number will be taken as hexadecimal one. If the base is 0
260 * and the number begin with a zero, number will be taken as octal one (as with
261 * base 8). Otherwise the base 0 is taken as decimal.
263 * @param nptr Pointer to string.
264 * @param[out] endptr If not NULL, function stores here pointer to the first
265 * invalid character.
266 * @param base Zero or number between 2 and 36 inclusive.
267 * @return Result of conversion.
269 long strtol(const char *nptr, char **endptr, int base)
271 return _strtosigned(nptr, endptr, base, LONG_MIN, LONG_MAX, &errno, false);
274 /** Convert initial part of string to unsigned long according to given base.
275 * The number may begin with an arbitrary number of whitespaces followed by
276 * optional sign (`+' or `-'). If the base is 0 or 16, the prefix `0x' may be
277 * inserted and the number will be taken as hexadecimal one. If the base is 0
278 * and the number begin with a zero, number will be taken as octal one (as with
279 * base 8). Otherwise the base 0 is taken as decimal.
281 * @param nptr Pointer to string.
282 * @param[out] endptr If not NULL, function stores here pointer to the first
283 * invalid character
284 * @param base Zero or number between 2 and 36 inclusive.
285 * @return Result of conversion.
287 unsigned long strtoul(const char *nptr, char **endptr, int base)
289 return _strtounsigned(nptr, endptr, base, ULONG_MAX, &errno, false);
292 long long strtoll(const char *nptr, char **endptr, int base)
294 return _strtosigned(nptr, endptr, base, LLONG_MIN, LLONG_MAX, &errno, false);
297 unsigned long long strtoull(const char *nptr, char **endptr, int base)
299 return _strtounsigned(nptr, endptr, base, ULLONG_MAX, &errno, false);
302 intmax_t strtoimax(const char *nptr, char **endptr, int base)
304 return _strtosigned(nptr, endptr, base, INTMAX_MIN, INTMAX_MAX, &errno, false);
307 uintmax_t strtoumax(const char *nptr, char **endptr, int base)
309 return _strtounsigned(nptr, endptr, base, UINTMAX_MAX, &errno, false);
312 int atoi(const char *nptr)
314 return (int)strtol(nptr, NULL, 10);
317 long atol(const char *nptr)
319 return strtol(nptr, NULL, 10);
322 long long atoll(const char *nptr)
324 return strtoll(nptr, NULL, 10);
327 /** Convert string to uint8_t.
329 * @param nptr Pointer to string.
330 * @param endptr If not NULL, pointer to the first invalid character
331 * is stored here.
332 * @param base Zero or number between 2 and 36 inclusive.
333 * @param strict Do not allow any trailing characters.
334 * @param result Result of the conversion.
336 * @return EOK if conversion was successful.
339 errno_t str_uint8_t(const char *nptr, const char **endptr, unsigned int base,
340 bool strict, uint8_t *result)
342 assert(result != NULL);
344 errno_t rc = EOK;
345 char *lendptr = (char *) nptr;
347 uintmax_t r = _strtounsigned(nptr, &lendptr, base, UINT8_MAX, &rc, true);
349 if (endptr)
350 *endptr = lendptr;
352 if (rc != EOK)
353 return rc;
355 if (strict && *lendptr != '\0')
356 return EINVAL;
358 *result = r;
359 return EOK;
362 /** Convert string to uint16_t.
364 * @param nptr Pointer to string.
365 * @param endptr If not NULL, pointer to the first invalid character
366 * is stored here.
367 * @param base Zero or number between 2 and 36 inclusive.
368 * @param strict Do not allow any trailing characters.
369 * @param result Result of the conversion.
371 * @return EOK if conversion was successful.
374 errno_t str_uint16_t(const char *nptr, const char **endptr, unsigned int base,
375 bool strict, uint16_t *result)
377 assert(result != NULL);
379 errno_t rc = EOK;
380 char *lendptr = (char *) nptr;
382 uintmax_t r = _strtounsigned(nptr, &lendptr, base, UINT16_MAX, &rc, true);
384 if (endptr)
385 *endptr = lendptr;
387 if (rc != EOK)
388 return rc;
390 if (strict && *lendptr != '\0')
391 return EINVAL;
393 *result = r;
394 return EOK;
397 /** Convert string to uint32_t.
399 * @param nptr Pointer to string.
400 * @param endptr If not NULL, pointer to the first invalid character
401 * is stored here.
402 * @param base Zero or number between 2 and 36 inclusive.
403 * @param strict Do not allow any trailing characters.
404 * @param result Result of the conversion.
406 * @return EOK if conversion was successful.
409 errno_t str_uint32_t(const char *nptr, const char **endptr, unsigned int base,
410 bool strict, uint32_t *result)
412 assert(result != NULL);
414 errno_t rc = EOK;
415 char *lendptr = (char *) nptr;
417 uintmax_t r = _strtounsigned(nptr, &lendptr, base, UINT32_MAX, &rc, true);
419 if (endptr)
420 *endptr = lendptr;
422 if (rc != EOK)
423 return rc;
425 if (strict && *lendptr != '\0')
426 return EINVAL;
428 *result = r;
429 return EOK;
432 /** Convert string to uint64_t.
434 * @param nptr Pointer to string.
435 * @param endptr If not NULL, pointer to the first invalid character
436 * is stored here.
437 * @param base Zero or number between 2 and 36 inclusive.
438 * @param strict Do not allow any trailing characters.
439 * @param result Result of the conversion.
441 * @return EOK if conversion was successful.
444 errno_t str_uint64_t(const char *nptr, const char **endptr, unsigned int base,
445 bool strict, uint64_t *result)
447 assert(result != NULL);
449 errno_t rc = EOK;
450 char *lendptr = (char *) nptr;
452 uintmax_t r = _strtounsigned(nptr, &lendptr, base, UINT64_MAX, &rc, true);
454 if (endptr)
455 *endptr = lendptr;
457 if (rc != EOK)
458 return rc;
460 if (strict && *lendptr != '\0')
461 return EINVAL;
463 *result = r;
464 return EOK;
467 /** Convert string to int64_t.
469 * @param nptr Pointer to string.
470 * @param endptr If not NULL, pointer to the first invalid character
471 * is stored here.
472 * @param base Zero or number between 2 and 36 inclusive.
473 * @param strict Do not allow any trailing characters.
474 * @param result Result of the conversion.
476 * @return EOK if conversion was successful.
479 errno_t str_int64_t(const char *nptr, const char **endptr, unsigned int base,
480 bool strict, int64_t *result)
482 assert(result != NULL);
484 errno_t rc = EOK;
485 char *lendptr = (char *) nptr;
487 intmax_t r = _strtosigned(nptr, &lendptr, base, INT64_MIN, INT64_MAX, &rc, true);
489 if (endptr)
490 *endptr = lendptr;
492 if (rc != EOK)
493 return rc;
495 if (strict && *lendptr != '\0')
496 return EINVAL;
498 *result = r;
499 return EOK;
502 /** Convert string to size_t.
504 * @param nptr Pointer to string.
505 * @param endptr If not NULL, pointer to the first invalid character
506 * is stored here.
507 * @param base Zero or number between 2 and 36 inclusive.
508 * @param strict Do not allow any trailing characters.
509 * @param result Result of the conversion.
511 * @return EOK if conversion was successful.
514 errno_t str_size_t(const char *nptr, const char **endptr, unsigned int base,
515 bool strict, size_t *result)
517 assert(result != NULL);
519 errno_t rc = EOK;
520 char *lendptr = (char *) nptr;
522 uintmax_t r = _strtounsigned(nptr, &lendptr, base, SIZE_MAX, &rc, true);
524 if (endptr)
525 *endptr = lendptr;
527 if (rc != EOK)
528 return rc;
530 if (strict && *lendptr != '\0')
531 return EINVAL;
533 *result = r;
534 return EOK;
537 /** @}