123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253 |
- // 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)}}
- }
- // FIXME: we should see the next note only 3 times and the following warning once, not twice
- // since it is independent of the template parameter 'I'.
- template <int I> struct S {
- char arr[I]; // expected-note 4 {{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)}} \
- expected-warning {{array index 3 is past the end of the array (which contains 3 elements)}}
- }
- void test_templates() {
- f<5>(); // expected-note {{in instantiation}}
- f<3>(); // 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}}
- }
|