blob: 5ddb4528c1b2349b5850b01aede4181e855c0117 [file] [log] [blame]
Ted Kremenekade31952011-03-12 06:14:28 +00001// RUN: %clang_cc1 -Wunused-variable -analyze -analyzer-checker=core,deadcode.DeadStores,deadcode.IdempotentOperations -fblocks -verify -Wno-unreachable-code -analyzer-opt-analyze-nested-blocks %s
Ted Kremenekade31952011-03-12 06:14:28 +00002// RUN: %clang_cc1 -Wunused-variable -analyze -analyzer-checker=core,deadcode.DeadStores,deadcode.IdempotentOperations -analyzer-store=region -analyzer-constraints=basic -fblocks -verify -Wno-unreachable-code -analyzer-opt-analyze-nested-blocks %s
3// RUN: %clang_cc1 -Wunused-variable -analyze -analyzer-checker=core,deadcode.DeadStores,deadcode.IdempotentOperations -analyzer-store=region -analyzer-constraints=range -fblocks -verify -Wno-unreachable-code -analyzer-opt-analyze-nested-blocks %s
Ted Kremenek8aefcbf2007-11-19 06:38:23 +00004
Ted Kremenek49a2fd22008-04-14 15:56:17 +00005void f1() {
Ted Kremenekebd42f42010-03-18 01:22:39 +00006 int k, y; // expected-warning{{unused variable 'k'}} expected-warning{{unused variable 'y'}}
Ted Kremenek0fdf06e2008-03-19 07:31:52 +00007 int abc=1;
Ted Kremenekebd42f42010-03-18 01:22:39 +00008 long idx=abc+3*5; // expected-warning {{never read}} expected-warning{{unused variable 'idx'}}
Ted Kremenekaa395ba2007-11-18 20:06:35 +00009}
Ted Kremenek8aefcbf2007-11-19 06:38:23 +000010
Ted Kremenek49a2fd22008-04-14 15:56:17 +000011void f2(void *b) {
Ted Kremenek8aefcbf2007-11-19 06:38:23 +000012 char *c = (char*)b; // no-warning
Ted Kremenekebd42f42010-03-18 01:22:39 +000013 char *d = b+1; // expected-warning {{never read}} expected-warning{{unused variable 'd'}}
Chris Lattner58f9e132010-09-05 00:04:01 +000014 printf("%s", c); // expected-warning{{implicitly declaring C library function 'printf' with type 'int (const char *, ...)'}} \
Douglas Gregora316e7b2009-02-14 00:32:47 +000015 // expected-note{{please include the header <stdio.h> or explicitly provide a declaration for 'printf'}}
Ted Kremenek8aefcbf2007-11-19 06:38:23 +000016}
Ted Kremenek74c43a02007-11-20 03:03:00 +000017
Chris Lattnere0303582010-01-09 20:43:19 +000018int f();
19
Ted Kremenek49a2fd22008-04-14 15:56:17 +000020void f3() {
Ted Kremenek0fdf06e2008-03-19 07:31:52 +000021 int r;
22 if ((r = f()) != 0) { // no-warning
23 int y = r; // no-warning
24 printf("the error is: %d\n", y);
25 }
Ted Kremenek74c43a02007-11-20 03:03:00 +000026}
Ted Kremenek49a2fd22008-04-14 15:56:17 +000027
28void f4(int k) {
29
30 k = 1;
31
32 if (k)
33 f1();
34
Ted Kremenek1a654b62008-06-20 21:45:25 +000035 k = 2; // expected-warning {{never read}}
Ted Kremenek49a2fd22008-04-14 15:56:17 +000036}
Ted Kremenek93fab7c2009-11-22 20:26:21 +000037
Ted Kremenekf87821c2008-04-15 18:37:29 +000038void f5() {
39
40 int x = 4; // no-warning
Ted Kremenekebd42f42010-03-18 01:22:39 +000041 int *p = &x; // expected-warning{{never read}} expected-warning{{unused variable 'p'}}
Ted Kremenekf87821c2008-04-15 18:37:29 +000042
Ted Kremeneka23157e2008-05-05 23:12:21 +000043}
44
Ted Kremenekf4e532b2011-02-12 00:17:19 +000045//
Ted Kremeneka23157e2008-05-05 23:12:21 +000046int f6() {
47
48 int x = 4;
Ted Kremenekf4e532b2011-02-12 00:17:19 +000049 ++x; // no-warning
Ted Kremeneka23157e2008-05-05 23:12:21 +000050 return 1;
51}
Ted Kremenek1a654b62008-06-20 21:45:25 +000052
53int f7(int *p) {
54 // This is allowed for defensive programming.
55 p = 0; // no-warning
56 return 1;
57}
58
Ted Kremenek93fab7c2009-11-22 20:26:21 +000059int f7b(int *p) {
60 // This is allowed for defensive programming.
61 p = (0); // no-warning
62 return 1;
63}
64
65int f7c(int *p) {
66 // This is allowed for defensive programming.
67 p = (void*) 0; // no-warning
68 return 1;
69}
70
71int f7d(int *p) {
72 // This is allowed for defensive programming.
73 p = (void*) (0); // no-warning
74 return 1;
75}
76
Ted Kremenek56b1f712011-01-13 20:58:56 +000077// Don't warn for dead stores in nested expressions. We have yet
78// to see a real bug in this scenario.
Ted Kremenek1a654b62008-06-20 21:45:25 +000079int f8(int *p) {
Daniel Dunbar4489fe12008-08-05 00:07:51 +000080 extern int *baz();
Ted Kremenek56b1f712011-01-13 20:58:56 +000081 if ((p = baz())) // no-warning
Ted Kremenek1a654b62008-06-20 21:45:25 +000082 return 1;
83 return 0;
84}
85
Ted Kremenek2cfac222008-07-23 21:16:38 +000086int f9() {
87 int x = 4;
88 x = x + 10; // expected-warning{{never read}}
89 return 1;
90}
91
Ted Kremenek2cfac222008-07-23 21:16:38 +000092int f10() {
93 int x = 4;
94 x = 10 + x; // expected-warning{{never read}}
95 return 1;
96}
97
Ted Kremenek8b00b6e2008-07-23 23:18:43 +000098int f11() {
99 int x = 4;
Ted Kremenek380277e2008-10-15 05:23:41 +0000100 return x++; // expected-warning{{never read}}
Ted Kremenek8b00b6e2008-07-23 23:18:43 +0000101}
102
Ted Kremenek380277e2008-10-15 05:23:41 +0000103int f11b() {
104 int x = 4;
Ted Kremenek7f5fce72009-01-20 00:47:45 +0000105 return ((((++x)))); // no-warning
Ted Kremenek380277e2008-10-15 05:23:41 +0000106}
107
Ted Kremenekfc7ff552008-07-25 04:47:34 +0000108int f12a(int y) {
Ted Kremenekebd42f42010-03-18 01:22:39 +0000109 int x = y; // expected-warning{{unused variable 'x'}}
Ted Kremenekfc7ff552008-07-25 04:47:34 +0000110 return 1;
111}
112int f12b(int y) {
113 int x __attribute__((unused)) = y; // no-warning
114 return 1;
115}
Ted Kremenekebd42f42010-03-18 01:22:39 +0000116int f12c(int y) {
117 // Allow initialiation of scalar variables by parameters as a form of
118 // defensive programming.
119 int x = y; // no-warning
120 x = 1;
121 return x;
122}
Ted Kremenek2cfac222008-07-23 21:16:38 +0000123
Ted Kremenekefe88f52008-08-06 23:26:31 +0000124// Filed with PR 2630. This code should produce no warnings.
125int f13(void)
126{
127 int a = 1;
128 int b, c = b = a + a;
129
130 if (b > 0)
131 return (0);
132
133 return (a + b + c);
134}
135
Ted Kremenekb497ebd2008-09-04 21:52:52 +0000136// Filed with PR 2763.
Ted Kremenek84fa6b92008-09-26 05:52:45 +0000137int f14(int count) {
Ted Kremenekb497ebd2008-09-04 21:52:52 +0000138 int index, nextLineIndex;
139 for (index = 0; index < count; index = nextLineIndex+1) {
140 nextLineIndex = index+1; // no-warning
141 continue;
142 }
143 return index;
144}
Ted Kremenek84fa6b92008-09-26 05:52:45 +0000145
146// Test case for <rdar://problem/6248086>
147void f15(unsigned x, unsigned y) {
148 int count = x * y; // no-warning
Ted Kremenekebd42f42010-03-18 01:22:39 +0000149 int z[count]; // expected-warning{{unused variable 'z'}}
Ted Kremenek84fa6b92008-09-26 05:52:45 +0000150}
151
Ted Kremenek56b1f712011-01-13 20:58:56 +0000152// Don't warn for dead stores in nested expressions. We have yet
153// to see a real bug in this scenario.
Ted Kremenek610a09e2008-09-26 22:58:57 +0000154int f16(int x) {
155 x = x * 2;
Ted Kremenek56b1f712011-01-13 20:58:56 +0000156 x = sizeof(int [x = (x || x + 1) * 2]) // expected-warning{{The left operand to '+' is always 0}} expected-warning{{The left operand to '*' is always 1}}
Ted Kremenekd2025e22008-09-26 23:05:47 +0000157 ? 5 : 8;
Ted Kremenek610a09e2008-09-26 22:58:57 +0000158 return x;
159}
160
Ted Kremenek3b587862009-01-09 22:15:01 +0000161// Self-assignments should not be flagged as dead stores.
Mike Stumpa5495ea2009-07-21 19:01:31 +0000162void f17() {
Ted Kremenek3b587862009-01-09 22:15:01 +0000163 int x = 1;
Tom Careef52bcb2010-08-24 21:09:07 +0000164 x = x;
Ted Kremenek3b587862009-01-09 22:15:01 +0000165}
Ted Kremenek7f5fce72009-01-20 00:47:45 +0000166
167// <rdar://problem/6506065>
168// The values of dead stores are only "consumed" in an enclosing expression
Mike Stumpcd7bf232009-07-17 01:04:31 +0000169// what that value is actually used. In other words, don't say "Although the
170// value stored to 'x' is used...".
Ted Kremenek7f5fce72009-01-20 00:47:45 +0000171int f18() {
172 int x = 0; // no-warning
173 if (1)
174 x = 10; // expected-warning{{Value stored to 'x' is never read}}
175 while (1)
176 x = 10; // expected-warning{{Value stored to 'x' is never read}}
Ted Kremenek848ec832011-02-11 23:24:26 +0000177 // unreachable.
Ted Kremenek7f5fce72009-01-20 00:47:45 +0000178 do
Ted Kremenek848ec832011-02-11 23:24:26 +0000179 x = 10; // no-warning
Ted Kremenek7f5fce72009-01-20 00:47:45 +0000180 while (1);
Ted Kremenek56b1f712011-01-13 20:58:56 +0000181 return (x = 10); // no-warning
Ted Kremenek7f5fce72009-01-20 00:47:45 +0000182}
Ted Kremenekd3098ee2009-02-09 18:01:00 +0000183
Ted Kremenek848ec832011-02-11 23:24:26 +0000184int f18_a() {
185 int x = 0; // no-warning
186 return (x = 10); // no-warning
187}
188
189void f18_b() {
190 int x = 0; // no-warning
191 if (1)
192 x = 10; // expected-warning{{Value stored to 'x' is never read}}
193}
194
195void f18_c() {
196 int x = 0;
197 while (1)
198 x = 10; // expected-warning{{Value stored to 'x' is never read}}
199}
200
201void f18_d() {
202 int x = 0; // no-warning
203 do
204 x = 10; // expected-warning{{Value stored to 'x' is never read}}
205 while (1);
206}
207
Ted Kremenekd3098ee2009-02-09 18:01:00 +0000208// PR 3514: false positive `dead initialization` warning for init to global
209// http://llvm.org/bugs/show_bug.cgi?id=3514
210extern const int MyConstant;
211int f19(void) {
212 int x = MyConstant; // no-warning
213 x = 1;
214 return x;
215}
216
Ted Kremenek28433ff2009-03-31 03:34:38 +0000217int f19b(void) { // This case is the same as f19.
Ted Kremenekd3098ee2009-02-09 18:01:00 +0000218 const int MyConstant = 0;
Ted Kremenek28433ff2009-03-31 03:34:38 +0000219 int x = MyConstant; // no-warning
Ted Kremenekd3098ee2009-02-09 18:01:00 +0000220 x = 1;
221 return x;
222}
Ted Kremenek632d1ec2009-03-23 22:30:58 +0000223
224void f20(void) {
225 int x = 1; // no-warning
226#pragma unused(x)
227}
228
Mike Stumpcd7bf232009-07-17 01:04:31 +0000229void halt() __attribute__((noreturn));
230int f21() {
231 int x = 4;
232
Ted Kremenekf4e532b2011-02-12 00:17:19 +0000233 x = x + 1; // expected-warning{{never read}}
Mike Stumpcd7bf232009-07-17 01:04:31 +0000234 if (1) {
235 halt();
236 (void)x;
237 }
238 return 1;
239}
Mike Stumpe5af3ce2009-07-20 23:24:15 +0000240
241int j;
242void f22() {
243 int x = 4;
244 int y1 = 4;
245 int y2 = 4;
246 int y3 = 4;
247 int y4 = 4;
248 int y5 = 4;
249 int y6 = 4;
250 int y7 = 4;
251 int y8 = 4;
252 int y9 = 4;
253 int y10 = 4;
Mike Stump5f203632009-07-21 00:38:52 +0000254 int y11 = 4;
255 int y12 = 4;
Mike Stumpfefb9f72009-07-21 01:12:51 +0000256 int y13 = 4;
257 int y14 = 4;
258 int y15 = 4;
Mike Stump8f9893a2009-07-21 01:27:50 +0000259 int y16 = 4;
260 int y17 = 4;
261 int y18 = 4;
Mike Stump22cd6582009-07-21 01:46:17 +0000262 int y19 = 4;
263 int y20 = 4;
Mike Stumpe5af3ce2009-07-20 23:24:15 +0000264
Ted Kremenekf4e532b2011-02-12 00:17:19 +0000265 x = x + 1; // expected-warning{{never read}}
Mike Stumpe5af3ce2009-07-20 23:24:15 +0000266 ++y1;
267 ++y2;
268 ++y3;
269 ++y4;
270 ++y5;
271 ++y6;
272 ++y7;
273 ++y8;
274 ++y9;
275 ++y10;
Mike Stump5f203632009-07-21 00:38:52 +0000276 ++y11;
277 ++y12;
Mike Stumpfefb9f72009-07-21 01:12:51 +0000278 ++y13;
279 ++y14;
280 ++y15;
Mike Stump8f9893a2009-07-21 01:27:50 +0000281 ++y16;
282 ++y17;
283 ++y18;
Mike Stump22cd6582009-07-21 01:46:17 +0000284 ++y19;
285 ++y20;
Mike Stumpe5af3ce2009-07-20 23:24:15 +0000286
287 switch (j) {
288 case 1:
289 if (0)
290 (void)x;
291 if (1) {
292 (void)y1;
293 return;
294 }
295 (void)x;
296 break;
297 case 2:
298 if (0)
299 (void)x;
300 else {
301 (void)y2;
302 return;
303 }
304 (void)x;
305 break;
306 case 3:
307 if (1) {
308 (void)y3;
309 return;
310 } else
311 (void)x;
312 (void)x;
313 break;
314 case 4:
315 0 ? : ((void)y4, ({ return; }));
316 (void)x;
317 break;
318 case 5:
319 1 ? : (void)x;
320 0 ? (void)x : ((void)y5, ({ return; }));
321 (void)x;
322 break;
323 case 6:
324 1 ? ((void)y6, ({ return; })) : (void)x;
325 (void)x;
326 break;
327 case 7:
328 (void)(0 && x);
329 (void)y7;
Argyrios Kyrtzidis1b2ad2f2010-09-19 23:03:35 +0000330 (void)(0 || (y8, ({ return; }), 1)); // expected-warning {{expression result unused}}
Mike Stumpe5af3ce2009-07-20 23:24:15 +0000331 (void)x;
332 break;
333 case 8:
Argyrios Kyrtzidis1b2ad2f2010-09-19 23:03:35 +0000334 (void)(1 && (y9, ({ return; }), 1)); // expected-warning {{expression result unused}}
Mike Stumpe5af3ce2009-07-20 23:24:15 +0000335 (void)x;
336 break;
337 case 9:
338 (void)(1 || x);
339 (void)y10;
Mike Stump5f203632009-07-21 00:38:52 +0000340 break;
341 case 10:
Mike Stump5f203632009-07-21 00:38:52 +0000342 while (0) {
343 (void)x;
344 }
Mike Stump8f9893a2009-07-21 01:27:50 +0000345 (void)y11;
Mike Stump5f203632009-07-21 00:38:52 +0000346 break;
Mike Stump8f9893a2009-07-21 01:27:50 +0000347 case 11:
348 while (1) {
349 (void)y12;
Mike Stumpfefb9f72009-07-21 01:12:51 +0000350 }
351 (void)x;
352 break;
Mike Stump8f9893a2009-07-21 01:27:50 +0000353 case 12:
354 do {
355 (void)y13;
356 } while (0);
357 (void)y14;
358 break;
359 case 13:
360 do {
361 (void)y15;
362 } while (1);
363 (void)x;
364 break;
Mike Stumpfefb9f72009-07-21 01:12:51 +0000365 case 14:
Mike Stump8f9893a2009-07-21 01:27:50 +0000366 for (;;) {
367 (void)y16;
368 }
369 (void)x;
370 break;
371 case 15:
372 for (;1;) {
373 (void)y17;
374 }
375 (void)x;
376 break;
377 case 16:
Mike Stumpfefb9f72009-07-21 01:12:51 +0000378 for (;0;) {
379 (void)x;
380 }
Mike Stump8f9893a2009-07-21 01:27:50 +0000381 (void)y18;
Mike Stumpfefb9f72009-07-21 01:12:51 +0000382 break;
Mike Stump22cd6582009-07-21 01:46:17 +0000383 case 17:
384 __builtin_choose_expr(0, (void)x, ((void)y19, ({ return; })));
385 (void)x;
386 break;
387 case 19:
388 __builtin_choose_expr(1, ((void)y20, ({ return; })), (void)x);
389 (void)x;
390 break;
Mike Stumpe5af3ce2009-07-20 23:24:15 +0000391 }
392}
Ted Kremenek3a976342009-11-26 06:55:36 +0000393
394void f23_aux(const char* s);
395void f23(int argc, char **argv) {
396 int shouldLog = (argc > 1); // no-warning
397 ^{
398 if (shouldLog) f23_aux("I did too use it!\n");
399 else f23_aux("I shouldn't log. Wait.. d'oh!\n");
400 }();
401}
402
403void f23_pos(int argc, char **argv) {
Ted Kremenekebd42f42010-03-18 01:22:39 +0000404 int shouldLog = (argc > 1); // expected-warning{{Value stored to 'shouldLog' during its initialization is never read}} expected-warning{{unused variable 'shouldLog'}}
Ted Kremenek3a976342009-11-26 06:55:36 +0000405 ^{
406 f23_aux("I did too use it!\n");
407 }();
408}
Ted Kremenek9a0459c2009-12-01 23:04:14 +0000409
410void f24_A(int y) {
411 // FIXME: One day this should be reported as dead since 'z = x + y' is dead.
412 int x = (y > 2); // no-warning
413 ^ {
Ted Kremenekebd42f42010-03-18 01:22:39 +0000414 int z = x + y; // expected-warning{{Value stored to 'z' during its initialization is never read}} expected-warning{{unused variable 'z'}}
Ted Kremenek9a0459c2009-12-01 23:04:14 +0000415 }();
416}
417
418void f24_B(int y) {
419 // FIXME: One day this should be reported as dead since 'x' is just overwritten.
420 __block int x = (y > 2); // no-warning
421 ^{
422 // FIXME: This should eventually be a dead store since it is never read either.
423 x = 5; // no-warning
424 }();
425}
426
427int f24_C(int y) {
428 // FIXME: One day this should be reported as dead since 'x' is just overwritten.
429 __block int x = (y > 2); // no-warning
430 ^{
431 x = 5; // no-warning
432 }();
433 return x;
434}
435
436int f24_D(int y) {
437 __block int x = (y > 2); // no-warning
438 ^{
439 if (y > 4)
440 x = 5; // no-warning
441 }();
442 return x;
443}
444
Ted Kremenek74635d82009-12-03 00:46:16 +0000445// This example shows that writing to a variable captured by a block means that it might
446// not be dead.
447int f25(int y) {
448 __block int x = (y > 2);
449 __block int z = 0;
450 void (^foo)() = ^{ z = x + y; };
451 x = 4; // no-warning
452 foo();
453 return z;
454}
455
456// This test is mostly the same as 'f25', but shows that the heuristic of pruning out dead
457// stores for variables that are just marked '__block' is overly conservative.
458int f25_b(int y) {
459 // FIXME: we should eventually report a dead store here.
460 __block int x = (y > 2);
461 __block int z = 0;
462 x = 4; // no-warning
463 return z;
464}
465
Ted Kremenek2cfe28b2010-03-10 00:18:11 +0000466int f26_nestedblocks() {
467 int z;
468 z = 1;
469 __block int y = 0;
470 ^{
471 int k;
472 k = 1; // expected-warning{{Value stored to 'k' is never read}}
473 ^{
474 y = z + 1;
475 }();
476 }();
477 return y;
478}
479
Ted Kremenek3f64a0e2010-05-21 20:30:15 +0000480// The FOREACH macro in QT uses 'break' statements within statement expressions
481// placed within the increment code of for loops.
482void rdar8014335() {
483 for (int i = 0 ; i != 10 ; ({ break; })) {
484 for ( ; ; ({ ++i; break; })) ;
485 // Note that the next value stored to 'i' is never executed
486 // because the next statement to be executed is the 'break'
487 // in the increment code of the first loop.
Ted Kremenek3e5637f2010-07-27 18:49:08 +0000488 i = i * 3; // expected-warning{{Value stored to 'i' is never read}} expected-warning{{The left operand to '*' is always 1}}
Ted Kremenek3f64a0e2010-05-21 20:30:15 +0000489 }
490}
491
Ted Kremenek334c1952010-08-17 21:00:06 +0000492// <rdar://problem/8320674> NullStmts followed by do...while() can lead to disconnected CFG
493//
494// This previously caused bogus dead-stores warnings because the body of the first do...while was
495// disconnected from the entry of the function.
496typedef struct { float r; float i; } s_rdar8320674;
497typedef struct { s_rdar8320674 x[1]; } s2_rdar8320674;
498
499void rdar8320674(s_rdar8320674 *z, unsigned y, s2_rdar8320674 *st, int m)
500{
501 s_rdar8320674 * z2;
502 s_rdar8320674 * tw1 = st->x;
503 s_rdar8320674 t;
504 z2 = z + m;
505 do{
506 ; ;
507 do{ (t).r = (*z2).r*(*tw1).r - (*z2).i*(*tw1).i; (t).i = (*z2).r*(*tw1).i + (*z2).i*(*tw1).r; }while(0);
508 tw1 += y;
509 do { (*z2).r=(*z).r-(t).r; (*z2).i=(*z).i-(t).i; }while(0);
510 do { (*z).r += (t).r; (*z).i += (t).i; }while(0);
511 ++z2;
512 ++z;
513 }while (--m);
514}
515
Ted Kremenek848ec832011-02-11 23:24:26 +0000516// Avoid dead stores resulting from an assignment (and use) being unreachable.
517void rdar8405222_aux(int i);
518void rdar8405222() {
519 const int show = 0;
520 int i = 0;
521
522 if (show)
523 i = 5; // no-warning
524
525 if (show)
526 rdar8405222_aux(i);
527}
528