2 This file is part of systemd.
4 Copyright 2010 Lennart Poettering
6 systemd is free software; you can redistribute it and/or modify it
7 under the terms of the GNU Lesser General Public License as published by
8 the Free Software Foundation; either version 2.1 of the License, or
9 (at your option) any later version.
11 systemd is distributed in the hope that it will be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 Lesser General Public License for more details.
16 You should have received a copy of the GNU Lesser General Public License
17 along with systemd; If not, see <http://www.gnu.org/licenses/>.
20 #include "alloc-util.h"
21 #include "hexdecoct.h"
23 #include "string-util.h"
25 /// Additional includes needed by elogind
26 #include "musl_missing.h"
28 static void test_hexchar(void) {
29 assert_se(hexchar(0xa) == 'a');
30 assert_se(hexchar(0x0) == '0');
33 static void test_unhexchar(void) {
34 assert_se(unhexchar('a') == 0xA);
35 assert_se(unhexchar('A') == 0xA);
36 assert_se(unhexchar('0') == 0x0);
39 static void test_base32hexchar(void) {
40 assert_se(base32hexchar(0) == '0');
41 assert_se(base32hexchar(9) == '9');
42 assert_se(base32hexchar(10) == 'A');
43 assert_se(base32hexchar(31) == 'V');
46 static void test_unbase32hexchar(void) {
47 assert_se(unbase32hexchar('0') == 0);
48 assert_se(unbase32hexchar('9') == 9);
49 assert_se(unbase32hexchar('A') == 10);
50 assert_se(unbase32hexchar('V') == 31);
51 assert_se(unbase32hexchar('=') == -EINVAL);
54 static void test_base64char(void) {
55 assert_se(base64char(0) == 'A');
56 assert_se(base64char(26) == 'a');
57 assert_se(base64char(63) == '/');
60 static void test_unbase64char(void) {
61 assert_se(unbase64char('A') == 0);
62 assert_se(unbase64char('Z') == 25);
63 assert_se(unbase64char('a') == 26);
64 assert_se(unbase64char('z') == 51);
65 assert_se(unbase64char('0') == 52);
66 assert_se(unbase64char('9') == 61);
67 assert_se(unbase64char('+') == 62);
68 assert_se(unbase64char('/') == 63);
69 assert_se(unbase64char('=') == -EINVAL);
72 static void test_octchar(void) {
73 assert_se(octchar(00) == '0');
74 assert_se(octchar(07) == '7');
77 static void test_unoctchar(void) {
78 assert_se(unoctchar('0') == 00);
79 assert_se(unoctchar('7') == 07);
82 static void test_decchar(void) {
83 assert_se(decchar(0) == '0');
84 assert_se(decchar(9) == '9');
87 static void test_undecchar(void) {
88 assert_se(undecchar('0') == 0);
89 assert_se(undecchar('9') == 9);
92 static void test_unhexmem(void) {
93 const char *hex = "efa2149213";
94 const char *hex_invalid = "efa214921o";
95 _cleanup_free_ char *hex2 = NULL;
96 _cleanup_free_ void *mem = NULL;
99 assert_se(unhexmem(hex_invalid, strlen(hex_invalid), &mem, &len) == -EINVAL);
100 assert_se(unhexmem(hex, strlen(hex) + 1, &mem, &len) == -EINVAL);
101 assert_se(unhexmem(hex, strlen(hex) - 1, &mem, &len) == -EINVAL);
102 assert_se(unhexmem(hex, strlen(hex), &mem, &len) == 0);
104 assert_se((hex2 = hexmem(mem, len)));
105 assert_se(streq(hex, hex2));
108 /* https://tools.ietf.org/html/rfc4648#section-10 */
109 static void test_base32hexmem(void) {
112 b32 = base32hexmem("", strlen(""), true);
114 assert_se(streq(b32, ""));
117 b32 = base32hexmem("f", strlen("f"), true);
119 assert_se(streq(b32, "CO======"));
122 b32 = base32hexmem("fo", strlen("fo"), true);
124 assert_se(streq(b32, "CPNG===="));
127 b32 = base32hexmem("foo", strlen("foo"), true);
129 assert_se(streq(b32, "CPNMU==="));
132 b32 = base32hexmem("foob", strlen("foob"), true);
134 assert_se(streq(b32, "CPNMUOG="));
137 b32 = base32hexmem("fooba", strlen("fooba"), true);
139 assert_se(streq(b32, "CPNMUOJ1"));
142 b32 = base32hexmem("foobar", strlen("foobar"), true);
144 assert_se(streq(b32, "CPNMUOJ1E8======"));
147 b32 = base32hexmem("", strlen(""), false);
149 assert_se(streq(b32, ""));
152 b32 = base32hexmem("f", strlen("f"), false);
154 assert_se(streq(b32, "CO"));
157 b32 = base32hexmem("fo", strlen("fo"), false);
159 assert_se(streq(b32, "CPNG"));
162 b32 = base32hexmem("foo", strlen("foo"), false);
164 assert_se(streq(b32, "CPNMU"));
167 b32 = base32hexmem("foob", strlen("foob"), false);
169 assert_se(streq(b32, "CPNMUOG"));
172 b32 = base32hexmem("fooba", strlen("fooba"), false);
174 assert_se(streq(b32, "CPNMUOJ1"));
177 b32 = base32hexmem("foobar", strlen("foobar"), false);
179 assert_se(streq(b32, "CPNMUOJ1E8"));
183 static void test_unbase32hexmem(void) {
187 assert_se(unbase32hexmem("", strlen(""), true, &mem, &len) == 0);
188 assert_se(streq(strndupa(mem, len), ""));
191 assert_se(unbase32hexmem("CO======", strlen("CO======"), true, &mem, &len) == 0);
192 assert_se(streq(strndupa(mem, len), "f"));
195 assert_se(unbase32hexmem("CPNG====", strlen("CPNG===="), true, &mem, &len) == 0);
196 assert_se(streq(strndupa(mem, len), "fo"));
199 assert_se(unbase32hexmem("CPNMU===", strlen("CPNMU==="), true, &mem, &len) == 0);
200 assert_se(streq(strndupa(mem, len), "foo"));
203 assert_se(unbase32hexmem("CPNMUOG=", strlen("CPNMUOG="), true, &mem, &len) == 0);
204 assert_se(streq(strndupa(mem, len), "foob"));
207 assert_se(unbase32hexmem("CPNMUOJ1", strlen("CPNMUOJ1"), true, &mem, &len) == 0);
208 assert_se(streq(strndupa(mem, len), "fooba"));
211 assert_se(unbase32hexmem("CPNMUOJ1E8======", strlen("CPNMUOJ1E8======"), true, &mem, &len) == 0);
212 assert_se(streq(strndupa(mem, len), "foobar"));
215 assert_se(unbase32hexmem("A", strlen("A"), true, &mem, &len) == -EINVAL);
216 assert_se(unbase32hexmem("A=======", strlen("A======="), true, &mem, &len) == -EINVAL);
217 assert_se(unbase32hexmem("AAA=====", strlen("AAA====="), true, &mem, &len) == -EINVAL);
218 assert_se(unbase32hexmem("AAAAAA==", strlen("AAAAAA=="), true, &mem, &len) == -EINVAL);
219 assert_se(unbase32hexmem("AB======", strlen("AB======"), true, &mem, &len) == -EINVAL);
220 assert_se(unbase32hexmem("AAAB====", strlen("AAAB===="), true, &mem, &len) == -EINVAL);
221 assert_se(unbase32hexmem("AAAAB===", strlen("AAAAB==="), true, &mem, &len) == -EINVAL);
222 assert_se(unbase32hexmem("AAAAAAB=", strlen("AAAAAAB="), true, &mem, &len) == -EINVAL);
224 assert_se(unbase32hexmem("XPNMUOJ1", strlen("CPNMUOJ1"), true, &mem, &len) == -EINVAL);
225 assert_se(unbase32hexmem("CXNMUOJ1", strlen("CPNMUOJ1"), true, &mem, &len) == -EINVAL);
226 assert_se(unbase32hexmem("CPXMUOJ1", strlen("CPNMUOJ1"), true, &mem, &len) == -EINVAL);
227 assert_se(unbase32hexmem("CPNXUOJ1", strlen("CPNMUOJ1"), true, &mem, &len) == -EINVAL);
228 assert_se(unbase32hexmem("CPNMXOJ1", strlen("CPNMUOJ1"), true, &mem, &len) == -EINVAL);
229 assert_se(unbase32hexmem("CPNMUXJ1", strlen("CPNMUOJ1"), true, &mem, &len) == -EINVAL);
230 assert_se(unbase32hexmem("CPNMUOX1", strlen("CPNMUOJ1"), true, &mem, &len) == -EINVAL);
231 assert_se(unbase32hexmem("CPNMUOJX", strlen("CPNMUOJ1"), true, &mem, &len) == -EINVAL);
233 assert_se(unbase32hexmem("", strlen(""), false, &mem, &len) == 0);
234 assert_se(streq(strndupa(mem, len), ""));
237 assert_se(unbase32hexmem("CO", strlen("CO"), false, &mem, &len) == 0);
238 assert_se(streq(strndupa(mem, len), "f"));
241 assert_se(unbase32hexmem("CPNG", strlen("CPNG"), false, &mem, &len) == 0);
242 assert_se(streq(strndupa(mem, len), "fo"));
245 assert_se(unbase32hexmem("CPNMU", strlen("CPNMU"), false, &mem, &len) == 0);
246 assert_se(streq(strndupa(mem, len), "foo"));
249 assert_se(unbase32hexmem("CPNMUOG", strlen("CPNMUOG"), false, &mem, &len) == 0);
250 assert_se(streq(strndupa(mem, len), "foob"));
253 assert_se(unbase32hexmem("CPNMUOJ1", strlen("CPNMUOJ1"), false, &mem, &len) == 0);
254 assert_se(streq(strndupa(mem, len), "fooba"));
257 assert_se(unbase32hexmem("CPNMUOJ1E8", strlen("CPNMUOJ1E8"), false, &mem, &len) == 0);
258 assert_se(streq(strndupa(mem, len), "foobar"));
261 assert_se(unbase32hexmem("CPNMUOG=", strlen("CPNMUOG="), false, &mem, &len) == -EINVAL);
262 assert_se(unbase32hexmem("CPNMUOJ1E8======", strlen("CPNMUOJ1E8======"), false, &mem, &len) == -EINVAL);
263 assert_se(unbase32hexmem("A", strlen("A"), false, &mem, &len) == -EINVAL);
264 assert_se(unbase32hexmem("A", strlen("A"), false, &mem, &len) == -EINVAL);
265 assert_se(unbase32hexmem("AAA", strlen("AAA"), false, &mem, &len) == -EINVAL);
266 assert_se(unbase32hexmem("AAAAAA", strlen("AAAAAA"), false, &mem, &len) == -EINVAL);
267 assert_se(unbase32hexmem("AB", strlen("AB"), false, &mem, &len) == -EINVAL);
268 assert_se(unbase32hexmem("AAAB", strlen("AAAB"), false, &mem, &len) == -EINVAL);
269 assert_se(unbase32hexmem("AAAAB", strlen("AAAAB"), false, &mem, &len) == -EINVAL);
270 assert_se(unbase32hexmem("AAAAAAB", strlen("AAAAAAB"), false, &mem, &len) == -EINVAL);
273 /* https://tools.ietf.org/html/rfc4648#section-10 */
274 static void test_base64mem(void) {
277 assert_se(base64mem("", strlen(""), &b64) == 0);
278 assert_se(streq(b64, ""));
281 assert_se(base64mem("f", strlen("f"), &b64) == 4);
282 assert_se(streq(b64, "Zg=="));
285 assert_se(base64mem("fo", strlen("fo"), &b64) == 4);
286 assert_se(streq(b64, "Zm8="));
289 assert_se(base64mem("foo", strlen("foo"), &b64) == 4);
290 assert_se(streq(b64, "Zm9v"));
293 assert_se(base64mem("foob", strlen("foob"), &b64) == 8);
294 assert_se(streq(b64, "Zm9vYg=="));
297 assert_se(base64mem("fooba", strlen("fooba"), &b64) == 8);
298 assert_se(streq(b64, "Zm9vYmE="));
301 assert_se(base64mem("foobar", strlen("foobar"), &b64) == 8);
302 assert_se(streq(b64, "Zm9vYmFy"));
306 static void test_unbase64mem(void) {
310 assert_se(unbase64mem("", strlen(""), &mem, &len) == 0);
311 assert_se(streq(strndupa(mem, len), ""));
314 assert_se(unbase64mem("Zg==", strlen("Zg=="), &mem, &len) == 0);
315 assert_se(streq(strndupa(mem, len), "f"));
318 assert_se(unbase64mem("Zm8=", strlen("Zm8="), &mem, &len) == 0);
319 assert_se(streq(strndupa(mem, len), "fo"));
322 assert_se(unbase64mem("Zm9v", strlen("Zm9v"), &mem, &len) == 0);
323 assert_se(streq(strndupa(mem, len), "foo"));
326 assert_se(unbase64mem("Zm9vYg==", strlen("Zm9vYg=="), &mem, &len) == 0);
327 assert_se(streq(strndupa(mem, len), "foob"));
330 assert_se(unbase64mem("Zm9vYmE=", strlen("Zm9vYmE="), &mem, &len) == 0);
331 assert_se(streq(strndupa(mem, len), "fooba"));
334 assert_se(unbase64mem("Zm9vYmFy", strlen("Zm9vYmFy"), &mem, &len) == 0);
335 assert_se(streq(strndupa(mem, len), "foobar"));
338 assert_se(unbase64mem("A", strlen("A"), &mem, &len) == -EINVAL);
339 assert_se(unbase64mem("A====", strlen("A===="), &mem, &len) == -EINVAL);
340 assert_se(unbase64mem("AAB==", strlen("AAB=="), &mem, &len) == -EINVAL);
341 assert_se(unbase64mem("AAAB=", strlen("AAAB="), &mem, &len) == -EINVAL);
344 static void test_hexdump(void) {
348 hexdump(stdout, NULL, 0);
349 hexdump(stdout, "", 0);
350 hexdump(stdout, "", 1);
351 hexdump(stdout, "x", 1);
352 hexdump(stdout, "x", 2);
353 hexdump(stdout, "foobar", 7);
354 hexdump(stdout, "f\nobar", 7);
355 hexdump(stdout, "xxxxxxxxxxxxxxxxxxxxyz", 23);
357 for (i = 0; i < ELEMENTSOF(data); i++)
360 hexdump(stdout, data, sizeof(data));
363 int main(int argc, char *argv[]) {
366 test_base32hexchar();
367 test_unbase32hexchar();
376 test_unbase32hexmem();