err.c revision 1.210 1 /* $NetBSD: err.c,v 1.210 2023/07/10 19:58:47 rillig Exp $ */
2
3 /*
4 * Copyright (c) 1994, 1995 Jochen Pohl
5 * All Rights Reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3. All advertising materials mentioning features or use of this software
16 * must display the following acknowledgement:
17 * This product includes software developed by Jochen Pohl for
18 * The NetBSD Project.
19 * 4. The name of the author may not be used to endorse or promote products
20 * derived from this software without specific prior written permission.
21 *
22 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
23 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
24 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
25 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
26 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
27 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
28 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
29 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
30 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
31 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 */
33
34 #if HAVE_NBTOOL_CONFIG_H
35 #include "nbtool_config.h"
36 #endif
37
38 #include <sys/cdefs.h>
39 #if defined(__RCSID)
40 __RCSID("$NetBSD: err.c,v 1.210 2023/07/10 19:58:47 rillig Exp $");
41 #endif
42
43 #include <limits.h>
44 #include <stdarg.h>
45 #include <stdlib.h>
46 #include <string.h>
47
48 #include "lint1.h"
49
50 bool seen_error;
51 bool seen_warning;
52
53 /* number of syntax errors */
54 int sytxerr;
55
56
57 static const char *const msgs[] = {
58 "empty declaration", /* 0 */
59 "old-style declaration; add 'int'", /* 1 */
60 "empty declaration", /* 2 */
61 "'%s' declared in argument declaration list", /* 3 */
62 "illegal type combination", /* 4 */
63 "modifying typedef with '%s'; only qualifiers allowed", /* 5 */
64 "use 'double' instead of 'long float'", /* 6 */
65 "only one storage class allowed", /* 7 */
66 "illegal storage class", /* 8 */
67 "only 'register' is valid as storage class in parameter", /* 9 */
68 "duplicate '%s'", /* 10 */
69 "bit-field initializer out of range", /* 11 */
70 "compiler takes size of function", /* 12 */
71 "incomplete enum type '%s'", /* 13 */
72 "", /* 14 */
73 "function returns illegal type '%s'", /* 15 */
74 "array of function is illegal", /* 16 */
75 "null dimension", /* 17 */
76 "illegal use of 'void'", /* 18 */
77 "void type for '%s'", /* 19 */
78 "negative array dimension (%d)", /* 20 */
79 "redeclaration of formal parameter '%s'", /* 21 */
80 "incomplete or misplaced function definition", /* 22 */
81 "undefined label '%s'", /* 23 */
82 "cannot initialize function '%s'", /* 24 */
83 "cannot initialize typedef '%s'", /* 25 */
84 "cannot initialize extern declaration '%s'", /* 26 */
85 "redeclaration of '%s'", /* 27 */
86 "redefinition of '%s'", /* 28 */
87 "'%s' was previously declared extern, becomes static", /* 29 */
88 "redeclaration of '%s'; ANSI C requires static", /* 30 */
89 "'%s' has incomplete type '%s'", /* 31 */
90 "type of argument '%s' defaults to 'int'", /* 32 */
91 "duplicate member name '%s'", /* 33 */
92 "nonportable bit-field type '%s'", /* 34 */
93 "illegal bit-field type '%s'", /* 35 */
94 "illegal bit-field size: %d", /* 36 */
95 "zero size bit-field", /* 37 */
96 "function illegal in structure or union", /* 38 */
97 "zero-sized array '%s' in struct is a C99 extension", /* 39 */
98 "", /* never used */ /* 40 */
99 "bit-field in union is very unusual", /* 41 */
100 "forward reference to enum type", /* 42 */
101 "redefinition of '%s' hides earlier one", /* 43 */
102 "declaration of '%s %s' introduces new type in ANSI C", /* 44 */
103 "base type is really '%s %s'", /* 45 */
104 "%s tag '%s' redeclared as %s", /* 46 */
105 "zero sized %s is a C99 feature", /* 47 */
106 "enumeration value '%s' overflows", /* 48 */
107 "anonymous struct/union members is a C11 feature", /* 49 */
108 "parameter '%s' has function type, should be pointer", /* 50 */
109 "parameter mismatch: %d declared, %d defined", /* 51 */
110 "cannot initialize parameter '%s'", /* 52 */
111 "declared argument '%s' is missing", /* 53 */
112 "trailing ',' prohibited in enum declaration", /* 54 */
113 "integral constant expression expected", /* 55 */
114 "integral constant too large", /* 56 */
115 "enumeration constant '%s' hides parameter", /* 57 */
116 "type of '%s' does not match prototype", /* 58 */
117 "formal parameter #%d lacks name", /* 59 */
118 "void must be sole parameter", /* 60 */
119 "void parameter '%s' cannot have name", /* 61 */
120 "function prototype parameters must have types", /* 62 */
121 "prototype does not match old-style definition", /* 63 */
122 "()-less function definition", /* 64 */
123 "'%s' has no named members", /* 65 */
124 "", /* 66 */
125 "cannot return incomplete type", /* 67 */
126 "typedef already qualified with '%s'", /* 68 */
127 "inappropriate qualifiers with 'void'", /* 69 */
128 "", /* unused */ /* 70 */
129 "too many characters in character constant", /* 71 */
130 "typedef declares no type name", /* 72 */
131 "empty character constant", /* 73 */
132 "no hex digits follow \\x", /* 74 */
133 "overflow in hex escape", /* 75 */
134 "character escape does not fit in character", /* 76 */
135 "bad octal digit '%c'", /* 77 */
136 "", /* unused */ /* 78 */
137 "dubious escape \\%c", /* 79 */
138 "dubious escape \\%o", /* 80 */
139 "\\a undefined in traditional C", /* 81 */
140 "\\x undefined in traditional C", /* 82 */
141 "storage class after type is obsolescent", /* 83 */
142 "ANSI C requires formal parameter before '...'", /* 84 */
143 "dubious tag declaration '%s %s'", /* 85 */
144 "automatic '%s' hides external declaration", /* 86 */
145 "static '%s' hides external declaration", /* 87 */
146 "typedef '%s' hides external declaration", /* 88 */
147 "typedef '%s' redeclared", /* 89 */
148 "inconsistent redeclaration of extern '%s'", /* 90 */
149 "declaration of '%s' hides parameter", /* 91 */
150 "inconsistent redeclaration of static '%s'", /* 92 */
151 "dubious static function '%s' at block level", /* 93 */
152 "function '%s' has illegal storage class", /* 94 */
153 "declaration of '%s' hides earlier one", /* 95 */
154 "cannot dereference non-pointer type '%s'", /* 96 */
155 "suffix 'U' is illegal in traditional C", /* 97 */
156 "suffixes 'F' and 'L' are illegal in traditional C", /* 98 */
157 "'%s' undefined", /* 99 */
158 "unary '+' is illegal in traditional C", /* 100 */
159 "type '%s' does not have member '%s'", /* 101 */
160 "illegal use of member '%s'", /* 102 */
161 "left operand of '.' must be struct or union, not '%s'", /* 103 */
162 "left operand of '->' must be pointer to struct or union, not '%s'", /* 104 */
163 "non-unique member requires struct/union %s", /* 105 */
164 "left operand of '->' must be pointer", /* 106 */
165 "operands of '%s' have incompatible types '%s' and '%s'", /* 107 */
166 "operand of '%s' has invalid type '%s'", /* 108 */
167 "void type illegal in expression", /* 109 */
168 "pointer to function is not allowed here", /* 110 */
169 "unacceptable operand of '%s'", /* 111 */
170 "cannot take address of bit-field", /* 112 */
171 "cannot take address of register '%s'", /* 113 */
172 "%soperand of '%s' must be lvalue", /* 114 */
173 "%soperand of '%s' must be modifiable lvalue", /* 115 */
174 "illegal pointer subtraction", /* 116 */
175 "bitwise '%s' on signed value possibly nonportable", /* 117 */
176 "semantics of '%s' change in ANSI C; use explicit cast", /* 118 */
177 "conversion of '%s' to '%s' is out of range", /* 119 */
178 "bitwise '%s' on signed value nonportable", /* 120 */
179 "negative shift", /* 121 */
180 "shift amount %llu is greater than bit-size %llu of '%s'", /* 122 */
181 "illegal combination of %s '%s' and %s '%s', op '%s'", /* 123 */
182 "illegal combination of '%s' and '%s', op '%s'", /* 124 */
183 "ANSI C forbids ordered comparisons of pointers to functions",/* 125 */
184 "incompatible types '%s' and '%s' in conditional", /* 126 */
185 "'&' before array or function: ignored", /* 127 */
186 "operands of '%s' have incompatible pointer types to '%s' and '%s'", /* 128 */
187 "expression has null effect", /* 129 */
188 "enum type mismatch: '%s' '%s' '%s'", /* 130 */
189 "conversion to '%s' may sign-extend incorrectly", /* 131 */
190 "conversion from '%s' to '%s' may lose accuracy", /* 132 */
191 "conversion of pointer to '%s' loses bits", /* 133 */
192 "conversion of pointer to '%s' may lose bits", /* 134 */
193 "converting '%s' to '%s' increases alignment from %u to %u", /* 135 */
194 "cannot do pointer arithmetic on operand of unknown size", /* 136 */
195 "", /* unused */ /* 137 */
196 "unknown operand size, op '%s'", /* 138 */
197 "division by 0", /* 139 */
198 "modulus by 0", /* 140 */
199 "operator '%s' produces integer overflow", /* 141 */
200 "operator '%s' produces floating point overflow", /* 142 */
201 "cannot take size/alignment of incomplete type", /* 143 */
202 "cannot take size/alignment of function type '%s'", /* 144 */
203 "cannot take size/alignment of bit-field", /* 145 */
204 "cannot take size/alignment of void", /* 146 */
205 "invalid cast from '%s' to '%s'", /* 147 */
206 "improper cast of void expression", /* 148 */
207 "cannot call '%s', must be a function", /* 149 */
208 "argument mismatch: %d %s passed, %d expected", /* 150 */
209 "void expressions may not be arguments, arg #%d", /* 151 */
210 "argument cannot have unknown size, arg #%d", /* 152 */
211 "converting '%s' to incompatible '%s' for argument %d", /* 153 */
212 "illegal combination of %s '%s' and %s '%s', arg #%d", /* 154 */
213 "passing '%s' to incompatible '%s', arg #%d", /* 155 */
214 "function expects '%s', passing '%s' for arg #%d", /* 156 */
215 "ANSI C treats constant as unsigned", /* 157 */
216 "'%s' may be used before set", /* 158 */
217 "assignment in conditional context", /* 159 */
218 "operator '==' found where '=' was expected", /* 160 */
219 "constant in conditional context", /* 161 */
220 "operator '%s' compares '%s' with '%s'", /* 162 */
221 "a cast does not yield an lvalue", /* 163 */
222 "assignment of negative constant to unsigned type", /* 164 */
223 "constant truncated by assignment", /* 165 */
224 "precision lost in bit-field assignment", /* 166 */
225 "array subscript cannot be negative: %ld", /* 167 */
226 "array subscript cannot be > %d: %ld", /* 168 */
227 "precedence confusion possible: parenthesize!", /* 169 */
228 "first operand of '?' must have scalar type", /* 170 */
229 "cannot assign to '%s' from '%s'", /* 171 */
230 "too many struct/union initializers", /* 172 */
231 "too many array initializers, expected %d", /* 173 */
232 "too many initializers", /* 174 */
233 "initialization of incomplete type '%s'", /* 175 */
234 "", /* no longer used */ /* 176 */
235 "non-constant initializer", /* 177 */
236 "initializer does not fit", /* 178 */
237 "cannot initialize struct/union with no named member", /* 179 */
238 "bit-field initializer does not fit", /* 180 */
239 "{}-enclosed initializer required", /* 181 */
240 "incompatible pointer types to '%s' and '%s'", /* 182 */
241 "illegal combination of %s '%s' and %s '%s'", /* 183 */
242 "illegal combination of '%s' and '%s'", /* 184 */
243 "cannot initialize '%s' from '%s'", /* 185 */
244 "bit-field initialization is illegal in traditional C", /* 186 */
245 "string literal too long (%lu) for target array (%lu)", /* 187 */
246 "no automatic aggregate initialization in traditional C", /* 188 */
247 "", /* no longer used */ /* 189 */
248 "empty array declaration for '%s'", /* 190 */
249 "'%s' set but not used in function '%s'", /* 191 */
250 "'%s' unused in function '%s'", /* 192 */
251 "statement not reached", /* 193 */
252 "label '%s' redefined", /* 194 */
253 "case not in switch", /* 195 */
254 "case label affected by conversion", /* 196 */
255 "non-constant case expression", /* 197 */
256 "non-integral case expression", /* 198 */
257 "duplicate case '%ld' in switch", /* 199 */
258 "duplicate case '%lu' in switch", /* 200 */
259 "default outside switch", /* 201 */
260 "duplicate default in switch", /* 202 */
261 "case label must be of type 'int' in traditional C", /* 203 */
262 "controlling expressions must have scalar type", /* 204 */
263 "switch expression must have integral type", /* 205 */
264 "enumeration value(s) not handled in switch", /* 206 */
265 "loop not entered at top", /* 207 */
266 "break outside loop or switch", /* 208 */
267 "continue outside loop", /* 209 */
268 "enum type mismatch between '%s' and '%s' in initialization", /* 210 */
269 "function has return type '%s' but returns '%s'", /* 211 */
270 "cannot return incomplete type", /* 212 */
271 "void function '%s' cannot return value", /* 213 */
272 "function '%s' expects to return value", /* 214 */
273 "function '%s' implicitly declared to return int", /* 215 */
274 "function '%s' has 'return expr' and 'return'", /* 216 */
275 "function '%s' falls off bottom without returning value", /* 217 */
276 "ANSI C treats constant as unsigned, op '%s'", /* 218 */
277 "concatenated strings are illegal in traditional C", /* 219 */
278 "fallthrough on case statement", /* 220 */
279 "initialization of unsigned with negative constant", /* 221 */
280 "conversion of negative constant to unsigned type", /* 222 */
281 "end-of-loop code not reached", /* 223 */
282 "cannot recover from previous errors", /* 224 */
283 "static function '%s' called but not defined", /* 225 */
284 "static variable '%s' unused", /* 226 */
285 "const object '%s' should have initializer", /* 227 */
286 "function cannot return const or volatile object", /* 228 */
287 "converting '%s' to '%s' is questionable", /* 229 */
288 "nonportable character comparison '%s'", /* 230 */
289 "parameter '%s' unused in function '%s'", /* 231 */
290 "label '%s' unused in function '%s'", /* 232 */
291 "struct '%s' never defined", /* 233 */
292 "union '%s' never defined", /* 234 */
293 "enum '%s' never defined", /* 235 */
294 "static function '%s' unused", /* 236 */
295 "redeclaration of formal parameter '%s'", /* 237 */
296 "initialization of union is illegal in traditional C", /* 238 */
297 "constant argument to '!'", /* 239 */
298 "", /* unused */ /* 240 */
299 "dubious operation '%s' on enum", /* 241 */
300 "combination of '%s' and '%s', op '%s'", /* 242 */
301 "operator '%s' assumes that '%s' is ordered", /* 243 */
302 "illegal structure pointer combination", /* 244 */
303 "incompatible structure pointers: '%s' '%s' '%s'", /* 245 */
304 "dubious conversion of enum to '%s'", /* 246 */
305 "pointer cast from '%s' to '%s' may be troublesome", /* 247 */
306 "floating-point constant out of range", /* 248 */
307 "syntax error '%s'", /* 249 */
308 "unknown character \\%o", /* 250 */
309 "malformed integer constant", /* 251 */
310 "integer constant out of range", /* 252 */
311 "unterminated character constant", /* 253 */
312 "newline in string or char constant", /* 254 */
313 "undefined or invalid '#' directive", /* 255 */
314 "unterminated comment", /* 256 */
315 "extra characters in lint comment", /* 257 */
316 "unterminated string constant", /* 258 */
317 "argument %d is converted from '%s' to '%s' due to prototype", /* 259 */
318 "previous declaration of '%s'", /* 260 */
319 "previous definition of '%s'", /* 261 */
320 "\\\" inside character constants undefined in traditional C", /* 262 */
321 "\\? undefined in traditional C", /* 263 */
322 "\\v undefined in traditional C", /* 264 */
323 "%s does not support 'long long'", /* 265 */
324 "'long double' is illegal in traditional C", /* 266 */
325 "shift amount %u equals bit-size of '%s'", /* 267 */
326 "variable '%s' declared inline", /* 268 */
327 "parameter '%s' declared inline", /* 269 */
328 "function prototypes are illegal in traditional C", /* 270 */
329 "switch expression must be of type 'int' in traditional C", /* 271 */
330 "empty translation unit", /* 272 */
331 "bit-field type '%s' invalid in ANSI C", /* 273 */
332 "ANSI C forbids comparison of %s with %s", /* 274 */
333 "cast discards 'const' from type '%s'", /* 275 */
334 "'__%s__' is illegal for type '%s'", /* 276 */
335 "initialization of '%s' with '%s'", /* 277 */
336 "combination of '%s' and '%s', arg #%d", /* 278 */
337 "combination of '%s' and '%s' in return", /* 279 */
338 "comment /* %s */ must be outside function", /* 280 */
339 "duplicate comment /* %s */", /* 281 */
340 "comment /* %s */ must precede function definition", /* 282 */
341 "argument number mismatch in comment /* %s */", /* 283 */
342 "fallthrough on default statement", /* 284 */
343 "prototype declaration", /* 285 */
344 "function definition is not a prototype", /* 286 */
345 "function declaration is not a prototype", /* 287 */
346 "dubious use of /* VARARGS */ with /* %s */", /* 288 */
347 "/* PRINTFLIKE */ and /* SCANFLIKE */ cannot be combined", /* 289 */
348 "static function '%s' declared but not defined", /* 290 */
349 "invalid multibyte character", /* 291 */
350 "cannot concatenate wide and regular string literals", /* 292 */
351 "argument %d must be 'char *' for PRINTFLIKE/SCANFLIKE", /* 293 */
352 "multi-character character constant", /* 294 */
353 "conversion of '%s' to '%s' is out of range, arg #%d", /* 295 */
354 "conversion of negative constant to unsigned type, arg #%d", /* 296 */
355 "conversion to '%s' may sign-extend incorrectly, arg #%d", /* 297 */
356 "conversion from '%s' to '%s' may lose accuracy, arg #%d", /* 298 */
357 "prototype does not match old-style definition, arg #%d", /* 299 */
358 "old-style definition", /* 300 */
359 "array of incomplete type", /* 301 */
360 "'%s' returns pointer to automatic object", /* 302 */
361 "ANSI C forbids conversion of %s to %s", /* 303 */
362 "ANSI C forbids conversion of %s to %s, arg #%d", /* 304 */
363 "ANSI C forbids conversion of %s to %s, op %s", /* 305 */
364 "constant truncated by conversion, op '%s'", /* 306 */
365 "static variable '%s' set but not used", /* 307 */
366 "invalid type for _Complex", /* 308 */
367 "extra bits set to 0 in conversion of '%s' to '%s', op '%s'", /* 309 */
368 "symbol renaming can't be used on function arguments", /* 310 */
369 "symbol renaming can't be used on automatic variables", /* 311 */
370 "%s does not support '//' comments", /* 312 */
371 "struct or union member name in initializer is a C99 feature",/* 313 */
372 "", /* never used */ /* 314 */
373 "GCC style struct or union member name in initializer", /* 315 */
374 "__FUNCTION__/__PRETTY_FUNCTION__ is a GCC extension", /* 316 */
375 "__func__ is a C99 feature", /* 317 */
376 "variable array dimension is a C99/GCC extension", /* 318 */
377 "compound literals are a C99/GCC extension", /* 319 */
378 "'({ ... })' is a GCC extension", /* 320 */
379 "array initializer with designators is a C99 feature", /* 321 */
380 "zero sized array is a C99 extension", /* 322 */
381 "continue in 'do ... while (0)' loop", /* 323 */
382 "suggest cast from '%s' to '%s' on op '%s' to avoid overflow", /* 324 */
383 "variable declaration in for loop", /* 325 */
384 "attribute '%s' ignored for '%s'", /* 326 */
385 "declarations after statements is a C99 feature", /* 327 */
386 "union cast is a GCC extension", /* 328 */
387 "type '%s' is not a member of '%s'", /* 329 */
388 "operand of '%s' must be bool, not '%s'", /* 330 */
389 "left operand of '%s' must be bool, not '%s'", /* 331 */
390 "right operand of '%s' must be bool, not '%s'", /* 332 */
391 "controlling expression must be bool, not '%s'", /* 333 */
392 "argument %d expects '%s', gets passed '%s'", /* 334 */
393 "operand of '%s' must not be bool", /* 335 */
394 "left operand of '%s' must not be bool", /* 336 */
395 "right operand of '%s' must not be bool", /* 337 */
396 "option '%c' should be handled in the switch", /* 338 */
397 "option '%c' should be listed in the options string", /* 339 */
398 "initialization with '[a...b]' is a GCC extension", /* 340 */
399 "argument to '%s' must be 'unsigned char' or EOF, not '%s'", /* 341 */
400 "argument to '%s' must be cast to 'unsigned char', not to '%s'", /* 342 */
401 "static array size is a C11 extension", /* 343 */
402 "bit-field of type plain 'int' has implementation-defined signedness", /* 344 */
403 "generic selection requires C11 or later", /* 345 */
404 "call to '%s' effectively discards 'const' from argument", /* 346 */
405 "redeclaration of '%s' with type '%s', expected '%s'", /* 347 */
406 "maximum value %d of '%s' does not match maximum array index %d", /* 348 */
407 "non type argument to alignof is a GCC extension", /* 349 */
408 "'_Atomic' requires C11 or later", /* 350 */
409 "missing%s header declaration for '%s'", /* 351 */
410 "nested 'extern' declaration of '%s'", /* 352 */
411 "empty initializer braces require C23 or later", /* 353 */
412 };
413
414 static bool is_suppressed[sizeof(msgs) / sizeof(msgs[0])];
415
416 static struct include_level {
417 const char *filename;
418 int lineno;
419 struct include_level *by;
420 } *includes;
421
422 void
423 suppress_messages(const char *p)
424 {
425 char *end;
426
427 for (; ch_isdigit(*p); p = end + 1) {
428 unsigned long id = strtoul(p, &end, 10);
429 if ((*end != '\0' && *end != ',') ||
430 id >= sizeof(msgs) / sizeof(msgs[0]) ||
431 msgs[id][0] == '\0')
432 break;
433
434 is_suppressed[id] = true;
435
436 if (*end == '\0')
437 return;
438 }
439 errx(1, "invalid message ID '%.*s'", (int)strcspn(p, ","), p);
440 }
441
442 void
443 update_location(const char *filename, int lineno, bool is_begin, bool is_end)
444 {
445 struct include_level *top;
446
447 top = includes;
448 if (is_begin && top != NULL)
449 top->lineno = curr_pos.p_line;
450
451 if (top == NULL || is_begin) {
452 top = xmalloc(sizeof(*top));
453 top->filename = filename;
454 top->lineno = lineno;
455 top->by = includes;
456 includes = top;
457 } else {
458 if (is_end) {
459 includes = top->by;
460 free(top);
461 top = includes;
462 }
463 if (top != NULL) {
464 top->filename = filename;
465 top->lineno = lineno;
466 }
467 }
468 }
469
470 static void
471 print_stack_trace(void)
472 {
473 const struct include_level *top;
474
475 if ((top = includes) == NULL)
476 return;
477 /*
478 * Skip the innermost include level since it is already listed in the
479 * diagnostic itself. Furthermore, its lineno is the line number of
480 * the last '#' line, not the current line.
481 */
482 for (top = top->by; top != NULL; top = top->by)
483 printf("\tincluded from %s(%d)\n", top->filename, top->lineno);
484 }
485
486 /*
487 * print a list of the messages with their ids
488 */
489 void
490 msglist(void)
491 {
492 size_t i;
493
494 for (i = 0; i < sizeof(msgs) / sizeof(msgs[0]); i++) {
495 if (msgs[i][0] != '\0')
496 printf("%zu\t%s\n", i, msgs[i]);
497 else
498 printf("---\t(no longer used)\n");
499 }
500 }
501
502 /*
503 * If Fflag is not set, lbasename() returns a pointer to the last
504 * component of the path, otherwise it returns the argument.
505 */
506 static const char *
507 lbasename(const char *path)
508 {
509
510 if (Fflag)
511 return path;
512
513 const char *base = path;
514 for (const char *p = path; *p != '\0'; p++)
515 if (*p == '/')
516 base = p + 1;
517 return base;
518 }
519
520 static FILE *
521 output_channel(void)
522 {
523 return yflag ? stderr : stdout;
524 }
525
526 static void
527 verror_at(int msgid, const pos_t *pos, va_list ap)
528 {
529
530 if (is_suppressed[msgid])
531 return;
532
533 FILE *out = output_channel();
534 (void)fprintf(out, "%s(%d): error: ",
535 lbasename(pos->p_file), pos->p_line);
536 (void)vfprintf(out, msgs[msgid], ap);
537 (void)fprintf(out, " [%d]\n", msgid);
538 seen_error = true;
539 print_stack_trace();
540 }
541
542 static void
543 vwarning_at(int msgid, const pos_t *pos, va_list ap)
544 {
545
546 if (is_suppressed[msgid])
547 return;
548
549 debug_step("%s: lwarn=%d msgid=%d", __func__, lwarn, msgid);
550 if (lwarn == LWARN_NONE || lwarn == msgid)
551 /* this warning is suppressed by a LINTED comment */
552 return;
553
554 FILE *out = output_channel();
555 (void)fprintf(out, "%s(%d): warning: ",
556 lbasename(pos->p_file), pos->p_line);
557 (void)vfprintf(out, msgs[msgid], ap);
558 (void)fprintf(out, " [%d]\n", msgid);
559 seen_warning = true;
560 print_stack_trace();
561 }
562
563 static void
564 vmessage_at(int msgid, const pos_t *pos, va_list ap)
565 {
566
567 if (is_suppressed[msgid])
568 return;
569
570 FILE *out = output_channel();
571 (void)fprintf(out, "%s(%d): ",
572 lbasename(pos->p_file), pos->p_line);
573 (void)vfprintf(out, msgs[msgid], ap);
574 (void)fprintf(out, " [%d]\n", msgid);
575 print_stack_trace();
576 }
577
578 void
579 (error_at)(int msgid, const pos_t *pos, ...)
580 {
581 va_list ap;
582
583 va_start(ap, pos);
584 verror_at(msgid, pos, ap);
585 va_end(ap);
586 }
587
588 void
589 (error)(int msgid, ...)
590 {
591 va_list ap;
592
593 va_start(ap, msgid);
594 verror_at(msgid, &curr_pos, ap);
595 va_end(ap);
596 }
597
598 void
599 assert_failed(const char *file, int line, const char *func, const char *cond)
600 {
601
602 /*
603 * After encountering a parse error in the grammar, lint often does
604 * not properly clean up its data structures, especially in 'dcs',
605 * the stack of declaration levels. This often leads to assertion
606 * failures. These cases are not interesting though, as the purpose
607 * of lint is to check syntactically valid code. In such a case,
608 * exit gracefully. This allows a fuzzer like afl to focus on more
609 * interesting cases instead of reporting nonsense translation units
610 * like 'f=({e:;}' or 'v(const(char););e(v){'.
611 */
612 if (sytxerr > 0)
613 norecover();
614
615 (void)fflush(stdout);
616 (void)fprintf(stderr,
617 "lint: assertion \"%s\" failed in %s at %s:%d near %s:%d\n",
618 cond, func, file, line,
619 lbasename(curr_pos.p_file), curr_pos.p_line);
620 print_stack_trace();
621 (void)fflush(stdout);
622 abort();
623 }
624
625 void
626 (warning_at)(int msgid, const pos_t *pos, ...)
627 {
628 va_list ap;
629
630 va_start(ap, pos);
631 vwarning_at(msgid, pos, ap);
632 va_end(ap);
633 }
634
635 void
636 (warning)(int msgid, ...)
637 {
638 va_list ap;
639
640 va_start(ap, msgid);
641 vwarning_at(msgid, &curr_pos, ap);
642 va_end(ap);
643 }
644
645 void
646 (message_at)(int msgid, const pos_t *pos, ...)
647 {
648 va_list ap;
649
650 va_start(ap, pos);
651 vmessage_at(msgid, pos, ap);
652 va_end(ap);
653 }
654
655 void
656 (c99ism)(int msgid, ...)
657 {
658 va_list ap;
659
660 if (allow_c99)
661 return;
662
663 va_start(ap, msgid);
664 int severity = (!allow_gcc ? 1 : 0) + (!allow_trad ? 1 : 0);
665 if (severity == 2)
666 verror_at(msgid, &curr_pos, ap);
667 if (severity == 1)
668 vwarning_at(msgid, &curr_pos, ap);
669 va_end(ap);
670 }
671
672 void
673 (c11ism)(int msgid, ...)
674 {
675 va_list ap;
676
677 /* FIXME: C11 mode has nothing to do with GCC mode. */
678 if (allow_c11 || allow_gcc)
679 return;
680 va_start(ap, msgid);
681 verror_at(msgid, &curr_pos, ap);
682 va_end(ap);
683 }
684
685 void
686 (c23ism)(int msgid, ...)
687 {
688 va_list ap;
689
690 if (allow_c23)
691 return;
692 va_start(ap, msgid);
693 verror_at(msgid, &curr_pos, ap);
694 va_end(ap);
695 }
696
697 bool
698 (gnuism)(int msgid, ...)
699 {
700 va_list ap;
701 int severity = (!allow_gcc ? 1 : 0) +
702 (!allow_trad && !allow_c99 ? 1 : 0);
703
704 va_start(ap, msgid);
705 if (severity == 2)
706 verror_at(msgid, &curr_pos, ap);
707 if (severity == 1)
708 vwarning_at(msgid, &curr_pos, ap);
709 va_end(ap);
710 return severity > 0;
711 }
712
713
714 static const char *queries[] = {
715 "", /* unused, to make queries 1-based */
716 "implicit conversion from floating point '%s' to integer '%s'", /* Q1 */
717 "cast from floating point '%s' to integer '%s'", /* Q2 */
718 "implicit conversion changes sign from '%s' to '%s'", /* Q3 */
719 "usual arithmetic conversion for '%s' from '%s' to '%s'", /* Q4 */
720 "pointer addition has integer on the left-hand side", /* Q5 */
721 "no-op cast from '%s' to '%s'", /* Q6 */
722 "redundant cast from '%s' to '%s' before assignment", /* Q7 */
723 "octal number '%.*s'", /* Q8 */
724 "parenthesized return value", /* Q9 */
725 "chained assignment with '%s' and '%s'", /* Q10 */
726 "static variable '%s' in function", /* Q11 */
727 "comma operator with types '%s' and '%s'", /* Q12 */
728 "redundant 'extern' in function declaration of '%s'", /* Q13 */
729 "comparison '%s' of 'char' with plain integer %d", /* Q14 */
730 "implicit conversion from integer 0 to pointer '%s'", /* Q15 */
731 };
732
733 bool any_query_enabled; /* for optimizing non-query scenarios */
734 static bool is_query_enabled[sizeof(queries) / sizeof(queries[0])];
735
736 void
737 (query_message)(int query_id, ...)
738 {
739
740 if (!is_query_enabled[query_id])
741 return;
742
743 va_list ap;
744 FILE *out = output_channel();
745 (void)fprintf(out, "%s(%d): ",
746 lbasename(curr_pos.p_file), curr_pos.p_line);
747 va_start(ap, query_id);
748 (void)vfprintf(out, queries[query_id], ap);
749 va_end(ap);
750 (void)fprintf(out, " [Q%d]\n", query_id);
751 print_stack_trace();
752 }
753
754 void
755 enable_queries(const char *p)
756 {
757 char *end;
758
759 for (; ch_isdigit(*p); p = end + 1) {
760 unsigned long id = strtoul(p, &end, 10);
761 if ((*end != '\0' && *end != ',') ||
762 id >= sizeof(queries) / sizeof(queries[0]) ||
763 queries[id][0] == '\0')
764 break;
765
766 any_query_enabled = true;
767 is_query_enabled[id] = true;
768
769 if (*end == '\0')
770 return;
771 }
772 errx(1, "invalid query ID '%.*s'", (int)strcspn(p, ","), p);
773 }
774