aboutsummaryrefslogtreecommitdiffstats
path: root/dos/strntoumax.c
blob: d8bc73bcce1869ed16d67bb9609c906b6744f754 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
/*
 * strntoumax.c
 *
 * The strntoumax() function and associated
 */

#include <stddef.h>
#include <stdint.h>
#include <ctype.h>

static inline int digitval(int ch)
{
    if (ch >= '0' && ch <= '9') {
	return ch - '0';
    } else if (ch >= 'A' && ch <= 'Z') {
	return ch - 'A' + 10;
    } else if (ch >= 'a' && ch <= 'z') {
	return ch - 'a' + 10;
    } else {
	return -1;
    }
}

uintmax_t strntoumax(const char *nptr, char **endptr, int base, size_t n)
{
    int minus = 0;
    uintmax_t v = 0;
    int d;

    while (n && isspace((unsigned char)*nptr)) {
	nptr++;
	n--;
    }

    /* Single optional + or - */
    if (n && *nptr == '-') {
	minus = 1;
	nptr++;
	n--;
    } else if (n && *nptr == '+') {
	nptr++;
    }

    if (base == 0) {
	if (n >= 2 && nptr[0] == '0' && (nptr[1] == 'x' || nptr[1] == 'X')) {
	    n -= 2;
	    nptr += 2;
	    base = 16;
	} else if (n >= 1 && nptr[0] == '0') {
	    n--;
	    nptr++;
	    base = 8;
	} else {
	    base = 10;
	}
    } else if (base == 16) {
	if (n >= 2 && nptr[0] == '0' && (nptr[1] == 'x' || nptr[1] == 'X')) {
	    n -= 2;
	    nptr += 2;
	}
    }

    while (n && (d = digitval(*nptr)) >= 0 && d < base) {
	v = v * base + d;
	n--;
	nptr++;
    }

    if (endptr)
	*endptr = (char *)nptr;

    return minus ? -v : v;
}