// RUN: %clang_analyze_cc1 -analyzer-checker=core,unix.Malloc,security.ArrayBound,debug.ExprInspection \ // RUN: -analyzer-config eagerly-assume=false -verify %s // When the checker security.ArrayBound encounters an array subscript operation // that _may be_ in bounds, it assumes that indexing _is_ in bound. This test // file validates these assumptions. void clang_analyzer_value(int); // Simple case: memory area with a static extent. extern int FiveInts[5]; void int_plus_one(int len) { (void)FiveInts[len + 1]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [-1, 3] }}} } void int_neutral(int len) { (void)FiveInts[len]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 4] }}} } void int_minus_one(int len) { (void)FiveInts[len - 1]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [1, 5] }}} } void unsigned_plus_one(unsigned len) { (void)FiveInts[len + 1]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 3] }}} } void unsigned_neutral(unsigned len) { (void)FiveInts[len]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 4] }}} } void unsigned_minus_one(unsigned len) { (void)FiveInts[len - 1]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [1, 5] }}} } void ll_plus_one(long long len) { (void)FiveInts[len + 1]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [-1, 3] }}} } void ll_neutral(long long len) { (void)FiveInts[len]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 4] }}} } void ll_minus_one(long long len) { (void)FiveInts[len - 1]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [1, 5] }}} } void ull_plus_one(unsigned long long len) { (void)FiveInts[len + 1]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 3] }}} } void ull_neutral(unsigned long long len) { (void)FiveInts[len]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 4] }}} } void ull_minus_one(unsigned long long len) { (void)FiveInts[len - 1]; // no-warning clang_analyzer_value(len); // expected-warning {{{ [1, 5] }}} } // Also try the same with a dynamically allocated memory block, because in the // past there were issues with the type/signedness of dynamic extent symbols. typedef __typeof(sizeof(int)) size_t; void *malloc(size_t); void free(void *); void dyn_int_plus_one(int len) { char *p = malloc(5); p[len + 1] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [-1, 3] }}} free(p); } void dyn_int_neutral(int len) { char *p = malloc(5); p[len] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 4] }}} free(p); } void dyn_int_minus_one(int len) { char *p = malloc(5); p[len - 1] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [1, 5] }}} free(p); } void dyn_unsigned_plus_one(unsigned len) { char *p = malloc(5); p[len + 1] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 3] }}} free(p); } void dyn_unsigned_neutral(unsigned len) { char *p = malloc(5); p[len] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 4] }}} free(p); } void dyn_unsigned_minus_one(unsigned len) { char *p = malloc(5); p[len - 1] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [1, 5] }}} free(p); } void dyn_ll_plus_one(long long len) { char *p = malloc(5); p[len + 1] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [-1, 3] }}} free(p); } void dyn_ll_neutral(long long len) { char *p = malloc(5); p[len] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 4] }}} free(p); } void dyn_ll_minus_one(long long len) { char *p = malloc(5); p[len - 1] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [1, 5] }}} free(p); } void dyn_ull_plus_one(unsigned long long len) { char *p = malloc(5); p[len + 1] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 3] }}} free(p); } void dyn_ull_neutral(unsigned long long len) { char *p = malloc(5); p[len] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [0, 4] }}} free(p); } void dyn_ull_minus_one(unsigned long long len) { char *p = malloc(5); p[len - 1] = 1; // no-warning clang_analyzer_value(len); // expected-warning {{{ [1, 5] }}} free(p); }