array-struct-region.c revision e0d24eb1060a213ec9820dc02c45f26b2d5b348b
1// RUN: %clang_cc1 -analyze -analyzer-checker=core,experimental.core,debug.ExprInspection -analyzer-store=region -analyzer-constraints=basic -analyzer-ipa=inlining -verify %s 2// RUN: %clang_cc1 -analyze -analyzer-checker=core,experimental.core,debug.ExprInspection -analyzer-store=region -analyzer-constraints=range -analyzer-ipa=inlining -verify %s 3 4void clang_analyzer_eval(int); 5 6int string_literal_init() { 7 char a[] = "abc"; 8 char b[2] = "abc"; // expected-warning{{too long}} 9 char c[5] = "abc"; 10 11 clang_analyzer_eval(a[1] == 'b'); // expected-warning{{TRUE}} 12 clang_analyzer_eval(b[1] == 'b'); // expected-warning{{TRUE}} 13 clang_analyzer_eval(c[1] == 'b'); // expected-warning{{TRUE}} 14 15 clang_analyzer_eval(a[3] == 0); // expected-warning{{TRUE}} 16 clang_analyzer_eval(c[3] == 0); // expected-warning{{TRUE}} 17 18 clang_analyzer_eval(c[4] == 0); // expected-warning{{TRUE}} 19 20 return 42; 21} 22 23void nested_compound_literals(int rad) { 24 int vec[6][2] = {{0.195, 0.02}, {0.383, 0.067}, {0.55, 0.169}, // expected-warning 6 {{implicit conversion from 'double' to 'int' changes value from}} 25 {0.831, 0.45}, {0.924, 0.617}, {0.98, 0.805}}; // expected-warning 6 {{implicit conversion from 'double' to 'int' changes value from}} 26 int a; 27 28 for (a = 0; a < 6; ++a) { 29 vec[a][0] *= rad; // no-warning 30 vec[a][1] *= rad; // no-warning 31 } 32} 33 34void nested_compound_literals_float(float rad) { 35 float vec[6][2] = {{0.195, 0.02}, {0.383, 0.067}, {0.55, 0.169}, 36 {0.831, 0.45}, {0.924, 0.617}, {0.98, 0.805}}; 37 int a; 38 39 for (a = 0; a < 6; ++a) { 40 vec[a][0] *= rad; // no-warning 41 vec[a][1] *= rad; // no-warning 42 } 43} 44 45 46void struct_as_array() { 47 struct simple { int x; int y; }; 48 struct simple a; 49 struct simple *p = &a; 50 51 p->x = 5; 52 clang_analyzer_eval(a.x == 5); // expected-warning{{TRUE}} 53 clang_analyzer_eval(p[0].x == 5); // expected-warning{{TRUE}} 54 55 p[0].y = 5; 56 clang_analyzer_eval(a.y == 5); // expected-warning{{TRUE}} 57 clang_analyzer_eval(p->y == 5); // expected-warning{{TRUE}} 58} 59 60 61// PR13264 / <rdar://problem/11802440> 62struct point { int x; int y; }; 63struct circle { struct point o; int r; }; 64struct circle get_circle() { 65 struct circle result; 66 result.r = 5; 67 result.o = (struct point){0, 0}; 68 return result; 69} 70 71void struct_in_struct() { 72 struct circle c; 73 c = get_circle(); 74 // This used to think c.r was undefined because c.o is a LazyCompoundVal. 75 clang_analyzer_eval(c.r == 5); // expected-warning{{TRUE}} 76} 77 78// We also test with floats because we don't model floats right now, 79// and the original bug report used a float. 80struct circle_f { struct point o; float r; }; 81struct circle_f get_circle_f() { 82 struct circle_f result; 83 result.r = 5.0; 84 result.o = (struct point){0, 0}; 85 return result; 86} 87 88float struct_in_struct_f() { 89 struct circle_f c; 90 c = get_circle_f(); 91 92 return c.r; // no-warning 93} 94 95 96int randomInt(); 97 98int testSymbolicInvalidation(int index) { 99 int vals[10]; 100 101 vals[0] = 42; 102 clang_analyzer_eval(vals[0] == 42); // expected-warning{{TRUE}} 103 104 vals[index] = randomInt(); 105 clang_analyzer_eval(vals[0] == 42); // expected-warning{{UNKNOWN}} 106 107 return vals[index]; // no-warning 108} 109 110int testConcreteInvalidation(int index) { 111 int vals[10]; 112 113 vals[index] = 42; 114 clang_analyzer_eval(vals[index] == 42); // expected-warning{{TRUE}} 115 vals[0] = randomInt(); 116 clang_analyzer_eval(vals[index] == 42); // expected-warning{{UNKNOWN}} 117 118 return vals[0]; // no-warning 119} 120 121 122typedef struct { 123 int x, y, z; 124} S; 125 126S makeS(); 127 128int testSymbolicInvalidationStruct(int index) { 129 S vals[10]; 130 131 vals[0].x = 42; 132 clang_analyzer_eval(vals[0].x == 42); // expected-warning{{TRUE}} 133 134 vals[index] = makeS(); 135 clang_analyzer_eval(vals[0].x == 42); // expected-warning{{UNKNOWN}} 136 137 return vals[index].x; // no-warning 138} 139 140int testConcreteInvalidationStruct(int index) { 141 S vals[10]; 142 143 vals[index].x = 42; 144 clang_analyzer_eval(vals[index].x == 42); // expected-warning{{TRUE}} 145 vals[0] = makeS(); 146 clang_analyzer_eval(vals[index].x == 42); // expected-warning{{UNKNOWN}} 147 148 return vals[0].x; // no-warning 149} 150 151typedef struct { 152 S a[5]; 153 S b[5]; 154} SS; 155 156int testSymbolicInvalidationDoubleStruct(int index) { 157 SS vals; 158 159 vals.a[0].x = 42; 160 vals.b[0].x = 42; 161 clang_analyzer_eval(vals.a[0].x == 42); // expected-warning{{TRUE}} 162 clang_analyzer_eval(vals.b[0].x == 42); // expected-warning{{TRUE}} 163 164 vals.a[index] = makeS(); 165 clang_analyzer_eval(vals.a[0].x == 42); // expected-warning{{UNKNOWN}} 166 clang_analyzer_eval(vals.b[0].x == 42); // expected-warning{{TRUE}} 167 168 return vals.b[index].x; // no-warning 169} 170 171int testConcreteInvalidationDoubleStruct(int index) { 172 SS vals; 173 174 vals.a[index].x = 42; 175 vals.b[index].x = 42; 176 clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{TRUE}} 177 clang_analyzer_eval(vals.b[index].x == 42); // expected-warning{{TRUE}} 178 179 vals.a[0] = makeS(); 180 clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{UNKNOWN}} 181 clang_analyzer_eval(vals.b[index].x == 42); // expected-warning{{TRUE}} 182 183 return vals.b[0].x; // no-warning 184} 185 186 187