blob: f20159da02997e8dc1b7c83c3ec9000855436146 [file] [log] [blame]
// 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);
}