usr/src/lib/libc/port/gen/strtoll.c
author raf
Mon, 10 Apr 2006 12:27:38 -0700
changeset 1778 6357a59054f7
parent 0 68f95e015346
child 6812 febeba71273d
permissions -rw-r--r--
6404383 select() behaviour changed in Solaris 10, breaking binary compatibility
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     1
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     2
 * CDDL HEADER START
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     3
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     4
 * The contents of this file are subject to the terms of the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     5
 * Common Development and Distribution License, Version 1.0 only
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     6
 * (the "License").  You may not use this file except in compliance
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     7
 * with the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     8
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     9
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    10
 * or http://www.opensolaris.org/os/licensing.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    11
 * See the License for the specific language governing permissions
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    12
 * and limitations under the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    13
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    14
 * When distributing Covered Code, include this CDDL HEADER in each
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    15
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    16
 * If applicable, add the following below this CDDL HEADER, with the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    17
 * fields enclosed by brackets "[]" replaced with your own identifying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    18
 * information: Portions Copyright [yyyy] [name of copyright owner]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    19
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    20
 * CDDL HEADER END
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    21
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    22
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    23
 * Copyright 2004 Sun Microsystems, Inc.  All rights reserved.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    24
 * Use is subject to license terms.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    25
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    26
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    27
#pragma ident	"%Z%%M%	%I%	%E% SMI"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    29
/*	Copyright (c) 1988 AT&T	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    30
/*	  All Rights Reserved  	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    32
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
#pragma weak strtoll = _strtoll
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    35
#include "synonyms.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
#include <errno.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
#include <ctype.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
#include <limits.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
#include <sys/types.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
#include <stdlib.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
#define	DIGIT(x)	\
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
	(isdigit(x) ? (x) - '0' : islower(x) ? (x) + 10 - 'a' : (x) + 10 - 'A')
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
#define	MBASE	('z' - 'a' + 1 + 10)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
 * The following macro is a local version of isalnum() which limits
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    49
 * alphabetic characters to the ranges a-z and A-Z; locale dependent
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
 * characters will not return 1. The members of a-z and A-Z are
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    51
 * assumed to be in ascending order and contiguous
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    53
#define	lisalnum(x)	\
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    54
	(isdigit(x) || ((x) >= 'a' && (x) <= 'z') || ((x) >= 'A' && (x) <= 'Z'))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    55
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    56
longlong_t
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    57
strtoll(const char *str, char **nptr, int base)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
	longlong_t val;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
	int c;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    61
	int xx, neg = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
	longlong_t	multmin;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
	longlong_t	limit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    64
	longlong_t	llong_min, llong_max;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
	const char **ptr = (const char **)nptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
	const unsigned char	*ustr = (const unsigned char *)str;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    68
	llong_min = LLONG_MIN;   /* from a local version of limits.h */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    69
	llong_max = LLONG_MAX;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
	if (ptr != (const char **)0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
		*ptr = (char *)ustr; /* in case no number is formed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
	if (base < 0 || base > MBASE || base == 1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
		errno = EINVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    75
		return (0); /* base is invalid -- should be a fatal error */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    76
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    77
	if (!isalnum(c = *ustr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    78
		while (isspace(c))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    79
			c = *++ustr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    80
		switch (c) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
		case '-':
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
			neg++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    83
			/* FALLTHROUGH */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
		case '+':
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
			c = *++ustr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
	if (base == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
		if (c != '0')
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
			base = 10;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    91
		else if (ustr[1] == 'x' || ustr[1] == 'X')
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    92
			base = 16;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    94
			base = 8;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    95
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
	 * for any base > 10, the digits incrementally following
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
	 *	9 are assumed to be "abc...z" or "ABC...Z"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    99
	if (!lisalnum(c) || (xx = DIGIT(c)) >= base)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
		return (0); /* no number formed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
	if (base == 16 && c == '0' && (ustr[1] == 'x' || ustr[1] == 'X') &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   102
		isxdigit(ustr[2]))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
		c = *(ustr += 2); /* skip over leading "0x" or "0X" */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   104
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
	/* this code assumes that abs(llong_min) >= abs(llong_max) */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   106
	if (neg)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   107
		limit = llong_min;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   108
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   109
		limit = -llong_max;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   110
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   111
	multmin = limit / base;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   112
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   113
	val = -DIGIT(c);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   114
	for (c = *++ustr; lisalnum(c) && (xx = DIGIT(c)) < base; ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   115
		/* accumulate neg avoids surprises near llong_max */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   116
		if (val < multmin)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   117
			goto overflow;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   118
		val *= base;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   119
		if (val < limit + xx)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   120
			goto overflow;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   121
		val -= xx;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   122
		c = *++ustr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   123
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   124
	if (ptr != (const char **)0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   125
		*ptr = (char *)ustr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   126
	return (neg ? val : -val);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   127
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   128
overflow:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   129
	for (c = *++ustr; lisalnum(c) && (xx = DIGIT(c)) < base; (c = *++ustr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   130
		;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   131
	if (ptr != (const char **)0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   132
		*ptr = (char *)ustr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   133
	errno = ERANGE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   134
	return (neg ? llong_min : llong_max);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   135
}