// RUN: %clang_cc1 -analyze -analyzer-checker=core,alpha.core,debug.ExprInspection -analyzer-constraints=range -verify %s void clang_analyzer_eval(int); int string_literal_init() { char a[] = "abc"; char b[2] = "abc"; // expected-warning{{too long}} char c[5] = "abc"; clang_analyzer_eval(a[1] == 'b'); // expected-warning{{TRUE}} clang_analyzer_eval(b[1] == 'b'); // expected-warning{{TRUE}} clang_analyzer_eval(c[1] == 'b'); // expected-warning{{TRUE}} clang_analyzer_eval(a[3] == 0); // expected-warning{{TRUE}} clang_analyzer_eval(c[3] == 0); // expected-warning{{TRUE}} clang_analyzer_eval(c[4] == 0); // expected-warning{{TRUE}} return 42; } void nested_compound_literals(int rad) { 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}} {0.831, 0.45}, {0.924, 0.617}, {0.98, 0.805}}; // expected-warning 6 {{implicit conversion from 'double' to 'int' changes value from}} int a; for (a = 0; a < 6; ++a) { vec[a][0] *= rad; // no-warning vec[a][1] *= rad; // no-warning } } void nested_compound_literals_float(float rad) { float vec[6][2] = {{0.195, 0.02}, {0.383, 0.067}, {0.55, 0.169}, {0.831, 0.45}, {0.924, 0.617}, {0.98, 0.805}}; int a; for (a = 0; a < 6; ++a) { vec[a][0] *= rad; // no-warning vec[a][1] *= rad; // no-warning } } void struct_as_array() { struct simple { int x; int y; }; struct simple a; struct simple *p = &a; p->x = 5; clang_analyzer_eval(a.x == 5); // expected-warning{{TRUE}} clang_analyzer_eval(p[0].x == 5); // expected-warning{{TRUE}} p[0].y = 5; clang_analyzer_eval(a.y == 5); // expected-warning{{TRUE}} clang_analyzer_eval(p->y == 5); // expected-warning{{TRUE}} } // PR13264 / <rdar://problem/11802440> struct point { int x; int y; }; struct circle { struct point o; int r; }; struct circle get_circle() { struct circle result; result.r = 5; result.o = (struct point){0, 0}; return result; } void struct_in_struct() { struct circle c; c = get_circle(); // This used to think c.r was undefined because c.o is a LazyCompoundVal. clang_analyzer_eval(c.r == 5); // expected-warning{{TRUE}} } // We also test with floats because we don't model floats right now, // and the original bug report used a float. struct circle_f { struct point o; float r; }; struct circle_f get_circle_f() { struct circle_f result; result.r = 5.0; result.o = (struct point){0, 0}; return result; } float struct_in_struct_f() { struct circle_f c; c = get_circle_f(); return c.r; // no-warning } int randomInt(); int testSymbolicInvalidation(int index) { int vals[10]; vals[0] = 42; clang_analyzer_eval(vals[0] == 42); // expected-warning{{TRUE}} vals[index] = randomInt(); clang_analyzer_eval(vals[0] == 42); // expected-warning{{UNKNOWN}} return vals[index]; // no-warning } int testConcreteInvalidation(int index) { int vals[10]; vals[index] = 42; clang_analyzer_eval(vals[index] == 42); // expected-warning{{TRUE}} vals[0] = randomInt(); clang_analyzer_eval(vals[index] == 42); // expected-warning{{UNKNOWN}} return vals[0]; // no-warning } typedef struct { int x, y, z; } S; S makeS(); int testSymbolicInvalidationStruct(int index) { S vals[10]; vals[0].x = 42; clang_analyzer_eval(vals[0].x == 42); // expected-warning{{TRUE}} vals[index] = makeS(); clang_analyzer_eval(vals[0].x == 42); // expected-warning{{UNKNOWN}} return vals[index].x; // no-warning } int testConcreteInvalidationStruct(int index) { S vals[10]; vals[index].x = 42; clang_analyzer_eval(vals[index].x == 42); // expected-warning{{TRUE}} vals[0] = makeS(); clang_analyzer_eval(vals[index].x == 42); // expected-warning{{UNKNOWN}} return vals[0].x; // no-warning } typedef struct { S a[5]; S b[5]; } SS; int testSymbolicInvalidationDoubleStruct(int index) { SS vals; vals.a[0].x = 42; vals.b[0].x = 42; clang_analyzer_eval(vals.a[0].x == 42); // expected-warning{{TRUE}} clang_analyzer_eval(vals.b[0].x == 42); // expected-warning{{TRUE}} vals.a[index] = makeS(); clang_analyzer_eval(vals.a[0].x == 42); // expected-warning{{UNKNOWN}} clang_analyzer_eval(vals.b[0].x == 42); // expected-warning{{TRUE}} return vals.b[index].x; // no-warning } int testConcreteInvalidationDoubleStruct(int index) { SS vals; vals.a[index].x = 42; vals.b[index].x = 42; clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{TRUE}} clang_analyzer_eval(vals.b[index].x == 42); // expected-warning{{TRUE}} vals.a[0] = makeS(); clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{UNKNOWN}} clang_analyzer_eval(vals.b[index].x == 42); // expected-warning{{TRUE}} return vals.b[0].x; // no-warning } int testNonOverlappingStructFieldsSimple() { S val; val.x = 1; val.y = 2; clang_analyzer_eval(val.x == 1); // expected-warning{{TRUE}} clang_analyzer_eval(val.y == 2); // expected-warning{{TRUE}} return val.z; // expected-warning{{garbage}} } int testNonOverlappingStructFieldsSymbolicBase(int index, int anotherIndex) { SS vals; vals.a[index].x = 42; vals.a[index].y = 42; clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{TRUE}} clang_analyzer_eval(vals.a[index].y == 42); // expected-warning{{TRUE}} vals.a[anotherIndex].x = 42; clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{UNKNOWN}} clang_analyzer_eval(vals.a[index].y == 42); // expected-warning{{TRUE}} // FIXME: False negative. No bind ever set a field 'z'. return vals.a[index].z; // no-warning } int testStructFieldChains(int index, int anotherIndex) { SS vals[4]; vals[index].a[0].x = 42; vals[anotherIndex].a[1].y = 42; clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}} clang_analyzer_eval(vals[anotherIndex].a[1].y == 42); // expected-warning{{TRUE}} // This doesn't affect anything in the 'a' array field. vals[anotherIndex].b[1].x = 42; clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}} clang_analyzer_eval(vals[anotherIndex].a[1].y == 42); // expected-warning{{TRUE}} clang_analyzer_eval(vals[anotherIndex].b[1].x == 42); // expected-warning{{TRUE}} // This doesn't affect anything in the 'b' array field. vals[index].a[anotherIndex].x = 42; clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{UNKNOWN}} clang_analyzer_eval(vals[anotherIndex].a[0].x == 42); // expected-warning{{UNKNOWN}} clang_analyzer_eval(vals[anotherIndex].a[1].y == 42); // expected-warning{{TRUE}} clang_analyzer_eval(vals[anotherIndex].b[1].x == 42); // expected-warning{{TRUE}} // FIXME: False negative. No bind ever set a field 'z'. return vals[index].a[0].z; // no-warning } int testStructFieldChainsNested(int index, int anotherIndex) { SS vals[4]; vals[index].a[0].x = 42; clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}} vals[index].b[0] = makeS(); clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}} vals[index].a[0] = makeS(); clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{UNKNOWN}} vals[index].a[0].x = 42; clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}} return 0; } typedef struct { int zoomLevel; struct point center; } Outer; extern int test13116945(struct point x); static void radar13116945(struct point centerCoordinate) { Outer zoomRegion; zoomRegion.zoomLevel = 0; zoomRegion.center = centerCoordinate; Outer r = zoomRegion; test13116945(r.center); // no-warning } typedef struct { char data[4]; } ShortString; typedef struct { ShortString str; int length; } ShortStringWrapper; void testArrayStructCopy() { ShortString s = { "abc" }; ShortString s2 = s; ShortString s3 = s2; clang_analyzer_eval(s3.data[0] == 'a'); // expected-warning{{TRUE}} clang_analyzer_eval(s3.data[1] == 'b'); // expected-warning{{TRUE}} clang_analyzer_eval(s3.data[2] == 'c'); // expected-warning{{TRUE}} s3.data[0] = 'z'; ShortString s4 = s3; clang_analyzer_eval(s4.data[0] == 'z'); // expected-warning{{TRUE}} clang_analyzer_eval(s4.data[1] == 'b'); // expected-warning{{TRUE}} clang_analyzer_eval(s4.data[2] == 'c'); // expected-warning{{TRUE}} } void testArrayStructCopyNested() { ShortString s = { "abc" }; ShortString s2 = s; ShortStringWrapper w = { s2, 0 }; clang_analyzer_eval(w.str.data[0] == 'a'); // expected-warning{{TRUE}} clang_analyzer_eval(w.str.data[1] == 'b'); // expected-warning{{TRUE}} clang_analyzer_eval(w.str.data[2] == 'c'); // expected-warning{{TRUE}} clang_analyzer_eval(w.length == 0); // expected-warning{{TRUE}} ShortStringWrapper w2 = w; clang_analyzer_eval(w2.str.data[0] == 'a'); // expected-warning{{TRUE}} clang_analyzer_eval(w2.str.data[1] == 'b'); // expected-warning{{TRUE}} clang_analyzer_eval(w2.str.data[2] == 'c'); // expected-warning{{TRUE}} clang_analyzer_eval(w2.length == 0); // expected-warning{{TRUE}} ShortStringWrapper w3 = w2; clang_analyzer_eval(w3.str.data[0] == 'a'); // expected-warning{{TRUE}} clang_analyzer_eval(w3.str.data[1] == 'b'); // expected-warning{{TRUE}} clang_analyzer_eval(w3.str.data[2] == 'c'); // expected-warning{{TRUE}} clang_analyzer_eval(w3.length == 0); // expected-warning{{TRUE}} } // -------------------- // False positives // -------------------- int testMixSymbolicAndConcrete(int index, int anotherIndex) { SS vals; vals.a[index].x = 42; vals.a[0].y = 42; // FIXME: Should be TRUE. clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{UNKNOWN}} // Should be TRUE; we set this explicitly. clang_analyzer_eval(vals.a[0].y == 42); // expected-warning{{TRUE}} vals.a[anotherIndex].y = 42; // Should be UNKNOWN; we set an 'x'. clang_analyzer_eval(vals.a[index].x == 42); // expected-warning{{UNKNOWN}} // FIXME: Should be TRUE. clang_analyzer_eval(vals.a[0].y == 42); // expected-warning{{UNKNOWN}} return vals.a[0].x; // no-warning } void testFieldChainIsNotEnough(int index) { SS vals[4]; vals[index].a[0].x = 42; clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{TRUE}} vals[index].a[1] = makeS(); // FIXME: Should be TRUE. clang_analyzer_eval(vals[index].a[0].x == 42); // expected-warning{{UNKNOWN}} }