]> git.draconx.ca Git - cdecl99.git/blob - t/testlib.c
10b27ad81c3ccef7a7a3bbebd83ba58fdfecc3cd
[cdecl99.git] / t / testlib.c
1 /*
2  *  Miscellaneous functions used by the cdecl99 test suite.
3  *  Copyright © 2011-2012, 2021-2024 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
26 #include "cdecl.h"
27
28 #include "help.h"
29 #include "test.h"
30 #include "intconv.h"
31
32 void print_error(const char *fmt, ...)
33 {
34         extern const char *progname;
35         va_list ap;
36
37         fprintf(stderr, "%s: ", progname);
38
39         va_start(ap, fmt);
40         vfprintf(stderr, fmt, ap);
41         va_end(ap);
42
43         fprintf(stderr, "\n");
44 }
45
46 void *realloc_nofail(void *ptr, size_t size)
47 {
48         void *p;
49         
50         p = realloc(ptr, size);
51         if (!p) {
52                 perror("failed to allocate memory");
53                 abort();
54         }
55
56         return p;
57 }
58
59 void *malloc_nofail(size_t size)
60 {
61         return realloc_nofail(NULL, size);
62 }
63
64 static unsigned intconv_base(const char **str)
65 {
66         if ((*str)[0] == '0') {
67                 if ((*str)[1] == 'X' || (*str)[1] == 'x') {
68                         *str += 2;
69                         return INTCONV_HEXADECIMAL;
70                 }
71
72                 return INTCONV_OCTAL;
73         }
74
75         return INTCONV_DECIMAL;
76 }
77
78 bool test_strtoumax(uintmax_t *out, const char *s, uintmax_t limit)
79 {
80         static const char idx[] = "0123456789abcdef0123456789ABCDEF";
81         unsigned base = intconv_base(&s);
82         uintmax_t v;
83         char *c, d;
84
85         for (v = 0; (d = *s++);) {
86                 if (!(c = strchr(idx, d)) || (d = (c-idx) & 0xf) >= base)
87                         return 0;
88
89                 if (!intconv_shift(&v, base, d) || v > limit)
90                         return 0;
91         }
92
93         *out = v;
94         return true;
95 }
96
97 void test_print_options(const struct option *lopts)
98 {
99         const struct option *opt;
100
101         puts("Options:");
102         for (opt = lopts; opt->val; opt++) {
103                 if (help_print_optstring(opt, "ARG", 20))
104                         putchar('\n');
105         }
106 }