256 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			256 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // RUN: %clang_cc1 -verify %s
 | |
| 
 | |
| int foo() {
 | |
|   int x[2]; // expected-note 4 {{array 'x' declared here}}
 | |
|   int y[2]; // expected-note 2 {{array 'y' declared here}}
 | |
|   int z[1]; // expected-note {{array 'z' declared here}}
 | |
|   int w[1][1]; // expected-note {{array 'w' declared here}}
 | |
|   int v[1][1][1]; // expected-note {{array 'v' declared here}}
 | |
|   int *p = &y[2]; // no-warning
 | |
|   (void) sizeof(x[2]); // no-warning
 | |
|   y[2] = 2; // expected-warning {{array index 2 is past the end of the array (which contains 2 elements)}}
 | |
|   z[1] = 'x'; // expected-warning {{array index 1 is past the end of the array (which contains 1 element)}}
 | |
|   w[0][2] = 0; // expected-warning {{array index 2 is past the end of the array (which contains 1 element)}}
 | |
|   v[0][0][2] = 0; // expected-warning {{array index 2 is past the end of the array (which contains 1 element)}}
 | |
|   return x[2] +  // expected-warning {{array index 2 is past the end of the array (which contains 2 elements)}}
 | |
|          y[-1] + // expected-warning {{array index -1 is before the beginning of the array}}
 | |
|          x[sizeof(x)] +  // expected-warning {{array index 8 is past the end of the array (which contains 2 elements)}}
 | |
|          x[sizeof(x) / sizeof(x[0])] +  // expected-warning {{array index 2 is past the end of the array (which contains 2 elements)}}
 | |
|          x[sizeof(x) / sizeof(x[0]) - 1] + // no-warning
 | |
|          x[sizeof(x[2])]; // expected-warning {{array index 4 is past the end of the array (which contains 2 elements)}}
 | |
| }
 | |
| 
 | |
| // This code example tests that -Warray-bounds works with arrays that
 | |
| // are template parameters.
 | |
| template <char *sz> class Qux {
 | |
|   bool test() { return sz[0] == 'a'; }
 | |
| };
 | |
| 
 | |
| void f1(int a[1]) {
 | |
|   int val = a[3]; // no warning for function argumnet
 | |
| }
 | |
| 
 | |
| void f2(const int (&a)[2]) { // expected-note {{declared here}}
 | |
|   int val = a[3];  // expected-warning {{array index 3 is past the end of the array (which contains 2 elements)}}
 | |
| }
 | |
| 
 | |
| void test() {
 | |
|   struct {
 | |
|     int a[0];
 | |
|   } s2;
 | |
|   s2.a[3] = 0; // no warning for 0-sized array
 | |
| 
 | |
|   union {
 | |
|     short a[2]; // expected-note 4 {{declared here}}
 | |
|     char c[4];
 | |
|   } u;
 | |
|   u.a[3] = 1; // expected-warning {{array index 3 is past the end of the array (which contains 2 elements)}}
 | |
|   u.c[3] = 1; // no warning
 | |
|   short *p = &u.a[2]; // no warning
 | |
|   p = &u.a[3]; // expected-warning {{array index 3 is past the end of the array (which contains 2 elements)}}
 | |
|   *(&u.a[2]) = 1; // expected-warning {{array index 2 is past the end of the array (which contains 2 elements)}}
 | |
|   *(&u.a[3]) = 1; // expected-warning {{array index 3 is past the end of the array (which contains 2 elements)}}
 | |
|   *(&u.c[3]) = 1; // no warning
 | |
| 
 | |
|   const int const_subscript = 3;
 | |
|   int array[2]; // expected-note {{declared here}}
 | |
|   array[const_subscript] = 0;  // expected-warning {{array index 3 is past the end of the array (which contains 2 elements)}}
 | |
| 
 | |
|   int *ptr;
 | |
|   ptr[3] = 0; // no warning for pointer references
 | |
|   int array2[] = { 0, 1, 2 }; // expected-note 2 {{declared here}}
 | |
| 
 | |
|   array2[3] = 0; // expected-warning {{array index 3 is past the end of the array (which contains 3 elements)}}
 | |
|   array2[2+2] = 0; // expected-warning {{array index 4 is past the end of the array (which contains 3 elements)}}
 | |
| 
 | |
|   const char *str1 = "foo";
 | |
|   char c1 = str1[5]; // no warning for pointers
 | |
| 
 | |
|   const char str2[] = "foo"; // expected-note {{declared here}}
 | |
|   char c2 = str2[5]; // expected-warning {{array index 5 is past the end of the array (which contains 4 elements)}}
 | |
| 
 | |
|   int (*array_ptr)[2];
 | |
|   (*array_ptr)[3] = 1; // expected-warning {{array index 3 is past the end of the array (which contains 2 elements)}}
 | |
| }
 | |
| 
 | |
| template <int I> struct S {
 | |
|   char arr[I]; // expected-note 3 {{declared here}}
 | |
| };
 | |
| template <int I> void f() {
 | |
|   S<3> s;
 | |
|   s.arr[4] = 0; // expected-warning 2 {{array index 4 is past the end of the array (which contains 3 elements)}}
 | |
|   s.arr[I] = 0; // expected-warning {{array index 5 is past the end of the array (which contains 3 elements)}}
 | |
| }
 | |
| 
 | |
| void test_templates() {
 | |
|   f<5>(); // expected-note {{in instantiation}}
 | |
| }
 | |
| 
 | |
| #define SIZE 10
 | |
| #define ARR_IN_MACRO(flag, arr, idx) flag ? arr[idx] : 1
 | |
| 
 | |
| int test_no_warn_macro_unreachable() {
 | |
|   int arr[SIZE]; // expected-note {{array 'arr' declared here}}
 | |
|   return ARR_IN_MACRO(0, arr, SIZE) + // no-warning
 | |
|          ARR_IN_MACRO(1, arr, SIZE); // expected-warning{{array index 10 is past the end of the array (which contains 10 elements)}}
 | |
| }
 | |
| 
 | |
| // This exhibited an assertion failure for a 32-bit build of Clang.
 | |
| int test_pr9240() {
 | |
|   short array[100]; // expected-note {{array 'array' declared here}}
 | |
|   return array[(unsigned long long) 100]; // expected-warning {{array index 100 is past the end of the array (which contains 100 elements)}}
 | |
| }
 | |
| 
 | |
| // PR 9284 - a template parameter can cause an array bounds access to be
 | |
| // infeasible.
 | |
| template <bool extendArray>
 | |
| void pr9284() {
 | |
|     int arr[3 + (extendArray ? 1 : 0)];
 | |
| 
 | |
|     if (extendArray)
 | |
|         arr[3] = 42; // no-warning
 | |
| }
 | |
| 
 | |
| template <bool extendArray>
 | |
| void pr9284b() {
 | |
|     int arr[3 + (extendArray ? 1 : 0)]; // expected-note {{array 'arr' declared here}}
 | |
| 
 | |
|     if (!extendArray)
 | |
|         arr[3] = 42; // expected-warning{{array index 3 is past the end of the array (which contains 3 elements)}}
 | |
| }
 | |
| 
 | |
| void test_pr9284() {
 | |
|     pr9284<true>();
 | |
|     pr9284<false>();
 | |
|     pr9284b<true>();
 | |
|     pr9284b<false>(); // expected-note{{in instantiation of function template specialization 'pr9284b<false>' requested here}}
 | |
| }
 | |
| 
 | |
| int test_pr9296() {
 | |
|     int array[2];
 | |
|     return array[true]; // no-warning
 | |
| }
 | |
| 
 | |
| int test_sizeof_as_condition(int flag) {
 | |
|   int arr[2] = { 0, 0 }; // expected-note {{array 'arr' declared here}}
 | |
|   if (flag) 
 | |
|     return sizeof(char) != sizeof(char) ? arr[2] : arr[1];
 | |
|   return sizeof(char) == sizeof(char) ? arr[2] : arr[1]; // expected-warning {{array index 2 is past the end of the array (which contains 2 elements)}}
 | |
| }
 | |
| 
 | |
| void test_switch() {
 | |
|   switch (4) {
 | |
|     case 1: {
 | |
|       int arr[2];
 | |
|       arr[2] = 1; // no-warning
 | |
|       break;
 | |
|     }
 | |
|     case 4: {
 | |
|       int arr[2]; // expected-note {{array 'arr' declared here}}
 | |
|       arr[2] = 1; // expected-warning {{array index 2 is past the end of the array (which contains 2 elements)}}
 | |
|       break;
 | |
|     }
 | |
|     default: {
 | |
|       int arr[2];
 | |
|       arr[2] = 1; // no-warning
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Test nested switch statements.
 | |
| enum enumA { enumA_A, enumA_B, enumA_C, enumA_D, enumA_E };
 | |
| enum enumB { enumB_X, enumB_Y, enumB_Z };
 | |
| static enum enumB myVal = enumB_X;
 | |
| void test_nested_switch() {
 | |
|   switch (enumA_E) { // expected-warning {{no case matching constant}}
 | |
|     switch (myVal) { // expected-warning {{enumeration values 'enumB_X' and 'enumB_Z' not handled in switch}}
 | |
|       case enumB_Y: ;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Test that if all the values of an enum covered, that the 'default' branch
 | |
| // is unreachable.
 | |
| enum Values { A, B, C, D };
 | |
| void test_all_enums_covered(enum Values v) {
 | |
|   int x[2];
 | |
|   switch (v) {
 | |
|   case A: return;
 | |
|   case B: return;
 | |
|   case C: return;
 | |
|   case D: return;
 | |
|   }
 | |
|   x[2] = 0; // no-warning
 | |
| }
 | |
| 
 | |
| namespace tailpad {
 | |
|   struct foo {
 | |
|     char c1[1]; // expected-note {{declared here}}
 | |
|     int x;
 | |
|     char c2[1];
 | |
|   };
 | |
| 
 | |
|   class baz {
 | |
|    public:
 | |
|     char c1[1]; // expected-note {{declared here}}
 | |
|     int x;
 | |
|     char c2[1];
 | |
|   };
 | |
| 
 | |
|   char bar(struct foo *F, baz *B) {
 | |
|     return F->c1[3] + // expected-warning {{array index 3 is past the end of the array (which contains 1 element)}}
 | |
|            F->c2[3] + // no warning, foo could have tail padding allocated.
 | |
|            B->c1[3] + // expected-warning {{array index 3 is past the end of the array (which contains 1 element)}}
 | |
|            B->c2[3]; // no warning, baz could have tail padding allocated.
 | |
|   }
 | |
| }
 | |
| 
 | |
| namespace metaprogramming {
 | |
| #define ONE 1
 | |
|   struct foo { char c[ONE]; }; // expected-note {{declared here}}
 | |
|   template <int N> struct bar { char c[N]; }; // expected-note {{declared here}}
 | |
| 
 | |
|   char test(foo *F, bar<1> *B) {
 | |
|     return F->c[3] + // expected-warning {{array index 3 is past the end of the array (which contains 1 element)}}
 | |
|            B->c[3]; // expected-warning {{array index 3 is past the end of the array (which contains 1 element)}}
 | |
|   }
 | |
| }
 | |
| 
 | |
| void bar(int x) {}
 | |
| int test_more() {
 | |
|   int foo[5]; // expected-note 5 {{array 'foo' declared here}}
 | |
|   bar(foo[5]); // expected-warning {{array index 5 is past the end of the array (which contains 5 elements)}}
 | |
|   ++foo[5]; // expected-warning {{array index 5 is past the end of the array (which contains 5 elements)}}
 | |
|   if (foo[6]) // expected-warning {{array index 6 is past the end of the array (which contains 5 elements)}}
 | |
|     return --foo[6]; // expected-warning {{array index 6 is past the end of the array (which contains 5 elements)}}
 | |
|   else
 | |
|     return foo[5]; // expected-warning {{array index 5 is past the end of the array (which contains 5 elements)}}
 | |
| }
 | |
| 
 | |
| void test_pr10771() {
 | |
|     double foo[4096];  // expected-note {{array 'foo' declared here}}
 | |
| 
 | |
|     ((char*)foo)[sizeof(foo) - 1] = '\0';  // no-warning
 | |
|     *(((char*)foo) + sizeof(foo) - 1) = '\0';  // no-warning
 | |
| 
 | |
|     ((char*)foo)[sizeof(foo)] = '\0';  // expected-warning {{array index 32768 is past the end of the array (which contains 32768 elements)}}
 | |
| 
 | |
|     // TODO: This should probably warn, too.
 | |
|     *(((char*)foo) + sizeof(foo)) = '\0';  // no-warning
 | |
| }
 | |
| 
 | |
| int test_pr11007_aux(const char * restrict, ...);
 | |
|   
 | |
| // Test checking with varargs.
 | |
| void test_pr11007() {
 | |
|   double a[5]; // expected-note {{array 'a' declared here}}
 | |
|   test_pr11007_aux("foo", a[1000]); // expected-warning {{array index 1000 is past the end of the array}}
 | |
| }
 | |
| 
 | |
| void test_rdar10916006(void)
 | |
| {
 | |
| 	int a[128]; // expected-note {{array 'a' declared here}}
 | |
| 	a[(unsigned char)'\xA1'] = 1; // expected-warning {{array index 161 is past the end of the array}}
 | |
| }
 | 
