xref: /freebsd/contrib/mandoc/compat_strtonum.c (revision 6d38604fc532a3fc060788e3ce40464b46047eaf)
1*6d38604fSBaptiste Daroussin /*	$Id: compat_strtonum.c,v 1.2 2020/06/15 01:37:15 schwarze Exp $	*/
2*6d38604fSBaptiste Daroussin /*	$OpenBSD: strtonum.c,v 1.8 2015/09/13 08:31:48 guenther Exp $	*/
361d06d6bSBaptiste Daroussin 
461d06d6bSBaptiste Daroussin /*
561d06d6bSBaptiste Daroussin  * Copyright (c) 2004 Ted Unangst and Todd Miller
661d06d6bSBaptiste Daroussin  * All rights reserved.
761d06d6bSBaptiste Daroussin  *
861d06d6bSBaptiste Daroussin  * Permission to use, copy, modify, and distribute this software for any
961d06d6bSBaptiste Daroussin  * purpose with or without fee is hereby granted, provided that the above
1061d06d6bSBaptiste Daroussin  * copyright notice and this permission notice appear in all copies.
1161d06d6bSBaptiste Daroussin  *
1261d06d6bSBaptiste Daroussin  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
1361d06d6bSBaptiste Daroussin  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
1461d06d6bSBaptiste Daroussin  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
1561d06d6bSBaptiste Daroussin  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
1661d06d6bSBaptiste Daroussin  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
1761d06d6bSBaptiste Daroussin  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
1861d06d6bSBaptiste Daroussin  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
1961d06d6bSBaptiste Daroussin  */
20*6d38604fSBaptiste Daroussin #include "config.h"
2161d06d6bSBaptiste Daroussin 
2261d06d6bSBaptiste Daroussin #include <errno.h>
2361d06d6bSBaptiste Daroussin #include <limits.h>
2461d06d6bSBaptiste Daroussin #include <stdlib.h>
2561d06d6bSBaptiste Daroussin 
2661d06d6bSBaptiste Daroussin #define	INVALID		1
2761d06d6bSBaptiste Daroussin #define	TOOSMALL	2
2861d06d6bSBaptiste Daroussin #define	TOOLARGE	3
2961d06d6bSBaptiste Daroussin 
3061d06d6bSBaptiste Daroussin long long
strtonum(const char * numstr,long long minval,long long maxval,const char ** errstrp)3161d06d6bSBaptiste Daroussin strtonum(const char *numstr, long long minval, long long maxval,
3261d06d6bSBaptiste Daroussin     const char **errstrp)
3361d06d6bSBaptiste Daroussin {
3461d06d6bSBaptiste Daroussin 	long long ll = 0;
3561d06d6bSBaptiste Daroussin 	int error = 0;
3661d06d6bSBaptiste Daroussin 	char *ep;
3761d06d6bSBaptiste Daroussin 	struct errval {
3861d06d6bSBaptiste Daroussin 		const char *errstr;
3961d06d6bSBaptiste Daroussin 		int err;
4061d06d6bSBaptiste Daroussin 	} ev[4] = {
4161d06d6bSBaptiste Daroussin 		{ NULL,		0 },
4261d06d6bSBaptiste Daroussin 		{ "invalid",	EINVAL },
4361d06d6bSBaptiste Daroussin 		{ "too small",	ERANGE },
4461d06d6bSBaptiste Daroussin 		{ "too large",	ERANGE },
4561d06d6bSBaptiste Daroussin 	};
4661d06d6bSBaptiste Daroussin 
4761d06d6bSBaptiste Daroussin 	ev[0].err = errno;
4861d06d6bSBaptiste Daroussin 	errno = 0;
4961d06d6bSBaptiste Daroussin 	if (minval > maxval) {
5061d06d6bSBaptiste Daroussin 		error = INVALID;
5161d06d6bSBaptiste Daroussin 	} else {
5261d06d6bSBaptiste Daroussin 		ll = strtoll(numstr, &ep, 10);
5361d06d6bSBaptiste Daroussin 		if (numstr == ep || *ep != '\0')
5461d06d6bSBaptiste Daroussin 			error = INVALID;
5561d06d6bSBaptiste Daroussin 		else if ((ll == LLONG_MIN && errno == ERANGE) || ll < minval)
5661d06d6bSBaptiste Daroussin 			error = TOOSMALL;
5761d06d6bSBaptiste Daroussin 		else if ((ll == LLONG_MAX && errno == ERANGE) || ll > maxval)
5861d06d6bSBaptiste Daroussin 			error = TOOLARGE;
5961d06d6bSBaptiste Daroussin 	}
6061d06d6bSBaptiste Daroussin 	if (errstrp != NULL)
6161d06d6bSBaptiste Daroussin 		*errstrp = ev[error].errstr;
6261d06d6bSBaptiste Daroussin 	errno = ev[error].err;
6361d06d6bSBaptiste Daroussin 	if (error)
6461d06d6bSBaptiste Daroussin 		ll = 0;
6561d06d6bSBaptiste Daroussin 
6661d06d6bSBaptiste Daroussin 	return (ll);
6761d06d6bSBaptiste Daroussin }
68