forked from OSchip/llvm-project
[analyzer] Improve modeling of static initializers.
Conversions between unrelated pointer types (e.g. char * and void *) involve bitcasts which were not properly modeled in case of static initializers. The patch fixes this problem. The problem was originally spotted by Artem Dergachev. Patched by Yuri Gribov! Differential Revision: http://reviews.llvm.org/D14652 llvm-svn: 253532
This commit is contained in:
parent
d4129b47d0
commit
61fcb521fa
|
@ -275,11 +275,17 @@ Optional<SVal> SValBuilder::getConstantVal(const Expr *E) {
|
||||||
|
|
||||||
case Stmt::ImplicitCastExprClass: {
|
case Stmt::ImplicitCastExprClass: {
|
||||||
const CastExpr *CE = cast<CastExpr>(E);
|
const CastExpr *CE = cast<CastExpr>(E);
|
||||||
if (CE->getCastKind() == CK_ArrayToPointerDecay) {
|
switch (CE->getCastKind()) {
|
||||||
Optional<SVal> ArrayVal = getConstantVal(CE->getSubExpr());
|
default:
|
||||||
if (!ArrayVal)
|
break;
|
||||||
|
case CK_ArrayToPointerDecay:
|
||||||
|
case CK_BitCast: {
|
||||||
|
const Expr *SE = CE->getSubExpr();
|
||||||
|
Optional<SVal> Val = getConstantVal(SE);
|
||||||
|
if (!Val)
|
||||||
return None;
|
return None;
|
||||||
return evalCast(*ArrayVal, CE->getType(), CE->getSubExpr()->getType());
|
return evalCast(*Val, CE->getType(), SE->getType());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// FALLTHROUGH
|
// FALLTHROUGH
|
||||||
}
|
}
|
||||||
|
|
|
@ -275,7 +275,7 @@ namespace DefaultArgs {
|
||||||
|
|
||||||
clang_analyzer_eval(defaultReferenceZero(1) == -1); // expected-warning{{TRUE}}
|
clang_analyzer_eval(defaultReferenceZero(1) == -1); // expected-warning{{TRUE}}
|
||||||
clang_analyzer_eval(defaultReferenceZero() == 0); // expected-warning{{TRUE}}
|
clang_analyzer_eval(defaultReferenceZero() == 0); // expected-warning{{TRUE}}
|
||||||
}
|
}
|
||||||
|
|
||||||
double defaultFloatReference(const double &i = 42) {
|
double defaultFloatReference(const double &i = 42) {
|
||||||
return -i;
|
return -i;
|
||||||
|
@ -300,6 +300,13 @@ namespace DefaultArgs {
|
||||||
clang_analyzer_eval(defaultString("xyz") == 'y'); // expected-warning{{TRUE}}
|
clang_analyzer_eval(defaultString("xyz") == 'y'); // expected-warning{{TRUE}}
|
||||||
clang_analyzer_eval(defaultString() == 'b'); // expected-warning{{TRUE}}
|
clang_analyzer_eval(defaultString() == 'b'); // expected-warning{{TRUE}}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const void * const void_string = "abc";
|
||||||
|
|
||||||
|
void testBitcastedString() {
|
||||||
|
clang_analyzer_eval(0 != void_string); // expected-warning{{TRUE}}
|
||||||
|
clang_analyzer_eval('b' == ((char *)void_string)[1]); // expected-warning{{TRUE}}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace OperatorNew {
|
namespace OperatorNew {
|
||||||
|
|
Loading…
Reference in New Issue