err.c revision 1.231 1 /* $NetBSD: err.c,v 1.231 2024/03/03 13:09:22 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.231 2024/03/03 13:09:22 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 parameter 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'; C90 or later require static", // 30
89 "'%s' has incomplete type '%s'", // 31
90 "type of parameter '%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 requires C99 or later", // 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 C90 or later", // 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 parameter '%s' is missing", // 53
112 "trailing ',' in enum declaration requires C99 or later", // 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 "C90 to C17 require 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 C90; 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 "pointers to functions can only be compared for equality", // 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 "C90 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 or constant initializer of type '%s' 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 "C90 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 operand 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 C90 or later", // 273
332 "C90 or later forbid 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 "parameter 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 "parameter %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 "conversion of %s to %s requires a cast", // 303
362 "conversion of %s to %s requires a cast, arg #%d", // 304
363 "conversion of %s to %s requires a cast, 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 parameters", // 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 requires C99 or later", // 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 "parameter %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 requires C11 or later", // 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 "'_Static_assert' requires C11 or later", // 354
413 "'_Static_assert' without message requires C23 or later", // 355
414 "short octal escape '%.*s' followed by digit '%c'", // 356
415 "hex escape '%.*s' mixes uppercase and lowercase digits", // 357
416 "hex escape '%.*s' has more than 2 digits", // 358
417 "missing new-style '\\177' or old-style number base", // 359
418 "missing new-style number base after '\\177'", // 360
419 "number base '%.*s' is %ju, must be 8, 10 or 16", // 361
420 "directive '%.*s' should not be escaped", // 362
421 "non-printing character '%.*s' in description '%.*s'", // 363
422 "missing bit position after '%.*s'", // 364
423 "missing field width after '%.*s'", // 365
424 "missing '\\0' at the end of '%.*s'", // 366
425 "empty description in '%.*s'", // 367
426 "missing comparison value after directive '%.*s'", // 368
427 "bit position '%.*s' in '%.*s' should be escaped as octal or hex", // 369
428 "field width '%.*s' in '%.*s' should be escaped as octal or hex", // 370
429 "bit position '%.*s' (%ju) in '%.*s' out of range %u..%u", // 371
430 "field width '%.*s' (%ju) in '%.*s' out of range 0..64", // 372
431 "bit field end %ju in '%.*s' out of range 0..64", // 373
432 "unknown directive '%.*s', must be one of 'bfF=:*'", // 374
433 "comparison value '%.*s' (%ju) exceeds maximum field value %ju", // 375
434 "'%.*s' overlaps earlier '%.*s' on bit %u", // 376
435 "redundant '\\0' at the end of the format", // 377
436 "directive '%.*s' is unreachable by input value", // 378
437 };
438
439 static bool is_suppressed[sizeof(msgs) / sizeof(msgs[0])];
440
441 static struct include_level {
442 const char *filename;
443 int lineno;
444 struct include_level *by;
445 } *includes;
446
447 void
448 suppress_messages(const char *p)
449 {
450 char *end;
451
452 for (; isdigit((unsigned char)*p); p = end + 1) {
453 unsigned long id = strtoul(p, &end, 10);
454 if ((*end != '\0' && *end != ',') ||
455 id >= sizeof(msgs) / sizeof(msgs[0]) ||
456 msgs[id][0] == '\0')
457 break;
458
459 is_suppressed[id] = true;
460
461 if (*end == '\0')
462 return;
463 }
464 errx(1, "invalid message ID '%.*s'", (int)strcspn(p, ","), p);
465 }
466
467 void
468 update_location(const char *filename, int lineno, bool is_begin, bool is_end)
469 {
470 struct include_level *top;
471
472 top = includes;
473 if (is_begin && top != NULL)
474 top->lineno = curr_pos.p_line;
475
476 if (top == NULL || is_begin) {
477 top = xmalloc(sizeof(*top));
478 top->filename = filename;
479 top->lineno = lineno;
480 top->by = includes;
481 includes = top;
482 } else {
483 if (is_end) {
484 includes = top->by;
485 free(top);
486 top = includes;
487 }
488 if (top != NULL) {
489 top->filename = filename;
490 top->lineno = lineno;
491 }
492 }
493 }
494
495 static void
496 print_stack_trace(void)
497 {
498 const struct include_level *top;
499
500 if ((top = includes) == NULL)
501 return;
502 /*
503 * Skip the innermost include level since it is already listed in the
504 * diagnostic itself. Furthermore, its lineno is the line number of
505 * the last '#' line, not the current line.
506 */
507 for (top = top->by; top != NULL; top = top->by)
508 printf("\tincluded from %s(%d)\n", top->filename, top->lineno);
509 }
510
511 /*
512 * If Fflag is not set, lbasename() returns a pointer to the last
513 * component of the path, otherwise it returns the argument.
514 */
515 static const char *
516 lbasename(const char *path)
517 {
518
519 if (Fflag)
520 return path;
521
522 const char *base = path;
523 for (const char *p = path; *p != '\0'; p++)
524 if (*p == '/')
525 base = p + 1;
526 return base;
527 }
528
529 static FILE *
530 output_channel(void)
531 {
532 return yflag ? stderr : stdout;
533 }
534
535 static void
536 verror_at(int msgid, const pos_t *pos, va_list ap)
537 {
538
539 if (is_suppressed[msgid])
540 return;
541
542 FILE *out = output_channel();
543 (void)fprintf(out, "%s(%d): error: ",
544 lbasename(pos->p_file), pos->p_line);
545 (void)vfprintf(out, msgs[msgid], ap);
546 (void)fprintf(out, " [%d]\n", msgid);
547 seen_error = true;
548 print_stack_trace();
549 }
550
551 static void
552 vwarning_at(int msgid, const pos_t *pos, va_list ap)
553 {
554
555 if (is_suppressed[msgid])
556 return;
557
558 debug_step("%s: lwarn=%d msgid=%d", __func__, lwarn, msgid);
559 if (lwarn == LWARN_NONE || lwarn == msgid)
560 /* this warning is suppressed by a LINTED comment */
561 return;
562
563 FILE *out = output_channel();
564 (void)fprintf(out, "%s(%d): warning: ",
565 lbasename(pos->p_file), pos->p_line);
566 (void)vfprintf(out, msgs[msgid], ap);
567 (void)fprintf(out, " [%d]\n", msgid);
568 seen_warning = true;
569 print_stack_trace();
570 }
571
572 static void
573 vmessage_at(int msgid, const pos_t *pos, va_list ap)
574 {
575
576 if (is_suppressed[msgid])
577 return;
578
579 FILE *out = output_channel();
580 (void)fprintf(out, "%s(%d): ",
581 lbasename(pos->p_file), pos->p_line);
582 (void)vfprintf(out, msgs[msgid], ap);
583 (void)fprintf(out, " [%d]\n", msgid);
584 print_stack_trace();
585 }
586
587 void
588 (error_at)(int msgid, const pos_t *pos, ...)
589 {
590 va_list ap;
591
592 va_start(ap, pos);
593 verror_at(msgid, pos, ap);
594 va_end(ap);
595 }
596
597 void
598 (error)(int msgid, ...)
599 {
600 va_list ap;
601
602 va_start(ap, msgid);
603 verror_at(msgid, &curr_pos, ap);
604 va_end(ap);
605 }
606
607 void
608 assert_failed(const char *file, int line, const char *func, const char *cond)
609 {
610
611 /*
612 * After encountering a parse error in the grammar, lint often does not
613 * properly clean up its data structures, especially in 'dcs', the
614 * stack of declaration levels. This often leads to assertion
615 * failures. These cases are not interesting though, as the purpose of
616 * lint is to check syntactically valid code. In such a case, exit
617 * gracefully. This allows a fuzzer like afl to focus on more
618 * interesting cases instead of reporting nonsense translation units
619 * like 'f=({e:;}' or 'v(const(char););e(v){'.
620 */
621 if (sytxerr > 0)
622 norecover();
623
624 (void)fflush(stdout);
625 (void)fprintf(stderr,
626 "lint: assertion \"%s\" failed in %s at %s:%d near %s:%d\n",
627 cond, func, file, line,
628 lbasename(curr_pos.p_file), curr_pos.p_line);
629 print_stack_trace();
630 (void)fflush(stdout);
631 abort();
632 }
633
634 void
635 (warning_at)(int msgid, const pos_t *pos, ...)
636 {
637 va_list ap;
638
639 va_start(ap, pos);
640 vwarning_at(msgid, pos, ap);
641 va_end(ap);
642 }
643
644 void
645 (warning)(int msgid, ...)
646 {
647 va_list ap;
648
649 va_start(ap, msgid);
650 vwarning_at(msgid, &curr_pos, ap);
651 va_end(ap);
652 }
653
654 void
655 (message_at)(int msgid, const pos_t *pos, ...)
656 {
657 va_list ap;
658
659 va_start(ap, pos);
660 vmessage_at(msgid, pos, ap);
661 va_end(ap);
662 }
663
664 void
665 (c99ism)(int msgid, ...)
666 {
667 va_list ap;
668
669 if (allow_c99)
670 return;
671
672 va_start(ap, msgid);
673 int severity = (!allow_gcc ? 1 : 0) + (!allow_trad ? 1 : 0);
674 if (severity == 2)
675 verror_at(msgid, &curr_pos, ap);
676 if (severity == 1)
677 vwarning_at(msgid, &curr_pos, ap);
678 va_end(ap);
679 }
680
681 void
682 (c11ism)(int msgid, ...)
683 {
684 va_list ap;
685
686 /* FIXME: C11 mode has nothing to do with GCC mode. */
687 if (allow_c11 || allow_gcc)
688 return;
689 va_start(ap, msgid);
690 verror_at(msgid, &curr_pos, ap);
691 va_end(ap);
692 }
693
694 void
695 (c23ism)(int msgid, ...)
696 {
697 va_list ap;
698
699 if (allow_c23)
700 return;
701 va_start(ap, msgid);
702 verror_at(msgid, &curr_pos, ap);
703 va_end(ap);
704 }
705
706 bool
707 (gnuism)(int msgid, ...)
708 {
709 va_list ap;
710 int severity = (!allow_gcc ? 1 : 0) +
711 (!allow_trad && !allow_c99 ? 1 : 0);
712
713 va_start(ap, msgid);
714 if (severity == 2)
715 verror_at(msgid, &curr_pos, ap);
716 if (severity == 1)
717 vwarning_at(msgid, &curr_pos, ap);
718 va_end(ap);
719 return severity > 0;
720 }
721
722
723 static const char *queries[] = {
724 "", /* unused, to make queries 1-based */
725 "implicit conversion from floating point '%s' to integer '%s'", // Q1
726 "cast from floating point '%s' to integer '%s'", // Q2
727 "implicit conversion changes sign from '%s' to '%s'", // Q3
728 "usual arithmetic conversion for '%s' from '%s' to '%s'", // Q4
729 "pointer addition has integer on the left-hand side", // Q5
730 "no-op cast from '%s' to '%s'", // Q6
731 "redundant cast from '%s' to '%s' before assignment", // Q7
732 "octal number '%.*s'", // Q8
733 "parenthesized return value", // Q9
734 "chained assignment with '%s' and '%s'", // Q10
735 "static variable '%s' in function", // Q11
736 "comma operator with types '%s' and '%s'", // Q12
737 "redundant 'extern' in function declaration of '%s'", // Q13
738 "comparison '%s' of 'char' with plain integer %d", // Q14
739 "implicit conversion from integer 0 to pointer '%s'", // Q15
740 "'%s' was declared 'static', now non-'static'", // Q16
741 "invisible character U+%04X in %s", // Q17
742 "const automatic variable '%s'", // Q18
743 };
744
745 bool any_query_enabled; /* for optimizing non-query scenarios */
746 bool is_query_enabled[sizeof(queries) / sizeof(queries[0])];
747
748 void
749 (query_message)(int query_id, ...)
750 {
751
752 if (!is_query_enabled[query_id])
753 return;
754
755 va_list ap;
756 FILE *out = output_channel();
757 (void)fprintf(out, "%s(%d): ",
758 lbasename(curr_pos.p_file), curr_pos.p_line);
759 va_start(ap, query_id);
760 (void)vfprintf(out, queries[query_id], ap);
761 va_end(ap);
762 (void)fprintf(out, " [Q%d]\n", query_id);
763 print_stack_trace();
764 }
765
766 void
767 enable_queries(const char *p)
768 {
769 char *end;
770
771 for (; isdigit((unsigned char)*p); p = end + 1) {
772 unsigned long id = strtoul(p, &end, 10);
773 if ((*end != '\0' && *end != ',') ||
774 id >= sizeof(queries) / sizeof(queries[0]) ||
775 queries[id][0] == '\0')
776 break;
777
778 any_query_enabled = true;
779 is_query_enabled[id] = true;
780
781 if (*end == '\0')
782 return;
783 }
784 errx(1, "invalid query ID '%.*s'", (int)strcspn(p, ","), p);
785 }
786