give stray-semicolon warning a name
[cparser] / warning.c
1 /*
2  * This file is part of cparser.
3  * Copyright (C) 2007-2009 Matthias Braun <matze@braunis.de>
4  *
5  * This program is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU General Public License
7  * as published by the Free Software Foundation; either version 2
8  * of the License, or (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, write to the Free Software
17  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
18  * 02111-1307, USA.
19  */
20 #include <assert.h>
21 #include <stdio.h>
22 #include <string.h>
23
24 #include "adt/util.h"
25 #include "warning.h"
26 #include "help.h"
27
28 static warning_switch_t warning[] = {
29         [WARN_ADDRESS]                       = { WARN_STATE_ON,   "address"                       },
30         [WARN_AGGREGATE_RETURN]              = { WARN_STATE_NONE, "aggregate-return"              },
31         [WARN_ATTRIBUTE]                     = { WARN_STATE_ON,   "attribute"                     },
32         [WARN_CAST_QUAL]                     = { WARN_STATE_NONE, "cast-qual",                    },
33         [WARN_CHAR_SUBSCRIPTS]               = { WARN_STATE_ON,   "char-subscripts",              },
34         [WARN_COMMENT]                       = { WARN_STATE_NONE, "comment",                      },
35         [WARN_CONVERSION]                    = { WARN_STATE_NONE, "conversion",                   },
36         [WARN_DECLARATION_AFTER_STATEMENT]   = { WARN_STATE_NONE, "declaration-after-statement",  },
37         [WARN_DEPRECATED_DECLARATIONS]       = { WARN_STATE_ON,   "deprecated-declarations",      },
38         [WARN_DIV_BY_ZERO]                   = { WARN_STATE_ON,   "div-by-zero",                  },
39         [WARN_EMPTY_BODY]                    = { WARN_STATE_NONE, "empty-body",                   },
40         [WARN_EMPTY_STATEMENT]               = { WARN_STATE_NONE, "empty-statement",              },
41         [WARN_ERROR]                         = { WARN_STATE_NONE, "error"                         },
42         [WARN_FATAL_ERRORS]                  = { WARN_STATE_NONE, "fatal-errors"                  },
43         [WARN_FLOAT_EQUAL]                   = { WARN_STATE_NONE, "float-equal",                  },
44         [WARN_FORMAT]                        = { WARN_STATE_ON,   "format"                        },
45         [WARN_IGNORED_QUALIFIERS]            = { WARN_STATE_ON,   "ignored-qualifiers"            },
46         [WARN_IMPLICIT_FUNCTION_DECLARATION] = { WARN_STATE_ON,   "implicit-function-declaration" },
47         [WARN_IMPLICIT_INT]                  = { WARN_STATE_ON,   "implicit-int"                  },
48         [WARN_LONG_LONG]                     = { WARN_STATE_NONE, "long-long"                     },
49         [WARN_MAIN]                          = { WARN_STATE_ON,   "main",                         },
50         [WARN_MISSING_DECLARATIONS]          = { WARN_STATE_NONE, "missing-declarations",         },
51         [WARN_MISSING_NORETURN]              = { WARN_STATE_NONE, "missing-noreturn",             },
52         [WARN_MISSING_PROTOTYPES]            = { WARN_STATE_NONE, "missing-prototypes",           },
53         [WARN_MULTICHAR]                     = { WARN_STATE_ON,   "multichar",                    },
54         [WARN_NESTED_EXTERNS]                = { WARN_STATE_NONE, "nested-externs"                },
55         [WARN_NONNULL]                       = { WARN_STATE_ON,   "nonnull",                      },
56         [WARN_OLD_STYLE_DEFINITION]          = { WARN_STATE_NONE, "old-style-definition",         },
57         [WARN_OTHER]                         = { WARN_STATE_ON,   "other"                         },
58         [WARN_PACKED]                        = { WARN_STATE_NONE, "packed",                       },
59         [WARN_PADDED]                        = { WARN_STATE_NONE, "padded",                       },
60         [WARN_PARENTHESES]                   = { WARN_STATE_NONE, "parentheses",                  },
61         [WARN_POINTER_ARITH]                 = { WARN_STATE_ON,   "pointer-arith",                },
62         [WARN_REDUNDANT_DECLS]               = { WARN_STATE_ON,   "redundant-decls",              },
63         [WARN_RETURN_TYPE]                   = { WARN_STATE_ON,   "return-type",                  },
64         [WARN_SHADOW]                        = { WARN_STATE_NONE, "shadow",                       },
65         [WARN_SHADOW_LOCAL]                  = { WARN_STATE_NONE, "shadow-local",                 },
66         [WARN_SIGN_COMPARE]                  = { WARN_STATE_NONE, "sign-compare",                 },
67         [WARN_STRAY_SEMICOLON]               = { WARN_STATE_ON,   "stray-semicolon",              },
68         [WARN_STRICT_PROTOTYPES]             = { WARN_STATE_ON,   "strict-prototypes"             },
69         [WARN_SWITCH_DEFAULT]                = { WARN_STATE_NONE, "switch-default",               },
70         [WARN_SWITCH_ENUM]                   = { WARN_STATE_NONE, "switch-enum",                  },
71         [WARN_TRADITIONAL]                   = { WARN_STATE_NONE, "traditional"                   },
72         [WARN_UNINITIALIZED]                 = { WARN_STATE_ON,   "uninitialized",                },
73         [WARN_UNKNOWN_PRAGMAS]               = { WARN_STATE_ON,   "unknown-pragmas",              },
74         [WARN_UNREACHABLE_CODE]              = { WARN_STATE_NONE, "unreachable-code"              },
75         [WARN_UNUSED_FUNCTION]               = { WARN_STATE_NONE, "unused-function",              },
76         [WARN_UNUSED_LABEL]                  = { WARN_STATE_NONE, "unused-label",                 },
77         [WARN_UNUSED_PARAMETER]              = { WARN_STATE_NONE, "unused-parameter",             },
78         [WARN_UNUSED_VALUE]                  = { WARN_STATE_ON,   "unused-value",                 },
79         [WARN_UNUSED_VARIABLE]               = { WARN_STATE_NONE, "unused-variable",              },
80         [WARN_WRITE_STRINGS]                 = { WARN_STATE_NONE, "write-strings",                },
81 };
82
83 warning_switch_t const *get_warn_switch(warning_t const w)
84 {
85         assert((size_t)w < lengthof(warning));
86         assert(warning[w].name);
87         return &warning[w];
88 }
89
90 void print_warning_opt_help(void)
91 {
92         /* TODO: write explanations */
93         for (warning_switch_t* i = warning; i != endof(warning); ++i) {
94                 put_help(i->name, "");
95         }
96 }
97
98 void set_warning_opt(const char *const opt)
99 {
100         /* Process prefixes: -W[no-][error=] */
101         char const *s     = opt;
102         bool const  no    = strncmp(s, "no-",    3) == 0 ? s += 3, true : false;
103         bool const  error = strncmp(s, "error=", 6) == 0 ? s += 6, true : false;
104
105         warn_state_t on  = WARN_STATE_NONE;
106         warn_state_t off = WARN_STATE_NONE;
107         if (!no || !error)
108                 on |= WARN_STATE_ON;
109         if (error) {
110                 on  |= WARN_STATE_ERROR;
111                 off |= WARN_STATE_NO_ERROR;
112         }
113         if (no) {
114                 warn_state_t const tmp = on;
115                 on  = off;
116                 off = tmp;
117         }
118
119         for (warning_switch_t* i = warning; i != endof(warning); ++i) {
120                 if (strcmp(i->name, s) == 0) {
121                         i->state = (i->state & ~off) | on;
122                         return;
123                 }
124         }
125
126         if (s[0] == '\0') { // -W is an alias for -Wextra
127                 goto extra;
128         }
129 #define OPTX(x)   else if (strcmp(s, x) == 0)
130 #define SET(y)    (void)(warning[y].state = (warning[y].state & ~off) | on)
131         OPTX("all") {
132                 /* Note: this switched on a lot more warnings than gcc's -Wall */
133                 SET(WARN_ADDRESS);
134                 SET(WARN_ATTRIBUTE);
135                 SET(WARN_CHAR_SUBSCRIPTS);
136                 SET(WARN_COMMENT);
137                 SET(WARN_EMPTY_STATEMENT);
138                 SET(WARN_FORMAT);
139                 SET(WARN_IMPLICIT_FUNCTION_DECLARATION);
140                 SET(WARN_IMPLICIT_INT);
141                 SET(WARN_MAIN);
142                 SET(WARN_MISSING_DECLARATIONS);
143                 SET(WARN_NONNULL);
144                 SET(WARN_OTHER);
145                 SET(WARN_PARENTHESES);
146                 SET(WARN_POINTER_ARITH);
147                 SET(WARN_REDUNDANT_DECLS);
148                 SET(WARN_RETURN_TYPE);
149                 SET(WARN_SHADOW_LOCAL);
150                 SET(WARN_SIGN_COMPARE);
151                 SET(WARN_STRICT_PROTOTYPES);
152                 SET(WARN_SWITCH_ENUM);
153                 SET(WARN_UNINITIALIZED);
154                 SET(WARN_UNKNOWN_PRAGMAS);
155                 SET(WARN_UNREACHABLE_CODE);
156                 SET(WARN_UNUSED_FUNCTION);
157                 SET(WARN_UNUSED_LABEL);
158                 SET(WARN_UNUSED_PARAMETER);
159                 SET(WARN_UNUSED_VALUE);
160                 SET(WARN_UNUSED_VARIABLE);
161         }
162         OPTX("extra") {
163 extra:
164                 /* TODO */
165                 // TODO SET(function_end_without_return);
166                 SET(WARN_EMPTY_STATEMENT);
167                 // TODO SET(incomplete_aggregate_init);
168                 // TODO SET(missing_field_initializers);
169                 // TODO SET(pointless_comparison);
170                 SET(WARN_SHADOW);
171                 SET(WARN_UNUSED_PARAMETER);
172                 SET(WARN_UNUSED_VALUE);
173         }
174         OPTX("implicit") {
175                 SET(WARN_IMPLICIT_FUNCTION_DECLARATION);
176                 SET(WARN_IMPLICIT_INT);
177         }
178         OPTX("unused") {
179                 SET(WARN_UNUSED_FUNCTION);
180                 SET(WARN_UNUSED_LABEL);
181                 SET(WARN_UNUSED_PARAMETER);
182                 SET(WARN_UNUSED_VALUE);
183                 SET(WARN_UNUSED_VARIABLE);
184         }
185 #undef SET
186 #undef OPT_X
187         else if (strcmp(opt /* sic */, "error-implicit-function-declaration") == 0) {
188                 /* GCC legacy: This way it only can be activated. */
189                 warning[WARN_IMPLICIT_FUNCTION_DECLARATION].state = WARN_STATE_ON | WARN_STATE_ERROR;
190                 return;
191         }
192         else {
193                 fprintf(stderr, "warning: ignoring unknown option -W%s\n", opt);
194         }
195 }
196
197 void disable_all_warnings(void)
198 {
199         for (warning_switch_t* i = warning; i != endof(warning); ++i) {
200                 if (i != &warning[WARN_ERROR] &&
201                     i != &warning[WARN_FATAL_ERRORS]) {
202                         i->state &= ~WARN_STATE_ON;
203                 }
204         }
205 }