2018-10-04 06:48:00 +08:00
|
|
|
// RUN: %clang_analyze_cc1 -std=c++14 -analyzer-checker=core,debug.ExprInspection -analyzer-store=region -verify %s
|
2018-08-30 06:43:31 +08:00
|
|
|
|
|
|
|
void clang_analyzer_eval(bool);
|
2013-09-19 02:58:58 +08:00
|
|
|
|
|
|
|
bool PR14634(int x) {
|
|
|
|
double y = (double)x;
|
|
|
|
return !y;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PR14634_implicit(int x) {
|
|
|
|
double y = (double)x;
|
|
|
|
return y;
|
|
|
|
}
|
2013-12-20 06:32:39 +08:00
|
|
|
|
|
|
|
void intAsBoolAsSwitchCondition(int c) {
|
2013-12-20 07:05:40 +08:00
|
|
|
switch ((bool)c) { // expected-warning {{switch condition has boolean value}}
|
2013-12-20 06:32:39 +08:00
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
}
|
2013-12-20 07:05:40 +08:00
|
|
|
|
|
|
|
switch ((int)(bool)c) { // no-warning
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
}
|
2013-12-20 06:32:39 +08:00
|
|
|
}
|
2018-05-05 05:39:25 +08:00
|
|
|
|
|
|
|
int *&castToIntPtrLValueRef(char *p) {
|
|
|
|
return (int *&)*(int *)p;
|
|
|
|
}
|
|
|
|
bool testCastToIntPtrLValueRef(char *p, int *s) {
|
|
|
|
return castToIntPtrLValueRef(p) != s; // no-crash
|
|
|
|
}
|
|
|
|
|
|
|
|
int *&&castToIntPtrRValueRef(char *p) {
|
|
|
|
return (int *&&)*(int *)p;
|
|
|
|
}
|
|
|
|
bool testCastToIntPtrRValueRef(char *p, int *s) {
|
|
|
|
return castToIntPtrRValueRef(p) != s; // no-crash
|
|
|
|
}
|
2018-07-17 08:42:35 +08:00
|
|
|
|
|
|
|
bool retrievePointerFromBoolean(int *p) {
|
|
|
|
bool q;
|
|
|
|
*reinterpret_cast<int **>(&q) = p;
|
|
|
|
return q;
|
|
|
|
}
|
2018-08-30 06:43:31 +08:00
|
|
|
|
|
|
|
namespace base_to_derived {
|
|
|
|
struct A {};
|
|
|
|
struct B : public A{};
|
|
|
|
|
|
|
|
void foo(A* a) {
|
|
|
|
B* b = (B* ) a;
|
|
|
|
A* a2 = (A *) b;
|
|
|
|
clang_analyzer_eval(a2 == a); // expected-warning{{TRUE}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace base_to_derived_double_inheritance {
|
|
|
|
struct A {
|
|
|
|
int x;
|
|
|
|
};
|
|
|
|
struct B {
|
|
|
|
int y;
|
|
|
|
};
|
|
|
|
struct C : A, B {};
|
|
|
|
|
|
|
|
void foo(B *b) {
|
|
|
|
C *c = (C *)b;
|
|
|
|
b->y = 1;
|
|
|
|
clang_analyzer_eval(c->x); // expected-warning{{UNKNOWN}}
|
|
|
|
clang_analyzer_eval(c->y); // expected-warning{{TRUE}}
|
|
|
|
}
|
2018-09-26 08:17:14 +08:00
|
|
|
} // namespace base_to_derived_double_inheritance
|
|
|
|
|
|
|
|
namespace base_to_derived_opaque_class {
|
|
|
|
class NotInt {
|
|
|
|
public:
|
|
|
|
operator int() { return !x; } // no-crash
|
|
|
|
int x;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct Opaque *OpaqueRef;
|
|
|
|
typedef void *VeryOpaqueRef;
|
|
|
|
|
|
|
|
class Transparent {
|
|
|
|
public:
|
|
|
|
int getNotInt() { return NI; }
|
|
|
|
NotInt NI;
|
|
|
|
};
|
|
|
|
|
|
|
|
class SubTransparent : public Transparent {};
|
|
|
|
|
|
|
|
SubTransparent *castToDerived(Transparent *TRef) {
|
|
|
|
return (SubTransparent *)TRef;
|
2018-08-30 06:43:31 +08:00
|
|
|
}
|
|
|
|
|
2018-09-26 08:17:14 +08:00
|
|
|
void foo(OpaqueRef ORef) {
|
|
|
|
castToDerived(reinterpret_cast<Transparent *>(ORef))->getNotInt();
|
|
|
|
}
|
|
|
|
|
|
|
|
void foo(VeryOpaqueRef ORef) {
|
|
|
|
castToDerived(reinterpret_cast<Transparent *>(ORef))->getNotInt();
|
|
|
|
}
|
|
|
|
} // namespace base_to_derived_opaque_class
|
2018-12-22 10:06:51 +08:00
|
|
|
|
|
|
|
namespace bool_to_nullptr {
|
|
|
|
struct S {
|
|
|
|
int *a[1];
|
|
|
|
bool b;
|
|
|
|
};
|
|
|
|
void foo(S s) {
|
|
|
|
s.b = true;
|
|
|
|
for (int i = 0; i < 2; ++i)
|
|
|
|
(void)(s.a[i] != nullptr); // no-crash
|
|
|
|
}
|
|
|
|
} // namespace bool_to_nullptr
|