forked from OSchip/llvm-project
UndefBranchChecker: more bug reporter helper information emit.
llvm-svn: 89643
This commit is contained in:
parent
39dbfd3c58
commit
5f76620b53
|
@ -204,96 +204,6 @@ public:
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
class VISIBILITY_HIDDEN ArgReport : public BuiltinBugReport {
|
|
||||||
const Stmt *Arg;
|
|
||||||
public:
|
|
||||||
ArgReport(BugType& bt, const char* desc, ExplodedNode *n,
|
|
||||||
const Stmt *arg)
|
|
||||||
: BuiltinBugReport(bt, desc, n), Arg(arg) {}
|
|
||||||
|
|
||||||
ArgReport(BugType& bt, const char *shortDesc, const char *desc,
|
|
||||||
ExplodedNode *n, const Stmt *arg)
|
|
||||||
: BuiltinBugReport(bt, shortDesc, desc, n), Arg(arg) {}
|
|
||||||
|
|
||||||
const Stmt *getArg() const { return Arg; }
|
|
||||||
};
|
|
||||||
|
|
||||||
class VISIBILITY_HIDDEN UndefBranch : public BuiltinBug {
|
|
||||||
struct VISIBILITY_HIDDEN FindUndefExpr {
|
|
||||||
GRStateManager& VM;
|
|
||||||
const GRState* St;
|
|
||||||
|
|
||||||
FindUndefExpr(GRStateManager& V, const GRState* S) : VM(V), St(S) {}
|
|
||||||
|
|
||||||
Expr* FindExpr(Expr* Ex) {
|
|
||||||
if (!MatchesCriteria(Ex))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
for (Stmt::child_iterator I=Ex->child_begin(), E=Ex->child_end();I!=E;++I)
|
|
||||||
if (Expr* ExI = dyn_cast_or_null<Expr>(*I)) {
|
|
||||||
Expr* E2 = FindExpr(ExI);
|
|
||||||
if (E2) return E2;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Ex;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool MatchesCriteria(Expr* Ex) { return St->getSVal(Ex).isUndef(); }
|
|
||||||
};
|
|
||||||
|
|
||||||
public:
|
|
||||||
UndefBranch(GRExprEngine *eng)
|
|
||||||
: BuiltinBug(eng,"Use of garbage value",
|
|
||||||
"Branch condition evaluates to an undefined or garbage value")
|
|
||||||
{}
|
|
||||||
|
|
||||||
void FlushReportsImpl(BugReporter& BR, GRExprEngine& Eng) {
|
|
||||||
for (GRExprEngine::undef_branch_iterator I=Eng.undef_branches_begin(),
|
|
||||||
E=Eng.undef_branches_end(); I!=E; ++I) {
|
|
||||||
|
|
||||||
// What's going on here: we want to highlight the subexpression of the
|
|
||||||
// condition that is the most likely source of the "uninitialized
|
|
||||||
// branch condition." We do a recursive walk of the condition's
|
|
||||||
// subexpressions and roughly look for the most nested subexpression
|
|
||||||
// that binds to Undefined. We then highlight that expression's range.
|
|
||||||
BlockEdge B = cast<BlockEdge>((*I)->getLocation());
|
|
||||||
Expr* Ex = cast<Expr>(B.getSrc()->getTerminatorCondition());
|
|
||||||
assert (Ex && "Block must have a terminator.");
|
|
||||||
|
|
||||||
// Get the predecessor node and check if is a PostStmt with the Stmt
|
|
||||||
// being the terminator condition. We want to inspect the state
|
|
||||||
// of that node instead because it will contain main information about
|
|
||||||
// the subexpressions.
|
|
||||||
assert (!(*I)->pred_empty());
|
|
||||||
|
|
||||||
// Note: any predecessor will do. They should have identical state,
|
|
||||||
// since all the BlockEdge did was act as an error sink since the value
|
|
||||||
// had to already be undefined.
|
|
||||||
ExplodedNode *N = *(*I)->pred_begin();
|
|
||||||
ProgramPoint P = N->getLocation();
|
|
||||||
const GRState* St = (*I)->getState();
|
|
||||||
|
|
||||||
if (PostStmt* PS = dyn_cast<PostStmt>(&P))
|
|
||||||
if (PS->getStmt() == Ex)
|
|
||||||
St = N->getState();
|
|
||||||
|
|
||||||
FindUndefExpr FindIt(Eng.getStateManager(), St);
|
|
||||||
Ex = FindIt.FindExpr(Ex);
|
|
||||||
|
|
||||||
ArgReport *R = new ArgReport(*this, desc.c_str(), *I, Ex);
|
|
||||||
R->addRange(Ex->getSourceRange());
|
|
||||||
BR.EmitReport(R);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void registerInitialVisitors(BugReporterContext& BRC,
|
|
||||||
const ExplodedNode* N,
|
|
||||||
BuiltinBugReport *R) {
|
|
||||||
registerTrackNullOrUndefValue(BRC, static_cast<ArgReport*>(R)->getArg(),
|
|
||||||
N);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
} // end clang namespace
|
} // end clang namespace
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
|
@ -21,6 +21,29 @@ namespace {
|
||||||
|
|
||||||
class VISIBILITY_HIDDEN UndefBranchChecker : public Checker {
|
class VISIBILITY_HIDDEN UndefBranchChecker : public Checker {
|
||||||
BuiltinBug *BT;
|
BuiltinBug *BT;
|
||||||
|
|
||||||
|
struct VISIBILITY_HIDDEN FindUndefExpr {
|
||||||
|
GRStateManager& VM;
|
||||||
|
const GRState* St;
|
||||||
|
|
||||||
|
FindUndefExpr(GRStateManager& V, const GRState* S) : VM(V), St(S) {}
|
||||||
|
|
||||||
|
Expr* FindExpr(Expr* Ex) {
|
||||||
|
if (!MatchesCriteria(Ex))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
for (Stmt::child_iterator I=Ex->child_begin(), E=Ex->child_end();I!=E;++I)
|
||||||
|
if (Expr* ExI = dyn_cast_or_null<Expr>(*I)) {
|
||||||
|
Expr* E2 = FindExpr(ExI);
|
||||||
|
if (E2) return E2;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Ex;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool MatchesCriteria(Expr* Ex) { return St->getSVal(Ex).isUndef(); }
|
||||||
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
UndefBranchChecker() : BT(0) {}
|
UndefBranchChecker() : BT(0) {}
|
||||||
static void *getTag();
|
static void *getTag();
|
||||||
|
@ -54,6 +77,37 @@ void UndefBranchChecker::VisitBranchCondition(GRBranchNodeBuilder &Builder,
|
||||||
EnhancedBugReport *R = new EnhancedBugReport(*BT, BT->getDescription(),N);
|
EnhancedBugReport *R = new EnhancedBugReport(*BT, BT->getDescription(),N);
|
||||||
R->addVisitorCreator(bugreporter::registerTrackNullOrUndefValue,
|
R->addVisitorCreator(bugreporter::registerTrackNullOrUndefValue,
|
||||||
Condition);
|
Condition);
|
||||||
|
|
||||||
|
// What's going on here: we want to highlight the subexpression of the
|
||||||
|
// condition that is the most likely source of the "uninitialized
|
||||||
|
// branch condition." We do a recursive walk of the condition's
|
||||||
|
// subexpressions and roughly look for the most nested subexpression
|
||||||
|
// that binds to Undefined. We then highlight that expression's range.
|
||||||
|
BlockEdge B = cast<BlockEdge>(N->getLocation());
|
||||||
|
Expr* Ex = cast<Expr>(B.getSrc()->getTerminatorCondition());
|
||||||
|
assert (Ex && "Block must have a terminator.");
|
||||||
|
|
||||||
|
// Get the predecessor node and check if is a PostStmt with the Stmt
|
||||||
|
// being the terminator condition. We want to inspect the state
|
||||||
|
// of that node instead because it will contain main information about
|
||||||
|
// the subexpressions.
|
||||||
|
assert (!N->pred_empty());
|
||||||
|
|
||||||
|
// Note: any predecessor will do. They should have identical state,
|
||||||
|
// since all the BlockEdge did was act as an error sink since the value
|
||||||
|
// had to already be undefined.
|
||||||
|
ExplodedNode *PrevN = *N->pred_begin();
|
||||||
|
ProgramPoint P = PrevN->getLocation();
|
||||||
|
const GRState* St = N->getState();
|
||||||
|
|
||||||
|
if (PostStmt* PS = dyn_cast<PostStmt>(&P))
|
||||||
|
if (PS->getStmt() == Ex)
|
||||||
|
St = PrevN->getState();
|
||||||
|
|
||||||
|
FindUndefExpr FindIt(Eng.getStateManager(), St);
|
||||||
|
Ex = FindIt.FindExpr(Ex);
|
||||||
|
R->addRange(Ex->getSourceRange());
|
||||||
|
|
||||||
Eng.getBugReporter().EmitReport(R);
|
Eng.getBugReporter().EmitReport(R);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue