main netmisc / compat / _strtoi.h
  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
 62static __TYPE
 63INT_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
153INT_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