Richard Trieu | 3bb8b56 | 2014-02-26 02:36:06 +0000 | [diff] [blame] | 1 | // RUN: %clang_cc1 -analyze -analyzer-checker=core -verify -std=c99 -Dbool=_Bool -Wno-bool-conversion %s |
| 2 | // RUN: %clang_cc1 -analyze -analyzer-checker=core -verify -x c++ -Wno-bool-conversion %s |
Zhongxing Xu | 87e7fc5 | 2010-06-08 10:00:00 +0000 | [diff] [blame] | 3 | |
Jordan Rose | dc16628 | 2013-05-01 18:19:59 +0000 | [diff] [blame] | 4 | typedef __INTPTR_TYPE__ intptr_t; |
Zhongxing Xu | 87e7fc5 | 2010-06-08 10:00:00 +0000 | [diff] [blame] | 5 | char const *p; |
| 6 | |
| 7 | void f0() { |
| 8 | char const str[] = "This will change"; |
Jordan Rose | e37ab50 | 2012-11-15 19:11:43 +0000 | [diff] [blame] | 9 | p = str; |
| 10 | } // expected-warning{{Address of stack memory associated with local variable 'str' is still referred to by the global variable 'p' upon returning to the caller. This will be a dangling reference}} |
Zhongxing Xu | 87e7fc5 | 2010-06-08 10:00:00 +0000 | [diff] [blame] | 11 | |
| 12 | void f1() { |
| 13 | char const str[] = "This will change"; |
| 14 | p = str; |
| 15 | p = 0; // no-warning |
| 16 | } |
Zhongxing Xu | 4200be5 | 2010-06-09 05:50:38 +0000 | [diff] [blame] | 17 | |
| 18 | void f2() { |
Jordan Rose | e37ab50 | 2012-11-15 19:11:43 +0000 | [diff] [blame] | 19 | p = (const char *) __builtin_alloca(12); |
Jordan Rose | dc16628 | 2013-05-01 18:19:59 +0000 | [diff] [blame] | 20 | } // expected-warning{{Address of stack memory allocated by call to alloca() on line 19 is still referred to by the global variable 'p' upon returning to the caller. This will be a dangling reference}} |
Ted Kremenek | 17504be | 2010-06-17 00:24:44 +0000 | [diff] [blame] | 21 | |
| 22 | // PR 7383 - previosly the stack address checker would crash on this example |
| 23 | // because it would attempt to do a direct load from 'pr7383_list'. |
| 24 | static int pr7383(__const char *__) |
| 25 | { |
| 26 | return 0; |
| 27 | } |
| 28 | extern __const char *__const pr7383_list[]; |
Ted Kremenek | 5df037e | 2010-06-17 04:21:37 +0000 | [diff] [blame] | 29 | |
| 30 | // Test that we catch multiple returns via globals when analyzing a function. |
| 31 | void test_multi_return() { |
| 32 | static int *a, *b; |
| 33 | int x; |
| 34 | a = &x; |
Jordan Rose | e37ab50 | 2012-11-15 19:11:43 +0000 | [diff] [blame] | 35 | b = &x; |
| 36 | } // expected-warning{{Address of stack memory associated with local variable 'x' is still referred to by the global variable 'a' upon returning}} expected-warning{{Address of stack memory associated with local variable 'x' is still referred to by the global variable 'b' upon returning}} |
Jordan Rose | dc16628 | 2013-05-01 18:19:59 +0000 | [diff] [blame] | 37 | |
| 38 | intptr_t returnAsNonLoc() { |
| 39 | int x; |
| 40 | return (intptr_t)&x; // expected-warning{{Address of stack memory associated with local variable 'x' returned to caller}} |
| 41 | } |
| 42 | |
| 43 | bool returnAsBool() { |
| 44 | int x; |
| 45 | return &x; // no-warning |
| 46 | } |
| 47 | |
| 48 | void assignAsNonLoc() { |
| 49 | extern intptr_t ip; |
| 50 | int x; |
| 51 | ip = (intptr_t)&x; |
| 52 | } // expected-warning{{Address of stack memory associated with local variable 'x' is still referred to by the global variable 'ip' upon returning}} |
| 53 | |
| 54 | void assignAsBool() { |
| 55 | extern bool b; |
| 56 | int x; |
| 57 | b = &x; |
| 58 | } // no-warning |