]> git.draconx.ca Git - cdecl99.git/blob - t/randomdecl.c
Avoid the use of for loop declarations.
[cdecl99.git] / t / randomdecl.c
1 /*
2  * Generate random C declarations for testing.
3  * Copyright © 2012, 2020, 2022-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 <https://www.gnu.org/licenses/>.
17  */
18
19 #include <config.h>
20 #include <stdio.h>
21 #include <stdlib.h>
22 #include <string.h>
23 #include <assert.h>
24 #include <stdbool.h>
25 #include <errno.h>
26 #include <getopt.h>
27 #include <cdecl.h>
28
29 #include "declgen.h"
30 #include "test.h"
31
32 #define PROGNAME "randomdecl"
33 static const char *progname = PROGNAME;
34 static const char sopts[] = "s:n:ECVH";
35 static const struct option lopts[] = {
36         { "seed",    1, NULL, 's' },
37         { "count",   1, NULL, 'n' },
38         { "cdecl",   0, NULL, 'C' },
39         { "english", 0, NULL, 'E' },
40         { "version", 0, NULL, 'V' },
41         { "help",    0, NULL, 'H' },
42         { 0 }
43 };
44
45 enum {
46         MODE_CDECL,
47         MODE_ENGLISH,
48 };
49
50 static void print_usage(FILE *f)
51 {
52         fprintf(f, "Usage: %s [options]\n", progname);
53 }
54
55 static void print_help(void)
56 {
57         const struct option *opt;
58
59         print_usage(stdout);
60         puts("Generate random C declarations for testing.\n");
61         test_print_options(lopts);
62 }
63
64 static struct cdecl *random_decl(struct test_rng *rng)
65 {
66         struct cdecl *decl;
67         unsigned flags = 0;
68
69         decl = malloc_nofail(sizeof *decl);
70         *decl = (struct cdecl) { 0 };
71
72         decl->declarators = gen_declarators(rng);
73         if (decl->declarators->type != CDECL_DECL_FUNCTION)
74                 flags |= GEN_NO_FUNCTION;
75         if (cdecl_is_abstract(decl->declarators))
76                 flags |= GEN_NO_STORAGE | GEN_NO_FUNCTION;
77         if (decl->declarators->type == CDECL_DECL_ARRAY
78             || decl->declarators->type == CDECL_DECL_IDENT)
79                 flags |= GEN_NO_VOID;
80
81         decl->specifiers = gen_declspecs(rng, flags);
82         return decl;
83 }
84
85 int main(int argc, char **argv)
86 {
87         const char *seed = "", *count_str = NULL;
88         unsigned long i, count = 0;
89         int opt, mode = MODE_CDECL;
90         struct test_rng *rng;
91         struct cdecl *decl;
92
93         if (argc > 0)
94                 progname = argv[0];
95
96         while ((opt = getopt_long(argc, argv, sopts, lopts, NULL)) != -1) {
97                 switch (opt) {
98                 case 's':
99                         seed = optarg;
100                         break;
101                 case 'n':
102                         count_str = optarg;
103                         break;
104                 case 'C':
105                         mode = MODE_CDECL;
106                         break;
107                 case 'E':
108                         mode = MODE_ENGLISH;
109                         break;
110                 case 'V':
111                         test_print_version(PROGNAME);
112                         return EXIT_SUCCESS;
113                 case 'H':
114                         print_help();
115                         return EXIT_SUCCESS;
116                 default:
117                         print_usage(stderr);
118                         return EXIT_FAILURE;
119                 }
120         }
121
122         if (count_str && !strict_strtoul(&count, count_str, 0)) {
123                 fprintf(stderr, "%s: invalid count: %s\n", progname, count_str);
124                 return EXIT_FAILURE;
125         }
126
127         rng = test_rng_alloc(seed);
128         if (!rng)
129                 return EXIT_FAILURE;
130
131         for (i = 0; !count || i < count; i++) {
132                 decl = random_decl(rng);
133
134                 if (mode == MODE_ENGLISH) {
135                         test_explain_decl(decl);
136                 } else {
137                         test_print_decl(decl);
138                 }
139
140                 gen_free_declspecs(decl->specifiers);
141                 gen_free_declarators(decl->declarators);
142                 free(decl);
143         }
144
145         test_rng_free(rng);
146
147         return EXIT_SUCCESS;
148 }