1 /*
2  * Copyright © 2005-2020 Rich Felker
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining
5  * a copy of this software and associated documentation files (the
6  * "Software"), to deal in the Software without restriction, including
7  * without limitation the rights to use, copy, modify, merge, publish,
8  * distribute, sublicense, and/or sell copies of the Software, and to
9  * permit persons to whom the Software is furnished to do so, subject to
10  * the following conditions:
11  *
12  * The above copyright notice and this permission notice shall be
13  * included in all copies or substantial portions of the Software.
14  *
15  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
16  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
17  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
18  * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
19  * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
20  * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
21  * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
22  */
23 #include <stdio.h>
24 #include <stdlib.h>
25 #include <string.h>
26 #include <errno.h>
27 
28 /* r = place to store result
29  * f = function call to test (or any expression)
30  * x = expected result
31  * m = message to print on failure (with formats for r & x)
32 **/
33 
34 #pragma GCC diagnostic ignored "-Wpragmas"
35 #pragma GCC diagnostic ignored "-Wunknown-warning-option"
36 #pragma GCC diagnostic ignored "-Wformat-extra-args"
37 #pragma GCC diagnostic ignored "-Wformat"
38 
39 #define TEST(r, f, x, m) ( \
40 msg = #f, ((r) = (f)) == (x) || \
41 (printf(__FILE__ ":%d: %s failed (" m ")\n", __LINE__, #f, r, x), err++, 0) )
42 
43 #define TEST2(r, f, x, m) ( \
44 ((r) = (f)) == (x) || \
45 (printf(__FILE__ ":%d: %s failed (" m ")\n", __LINE__, msg, r, x), err++, 0) )
46 
test_strtol(void)47 static int test_strtol(void)
48 {
49 	int i;
50 	long l;
51 	unsigned long ul;
52 	long long ll;
53 	unsigned long long ull;
54 	char *msg="";
55 	int err=0;
56 	char *s, *c;
57 
58         (void) ll;
59         (void) ull;
60 	TEST(l, atol("2147483647"), 2147483647L, "max 32bit signed %ld != %ld");
61 
62 	TEST(l, strtol("2147483647", 0, 0), 2147483647L, "max 32bit signed %ld != %ld");
63 	TEST(ul, strtoul("4294967295", 0, 0), 4294967295UL, "max 32bit unsigned %lu != %lu");
64 
65 	if (sizeof(long) == 4) {
66 		errno = 0;
67 		TEST(l, strtol(s="2147483648", &c, 0), 2147483647L, "uncaught overflow %ld != %ld");
68 		TEST2(i, c-s, 10, "wrong final position %d != %d");
69 		TEST2(i, errno, ERANGE, "missing errno %d != %d");
70 		errno = 0;
71 		TEST(l, strtol(s="-2147483649", &c, 0), -2147483647L-1, "uncaught overflow %ld != %ld");
72 		TEST2(i, c-s, 11, "wrong final position %d != %d");
73 		TEST2(i, errno, ERANGE, "missing errno %d != %d");
74 		errno = 0;
75 		TEST(ul, strtoul(s="4294967296", &c, 0), 4294967295UL, "uncaught overflow %lu != %lu");
76 		TEST2(i, c-s, 10, "wrong final position %d != %d");
77 		TEST2(i, errno, ERANGE, "missing errno %d != %d");
78 		errno = 0;
79 		TEST(ul, strtoul(s="-1", &c, 0), -1UL, "rejected negative %lu != %lu");
80 		TEST2(i, c-s, 2, "wrong final position %d != %d");
81 		TEST2(i, errno, 0, "spurious errno %d != %d");
82 		errno = 0;
83 		TEST(ul, strtoul(s="-2", &c, 0), -2UL, "rejected negative %lu != %lu");
84 		TEST2(i, c-s, 2, "wrong final position %d != %d");
85 		TEST2(i, errno, 0, "spurious errno %d != %d");
86 		errno = 0;
87 		TEST(ul, strtoul(s="-2147483648", &c, 0), -2147483648UL, "rejected negative %lu != %lu");
88 		TEST2(i, c-s, 11, "wrong final position %d != %d");
89 		TEST2(i, errno, 0, "spurious errno %d != %d");
90 		errno = 0;
91 		TEST(ul, strtoul(s="-2147483649", &c, 0), -2147483649UL, "rejected negative %lu != %lu");
92 		TEST2(i, c-s, 11, "wrong final position %d != %d");
93 		TEST2(i, errno, 0, "spurious errno %d != %d");
94 	} else {
95 		errno = 0;
96 		TEST(l, strtol(s="9223372036854775808", &c, 0), 9223372036854775807L, "uncaught overflow %ld != %ld");
97 		TEST2(i, c-s, 19, "wrong final position %d != %d");
98 		TEST2(i, errno, ERANGE, "missing errno %d != %d");
99 		errno = 0;
100 		TEST(l, strtol(s="-9223372036854775809", &c, 0), -9223372036854775807L-1, "uncaught overflow %ld != %ld");
101 		TEST2(i, c-s, 20, "wrong final position %d != %d");
102 		TEST2(i, errno, ERANGE, "missing errno %d != %d");
103 		errno = 0;
104 		TEST(ul, strtoul(s="18446744073709551616", &c, 0), 18446744073709551615UL, "uncaught overflow %lu != %lu");
105 		TEST2(i, c-s, 20, "wrong final position %d != %d");
106 		TEST2(i, errno, ERANGE, "missing errno %d != %d");
107 		errno = 0;
108 		TEST(ul, strtoul(s="-1", &c, 0), -1UL, "rejected negative %lu != %lu");
109 		TEST2(i, c-s, 2, "wrong final position %d != %d");
110 		TEST2(i, errno, 0, "spurious errno %d != %d");
111 		errno = 0;
112 		TEST(ul, strtoul(s="-2", &c, 0), -2UL, "rejected negative %lu != %lu");
113 		TEST2(i, c-s, 2, "wrong final position %d != %d");
114 		TEST2(i, errno, 0, "spurious errno %d != %d");
115 		errno = 0;
116 		TEST(ul, strtoul(s="-9223372036854775808", &c, 0), -9223372036854775808UL, "rejected negative %lu != %lu");
117 		TEST2(i, c-s, 20, "wrong final position %d != %d");
118 		TEST2(i, errno, 0, "spurious errno %d != %d");
119 		errno = 0;
120 		TEST(ul, strtoul(s="-9223372036854775809", &c, 0), -9223372036854775809UL, "rejected negative %lu != %lu");
121 		TEST2(i, c-s, 20, "wrong final position %d != %d");
122 		TEST2(i, errno, 0, "spurious errno %d != %d");
123 	}
124 
125 	TEST(l, strtol("z", 0, 36), 35, "%ld != %ld");
126 	TEST(l, strtol("00010010001101000101011001111000", 0, 2), 0x12345678, "%ld != %ld");
127 	TEST(l, strtol(s="0F5F", &c, 16), 0x0f5f, "%ld != %ld");
128 
129 	TEST(l, strtol(s="0xz", &c, 16), 0, "%ld != %ld");
130 	TEST2(i, c-s, 1, "wrong final position %ld != %ld");
131 
132 	TEST(l, strtol(s="0x1234", &c, 16), 0x1234, "%ld != %ld");
133 	TEST2(i, c-s, 6, "wrong final position %ld != %ld");
134 
135         char delim_buf[6] = "09af:";
136 
137         for (int j = 0; j < 256; j++) {
138             delim_buf[4] = j;
139             if (('0' <= j && j <= '9') ||
140                 ('A' <= j && j <= 'F') ||
141                 ('a' <= j && j <= 'f'))
142             {
143                 long k;
144                 if ('0' <= j && j <= '9')
145                     k = j - '0';
146                 else if ('A' <= j && j <= 'Z')
147                     k = j - 'A' + 10;
148                 else if ('a' <= j && j <= 'z')
149                     k = j - 'a' + 10;
150                 else
151                     k = 0xffffffff;
152                 TEST(l, strtol(s=delim_buf, &c, 16), 0x09af0 | k, "%ld != %ld");
153                 TEST2(i, c-s, 5, "wrong final position %ld != %ld");
154             } else {
155                 TEST(l, strtol(s=delim_buf, &c, 16), 0x09af, "%ld != %ld");
156                 TEST2(i, c-s, 4, "wrong final position %ld != %ld");
157             }
158         }
159 
160 	errno = 0;
161 	c = NULL;
162 	TEST(l, strtol(s="123", &c, 37), 0, "%ld != %ld");
163 
164         /*
165           The value of 'c' is undefined when base is invalid, so this
166           test isn't valid:
167            TEST2(i, c-s, 0, "wrong final position %d != %d");
168         */
169 	TEST2(i, errno, EINVAL, "%d != %d");
170 
171 	TEST(l, strtol(s="  15437", &c, 8), 015437, "%ld != %ld");
172 	TEST2(i, c-s, 7, "wrong final position %d != %d");
173 
174 	TEST(l, strtol(s="  1", &c, 0), 1, "%ld != %ld");
175 	TEST2(i, c-s, 3, "wrong final position %d != %d");
176 
177 	return err;
178 }
179 
180 #define TEST_NAME strtol
181 #include "testcase.h"
182