Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / clang / test / Analysis / array-struct-region.c
blobfbdc6918cad50528e867131b2a38846ba8e5ceec
1 // RUN: %clang_analyze_cc1 -analyzer-checker=core,alpha.core,debug.ExprInspection -verify -analyzer-config eagerly-assume=false %s
3 void clang_analyzer_eval(int);
5 int string_literal_init(void) {
6 char a[] = "abc";
7 char b[2] = "abc"; // expected-warning{{too long}}
8 char c[5] = "abc";
10 clang_analyzer_eval(a[1] == 'b'); // expected-warning{{TRUE}}
11 clang_analyzer_eval(b[1] == 'b'); // expected-warning{{TRUE}}
12 clang_analyzer_eval(c[1] == 'b'); // expected-warning{{TRUE}}
14 clang_analyzer_eval(a[3] == 0); // expected-warning{{TRUE}}
15 clang_analyzer_eval(c[3] == 0); // expected-warning{{TRUE}}
17 clang_analyzer_eval(c[4] == 0); // expected-warning{{TRUE}}
19 return 42;
22 void nested_compound_literals(int rad) {
23 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}}
24 {0.831, 0.45}, {0.924, 0.617}, {0.98, 0.805}}; // expected-warning 6 {{implicit conversion from 'double' to 'int' changes value from}}
25 int a;
27 for (a = 0; a < 6; ++a) {
28 vec[a][0] *= rad; // no-warning
29 vec[a][1] *= rad; // no-warning
33 void nested_compound_literals_float(float rad) {
34 float vec[6][2] = {{0.195, 0.02}, {0.383, 0.067}, {0.55, 0.169},
35 {0.831, 0.45}, {0.924, 0.617}, {0.98, 0.805}};
36 int a;
38 for (a = 0; a < 6; ++a) {
39 vec[a][0] *= rad; // no-warning
40 vec[a][1] *= rad; // no-warning
45 void struct_as_array(void) {
46 struct simple { int x; int y; };
47 struct simple a;
48 struct simple *p = &a;
50 p->x = 5;
51 clang_analyzer_eval(a.x == 5); // expected-warning{{TRUE}}
52 clang_analyzer_eval(p[0].x == 5); // expected-warning{{TRUE}}
54 p[0].y = 5;
55 clang_analyzer_eval(a.y == 5); // expected-warning{{TRUE}}
56 clang_analyzer_eval(p->y == 5); // expected-warning{{TRUE}}
60 // PR13264
61 struct point { int x; int y; };
62 struct circle { struct point o; int r; };
63 struct circle get_circle(void) {
64 struct circle result;
65 result.r = 5;
66 result.o = (struct point){0, 0};
67 return result;
70 void struct_in_struct(void) {
71 struct circle c;
72 c = get_circle();
73 // This used to think c.r was undefined because c.o is a LazyCompoundVal.
74 clang_analyzer_eval(c.r == 5); // expected-warning{{TRUE}}
77 // We also test with floats because we don't model floats right now,
78 // and the original bug report used a float.
79 struct circle_f { struct point o; float r; };
80 struct circle_f get_circle_f(void) {
81 struct circle_f result;
82 result.r = 5.0;
83 result.o = (struct point){0, 0};
84 return result;
87 float struct_in_struct_f(void) {
88 struct circle_f c;
89 c = get_circle_f();
91 return c.r; // no-warning
95 int randomInt(void);
97 int testSymbolicInvalidation(int index) {
98 int vals[10];
100 vals[0] = 42;
101 clang_analyzer_eval(vals[0] == 42); // expected-warning{{TRUE}}
103 vals[index] = randomInt();
104 clang_analyzer_eval(vals[0] == 42); // expected-warning{{UNKNOWN}}
106 return vals[index]; // no-warning
109 int testConcreteInvalidation(int index) {
110 int vals[10];
112 vals[index] = 42;
113 clang_analyzer_eval(vals[index] == 42); // expected-warning{{TRUE}}
114 vals[0] = randomInt();
115 clang_analyzer_eval(vals[index] == 42); // expected-warning{{UNKNOWN}}
117 return vals[0]; // no-warning
121 typedef struct {
122 int x, y, z;
123 } S;
125 S makeS(void);
127 int testSymbolicInvalidationStruct(int index) {
128 S vals[10];
130 vals[0].x = 42;
131 clang_analyzer_eval(vals[0].x == 42); // expected-warning{{TRUE}}
133 vals[index] = makeS();
134 clang_analyzer_eval(vals[0].x == 42); // expected-warning{{UNKNOWN}}
136 return vals[index].x; // no-warning
139 int testConcreteInvalidationStruct(int index) {
140 S vals[10];
142 vals[index].x = 42;
143 clang_analyzer_eval(vals[index].x == 42); // expected-warning{{TRUE}}
144 vals[0] = makeS();
145 clang_analyzer_eval(vals[index].x == 42); // expected-warning{{UNKNOWN}}
147 return vals[0].x; // no-warning
150 typedef struct {
151 S a[5];
152 S b[5];
153 } SS;
155 int testSymbolicInvalidationDoubleStruct(int index) {
156 SS vals;
158 vals.a[0].x = 42;
159 vals.b[0].x = 42;
160 clang_analyzer_eval(vals.a[0].x == 42); // expected-warning{{TRUE}}
161 clang_analyzer_eval(vals.b[0].x == 42); // expected-warning{{TRUE}}
163 vals.a[index] = makeS();
164 clang_analyzer_eval(vals.a[0].x == 42); // expected-warning{{UNKNOWN}}
165 clang_analyzer_eval(vals.b[0].x == 42); // expected-warning{{TRUE}}
167 return vals.b[index].x; // no-warning
170 int testConcreteInvalidationDoubleStruct(int index) {
171 SS vals;
173 vals.a[index].x = 42;
174 vals.b[index].x = 42;
175 clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{TRUE}}
176 clang_analyzer_eval(vals.b[index].x == 42); // expected-warning{{TRUE}}
178 vals.a[0] = makeS();
179 clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{UNKNOWN}}
180 clang_analyzer_eval(vals.b[index].x == 42); // expected-warning{{TRUE}}
182 return vals.b[0].x; // no-warning
186 int testNonOverlappingStructFieldsSimple(void) {
187 S val;
189 val.x = 1;
190 val.y = 2;
191 clang_analyzer_eval(val.x == 1); // expected-warning{{TRUE}}
192 clang_analyzer_eval(val.y == 2); // expected-warning{{TRUE}}
194 return val.z; // expected-warning{{garbage}}
197 int testNonOverlappingStructFieldsSymbolicBase(int index, int anotherIndex) {
198 SS vals;
200 vals.a[index].x = 42;
201 vals.a[index].y = 42;
202 clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{TRUE}}
203 clang_analyzer_eval(vals.a[index].y == 42); // expected-warning{{TRUE}}
205 vals.a[anotherIndex].x = 42;
206 clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{UNKNOWN}}
207 clang_analyzer_eval(vals.a[index].y == 42); // expected-warning{{TRUE}}
209 // FIXME: False negative. No bind ever set a field 'z'.
210 return vals.a[index].z; // no-warning
213 int testStructFieldChains(int index, int anotherIndex) {
214 SS vals[4];
216 vals[index].a[0].x = 42;
217 vals[anotherIndex].a[1].y = 42;
218 clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}}
219 clang_analyzer_eval(vals[anotherIndex].a[1].y == 42); // expected-warning{{TRUE}}
221 // This doesn't affect anything in the 'a' array field.
222 vals[anotherIndex].b[1].x = 42;
223 clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}}
224 clang_analyzer_eval(vals[anotherIndex].a[1].y == 42); // expected-warning{{TRUE}}
225 clang_analyzer_eval(vals[anotherIndex].b[1].x == 42); // expected-warning{{TRUE}}
227 // This doesn't affect anything in the 'b' array field.
228 vals[index].a[anotherIndex].x = 42;
229 clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{UNKNOWN}}
230 clang_analyzer_eval(vals[anotherIndex].a[0].x == 42); // expected-warning{{UNKNOWN}}
231 clang_analyzer_eval(vals[anotherIndex].a[1].y == 42); // expected-warning{{TRUE}}
232 clang_analyzer_eval(vals[anotherIndex].b[1].x == 42); // expected-warning{{TRUE}}
234 // FIXME: False negative. No bind ever set a field 'z'.
235 return vals[index].a[0].z; // no-warning
238 int testStructFieldChainsNested(int index, int anotherIndex) {
239 SS vals[4];
241 vals[index].a[0].x = 42;
242 clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}}
244 vals[index].b[0] = makeS();
245 clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}}
247 vals[index].a[0] = makeS();
248 clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{UNKNOWN}}
250 vals[index].a[0].x = 42;
251 clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}}
253 return 0;
256 typedef struct {
257 int zoomLevel;
258 struct point center;
259 } Outer;
261 extern int test13116945(struct point x);
262 static void radar13116945(struct point centerCoordinate) {
263 Outer zoomRegion;
264 zoomRegion.zoomLevel = 0;
265 zoomRegion.center = centerCoordinate;
266 Outer r = zoomRegion;
267 test13116945(r.center); // no-warning
271 typedef struct {
272 char data[4];
273 } ShortString;
275 typedef struct {
276 ShortString str;
277 int length;
278 } ShortStringWrapper;
280 void testArrayStructCopy(void) {
281 ShortString s = { "abc" };
282 ShortString s2 = s;
283 ShortString s3 = s2;
285 clang_analyzer_eval(s3.data[0] == 'a'); // expected-warning{{TRUE}}
286 clang_analyzer_eval(s3.data[1] == 'b'); // expected-warning{{TRUE}}
287 clang_analyzer_eval(s3.data[2] == 'c'); // expected-warning{{TRUE}}
289 s3.data[0] = 'z';
290 ShortString s4 = s3;
292 clang_analyzer_eval(s4.data[0] == 'z'); // expected-warning{{TRUE}}
293 clang_analyzer_eval(s4.data[1] == 'b'); // expected-warning{{TRUE}}
294 clang_analyzer_eval(s4.data[2] == 'c'); // expected-warning{{TRUE}}
297 void testArrayStructCopyNested(void) {
298 ShortString s = { "abc" };
299 ShortString s2 = s;
301 ShortStringWrapper w = { s2, 0 };
303 clang_analyzer_eval(w.str.data[0] == 'a'); // expected-warning{{TRUE}}
304 clang_analyzer_eval(w.str.data[1] == 'b'); // expected-warning{{TRUE}}
305 clang_analyzer_eval(w.str.data[2] == 'c'); // expected-warning{{TRUE}}
306 clang_analyzer_eval(w.length == 0); // expected-warning{{TRUE}}
308 ShortStringWrapper w2 = w;
309 clang_analyzer_eval(w2.str.data[0] == 'a'); // expected-warning{{TRUE}}
310 clang_analyzer_eval(w2.str.data[1] == 'b'); // expected-warning{{TRUE}}
311 clang_analyzer_eval(w2.str.data[2] == 'c'); // expected-warning{{TRUE}}
312 clang_analyzer_eval(w2.length == 0); // expected-warning{{TRUE}}
314 ShortStringWrapper w3 = w2;
315 clang_analyzer_eval(w3.str.data[0] == 'a'); // expected-warning{{TRUE}}
316 clang_analyzer_eval(w3.str.data[1] == 'b'); // expected-warning{{TRUE}}
317 clang_analyzer_eval(w3.str.data[2] == 'c'); // expected-warning{{TRUE}}
318 clang_analyzer_eval(w3.length == 0); // expected-warning{{TRUE}}
321 // --------------------
322 // False positives
323 // --------------------
325 int testMixSymbolicAndConcrete(int index, int anotherIndex) {
326 SS vals;
328 vals.a[index].x = 42;
329 vals.a[0].y = 42;
331 // FIXME: Should be TRUE.
332 clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{UNKNOWN}}
333 // Should be TRUE; we set this explicitly.
334 clang_analyzer_eval(vals.a[0].y == 42); // expected-warning{{TRUE}}
336 vals.a[anotherIndex].y = 42;
338 // Should be UNKNOWN; we set an 'x'.
339 clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{UNKNOWN}}
340 // FIXME: Should be TRUE.
341 clang_analyzer_eval(vals.a[0].y == 42); // expected-warning{{UNKNOWN}}
343 return vals.a[0].x; // no-warning
346 void testFieldChainIsNotEnough(int index) {
347 SS vals[4];
349 vals[index].a[0].x = 42;
350 clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}}
352 vals[index].a[1] = makeS();
353 // FIXME: Should be TRUE.
354 clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{UNKNOWN}}