Commit | Line | Data |
---|---|---|
460b9539 | 1 | /* |
2 | * This file is part of DisOrder. | |
39831a99 | 3 | * Copyright (C) 2005, 2007, 2008 Richard Kettlewell |
460b9539 | 4 | * |
e7eb3a27 | 5 | * This program is free software: you can redistribute it and/or modify |
460b9539 | 6 | * it under the terms of the GNU General Public License as published by |
e7eb3a27 | 7 | * the Free Software Foundation, either version 3 of the License, or |
460b9539 | 8 | * (at your option) any later version. |
e7eb3a27 RK |
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 | * | |
460b9539 | 15 | * You should have received a copy of the GNU General Public License |
e7eb3a27 | 16 | * along with this program. If not, see <http://www.gnu.org/licenses/>. |
460b9539 | 17 | */ |
59cf25c4 | 18 | /** @file libtests/test.c @brief Library tests */ |
460b9539 | 19 | |
b90f122b | 20 | #include "test.h" |
2257512d RK |
21 | #include "version.h" |
22 | #include <getopt.h> | |
460b9539 | 23 | |
cc3ede61 RK |
24 | /** @brief Count of tests */ |
25 | long long tests; | |
26 | ||
27 | /** @brief Count of errors */ | |
28 | long long errors; | |
29 | ||
30 | /** @brief If set, first error will fail whole test */ | |
b90f122b | 31 | int fail_first; |
cc3ede61 RK |
32 | |
33 | /** @brief Verbose mode */ | |
2257512d | 34 | int verbose; |
cc3ede61 RK |
35 | |
36 | /** @brief If set, test will return 'skipped' indicator */ | |
cca956b1 | 37 | int skipped; |
460b9539 | 38 | |
cc3ede61 RK |
39 | /** @brief Count up an error |
40 | * | |
41 | * If @ref fail_first is set then the test run is aborted. | |
42 | */ | |
b90f122b | 43 | void count_error(void) { |
bb48024f RK |
44 | ++errors; |
45 | if(fail_first) | |
46 | abort(); | |
47 | } | |
460b9539 | 48 | |
cc3ede61 RK |
49 | /** @brief Render a string into printable ASCII |
50 | * @param s String to format | |
51 | * @return Allocated copy of formatted string | |
52 | * | |
53 | * Replaces any non-ASCII characters with a hex escape. | |
54 | */ | |
b90f122b | 55 | const char *format(const char *s) { |
460b9539 | 56 | struct dynstr d; |
57 | int c; | |
58 | char buf[10]; | |
2257512d | 59 | |
460b9539 | 60 | dynstr_init(&d); |
61 | while((c = (unsigned char)*s++)) { | |
62 | if(c >= ' ' && c <= '~') | |
63 | dynstr_append(&d, c); | |
64 | else { | |
65 | sprintf(buf, "\\x%02X", (unsigned)c); | |
66 | dynstr_append_string(&d, buf); | |
67 | } | |
68 | } | |
69 | dynstr_terminate(&d); | |
70 | return d.vec; | |
71 | } | |
72 | ||
cc3ede61 RK |
73 | /** @brief Format a UTF-32 string into hex |
74 | * @param s String to format | |
75 | * @return Allocated copy of formatted string | |
76 | * | |
77 | * Returns the hex codes of @p s separated by spaces. | |
78 | */ | |
b90f122b | 79 | const char *format_utf32(const uint32_t *s) { |
e5a5a138 RK |
80 | struct dynstr d; |
81 | uint32_t c; | |
82 | char buf[64]; | |
2257512d | 83 | |
e5a5a138 RK |
84 | dynstr_init(&d); |
85 | while((c = *s++)) { | |
16506c9d RK |
86 | sprintf(buf, " %04lX", (long)c); |
87 | dynstr_append_string(&d, buf); | |
e5a5a138 RK |
88 | } |
89 | dynstr_terminate(&d); | |
90 | return d.vec; | |
91 | } | |
92 | ||
cc3ede61 RK |
93 | /** @brief Convert a string of hex codes to a UTF-32 string |
94 | * @param s String of hex codes, separated by spaces | |
95 | * @return Allocated string, 0-terminated | |
96 | */ | |
b90f122b | 97 | uint32_t *ucs4parse(const char *s) { |
460b9539 | 98 | struct dynstr_ucs4 d; |
99 | char *e; | |
100 | ||
101 | dynstr_ucs4_init(&d); | |
102 | while(*s) { | |
103 | errno = 0; | |
104 | dynstr_ucs4_append(&d, strtoul(s, &e, 0)); | |
2e9ba080 RK |
105 | if(errno) |
106 | disorder_fatal(errno, "strtoul (%s)", s); | |
460b9539 | 107 | s = e; |
108 | } | |
109 | dynstr_ucs4_terminate(&d); | |
110 | return d.vec; | |
111 | } | |
112 | ||
cc3ede61 RK |
113 | /** @brief Format a string like asprintf() |
114 | * @param fmt Format string, per printf(3) | |
115 | * @param ... Arguments | |
116 | * @return Formatted string or null pointer on error | |
117 | */ | |
b90f122b | 118 | const char *do_printf(const char *fmt, ...) { |
0c740e59 RK |
119 | va_list ap; |
120 | char *s; | |
121 | int rc; | |
122 | ||
123 | va_start(ap, fmt); | |
124 | rc = byte_vasprintf(&s, fmt, ap); | |
125 | va_end(ap); | |
126 | if(rc < 0) | |
127 | return 0; | |
128 | return s; | |
129 | } | |
130 | ||
cc3ede61 | 131 | /** @brief Jump buffer for exitfn() testing */ |
3943e9ec RK |
132 | jmp_buf fatal_env; |
133 | ||
cc3ede61 RK |
134 | /** @brief exitfn() callback for testing |
135 | * @param rc Value to return from setjmp() | |
136 | * Jumps to @ref fatal_env | |
137 | */ | |
3943e9ec RK |
138 | void test_exitfn(int rc) { |
139 | assert(rc != 0); | |
140 | longjmp(fatal_env, rc); | |
141 | } | |
142 | ||
2257512d RK |
143 | static const struct option options[] = { |
144 | { "verbose", no_argument, 0, 'v' }, | |
145 | { "fail-first", no_argument, 0, 'F' }, | |
146 | { "help", no_argument, 0, 'h' }, | |
147 | { "version", no_argument, 0, 'V' }, | |
148 | }; | |
149 | ||
150 | /* display usage message and terminate */ | |
151 | static void help(void) { | |
152 | xprintf("Usage:\n" | |
153 | " %s [OPTIONS]\n" | |
154 | "Options:\n" | |
155 | " --help, -h Display usage message\n" | |
156 | " --version, -V Display version number\n" | |
157 | " --verbose, -v Verbose output\n" | |
158 | " --fail-first, -F Stop on first failure\n", | |
159 | progname); | |
160 | xfclose(stdout); | |
161 | exit(0); | |
162 | } | |
163 | ||
cc3ede61 RK |
164 | /** @brief Standard test program initialization |
165 | * @param argc Argument count | |
166 | * @param argv Arguments | |
167 | */ | |
2257512d RK |
168 | void test_init(int argc, char **argv) { |
169 | int n; | |
170 | ||
171 | set_progname(argv); | |
172 | mem_init(); | |
173 | while((n = getopt_long(argc, argv, "vFhV", options, 0)) >= 0) { | |
174 | switch(n) { | |
175 | case 'v': verbose = 1; break; | |
176 | case 'F': fail_first = 1; break; | |
177 | case 'h': help(); | |
178 | case 'V': version(progname); | |
179 | default: exit(1); | |
180 | } | |
181 | } | |
182 | if(getenv("FAIL_FIRST")) | |
183 | fail_first = 1; | |
184 | } | |
185 | ||
186 | ||
460b9539 | 187 | /* |
188 | Local Variables: | |
189 | c-basic-offset:2 | |
190 | comment-column:40 | |
56fd389c RK |
191 | fill-column:79 |
192 | indent-tabs-mode:nil | |
460b9539 | 193 | End: |
194 | */ |