]> git.draconx.ca Git - cdecl99.git/blob - t/testlib.c
tests: Avoid dependency on strtoumax.
[cdecl99.git] / t / testlib.c
1 /*
2  *  Miscellaneous functions used by the cdecl99 test suite.
3  *  Copyright © 2011-2012, 2021-2023 Nick Bowler
4  *
5  *  This program is free software: you can redistribute it and/or modify
6  *  it under the terms of the GNU General Public License as published by
7  *  the Free Software Foundation, either version 3 of the License, or
8  *  (at your option) any later version.
9  *
10  *  This program is distributed in the hope that it will be useful,
11  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
12  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  *  GNU General Public License for more details.
14  *
15  *  You should have received a copy of the GNU General Public License
16  *  along with this program.  If not, see <http://www.gnu.org/licenses/>.
17  */
18
19 #include <config.h>
20 #include <stdio.h>
21 #include <stdlib.h>
22 #include <string.h>
23 #include <errno.h>
24 #include <getopt.h>
25 #include <cdecl.h>
26
27 #include "help.h"
28 #include "test.h"
29 #include "intconv.h"
30
31 void *realloc_nofail(void *ptr, size_t size)
32 {
33         void *p;
34         
35         p = realloc(ptr, size);
36         if (!p) {
37                 perror("failed to allocate memory");
38                 abort();
39         }
40
41         return p;
42 }
43
44 void *malloc_nofail(size_t size)
45 {
46         return realloc_nofail(NULL, size);
47 }
48
49 static unsigned intconv_base(const char **str)
50 {
51         if ((*str)[0] == '0') {
52                 if ((*str)[1] == 'X' || (*str)[1] == 'x') {
53                         *str += 2;
54                         return INTCONV_HEXADECIMAL;
55                 }
56
57                 return INTCONV_OCTAL;
58         }
59
60         return INTCONV_DECIMAL;
61 }
62
63 bool test_strtoumax(uintmax_t *out, const char *s, uintmax_t limit)
64 {
65         static const char idx[] = "0123456789abcdef0123456789ABCDEF";
66         unsigned base = intconv_base(&s);
67         uintmax_t v;
68         char *c, d;
69
70         for (v = 0; (d = *s++);) {
71                 if (!(c = strchr(idx, d)) || (d = (c-idx) & 0xf) >= base)
72                         return 0;
73
74                 if (!intconv_shift(&v, base, d) || v > limit)
75                         return 0;
76         }
77
78         *out = v;
79         return true;
80 }
81
82 void test_print_options(const struct option *lopts)
83 {
84         const struct option *opt;
85
86         puts("Options:");
87         for (opt = lopts; opt->val; opt++) {
88                 if (help_print_optstring(opt, "ARG", 20))
89                         putchar('\n');
90         }
91 }