Home | History | Annotate | Line # | Download | only in stdlib
      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