forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			356 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			356 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
| // 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}}
 | |
| }
 |