]> git.draconx.ca Git - cdecl99.git/blob - test/randomdecl.c
Free memory in randomdecl.
[cdecl99.git] / test / randomdecl.c
1 /*
2  *  Generate random C declarations for testing.
3  *  Copyright © 2011 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 <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:VH";
35 static const struct option lopts[] = {
36         { "seed",    1, NULL, 's' },
37         { "count",   1, NULL, 'n' },
38         { "version", 0, NULL, 'V' },
39         { "help",    0, NULL, 'H' },
40         { 0 }
41 };
42
43 static void print_usage(FILE *f)
44 {
45         fprintf(f, "Usage: %s [options]\n", progname);
46 }
47
48 static void print_help(void)
49 {
50         print_usage(stdout);
51         puts("Detailed help coming soon.");
52 }
53
54 static struct cdecl *random_decl(struct gen_rng *rng)
55 {
56         struct cdecl *decl;
57         unsigned flags = 0;
58
59         decl = malloc_nofail(sizeof *decl);
60         *decl = (struct cdecl) { 0 };
61
62         decl->declarators = gen_declarators(rng);
63         if (decl->declarators->type != CDECL_DECL_FUNCTION)
64                 flags |= GEN_NO_FUNCTION;
65         if (cdecl_is_abstract(decl->declarators))
66                 flags |= GEN_NO_STORAGE | GEN_NO_FUNCTION;
67         if (decl->declarators->type == CDECL_DECL_ARRAY
68             || decl->declarators->type == CDECL_DECL_IDENT)
69                 flags |= GEN_NO_VOID;
70
71         decl->specifiers = gen_declspecs(rng, flags);
72         return decl;
73 }
74
75 int main(int argc, char **argv)
76 {
77         const char *seed = "", *count_str = NULL;
78         struct gen_rng *rng;
79         struct cdecl *decl;
80         unsigned long count = 0;
81         int opt;
82
83         if (argc > 0)
84                 progname = argv[0];
85
86         while ((opt = getopt_long(argc, argv, sopts, lopts, NULL)) != -1) {
87                 switch (opt) {
88                 case 's':
89                         seed = optarg;
90                         break;
91                 case 'n':
92                         count_str = optarg;
93                         break;
94                 case 'V':
95                         test_print_version(PROGNAME);
96                         return EXIT_SUCCESS;
97                 case 'H':
98                         print_help();
99                         return EXIT_SUCCESS;
100                 default:
101                         print_usage(stderr);
102                         return EXIT_FAILURE;
103                 }
104         }
105
106         if (count_str && !strict_strtoul(&count, count_str, 0)) {
107                 fprintf(stderr, "%s: invalid count: %s\n", progname, count_str);
108                 return EXIT_FAILURE;
109         }
110
111         rng = gen_alloc_rng(seed);
112         if (!rng)
113                 return EXIT_FAILURE;
114
115         for (unsigned long i = 0; !count || i < count; i++) {
116                 decl = random_decl(rng);
117                 test_print_decl(decl);
118
119                 gen_free_declspecs(decl->specifiers);
120                 gen_free_declarators(decl->declarators);
121                 free(decl);
122         }
123
124         gen_free_rng(rng);
125
126         return EXIT_SUCCESS;
127 }