2017-03-04 02:02:02 +08:00
|
|
|
// RUN: %clang_analyze_cc1 -std=c++11 -analyzer-checker=debug.ExprInspection %s
|
2013-04-09 10:30:33 +08:00
|
|
|
|
|
|
|
void clang_analyzer_eval(bool);
|
|
|
|
|
|
|
|
enum class Foo {
|
|
|
|
Zero
|
|
|
|
};
|
|
|
|
|
|
|
|
bool pr15703(int x) {
|
|
|
|
return Foo::Zero == (Foo)x; // don't crash
|
|
|
|
}
|
|
|
|
|
|
|
|
void testCasting(int i) {
|
|
|
|
Foo f = static_cast<Foo>(i);
|
|
|
|
int j = static_cast<int>(f);
|
|
|
|
if (i == 0)
|
|
|
|
{
|
|
|
|
clang_analyzer_eval(f == Foo::Zero); // expected-warning{{TRUE}}
|
|
|
|
clang_analyzer_eval(j == 0); // expected-warning{{TRUE}}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
clang_analyzer_eval(f == Foo::Zero); // expected-warning{{FALSE}}
|
|
|
|
clang_analyzer_eval(j == 0); // expected-warning{{FALSE}}
|
|
|
|
}
|
|
|
|
}
|
2017-04-21 09:05:26 +08:00
|
|
|
|
|
|
|
enum class EnumBool : bool {
|
|
|
|
F = false,
|
|
|
|
T = true
|
|
|
|
};
|
|
|
|
|
|
|
|
bool testNoCrashOnSwitchEnumBool(EnumBool E) {
|
|
|
|
switch (E) {
|
|
|
|
case EnumBool::F:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2017-07-11 08:30:14 +08:00
|
|
|
|
|
|
|
bool testNoCrashOnSwitchEnumBoolConstant() {
|
|
|
|
EnumBool E = EnumBool::F;
|
|
|
|
switch (E) {
|
|
|
|
case EnumBool::F:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef __INTPTR_TYPE__ intptr_t;
|
|
|
|
bool testNoCrashOnSwitchEnumBoolConstantCastedFromNullptr() {
|
|
|
|
EnumBool E = static_cast<EnumBool>((intptr_t)nullptr);
|
|
|
|
switch (E) {
|
|
|
|
case EnumBool::F:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool testNoCrashOnSwitchEnumBoolConstantCastedFromPtr() {
|
|
|
|
int X;
|
|
|
|
intptr_t P = (intptr_t)&X;
|
|
|
|
EnumBool E = static_cast<EnumBool>(P);
|
|
|
|
switch (E) {
|
|
|
|
case EnumBool::F:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|