
Casting a pointer to a suitably large integral type by reinterpret-cast should result in the same value as by using the `__builtin_bit_cast()`. The compiler exploits this: https://godbolt.org/z/zMP3sG683 However, the analyzer does not bind the same symbolic value to these expressions, resulting in weird situations, such as failing equality checks and even results in crashes: https://godbolt.org/z/oeMP7cj8q Previously, in the `RegionStoreManager::getBinding()` even if `T` was non-null, we replaced it with `TVR->getValueType()` in case the `MR` was `TypedValueRegion`. It doesn't make much sense to auto-detect the type if the type is already given. By not doing the auto-detection, we would just do the right thing and perform the load by that type. This means that we will cast the value to that type. So, in this patch, I'm proposing to do auto-detection only if the type was null. Here is a snippet of code, annotated by the previous and new dump values. `LocAsInteger` should wrap the `SymRegion`, since we want to load the address as if it was an integer. In none of the following cases should type auto-detection be triggered, hence we should eventually reach an `evalCast()` to lazily cast the loaded value into that type. ```lang=C++ void LValueToRValueBitCast_dumps(void *p, char (*array)[8]) { clang_analyzer_dump(p); // remained: &SymRegion{reg_$0<void * p>} clang_analyzer_dump(array); // remained: {{&SymRegion{reg_$1<char (*)[8] array>} clang_analyzer_dump((unsigned long)p); // remained: {{&SymRegion{reg_$0<void * p>} [as 64 bit integer]}} clang_analyzer_dump(__builtin_bit_cast(unsigned long, p)); <--------- change #1 // previously: {{&SymRegion{reg_$0<void * p>}}} // now: {{&SymRegion{reg_$0<void * p>} [as 64 bit integer]}} clang_analyzer_dump((unsigned long)array); // remained: {{&SymRegion{reg_$1<char (*)[8] array>} [as 64 bit integer]}} clang_analyzer_dump(__builtin_bit_cast(unsigned long, array)); <--------- change #2 // previously: {{&SymRegion{reg_$1<char (*)[8] array>}}} // now: {{&SymRegion{reg_$1<char (*)[8] array>} [as 64 bit integer]}} } ``` Reviewed By: xazax.hun Differential Revision: https://reviews.llvm.org/D136603
168 lines
4.0 KiB
C++
168 lines
4.0 KiB
C++
// RUN: %clang_analyze_cc1 -Wno-unused-value -std=c++14 -verify %s -triple x86_64-pc-linux-gnu \
|
|
// RUN: -analyzer-checker=core,debug.ExprInspection,alpha.core.PointerArithm
|
|
|
|
// RUN: %clang_analyze_cc1 -Wno-unused-value -std=c++14 -verify %s -triple x86_64-pc-linux-gnu \
|
|
// RUN: -analyzer-config support-symbolic-integer-casts=true \
|
|
// RUN: -analyzer-checker=core,debug.ExprInspection,alpha.core.PointerArithm
|
|
|
|
template <typename T> void clang_analyzer_dump(T);
|
|
|
|
struct X {
|
|
int *p;
|
|
int zero;
|
|
void foo () {
|
|
reset(p - 1);
|
|
}
|
|
void reset(int *in) {
|
|
while (in != p) // Loop must be entered.
|
|
zero = 1;
|
|
}
|
|
};
|
|
|
|
int test (int *in) {
|
|
X littleX;
|
|
littleX.zero = 0;
|
|
littleX.p = in;
|
|
littleX.foo();
|
|
return 5/littleX.zero; // no-warning
|
|
}
|
|
|
|
|
|
class Base {};
|
|
class Derived : public Base {};
|
|
|
|
void checkPolymorphicUse() {
|
|
Derived d[10];
|
|
|
|
Base *p = d;
|
|
++p; // expected-warning{{Pointer arithmetic on a pointer to base class is dangerous}}
|
|
}
|
|
|
|
void checkBitCasts() {
|
|
long l;
|
|
char *p = (char*)&l;
|
|
p = p+2;
|
|
}
|
|
|
|
void checkBasicarithmetic(int i) {
|
|
int t[10];
|
|
int *p = t;
|
|
++p;
|
|
int a = 5;
|
|
p = &a;
|
|
++p; // expected-warning{{Pointer arithmetic on non-array variables relies on memory layout, which is dangerous}}
|
|
p = p + 2; // expected-warning{{}}
|
|
p = 2 + p; // expected-warning{{}}
|
|
p += 2; // expected-warning{{}}
|
|
a += p[2]; // expected-warning{{}}
|
|
p = i*0 + p;
|
|
p = p + i*0;
|
|
p += i*0;
|
|
}
|
|
|
|
void checkArithOnSymbolic(int*p) {
|
|
++p;
|
|
p = p + 2;
|
|
p = 2 + p;
|
|
p += 2;
|
|
(void)p[2];
|
|
}
|
|
|
|
struct S {
|
|
int t[10];
|
|
};
|
|
|
|
void arrayInStruct() {
|
|
S s;
|
|
int * p = s.t;
|
|
++p;
|
|
S *sp = new S;
|
|
p = sp->t;
|
|
++p;
|
|
delete sp;
|
|
}
|
|
|
|
void checkNew() {
|
|
int *p = new int;
|
|
p[1] = 1; // expected-warning{{}}
|
|
}
|
|
|
|
void InitState(int* state) {
|
|
state[1] = 1; // expected-warning{{}}
|
|
}
|
|
|
|
int* getArray(int size) {
|
|
if (size == 0)
|
|
return new int;
|
|
return new int[5];
|
|
}
|
|
|
|
void checkConditionalArray() {
|
|
int* maybeArray = getArray(0);
|
|
InitState(maybeArray);
|
|
}
|
|
|
|
void checkMultiDimansionalArray() {
|
|
int a[5][5];
|
|
*(*(a+1)+2) = 2;
|
|
}
|
|
|
|
unsigned ptrSubtractionNoCrash(char *Begin, char *End) {
|
|
auto N = End - Begin;
|
|
if (Begin)
|
|
return 0;
|
|
return N;
|
|
}
|
|
|
|
// Bug 34309
|
|
bool ptrAsIntegerSubtractionNoCrash(__UINTPTR_TYPE__ x, char *p) {
|
|
__UINTPTR_TYPE__ y = (__UINTPTR_TYPE__)p - 1;
|
|
return y == x;
|
|
}
|
|
|
|
// Bug 34374
|
|
bool integerAsPtrSubtractionNoCrash(char *p, __UINTPTR_TYPE__ m) {
|
|
auto n = p - reinterpret_cast<char*>((__UINTPTR_TYPE__)1);
|
|
return n == m;
|
|
}
|
|
|
|
namespace Bug_55934 {
|
|
struct header {
|
|
unsigned a : 1;
|
|
unsigned b : 1;
|
|
};
|
|
struct parse_t {
|
|
unsigned bits0 : 1;
|
|
unsigned bits2 : 2; // <-- header
|
|
unsigned bits4 : 4;
|
|
};
|
|
int parse(parse_t *p) {
|
|
unsigned copy = p->bits2;
|
|
clang_analyzer_dump(copy);
|
|
// expected-warning@-1 {{reg_$1<unsigned int Element{SymRegion{reg_$0<parse_t * p>},0 S64b,struct Bug_55934::parse_t}.bits2>}}
|
|
header *bits = (header *)©
|
|
clang_analyzer_dump(bits->b);
|
|
// expected-warning@-1 {{derived_$2{reg_$1<unsigned int Element{SymRegion{reg_$0<parse_t * p>},0 S64b,struct Bug_55934::parse_t}.bits2>,Element{copy,0 S64b,struct Bug_55934::header}.b}}}
|
|
return bits->b; // no-warning
|
|
}
|
|
} // namespace Bug_55934
|
|
|
|
void LValueToRValueBitCast_dumps(void *p, char (*array)[8]) {
|
|
clang_analyzer_dump(p);
|
|
clang_analyzer_dump(array);
|
|
// expected-warning@-2 {{&SymRegion{reg_$0<void * p>}}}
|
|
// expected-warning@-2 {{&SymRegion{reg_$1<char (*)[8] array>}}}
|
|
clang_analyzer_dump((unsigned long)p);
|
|
clang_analyzer_dump(__builtin_bit_cast(unsigned long, p));
|
|
// expected-warning@-2 {{&SymRegion{reg_$0<void * p>} [as 64 bit integer]}}
|
|
// expected-warning@-2 {{&SymRegion{reg_$0<void * p>} [as 64 bit integer]}}
|
|
clang_analyzer_dump((unsigned long)array);
|
|
clang_analyzer_dump(__builtin_bit_cast(unsigned long, array));
|
|
// expected-warning@-2 {{&SymRegion{reg_$1<char (*)[8] array>} [as 64 bit integer]}}
|
|
// expected-warning@-2 {{&SymRegion{reg_$1<char (*)[8] array>} [as 64 bit integer]}}
|
|
}
|
|
|
|
unsigned long ptr_arithmetic(void *p) {
|
|
return __builtin_bit_cast(unsigned long, p) + 1; // no-crash
|
|
}
|