2008-04-03 12:42:52 +08:00
|
|
|
// BugReporter.cpp - Generate PathDiagnostics for Bugs ------------*- C++ -*--//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file defines BugReporter, a utility class for generating
|
2009-06-26 08:05:51 +08:00
|
|
|
// PathDiagnostics.
|
2008-04-03 12:42:52 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-02-10 09:03:03 +08:00
|
|
|
#include "clang/StaticAnalyzer/Core/BugReporter/BugReporter.h"
|
|
|
|
#include "clang/StaticAnalyzer/Core/BugReporter/BugType.h"
|
|
|
|
#include "clang/StaticAnalyzer/Core/PathSensitive/ExprEngine.h"
|
2008-04-03 12:42:52 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2009-07-17 02:13:04 +08:00
|
|
|
#include "clang/Analysis/CFG.h"
|
2008-04-03 12:42:52 +08:00
|
|
|
#include "clang/AST/Expr.h"
|
2009-03-28 04:55:39 +08:00
|
|
|
#include "clang/AST/ParentMap.h"
|
2009-04-26 09:32:48 +08:00
|
|
|
#include "clang/AST/StmtObjC.h"
|
|
|
|
#include "clang/Basic/SourceManager.h"
|
2008-04-03 12:42:52 +08:00
|
|
|
#include "clang/Analysis/ProgramPoint.h"
|
2011-02-10 09:03:03 +08:00
|
|
|
#include "clang/StaticAnalyzer/Core/BugReporter/PathDiagnostic.h"
|
2008-08-24 06:23:37 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2008-06-18 13:34:07 +08:00
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2009-02-05 07:49:09 +08:00
|
|
|
#include "llvm/ADT/STLExtras.h"
|
2009-03-28 04:55:39 +08:00
|
|
|
#include "llvm/ADT/OwningPtr.h"
|
2009-03-13 07:41:59 +08:00
|
|
|
#include <queue>
|
2008-04-03 12:42:52 +08:00
|
|
|
|
|
|
|
using namespace clang;
|
2010-12-23 15:20:52 +08:00
|
|
|
using namespace ento;
|
2008-04-03 12:42:52 +08:00
|
|
|
|
2009-05-07 05:39:49 +08:00
|
|
|
BugReporterVisitor::~BugReporterVisitor() {}
|
|
|
|
BugReporterContext::~BugReporterContext() {
|
|
|
|
for (visitor_iterator I = visitor_begin(), E = visitor_end(); I != E; ++I)
|
|
|
|
if ((*I)->isOwnedByReporterContext()) delete *I;
|
|
|
|
}
|
|
|
|
|
2010-03-21 02:01:57 +08:00
|
|
|
void BugReporterContext::addVisitor(BugReporterVisitor* visitor) {
|
|
|
|
if (!visitor)
|
|
|
|
return;
|
|
|
|
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
visitor->Profile(ID);
|
|
|
|
void *InsertPos;
|
|
|
|
|
2010-03-21 12:38:40 +08:00
|
|
|
if (CallbacksSet.FindNodeOrInsertPos(ID, InsertPos)) {
|
|
|
|
delete visitor;
|
2010-03-21 02:01:57 +08:00
|
|
|
return;
|
2010-03-21 12:38:40 +08:00
|
|
|
}
|
2010-03-21 02:01:57 +08:00
|
|
|
|
|
|
|
CallbacksSet.InsertNode(visitor, InsertPos);
|
2010-11-24 08:54:37 +08:00
|
|
|
Callbacks = F.add(visitor, Callbacks);
|
2010-03-21 02:01:57 +08:00
|
|
|
}
|
|
|
|
|
2009-02-05 07:49:09 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2009-04-01 07:00:32 +08:00
|
|
|
// Helper routines for walking the ExplodedGraph and fetching statements.
|
2009-02-05 07:49:09 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2008-04-03 12:42:52 +08:00
|
|
|
|
2010-04-06 11:01:56 +08:00
|
|
|
static inline const Stmt* GetStmt(const ProgramPoint &P) {
|
2009-08-18 09:05:30 +08:00
|
|
|
if (const StmtPoint* SP = dyn_cast<StmtPoint>(&P))
|
|
|
|
return SP->getStmt();
|
2009-02-24 06:44:26 +08:00
|
|
|
else if (const BlockEdge* BE = dyn_cast<BlockEdge>(&P))
|
2008-04-03 12:42:52 +08:00
|
|
|
return BE->getSrc()->getTerminator();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-24 06:44:26 +08:00
|
|
|
return 0;
|
2008-04-03 12:42:52 +08:00
|
|
|
}
|
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
static inline const ExplodedNode*
|
|
|
|
GetPredecessorNode(const ExplodedNode* N) {
|
2009-02-24 06:44:26 +08:00
|
|
|
return N->pred_empty() ? NULL : *(N->pred_begin());
|
2008-04-08 07:35:17 +08:00
|
|
|
}
|
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
static inline const ExplodedNode*
|
|
|
|
GetSuccessorNode(const ExplodedNode* N) {
|
2009-02-24 06:44:26 +08:00
|
|
|
return N->succ_empty() ? NULL : *(N->succ_begin());
|
2008-04-18 07:44:37 +08:00
|
|
|
}
|
2008-04-26 03:01:27 +08:00
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
static const Stmt* GetPreviousStmt(const ExplodedNode* N) {
|
2009-02-24 06:44:26 +08:00
|
|
|
for (N = GetPredecessorNode(N); N; N = GetPredecessorNode(N))
|
2009-07-23 06:35:28 +08:00
|
|
|
if (const Stmt *S = GetStmt(N->getLocation()))
|
2009-02-24 06:44:26 +08:00
|
|
|
return S;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-24 06:44:26 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
static const Stmt* GetNextStmt(const ExplodedNode* N) {
|
2009-02-24 06:44:26 +08:00
|
|
|
for (N = GetSuccessorNode(N); N; N = GetSuccessorNode(N))
|
2009-07-23 06:35:28 +08:00
|
|
|
if (const Stmt *S = GetStmt(N->getLocation())) {
|
2009-03-29 01:33:57 +08:00
|
|
|
// Check if the statement is '?' or '&&'/'||'. These are "merges",
|
|
|
|
// not actual statement points.
|
|
|
|
switch (S->getStmtClass()) {
|
|
|
|
case Stmt::ChooseExprClass:
|
2011-02-17 18:25:35 +08:00
|
|
|
case Stmt::BinaryConditionalOperatorClass: continue;
|
2009-03-29 01:33:57 +08:00
|
|
|
case Stmt::ConditionalOperatorClass: continue;
|
|
|
|
case Stmt::BinaryOperatorClass: {
|
2010-08-25 19:45:40 +08:00
|
|
|
BinaryOperatorKind Op = cast<BinaryOperator>(S)->getOpcode();
|
|
|
|
if (Op == BO_LAnd || Op == BO_LOr)
|
2009-03-29 01:33:57 +08:00
|
|
|
continue;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-28 08:07:15 +08:00
|
|
|
// Some expressions don't have locations.
|
|
|
|
if (S->getLocStart().isInvalid())
|
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-24 06:44:26 +08:00
|
|
|
return S;
|
2009-03-29 01:33:57 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-24 06:44:26 +08:00
|
|
|
return 0;
|
2008-04-18 07:44:37 +08:00
|
|
|
}
|
|
|
|
|
2009-07-23 06:35:28 +08:00
|
|
|
static inline const Stmt*
|
2009-09-09 23:08:12 +08:00
|
|
|
GetCurrentOrPreviousStmt(const ExplodedNode* N) {
|
2009-07-23 06:35:28 +08:00
|
|
|
if (const Stmt *S = GetStmt(N->getLocation()))
|
2009-02-24 06:44:26 +08:00
|
|
|
return S;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-24 06:44:26 +08:00
|
|
|
return GetPreviousStmt(N);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-23 06:35:28 +08:00
|
|
|
static inline const Stmt*
|
2009-09-09 23:08:12 +08:00
|
|
|
GetCurrentOrNextStmt(const ExplodedNode* N) {
|
2009-07-23 06:35:28 +08:00
|
|
|
if (const Stmt *S = GetStmt(N->getLocation()))
|
2009-02-24 06:44:26 +08:00
|
|
|
return S;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-24 06:44:26 +08:00
|
|
|
return GetNextStmt(N);
|
2009-01-24 08:55:43 +08:00
|
|
|
}
|
|
|
|
|
2009-02-24 06:44:26 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2009-04-01 07:00:32 +08:00
|
|
|
// PathDiagnosticBuilder and its associated routines and helper objects.
|
2009-02-24 06:44:26 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2009-02-24 07:13:51 +08:00
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
typedef llvm::DenseMap<const ExplodedNode*,
|
|
|
|
const ExplodedNode*> NodeBackMap;
|
2009-04-01 04:22:36 +08:00
|
|
|
|
2009-03-27 13:06:10 +08:00
|
|
|
namespace {
|
2009-11-28 14:07:30 +08:00
|
|
|
class NodeMapClosure : public BugReport::NodeResolver {
|
2009-04-01 04:22:36 +08:00
|
|
|
NodeBackMap& M;
|
|
|
|
public:
|
|
|
|
NodeMapClosure(NodeBackMap *m) : M(*m) {}
|
|
|
|
~NodeMapClosure() {}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode* getOriginalNode(const ExplodedNode* N) {
|
2009-04-01 04:22:36 +08:00
|
|
|
NodeBackMap::iterator I = M.find(N);
|
|
|
|
return I == M.end() ? 0 : I->second;
|
|
|
|
}
|
|
|
|
};
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-28 14:07:30 +08:00
|
|
|
class PathDiagnosticBuilder : public BugReporterContext {
|
2009-04-01 04:22:36 +08:00
|
|
|
BugReport *R;
|
2009-03-27 13:06:10 +08:00
|
|
|
PathDiagnosticClient *PDC;
|
2009-03-28 04:55:39 +08:00
|
|
|
llvm::OwningPtr<ParentMap> PM;
|
2009-04-01 04:22:36 +08:00
|
|
|
NodeMapClosure NMC;
|
2009-09-09 23:08:12 +08:00
|
|
|
public:
|
2009-05-07 05:39:49 +08:00
|
|
|
PathDiagnosticBuilder(GRBugReporter &br,
|
2009-09-09 23:08:12 +08:00
|
|
|
BugReport *r, NodeBackMap *Backmap,
|
2009-05-07 05:39:49 +08:00
|
|
|
PathDiagnosticClient *pdc)
|
|
|
|
: BugReporterContext(br),
|
2009-09-09 23:08:12 +08:00
|
|
|
R(r), PDC(pdc), NMC(Backmap) {
|
2009-05-07 05:39:49 +08:00
|
|
|
addVisitor(R);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
PathDiagnosticLocation ExecutionContinues(const ExplodedNode* N);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 04:55:39 +08:00
|
|
|
PathDiagnosticLocation ExecutionContinues(llvm::raw_string_ostream& os,
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode* N);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-09-16 11:50:38 +08:00
|
|
|
Decl const &getCodeDecl() { return R->getErrorNode()->getCodeDecl(); }
|
2009-09-10 13:44:00 +08:00
|
|
|
|
2010-09-16 11:50:38 +08:00
|
|
|
ParentMap& getParentMap() { return R->getErrorNode()->getParentMap(); }
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-02 01:18:21 +08:00
|
|
|
const Stmt *getParent(const Stmt *S) {
|
|
|
|
return getParentMap().getParent(S);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-07 05:39:49 +08:00
|
|
|
virtual NodeMapClosure& getNodeResolver() { return NMC; }
|
2009-04-18 08:02:19 +08:00
|
|
|
|
2009-03-28 05:16:25 +08:00
|
|
|
PathDiagnosticLocation getEnclosingStmtLocation(const Stmt *S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 04:22:36 +08:00
|
|
|
PathDiagnosticClient::PathGenerationScheme getGenerationScheme() const {
|
|
|
|
return PDC ? PDC->getGenerationScheme() : PathDiagnosticClient::Extensive;
|
|
|
|
}
|
|
|
|
|
2009-03-27 13:06:10 +08:00
|
|
|
bool supportsLogicalOpControlFlow() const {
|
|
|
|
return PDC ? PDC->supportsLogicalOpControlFlow() : true;
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-03-27 13:06:10 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
2009-03-28 04:55:39 +08:00
|
|
|
PathDiagnosticLocation
|
2009-08-06 09:32:16 +08:00
|
|
|
PathDiagnosticBuilder::ExecutionContinues(const ExplodedNode* N) {
|
2009-07-23 06:35:28 +08:00
|
|
|
if (const Stmt *S = GetNextStmt(N))
|
2009-05-07 05:39:49 +08:00
|
|
|
return PathDiagnosticLocation(S, getSourceManager());
|
2009-03-28 04:55:39 +08:00
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
return FullSourceLoc(N->getLocationContext()->getDecl()->getBodyRBrace(),
|
2009-08-19 20:50:00 +08:00
|
|
|
getSourceManager());
|
2009-03-13 02:41:53 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 04:55:39 +08:00
|
|
|
PathDiagnosticLocation
|
2009-03-27 13:06:10 +08:00
|
|
|
PathDiagnosticBuilder::ExecutionContinues(llvm::raw_string_ostream& os,
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode* N) {
|
2009-03-27 13:06:10 +08:00
|
|
|
|
2008-05-07 02:11:09 +08:00
|
|
|
// Slow, but probably doesn't matter.
|
2009-02-24 06:44:26 +08:00
|
|
|
if (os.str().empty())
|
|
|
|
os << ' ';
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 04:55:39 +08:00
|
|
|
const PathDiagnosticLocation &Loc = ExecutionContinues(N);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 04:55:39 +08:00
|
|
|
if (Loc.asStmt())
|
2009-02-24 06:44:26 +08:00
|
|
|
os << "Execution continues on line "
|
2011-07-26 05:09:52 +08:00
|
|
|
<< getSourceManager().getExpansionLineNumber(Loc.asLocation())
|
2009-05-07 05:39:49 +08:00
|
|
|
<< '.';
|
2009-12-05 04:34:31 +08:00
|
|
|
else {
|
|
|
|
os << "Execution jumps to the end of the ";
|
|
|
|
const Decl *D = N->getLocationContext()->getDecl();
|
|
|
|
if (isa<ObjCMethodDecl>(D))
|
|
|
|
os << "method";
|
|
|
|
else if (isa<FunctionDecl>(D))
|
|
|
|
os << "function";
|
|
|
|
else {
|
|
|
|
assert(isa<BlockDecl>(D));
|
|
|
|
os << "anonymous block";
|
|
|
|
}
|
|
|
|
os << '.';
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-13 02:41:53 +08:00
|
|
|
return Loc;
|
2008-05-07 02:11:09 +08:00
|
|
|
}
|
|
|
|
|
2009-05-15 09:50:15 +08:00
|
|
|
static bool IsNested(const Stmt *S, ParentMap &PM) {
|
|
|
|
if (isa<Expr>(S) && PM.isConsumedExpr(cast<Expr>(S)))
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-15 09:50:15 +08:00
|
|
|
const Stmt *Parent = PM.getParentIgnoreParens(S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-15 09:50:15 +08:00
|
|
|
if (Parent)
|
|
|
|
switch (Parent->getStmtClass()) {
|
|
|
|
case Stmt::ForStmtClass:
|
|
|
|
case Stmt::DoStmtClass:
|
|
|
|
case Stmt::WhileStmtClass:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
return false;
|
2009-05-15 09:50:15 +08:00
|
|
|
}
|
|
|
|
|
2009-03-28 05:16:25 +08:00
|
|
|
PathDiagnosticLocation
|
|
|
|
PathDiagnosticBuilder::getEnclosingStmtLocation(const Stmt *S) {
|
|
|
|
assert(S && "Null Stmt* passed to getEnclosingStmtLocation");
|
2009-09-09 23:08:12 +08:00
|
|
|
ParentMap &P = getParentMap();
|
2009-05-07 05:39:49 +08:00
|
|
|
SourceManager &SMgr = getSourceManager();
|
2009-05-12 06:19:32 +08:00
|
|
|
|
2009-05-15 09:50:15 +08:00
|
|
|
while (IsNested(S, P)) {
|
2009-05-12 03:50:47 +08:00
|
|
|
const Stmt *Parent = P.getParentIgnoreParens(S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 11:37:59 +08:00
|
|
|
if (!Parent)
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 11:37:59 +08:00
|
|
|
switch (Parent->getStmtClass()) {
|
2009-04-01 14:13:56 +08:00
|
|
|
case Stmt::BinaryOperatorClass: {
|
|
|
|
const BinaryOperator *B = cast<BinaryOperator>(Parent);
|
|
|
|
if (B->isLogicalOp())
|
|
|
|
return PathDiagnosticLocation(S, SMgr);
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-03-28 11:37:59 +08:00
|
|
|
case Stmt::CompoundStmtClass:
|
|
|
|
case Stmt::StmtExprClass:
|
2009-03-28 12:08:14 +08:00
|
|
|
return PathDiagnosticLocation(S, SMgr);
|
|
|
|
case Stmt::ChooseExprClass:
|
|
|
|
// Similar to '?' if we are referring to condition, just have the edge
|
|
|
|
// point to the entire choose expression.
|
|
|
|
if (cast<ChooseExpr>(Parent)->getCond() == S)
|
|
|
|
return PathDiagnosticLocation(Parent, SMgr);
|
|
|
|
else
|
2009-09-09 23:08:12 +08:00
|
|
|
return PathDiagnosticLocation(S, SMgr);
|
2011-02-17 18:25:35 +08:00
|
|
|
case Stmt::BinaryConditionalOperatorClass:
|
2009-03-28 12:08:14 +08:00
|
|
|
case Stmt::ConditionalOperatorClass:
|
|
|
|
// For '?', if we are referring to condition, just have the edge point
|
|
|
|
// to the entire '?' expression.
|
2011-02-17 18:25:35 +08:00
|
|
|
if (cast<AbstractConditionalOperator>(Parent)->getCond() == S)
|
2009-03-28 12:08:14 +08:00
|
|
|
return PathDiagnosticLocation(Parent, SMgr);
|
|
|
|
else
|
2009-09-09 23:08:12 +08:00
|
|
|
return PathDiagnosticLocation(S, SMgr);
|
2009-03-28 11:37:59 +08:00
|
|
|
case Stmt::DoStmtClass:
|
2009-09-09 23:08:12 +08:00
|
|
|
return PathDiagnosticLocation(S, SMgr);
|
2009-03-28 11:37:59 +08:00
|
|
|
case Stmt::ForStmtClass:
|
|
|
|
if (cast<ForStmt>(Parent)->getBody() == S)
|
2009-09-09 23:08:12 +08:00
|
|
|
return PathDiagnosticLocation(S, SMgr);
|
|
|
|
break;
|
2009-03-28 11:37:59 +08:00
|
|
|
case Stmt::IfStmtClass:
|
|
|
|
if (cast<IfStmt>(Parent)->getCond() != S)
|
|
|
|
return PathDiagnosticLocation(S, SMgr);
|
2009-04-28 12:23:15 +08:00
|
|
|
break;
|
2009-03-28 11:37:59 +08:00
|
|
|
case Stmt::ObjCForCollectionStmtClass:
|
|
|
|
if (cast<ObjCForCollectionStmt>(Parent)->getBody() == S)
|
|
|
|
return PathDiagnosticLocation(S, SMgr);
|
|
|
|
break;
|
|
|
|
case Stmt::WhileStmtClass:
|
|
|
|
if (cast<WhileStmt>(Parent)->getCond() != S)
|
|
|
|
return PathDiagnosticLocation(S, SMgr);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2009-03-28 05:16:25 +08:00
|
|
|
S = Parent;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 05:16:25 +08:00
|
|
|
assert(S && "Cannot have null Stmt for PathDiagnosticLocation");
|
2009-05-12 06:19:32 +08:00
|
|
|
|
|
|
|
// Special case: DeclStmts can appear in for statement declarations, in which
|
|
|
|
// case the ForStmt is the context.
|
|
|
|
if (isa<DeclStmt>(S)) {
|
|
|
|
if (const Stmt *Parent = P.getParent(S)) {
|
|
|
|
switch (Parent->getStmtClass()) {
|
|
|
|
case Stmt::ForStmtClass:
|
|
|
|
case Stmt::ObjCForCollectionStmtClass:
|
|
|
|
return PathDiagnosticLocation(Parent, SMgr);
|
|
|
|
default:
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
|
|
|
}
|
2009-05-12 06:19:32 +08:00
|
|
|
}
|
|
|
|
else if (isa<BinaryOperator>(S)) {
|
|
|
|
// Special case: the binary operator represents the initialization
|
|
|
|
// code in a for statement (this can happen when the variable being
|
|
|
|
// initialized is an old variable.
|
|
|
|
if (const ForStmt *FS =
|
|
|
|
dyn_cast_or_null<ForStmt>(P.getParentIgnoreParens(S))) {
|
|
|
|
if (FS->getInit() == S)
|
|
|
|
return PathDiagnosticLocation(FS, SMgr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-28 05:16:25 +08:00
|
|
|
return PathDiagnosticLocation(S, SMgr);
|
|
|
|
}
|
|
|
|
|
2009-02-05 07:49:09 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2009-04-01 07:00:32 +08:00
|
|
|
// ScanNotableSymbols: closure-like callback for scanning Store bindings.
|
2009-02-05 07:49:09 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
static const VarDecl*
|
2009-08-06 09:32:16 +08:00
|
|
|
GetMostRecentVarDeclBinding(const ExplodedNode* N,
|
2009-04-01 07:00:32 +08:00
|
|
|
GRStateManager& VMgr, SVal X) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
for ( ; N ; N = N->pred_empty() ? 0 : *N->pred_begin()) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
ProgramPoint P = N->getLocation();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!isa<PostStmt>(P))
|
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-23 06:35:28 +08:00
|
|
|
const DeclRefExpr* DR = dyn_cast<DeclRefExpr>(cast<PostStmt>(P).getStmt());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!DR)
|
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-02-09 00:18:51 +08:00
|
|
|
SVal Y = N->getState()->getSVal(DR);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (X != Y)
|
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-23 06:35:28 +08:00
|
|
|
const VarDecl* VD = dyn_cast<VarDecl>(DR->getDecl());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!VD)
|
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
return VD;
|
2009-02-05 07:49:09 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
return 0;
|
2008-05-02 06:50:36 +08:00
|
|
|
}
|
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
namespace {
|
2009-11-28 14:07:30 +08:00
|
|
|
class NotableSymbolHandler
|
2009-04-01 07:00:32 +08:00
|
|
|
: public StoreManager::BindingsHandler {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
SymbolRef Sym;
|
|
|
|
const GRState* PrevSt;
|
|
|
|
const Stmt* S;
|
|
|
|
GRStateManager& VMgr;
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode* Pred;
|
2009-09-09 23:08:12 +08:00
|
|
|
PathDiagnostic& PD;
|
2009-04-01 07:00:32 +08:00
|
|
|
BugReporter& BR;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
public:
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
NotableSymbolHandler(SymbolRef sym, const GRState* prevst, const Stmt* s,
|
2009-08-06 09:32:16 +08:00
|
|
|
GRStateManager& vmgr, const ExplodedNode* pred,
|
2009-04-01 07:00:32 +08:00
|
|
|
PathDiagnostic& pd, BugReporter& br)
|
|
|
|
: Sym(sym), PrevSt(prevst), S(s), VMgr(vmgr), Pred(pred), PD(pd), BR(br) {}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
bool HandleBinding(StoreManager& SMgr, Store store, const MemRegion* R,
|
|
|
|
SVal V) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
SymbolRef ScanSym = V.getAsSymbol();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (ScanSym != Sym)
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
// Check if the previous state has this binding.
|
2010-02-09 00:18:51 +08:00
|
|
|
SVal X = PrevSt->getSVal(loc::MemRegionVal(R));
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (X == V) // Same binding?
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Different binding. Only handle assignments for now. We don't pull
|
2009-09-09 23:08:12 +08:00
|
|
|
// this check out of the loop because we will eventually handle other
|
2009-04-01 07:00:32 +08:00
|
|
|
// cases.
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
VarDecl *VD = 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (const BinaryOperator* B = dyn_cast<BinaryOperator>(S)) {
|
|
|
|
if (!B->isAssignmentOp())
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// What variable did we assign to?
|
|
|
|
DeclRefExpr* DR = dyn_cast<DeclRefExpr>(B->getLHS()->IgnoreParenCasts());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!DR)
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
VD = dyn_cast<VarDecl>(DR->getDecl());
|
|
|
|
}
|
|
|
|
else if (const DeclStmt* DS = dyn_cast<DeclStmt>(S)) {
|
|
|
|
// FIXME: Eventually CFGs won't have DeclStmts. Right now we
|
|
|
|
// assume that each DeclStmt has a single Decl. This invariant
|
2011-04-15 13:22:18 +08:00
|
|
|
// holds by construction in the CFG.
|
2009-04-01 07:00:32 +08:00
|
|
|
VD = dyn_cast<VarDecl>(*DS->decl_begin());
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!VD)
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// What is the most recently referenced variable with this binding?
|
|
|
|
const VarDecl* MostRecent = GetMostRecentVarDeclBinding(Pred, VMgr, V);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!MostRecent)
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Create the diagnostic.
|
|
|
|
FullSourceLoc L(S->getLocStart(), BR.getSourceManager());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-02-17 05:13:32 +08:00
|
|
|
if (Loc::isLocType(VD->getType())) {
|
2009-04-01 07:00:32 +08:00
|
|
|
std::string msg = "'" + std::string(VD->getNameAsString()) +
|
|
|
|
"' now aliases '" + MostRecent->getNameAsString() + "'";
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
PD.push_front(new PathDiagnosticEventPiece(L, msg));
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-04-01 07:00:32 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
static void HandleNotableSymbol(const ExplodedNode* N,
|
2009-04-01 07:00:32 +08:00
|
|
|
const Stmt* S,
|
|
|
|
SymbolRef Sym, BugReporter& BR,
|
|
|
|
PathDiagnostic& PD) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode* Pred = N->pred_empty() ? 0 : *N->pred_begin();
|
2009-04-01 07:00:32 +08:00
|
|
|
const GRState* PrevSt = Pred ? Pred->getState() : 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!PrevSt)
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Look at the region bindings of the current state that map to the
|
|
|
|
// specified symbol. Are any of them not in the previous state?
|
|
|
|
GRStateManager& VMgr = cast<GRBugReporter>(BR).getStateManager();
|
|
|
|
NotableSymbolHandler H(Sym, PrevSt, S, VMgr, Pred, PD, BR);
|
|
|
|
cast<GRBugReporter>(BR).getStateManager().iterBindings(N->getState(), H);
|
2008-04-03 12:42:52 +08:00
|
|
|
}
|
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
namespace {
|
2009-11-28 14:07:30 +08:00
|
|
|
class ScanNotableSymbols
|
2009-04-01 07:00:32 +08:00
|
|
|
: public StoreManager::BindingsHandler {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
llvm::SmallSet<SymbolRef, 10> AlreadyProcessed;
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode* N;
|
2009-07-23 06:35:28 +08:00
|
|
|
const Stmt* S;
|
2009-04-01 07:00:32 +08:00
|
|
|
GRBugReporter& BR;
|
|
|
|
PathDiagnostic& PD;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
public:
|
2009-08-06 09:32:16 +08:00
|
|
|
ScanNotableSymbols(const ExplodedNode* n, const Stmt* s,
|
2009-07-23 06:35:28 +08:00
|
|
|
GRBugReporter& br, PathDiagnostic& pd)
|
2009-04-01 07:00:32 +08:00
|
|
|
: N(n), S(s), BR(br), PD(pd) {}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
bool HandleBinding(StoreManager& SMgr, Store store,
|
|
|
|
const MemRegion* R, SVal V) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
SymbolRef ScanSym = V.getAsSymbol();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!ScanSym)
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!BR.isNotable(ScanSym))
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (AlreadyProcessed.count(ScanSym))
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
AlreadyProcessed.insert(ScanSym);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
HandleNotableSymbol(N, S, ScanSym, BR, PD);
|
|
|
|
return true;
|
2008-05-02 06:50:36 +08:00
|
|
|
}
|
2009-04-01 07:00:32 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
2008-04-10 05:41:14 +08:00
|
|
|
|
2009-02-05 07:49:09 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2009-04-01 07:00:32 +08:00
|
|
|
// "Minimal" path diagnostic generation algorithm.
|
2009-02-05 07:49:09 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
static void CompactPathDiagnostic(PathDiagnostic &PD, const SourceManager& SM);
|
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
static void GenerateMinimalPathDiagnostic(PathDiagnostic& PD,
|
|
|
|
PathDiagnosticBuilder &PDB,
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode *N) {
|
2009-05-07 05:39:49 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
SourceManager& SMgr = PDB.getSourceManager();
|
2009-09-09 23:08:12 +08:00
|
|
|
const ExplodedNode* NextNode = N->pred_empty()
|
2009-04-01 07:00:32 +08:00
|
|
|
? NULL : *(N->pred_begin());
|
|
|
|
while (NextNode) {
|
2009-09-09 23:08:12 +08:00
|
|
|
N = NextNode;
|
2009-04-01 07:00:32 +08:00
|
|
|
NextNode = GetPredecessorNode(N);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-03 12:42:52 +08:00
|
|
|
ProgramPoint P = N->getLocation();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-03 12:42:52 +08:00
|
|
|
if (const BlockEdge* BE = dyn_cast<BlockEdge>(&P)) {
|
2010-07-20 14:22:24 +08:00
|
|
|
const CFGBlock* Src = BE->getSrc();
|
|
|
|
const CFGBlock* Dst = BE->getDst();
|
|
|
|
const Stmt* T = Src->getTerminator();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-03 12:42:52 +08:00
|
|
|
if (!T)
|
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-13 02:41:53 +08:00
|
|
|
FullSourceLoc Start(T->getLocStart(), SMgr);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-03 12:42:52 +08:00
|
|
|
switch (T->getStmtClass()) {
|
|
|
|
default:
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-03 12:42:52 +08:00
|
|
|
case Stmt::GotoStmtClass:
|
2009-09-09 23:08:12 +08:00
|
|
|
case Stmt::IndirectGotoStmtClass: {
|
2009-07-23 06:35:28 +08:00
|
|
|
const Stmt* S = GetNextStmt(N);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-03 12:42:52 +08:00
|
|
|
if (!S)
|
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-11 07:56:07 +08:00
|
|
|
std::string sbuf;
|
2009-09-09 23:08:12 +08:00
|
|
|
llvm::raw_string_ostream os(sbuf);
|
2009-03-28 05:16:25 +08:00
|
|
|
const PathDiagnosticLocation &End = PDB.getEnclosingStmtLocation(S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 04:55:39 +08:00
|
|
|
os << "Control jumps to line "
|
2011-07-26 05:09:52 +08:00
|
|
|
<< End.asLocation().getExpansionLineNumber();
|
2009-03-13 02:41:53 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
|
|
|
os.str()));
|
2008-04-03 12:42:52 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
case Stmt::SwitchStmtClass: {
|
2008-04-03 12:42:52 +08:00
|
|
|
// Figure out what case arm we took.
|
2009-02-11 07:56:07 +08:00
|
|
|
std::string sbuf;
|
|
|
|
llvm::raw_string_ostream os(sbuf);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-07-20 14:22:24 +08:00
|
|
|
if (const Stmt* S = Dst->getLabel()) {
|
2009-03-28 04:55:39 +08:00
|
|
|
PathDiagnosticLocation End(S, SMgr);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-24 07:35:07 +08:00
|
|
|
switch (S->getStmtClass()) {
|
2009-03-13 02:41:53 +08:00
|
|
|
default:
|
|
|
|
os << "No cases match in the switch statement. "
|
2009-04-01 07:00:32 +08:00
|
|
|
"Control jumps to line "
|
2011-07-26 05:09:52 +08:00
|
|
|
<< End.asLocation().getExpansionLineNumber();
|
2009-03-13 02:41:53 +08:00
|
|
|
break;
|
|
|
|
case Stmt::DefaultStmtClass:
|
|
|
|
os << "Control jumps to the 'default' case at line "
|
2011-07-26 05:09:52 +08:00
|
|
|
<< End.asLocation().getExpansionLineNumber();
|
2009-03-13 02:41:53 +08:00
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-13 02:41:53 +08:00
|
|
|
case Stmt::CaseStmtClass: {
|
2009-09-09 23:08:12 +08:00
|
|
|
os << "Control jumps to 'case ";
|
2010-07-20 14:22:24 +08:00
|
|
|
const CaseStmt* Case = cast<CaseStmt>(S);
|
|
|
|
const Expr* LHS = Case->getLHS()->IgnoreParenCasts();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
// Determine if it is an enum.
|
2009-03-13 02:41:53 +08:00
|
|
|
bool GetRawInt = true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-07-20 14:22:24 +08:00
|
|
|
if (const DeclRefExpr* DR = dyn_cast<DeclRefExpr>(LHS)) {
|
2009-03-13 02:41:53 +08:00
|
|
|
// FIXME: Maybe this should be an assertion. Are there cases
|
|
|
|
// were it is not an EnumConstantDecl?
|
2010-07-20 14:22:24 +08:00
|
|
|
const EnumConstantDecl* D =
|
|
|
|
dyn_cast<EnumConstantDecl>(DR->getDecl());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-13 02:41:53 +08:00
|
|
|
if (D) {
|
|
|
|
GetRawInt = false;
|
2010-04-17 17:33:03 +08:00
|
|
|
os << D;
|
2009-03-13 02:41:53 +08:00
|
|
|
}
|
2008-04-24 07:35:07 +08:00
|
|
|
}
|
2009-04-27 03:04:51 +08:00
|
|
|
|
|
|
|
if (GetRawInt)
|
2009-05-07 05:39:49 +08:00
|
|
|
os << LHS->EvaluateAsInt(PDB.getASTContext());
|
2009-04-27 03:04:51 +08:00
|
|
|
|
2009-03-28 04:55:39 +08:00
|
|
|
os << ":' at line "
|
2011-07-26 05:09:52 +08:00
|
|
|
<< End.asLocation().getExpansionLineNumber();
|
2009-03-13 02:41:53 +08:00
|
|
|
break;
|
|
|
|
}
|
2008-04-03 12:42:52 +08:00
|
|
|
}
|
2009-03-28 04:55:39 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
|
|
|
os.str()));
|
2008-04-03 12:42:52 +08:00
|
|
|
}
|
2008-04-25 09:29:56 +08:00
|
|
|
else {
|
2008-09-13 02:17:46 +08:00
|
|
|
os << "'Default' branch taken. ";
|
2009-09-09 23:08:12 +08:00
|
|
|
const PathDiagnosticLocation &End = PDB.ExecutionContinues(os, N);
|
2009-03-28 04:55:39 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
|
|
|
os.str()));
|
2008-04-25 09:29:56 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-03 12:42:52 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-26 03:01:27 +08:00
|
|
|
case Stmt::BreakStmtClass:
|
|
|
|
case Stmt::ContinueStmtClass: {
|
2009-02-11 07:56:07 +08:00
|
|
|
std::string sbuf;
|
|
|
|
llvm::raw_string_ostream os(sbuf);
|
2009-03-28 04:55:39 +08:00
|
|
|
PathDiagnosticLocation End = PDB.ExecutionContinues(os, N);
|
2009-03-13 02:41:53 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
|
|
|
os.str()));
|
2008-04-26 03:01:27 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Determine control-flow for ternary '?'.
|
2011-02-17 18:25:35 +08:00
|
|
|
case Stmt::BinaryConditionalOperatorClass:
|
2008-04-08 07:35:17 +08:00
|
|
|
case Stmt::ConditionalOperatorClass: {
|
2009-02-11 07:56:07 +08:00
|
|
|
std::string sbuf;
|
|
|
|
llvm::raw_string_ostream os(sbuf);
|
2009-03-28 12:08:14 +08:00
|
|
|
os << "'?' condition is ";
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-08 07:35:17 +08:00
|
|
|
if (*(Src->succ_begin()+1) == Dst)
|
2009-03-13 02:41:53 +08:00
|
|
|
os << "false";
|
2008-04-08 07:35:17 +08:00
|
|
|
else
|
2009-03-13 02:41:53 +08:00
|
|
|
os << "true";
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 04:55:39 +08:00
|
|
|
PathDiagnosticLocation End = PDB.ExecutionContinues(N);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 12:08:14 +08:00
|
|
|
if (const Stmt *S = End.asStmt())
|
|
|
|
End = PDB.getEnclosingStmtLocation(S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-27 13:06:10 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
|
|
|
os.str()));
|
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Determine control-flow for short-circuited '&&' and '||'.
|
2009-03-27 13:06:10 +08:00
|
|
|
case Stmt::BinaryOperatorClass: {
|
|
|
|
if (!PDB.supportsLogicalOpControlFlow())
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-07-20 14:22:24 +08:00
|
|
|
const BinaryOperator *B = cast<BinaryOperator>(T);
|
2009-03-27 13:06:10 +08:00
|
|
|
std::string sbuf;
|
|
|
|
llvm::raw_string_ostream os(sbuf);
|
|
|
|
os << "Left side of '";
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-25 19:45:40 +08:00
|
|
|
if (B->getOpcode() == BO_LAnd) {
|
2009-03-29 01:33:57 +08:00
|
|
|
os << "&&" << "' is ";
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-29 01:33:57 +08:00
|
|
|
if (*(Src->succ_begin()+1) == Dst) {
|
|
|
|
os << "false";
|
|
|
|
PathDiagnosticLocation End(B->getLHS(), SMgr);
|
|
|
|
PathDiagnosticLocation Start(B->getOperatorLoc(), SMgr);
|
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
|
|
|
os.str()));
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-03-29 01:33:57 +08:00
|
|
|
else {
|
|
|
|
os << "true";
|
|
|
|
PathDiagnosticLocation Start(B->getLHS(), SMgr);
|
|
|
|
PathDiagnosticLocation End = PDB.ExecutionContinues(N);
|
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
|
|
|
os.str()));
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-03-27 13:06:10 +08:00
|
|
|
}
|
|
|
|
else {
|
2010-08-25 19:45:40 +08:00
|
|
|
assert(B->getOpcode() == BO_LOr);
|
2009-03-29 01:33:57 +08:00
|
|
|
os << "||" << "' is ";
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-29 01:33:57 +08:00
|
|
|
if (*(Src->succ_begin()+1) == Dst) {
|
|
|
|
os << "false";
|
|
|
|
PathDiagnosticLocation Start(B->getLHS(), SMgr);
|
|
|
|
PathDiagnosticLocation End = PDB.ExecutionContinues(N);
|
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
2009-09-09 23:08:12 +08:00
|
|
|
os.str()));
|
2009-03-29 01:33:57 +08:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
os << "true";
|
|
|
|
PathDiagnosticLocation End(B->getLHS(), SMgr);
|
|
|
|
PathDiagnosticLocation Start(B->getOperatorLoc(), SMgr);
|
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
2009-09-09 23:08:12 +08:00
|
|
|
os.str()));
|
2009-03-29 01:33:57 +08:00
|
|
|
}
|
2009-03-27 13:06:10 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-04-08 07:35:17 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
case Stmt::DoStmtClass: {
|
2008-04-08 07:35:17 +08:00
|
|
|
if (*(Src->succ_begin()) == Dst) {
|
2009-02-11 07:56:07 +08:00
|
|
|
std::string sbuf;
|
|
|
|
llvm::raw_string_ostream os(sbuf);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-09-13 02:17:46 +08:00
|
|
|
os << "Loop condition is true. ";
|
2009-03-28 05:16:25 +08:00
|
|
|
PathDiagnosticLocation End = PDB.ExecutionContinues(os, N);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 05:16:25 +08:00
|
|
|
if (const Stmt *S = End.asStmt())
|
|
|
|
End = PDB.getEnclosingStmtLocation(S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-13 02:41:53 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
|
|
|
os.str()));
|
|
|
|
}
|
|
|
|
else {
|
2009-03-28 04:55:39 +08:00
|
|
|
PathDiagnosticLocation End = PDB.ExecutionContinues(N);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-28 05:16:25 +08:00
|
|
|
if (const Stmt *S = End.asStmt())
|
|
|
|
End = PDB.getEnclosingStmtLocation(S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
|
|
|
"Loop condition is false. Exiting loop"));
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
case Stmt::WhileStmtClass:
|
2009-09-09 23:08:12 +08:00
|
|
|
case Stmt::ForStmtClass: {
|
2009-04-01 07:00:32 +08:00
|
|
|
if (*(Src->succ_begin()+1) == Dst) {
|
|
|
|
std::string sbuf;
|
|
|
|
llvm::raw_string_ostream os(sbuf);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
os << "Loop condition is false. ";
|
|
|
|
PathDiagnosticLocation End = PDB.ExecutionContinues(os, N);
|
|
|
|
if (const Stmt *S = End.asStmt())
|
|
|
|
End = PDB.getEnclosingStmtLocation(S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
|
|
|
os.str()));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
PathDiagnosticLocation End = PDB.ExecutionContinues(N);
|
|
|
|
if (const Stmt *S = End.asStmt())
|
|
|
|
End = PDB.getEnclosingStmtLocation(S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
2009-04-01 14:13:56 +08:00
|
|
|
"Loop condition is true. Entering loop body"));
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
case Stmt::IfStmtClass: {
|
|
|
|
PathDiagnosticLocation End = PDB.ExecutionContinues(N);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (const Stmt *S = End.asStmt())
|
|
|
|
End = PDB.getEnclosingStmtLocation(S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (*(Src->succ_begin()+1) == Dst)
|
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
2009-04-01 14:13:56 +08:00
|
|
|
"Taking false branch"));
|
2009-09-09 23:08:12 +08:00
|
|
|
else
|
2009-04-01 07:00:32 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(Start, End,
|
2009-04-01 14:13:56 +08:00
|
|
|
"Taking true branch"));
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-07 08:45:33 +08:00
|
|
|
if (NextNode) {
|
|
|
|
for (BugReporterContext::visitor_iterator I = PDB.visitor_begin(),
|
|
|
|
E = PDB.visitor_end(); I!=E; ++I) {
|
|
|
|
if (PathDiagnosticPiece* p = (*I)->VisitNode(N, NextNode, PDB))
|
|
|
|
PD.push_front(p);
|
|
|
|
}
|
2009-05-07 05:39:49 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
if (const PostStmt* PS = dyn_cast<PostStmt>(&P)) {
|
2009-04-01 07:00:32 +08:00
|
|
|
// Scan the region bindings, and see if a "notable" symbol has a new
|
|
|
|
// lval binding.
|
|
|
|
ScanNotableSymbols SNS(N, PS->getStmt(), PDB.getBugReporter(), PD);
|
|
|
|
PDB.getStateManager().iterBindings(N->getState(), SNS);
|
|
|
|
}
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
// After constructing the full PathDiagnostic, do a pass over it to compact
|
|
|
|
// PathDiagnosticPieces that occur within a macro.
|
|
|
|
CompactPathDiagnostic(PD, PDB.getSourceManager());
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
|
|
|
|
2009-04-01 14:13:56 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// "Extensive" PathDiagnostic generation.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
static bool IsControlFlowExpr(const Stmt *S) {
|
|
|
|
const Expr *E = dyn_cast<Expr>(S);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 14:13:56 +08:00
|
|
|
if (!E)
|
|
|
|
return false;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
E = E->IgnoreParenCasts();
|
|
|
|
|
2011-02-17 18:25:35 +08:00
|
|
|
if (isa<AbstractConditionalOperator>(E))
|
2009-04-01 14:13:56 +08:00
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 14:13:56 +08:00
|
|
|
if (const BinaryOperator *B = dyn_cast<BinaryOperator>(E))
|
|
|
|
if (B->isLogicalOp())
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
return false;
|
2009-04-01 14:13:56 +08:00
|
|
|
}
|
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
namespace {
|
2009-11-28 14:07:30 +08:00
|
|
|
class ContextLocation : public PathDiagnosticLocation {
|
2009-05-02 00:08:09 +08:00
|
|
|
bool IsDead;
|
|
|
|
public:
|
|
|
|
ContextLocation(const PathDiagnosticLocation &L, bool isdead = false)
|
|
|
|
: PathDiagnosticLocation(L), IsDead(isdead) {}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
void markDead() { IsDead = true; }
|
2009-05-02 00:08:09 +08:00
|
|
|
bool isDead() const { return IsDead; }
|
|
|
|
};
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-28 14:07:30 +08:00
|
|
|
class EdgeBuilder {
|
2009-05-02 00:08:09 +08:00
|
|
|
std::vector<ContextLocation> CLocs;
|
|
|
|
typedef std::vector<ContextLocation>::iterator iterator;
|
2009-04-07 07:06:54 +08:00
|
|
|
PathDiagnostic &PD;
|
|
|
|
PathDiagnosticBuilder &PDB;
|
|
|
|
PathDiagnosticLocation PrevLoc;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-02 00:08:09 +08:00
|
|
|
bool IsConsumedExpr(const PathDiagnosticLocation &L);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
bool containsLocation(const PathDiagnosticLocation &Container,
|
|
|
|
const PathDiagnosticLocation &Containee);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
PathDiagnosticLocation getContextLocation(const PathDiagnosticLocation &L);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-12 05:42:34 +08:00
|
|
|
PathDiagnosticLocation cleanUpLocation(PathDiagnosticLocation L,
|
|
|
|
bool firstCharOnly = false) {
|
2009-05-12 03:50:47 +08:00
|
|
|
if (const Stmt *S = L.asStmt()) {
|
2009-05-12 05:42:34 +08:00
|
|
|
const Stmt *Original = S;
|
2009-05-12 03:50:47 +08:00
|
|
|
while (1) {
|
|
|
|
// Adjust the location for some expressions that are best referenced
|
|
|
|
// by one of their subexpressions.
|
2009-05-12 05:42:34 +08:00
|
|
|
switch (S->getStmtClass()) {
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
case Stmt::ParenExprClass:
|
2011-04-15 08:35:48 +08:00
|
|
|
case Stmt::GenericSelectionExprClass:
|
|
|
|
S = cast<Expr>(S)->IgnoreParens();
|
2009-05-12 05:42:34 +08:00
|
|
|
firstCharOnly = true;
|
|
|
|
continue;
|
2011-02-17 18:25:35 +08:00
|
|
|
case Stmt::BinaryConditionalOperatorClass:
|
2009-05-12 05:42:34 +08:00
|
|
|
case Stmt::ConditionalOperatorClass:
|
2011-02-17 18:25:35 +08:00
|
|
|
S = cast<AbstractConditionalOperator>(S)->getCond();
|
2009-05-12 05:42:34 +08:00
|
|
|
firstCharOnly = true;
|
|
|
|
continue;
|
|
|
|
case Stmt::ChooseExprClass:
|
|
|
|
S = cast<ChooseExpr>(S)->getCond();
|
|
|
|
firstCharOnly = true;
|
|
|
|
continue;
|
|
|
|
case Stmt::BinaryOperatorClass:
|
|
|
|
S = cast<BinaryOperator>(S)->getLHS();
|
|
|
|
firstCharOnly = true;
|
|
|
|
continue;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-12 05:42:34 +08:00
|
|
|
break;
|
2009-05-12 03:50:47 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-12 05:42:34 +08:00
|
|
|
if (S != Original)
|
|
|
|
L = PathDiagnosticLocation(S, L.getManager());
|
2009-05-12 03:50:47 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-12 05:42:34 +08:00
|
|
|
if (firstCharOnly)
|
|
|
|
L = PathDiagnosticLocation(L.asLocation());
|
|
|
|
|
2009-05-12 03:50:47 +08:00
|
|
|
return L;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
void popLocation() {
|
2009-05-02 00:08:09 +08:00
|
|
|
if (!CLocs.back().isDead() && CLocs.back().asLocation().isFileID()) {
|
2009-04-23 04:36:26 +08:00
|
|
|
// For contexts, we only one the first character as the range.
|
2009-05-15 10:46:13 +08:00
|
|
|
rawAddEdge(cleanUpLocation(CLocs.back(), true));
|
2009-04-23 04:36:26 +08:00
|
|
|
}
|
2009-04-07 07:06:54 +08:00
|
|
|
CLocs.pop_back();
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
public:
|
|
|
|
EdgeBuilder(PathDiagnostic &pd, PathDiagnosticBuilder &pdb)
|
|
|
|
: PD(pd), PDB(pdb) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-23 02:16:20 +08:00
|
|
|
// If the PathDiagnostic already has pieces, add the enclosing statement
|
|
|
|
// of the first piece as a context as well.
|
2009-04-07 07:06:54 +08:00
|
|
|
if (!PD.empty()) {
|
|
|
|
PrevLoc = PD.begin()->getLocation();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
if (const Stmt *S = PrevLoc.asStmt())
|
2009-05-06 07:13:38 +08:00
|
|
|
addExtendedContext(PDB.getEnclosingStmtLocation(S).asStmt());
|
2009-04-07 07:06:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
~EdgeBuilder() {
|
|
|
|
while (!CLocs.empty()) popLocation();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-23 02:16:20 +08:00
|
|
|
// Finally, add an initial edge from the start location of the first
|
|
|
|
// statement (if it doesn't already exist).
|
2009-04-27 04:35:05 +08:00
|
|
|
// FIXME: Should handle CXXTryStmt if analyser starts supporting C++.
|
|
|
|
if (const CompoundStmt *CS =
|
2010-07-07 19:31:34 +08:00
|
|
|
dyn_cast_or_null<CompoundStmt>(PDB.getCodeDecl().getBody()))
|
2009-04-23 02:16:20 +08:00
|
|
|
if (!CS->body_empty()) {
|
|
|
|
SourceLocation Loc = (*CS->body_begin())->getLocStart();
|
|
|
|
rawAddEdge(PathDiagnosticLocation(Loc, PDB.getSourceManager()));
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void addEdge(PathDiagnosticLocation NewLoc, bool alwaysAdd = false);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-28 12:23:15 +08:00
|
|
|
void rawAddEdge(PathDiagnosticLocation NewLoc);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
void addContext(const Stmt *S);
|
2009-05-06 07:13:38 +08:00
|
|
|
void addExtendedContext(const Stmt *S);
|
2009-09-09 23:08:12 +08:00
|
|
|
};
|
2009-04-07 07:06:54 +08:00
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
|
|
|
|
PathDiagnosticLocation
|
|
|
|
EdgeBuilder::getContextLocation(const PathDiagnosticLocation &L) {
|
|
|
|
if (const Stmt *S = L.asStmt()) {
|
|
|
|
if (IsControlFlowExpr(S))
|
|
|
|
return L;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
return PDB.getEnclosingStmtLocation(S);
|
2009-04-07 07:06:54 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
return L;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool EdgeBuilder::containsLocation(const PathDiagnosticLocation &Container,
|
|
|
|
const PathDiagnosticLocation &Containee) {
|
|
|
|
|
|
|
|
if (Container == Containee)
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
if (Container.asDecl())
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
if (const Stmt *S = Containee.asStmt())
|
|
|
|
if (const Stmt *ContainerS = Container.asStmt()) {
|
|
|
|
while (S) {
|
|
|
|
if (S == ContainerS)
|
|
|
|
return true;
|
|
|
|
S = PDB.getParent(S);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Less accurate: compare using source ranges.
|
|
|
|
SourceRange ContainerR = Container.asRange();
|
|
|
|
SourceRange ContaineeR = Containee.asRange();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
SourceManager &SM = PDB.getSourceManager();
|
2011-07-26 00:49:02 +08:00
|
|
|
SourceLocation ContainerRBeg = SM.getExpansionLoc(ContainerR.getBegin());
|
|
|
|
SourceLocation ContainerREnd = SM.getExpansionLoc(ContainerR.getEnd());
|
|
|
|
SourceLocation ContaineeRBeg = SM.getExpansionLoc(ContaineeR.getBegin());
|
|
|
|
SourceLocation ContaineeREnd = SM.getExpansionLoc(ContaineeR.getEnd());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-07-26 05:09:52 +08:00
|
|
|
unsigned ContainerBegLine = SM.getExpansionLineNumber(ContainerRBeg);
|
|
|
|
unsigned ContainerEndLine = SM.getExpansionLineNumber(ContainerREnd);
|
|
|
|
unsigned ContaineeBegLine = SM.getExpansionLineNumber(ContaineeRBeg);
|
|
|
|
unsigned ContaineeEndLine = SM.getExpansionLineNumber(ContaineeREnd);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
assert(ContainerBegLine <= ContainerEndLine);
|
2009-09-09 23:08:12 +08:00
|
|
|
assert(ContaineeBegLine <= ContaineeEndLine);
|
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
return (ContainerBegLine <= ContaineeBegLine &&
|
|
|
|
ContainerEndLine >= ContaineeEndLine &&
|
|
|
|
(ContainerBegLine != ContaineeBegLine ||
|
2011-07-26 04:57:57 +08:00
|
|
|
SM.getExpansionColumnNumber(ContainerRBeg) <=
|
|
|
|
SM.getExpansionColumnNumber(ContaineeRBeg)) &&
|
2009-04-07 07:06:54 +08:00
|
|
|
(ContainerEndLine != ContaineeEndLine ||
|
2011-07-26 04:57:57 +08:00
|
|
|
SM.getExpansionColumnNumber(ContainerREnd) >=
|
|
|
|
SM.getExpansionColumnNumber(ContainerREnd)));
|
2009-04-07 07:06:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void EdgeBuilder::rawAddEdge(PathDiagnosticLocation NewLoc) {
|
|
|
|
if (!PrevLoc.isValid()) {
|
|
|
|
PrevLoc = NewLoc;
|
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-12 03:50:47 +08:00
|
|
|
const PathDiagnosticLocation &NewLocClean = cleanUpLocation(NewLoc);
|
|
|
|
const PathDiagnosticLocation &PrevLocClean = cleanUpLocation(PrevLoc);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-12 03:50:47 +08:00
|
|
|
if (NewLocClean.asLocation() == PrevLocClean.asLocation())
|
2009-04-07 07:06:54 +08:00
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
// FIXME: Ignore intra-macro edges for now.
|
2011-07-26 00:49:02 +08:00
|
|
|
if (NewLocClean.asLocation().getExpansionLoc() ==
|
|
|
|
PrevLocClean.asLocation().getExpansionLoc())
|
2009-04-07 07:06:54 +08:00
|
|
|
return;
|
|
|
|
|
2009-05-12 03:50:47 +08:00
|
|
|
PD.push_front(new PathDiagnosticControlFlowPiece(NewLocClean, PrevLocClean));
|
|
|
|
PrevLoc = NewLoc;
|
2009-04-07 07:06:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void EdgeBuilder::addEdge(PathDiagnosticLocation NewLoc, bool alwaysAdd) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-23 02:16:20 +08:00
|
|
|
if (!alwaysAdd && NewLoc.asLocation().isMacroID())
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
const PathDiagnosticLocation &CLoc = getContextLocation(NewLoc);
|
|
|
|
|
|
|
|
while (!CLocs.empty()) {
|
2009-05-02 00:08:09 +08:00
|
|
|
ContextLocation &TopContextLoc = CLocs.back();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
// Is the top location context the same as the one for the new location?
|
|
|
|
if (TopContextLoc == CLoc) {
|
2009-05-02 00:08:09 +08:00
|
|
|
if (alwaysAdd) {
|
2009-05-05 02:15:17 +08:00
|
|
|
if (IsConsumedExpr(TopContextLoc) &&
|
|
|
|
!IsControlFlowExpr(TopContextLoc.asStmt()))
|
2009-05-02 00:08:09 +08:00
|
|
|
TopContextLoc.markDead();
|
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
rawAddEdge(NewLoc);
|
2009-05-02 00:08:09 +08:00
|
|
|
}
|
2009-04-07 07:06:54 +08:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (containsLocation(TopContextLoc, CLoc)) {
|
2009-05-02 00:08:09 +08:00
|
|
|
if (alwaysAdd) {
|
2009-04-07 07:06:54 +08:00
|
|
|
rawAddEdge(NewLoc);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-05 02:15:17 +08:00
|
|
|
if (IsConsumedExpr(CLoc) && !IsControlFlowExpr(CLoc.asStmt())) {
|
2009-05-02 00:08:09 +08:00
|
|
|
CLocs.push_back(ContextLocation(CLoc, true));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
CLocs.push_back(CLoc);
|
2009-09-09 23:08:12 +08:00
|
|
|
return;
|
2009-04-07 07:06:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Context does not contain the location. Flush it.
|
|
|
|
popLocation();
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-23 04:36:26 +08:00
|
|
|
// If we reach here, there is no enclosing context. Just add the edge.
|
|
|
|
rawAddEdge(NewLoc);
|
2009-04-07 07:06:54 +08:00
|
|
|
}
|
|
|
|
|
2009-05-02 00:08:09 +08:00
|
|
|
bool EdgeBuilder::IsConsumedExpr(const PathDiagnosticLocation &L) {
|
|
|
|
if (const Expr *X = dyn_cast_or_null<Expr>(L.asStmt()))
|
|
|
|
return PDB.getParentMap().isConsumedExpr(X) && !IsControlFlowExpr(X);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-02 00:08:09 +08:00
|
|
|
return false;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-06 07:13:38 +08:00
|
|
|
void EdgeBuilder::addExtendedContext(const Stmt *S) {
|
|
|
|
if (!S)
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
const Stmt *Parent = PDB.getParent(S);
|
2009-05-06 07:13:38 +08:00
|
|
|
while (Parent) {
|
|
|
|
if (isa<CompoundStmt>(Parent))
|
|
|
|
Parent = PDB.getParent(Parent);
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Parent) {
|
|
|
|
switch (Parent->getStmtClass()) {
|
|
|
|
case Stmt::DoStmtClass:
|
|
|
|
case Stmt::ObjCAtSynchronizedStmtClass:
|
|
|
|
addContext(Parent);
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-06 07:13:38 +08:00
|
|
|
addContext(S);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
void EdgeBuilder::addContext(const Stmt *S) {
|
|
|
|
if (!S)
|
|
|
|
return;
|
|
|
|
|
|
|
|
PathDiagnosticLocation L(S, PDB.getSourceManager());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-07 07:06:54 +08:00
|
|
|
while (!CLocs.empty()) {
|
|
|
|
const PathDiagnosticLocation &TopContextLoc = CLocs.back();
|
|
|
|
|
|
|
|
// Is the top location context the same as the one for the new location?
|
|
|
|
if (TopContextLoc == L)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (containsLocation(TopContextLoc, L)) {
|
|
|
|
CLocs.push_back(L);
|
2009-09-09 23:08:12 +08:00
|
|
|
return;
|
2009-04-07 07:06:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Context does not contain the location. Flush it.
|
|
|
|
popLocation();
|
|
|
|
}
|
|
|
|
|
|
|
|
CLocs.push_back(L);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void GenerateExtensivePathDiagnostic(PathDiagnostic& PD,
|
|
|
|
PathDiagnosticBuilder &PDB,
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode *N) {
|
2009-04-07 07:06:54 +08:00
|
|
|
EdgeBuilder EB(PD, PDB);
|
|
|
|
|
2010-03-23 13:13:26 +08:00
|
|
|
const ExplodedNode* NextNode = N->pred_empty() ? NULL : *(N->pred_begin());
|
2009-04-07 07:06:54 +08:00
|
|
|
while (NextNode) {
|
|
|
|
N = NextNode;
|
|
|
|
NextNode = GetPredecessorNode(N);
|
|
|
|
ProgramPoint P = N->getLocation();
|
|
|
|
|
2009-05-07 08:45:33 +08:00
|
|
|
do {
|
|
|
|
// Block edges.
|
|
|
|
if (const BlockEdge *BE = dyn_cast<BlockEdge>(&P)) {
|
|
|
|
const CFGBlock &Blk = *BE->getSrc();
|
|
|
|
const Stmt *Term = Blk.getTerminator();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-07 08:45:33 +08:00
|
|
|
// Are we jumping to the head of a loop? Add a special diagnostic.
|
2009-05-15 09:50:15 +08:00
|
|
|
if (const Stmt *Loop = BE->getDst()->getLoopTarget()) {
|
2009-05-07 08:45:33 +08:00
|
|
|
PathDiagnosticLocation L(Loop, PDB.getSourceManager());
|
2009-05-15 09:50:15 +08:00
|
|
|
const CompoundStmt *CS = NULL;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-15 09:50:15 +08:00
|
|
|
if (!Term) {
|
|
|
|
if (const ForStmt *FS = dyn_cast<ForStmt>(Loop))
|
|
|
|
CS = dyn_cast<CompoundStmt>(FS->getBody());
|
|
|
|
else if (const WhileStmt *WS = dyn_cast<WhileStmt>(Loop))
|
2009-09-09 23:08:12 +08:00
|
|
|
CS = dyn_cast<CompoundStmt>(WS->getBody());
|
2009-05-15 09:50:15 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-07 08:45:33 +08:00
|
|
|
PathDiagnosticEventPiece *p =
|
|
|
|
new PathDiagnosticEventPiece(L,
|
2009-05-15 10:46:13 +08:00
|
|
|
"Looping back to the head of the loop");
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-07 08:45:33 +08:00
|
|
|
EB.addEdge(p->getLocation(), true);
|
|
|
|
PD.push_front(p);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-15 09:50:15 +08:00
|
|
|
if (CS) {
|
2009-05-15 10:46:13 +08:00
|
|
|
PathDiagnosticLocation BL(CS->getRBracLoc(),
|
|
|
|
PDB.getSourceManager());
|
|
|
|
BL = PathDiagnosticLocation(BL.asLocation());
|
|
|
|
EB.addEdge(BL);
|
2009-05-07 08:45:33 +08:00
|
|
|
}
|
2009-04-28 12:23:15 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-15 09:50:15 +08:00
|
|
|
if (Term)
|
|
|
|
EB.addContext(Term);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-07 08:45:33 +08:00
|
|
|
break;
|
2009-04-28 12:23:15 +08:00
|
|
|
}
|
2009-04-07 07:06:54 +08:00
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
if (const BlockEntrance *BE = dyn_cast<BlockEntrance>(&P)) {
|
2011-03-01 11:15:10 +08:00
|
|
|
if (const CFGStmt *S = BE->getFirstElement().getAs<CFGStmt>()) {
|
|
|
|
const Stmt *stmt = S->getStmt();
|
|
|
|
if (IsControlFlowExpr(stmt)) {
|
2010-09-16 09:25:47 +08:00
|
|
|
// Add the proper context for '&&', '||', and '?'.
|
2011-03-01 11:15:10 +08:00
|
|
|
EB.addContext(stmt);
|
2010-09-16 09:25:47 +08:00
|
|
|
}
|
|
|
|
else
|
2011-03-01 11:15:10 +08:00
|
|
|
EB.addExtendedContext(PDB.getEnclosingStmtLocation(stmt).asStmt());
|
2009-05-07 08:45:33 +08:00
|
|
|
}
|
2010-09-16 09:25:47 +08:00
|
|
|
|
2009-05-07 08:45:33 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while (0);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-07 08:45:33 +08:00
|
|
|
if (!NextNode)
|
2009-04-07 07:06:54 +08:00
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-07 05:39:49 +08:00
|
|
|
for (BugReporterContext::visitor_iterator I = PDB.visitor_begin(),
|
|
|
|
E = PDB.visitor_end(); I!=E; ++I) {
|
|
|
|
if (PathDiagnosticPiece* p = (*I)->VisitNode(N, NextNode, PDB)) {
|
|
|
|
const PathDiagnosticLocation &Loc = p->getLocation();
|
|
|
|
EB.addEdge(Loc, true);
|
|
|
|
PD.push_front(p);
|
|
|
|
if (const Stmt *S = Loc.asStmt())
|
2009-09-09 23:08:12 +08:00
|
|
|
EB.addExtendedContext(PDB.getEnclosingStmtLocation(S).asStmt());
|
2009-05-07 05:39:49 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-04-07 07:06:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Methods for BugType and subclasses.
|
|
|
|
//===----------------------------------------------------------------------===//
|
2011-02-23 08:16:01 +08:00
|
|
|
BugType::~BugType() { }
|
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
void BugType::FlushReports(BugReporter &BR) {}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Methods for BugReport and subclasses.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
BugReport::~BugReport() {}
|
|
|
|
RangedBugReport::~RangedBugReport() {}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
const Stmt* BugReport::getStmt() const {
|
2010-09-16 11:50:38 +08:00
|
|
|
ProgramPoint ProgP = ErrorNode->getLocation();
|
2009-07-23 06:35:28 +08:00
|
|
|
const Stmt *S = NULL;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (BlockEntrance* BE = dyn_cast<BlockEntrance>(&ProgP)) {
|
2009-08-20 09:23:34 +08:00
|
|
|
CFGBlock &Exit = ProgP.getLocationContext()->getCFG()->getExit();
|
2009-08-18 16:46:04 +08:00
|
|
|
if (BE->getBlock() == &Exit)
|
2010-09-16 11:50:38 +08:00
|
|
|
S = GetPreviousStmt(ErrorNode);
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
2009-07-23 06:35:28 +08:00
|
|
|
if (!S)
|
2009-09-09 23:08:12 +08:00
|
|
|
S = GetStmt(ProgP);
|
|
|
|
|
|
|
|
return S;
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
PathDiagnosticPiece*
|
2009-05-07 05:39:49 +08:00
|
|
|
BugReport::getEndPath(BugReporterContext& BRC,
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode* EndPathNode) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-08-18 16:46:04 +08:00
|
|
|
const Stmt* S = getStmt();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!S)
|
|
|
|
return NULL;
|
2009-05-12 07:50:59 +08:00
|
|
|
|
2010-12-04 09:12:15 +08:00
|
|
|
BugReport::ranges_iterator Beg, End;
|
|
|
|
llvm::tie(Beg, End) = getRanges();
|
2009-05-12 07:50:59 +08:00
|
|
|
PathDiagnosticLocation L(S, BRC.getSourceManager());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-12 07:50:59 +08:00
|
|
|
// Only add the statement itself as a range if we didn't specify any
|
|
|
|
// special ranges for this report.
|
|
|
|
PathDiagnosticPiece* P = new PathDiagnosticEventPiece(L, getDescription(),
|
|
|
|
Beg == End);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
for (; Beg != End; ++Beg)
|
|
|
|
P->addRange(*Beg);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
return P;
|
|
|
|
}
|
|
|
|
|
2010-12-04 09:12:15 +08:00
|
|
|
std::pair<BugReport::ranges_iterator, BugReport::ranges_iterator>
|
|
|
|
BugReport::getRanges() const {
|
2009-08-18 16:46:04 +08:00
|
|
|
if (const Expr* E = dyn_cast_or_null<Expr>(getStmt())) {
|
2009-04-01 07:00:32 +08:00
|
|
|
R = E->getSourceRange();
|
|
|
|
assert(R.isValid());
|
2010-12-04 09:12:15 +08:00
|
|
|
return std::make_pair(&R, &R+1);
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
|
|
|
else
|
2010-12-04 09:12:15 +08:00
|
|
|
return std::make_pair(ranges_iterator(), ranges_iterator());
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
SourceLocation BugReport::getLocation() const {
|
2010-09-16 11:50:38 +08:00
|
|
|
if (ErrorNode)
|
|
|
|
if (const Stmt* S = GetCurrentOrPreviousStmt(ErrorNode)) {
|
2009-04-01 07:00:32 +08:00
|
|
|
// For member expressions, return the location of the '.' or '->'.
|
2009-09-12 06:07:28 +08:00
|
|
|
if (const MemberExpr *ME = dyn_cast<MemberExpr>(S))
|
2009-04-01 07:00:32 +08:00
|
|
|
return ME->getMemberLoc();
|
2009-09-12 06:07:28 +08:00
|
|
|
// For binary operators, return the location of the operator.
|
|
|
|
if (const BinaryOperator *B = dyn_cast<BinaryOperator>(S))
|
|
|
|
return B->getOperatorLoc();
|
2009-04-01 07:00:32 +08:00
|
|
|
|
|
|
|
return S->getLocStart();
|
|
|
|
}
|
|
|
|
|
|
|
|
return FullSourceLoc();
|
|
|
|
}
|
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
PathDiagnosticPiece* BugReport::VisitNode(const ExplodedNode* N,
|
|
|
|
const ExplodedNode* PrevN,
|
2009-05-07 05:39:49 +08:00
|
|
|
BugReporterContext &BRC) {
|
2009-04-01 07:00:32 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Methods for BugReporter and subclasses.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
BugReportEquivClass::~BugReportEquivClass() {
|
2009-05-07 05:39:49 +08:00
|
|
|
for (iterator I=begin(), E=end(); I!=E; ++I) delete *I;
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
|
|
|
|
2009-09-05 14:06:49 +08:00
|
|
|
GRBugReporter::~GRBugReporter() { }
|
2009-04-01 07:00:32 +08:00
|
|
|
BugReporterData::~BugReporterData() {}
|
|
|
|
|
2009-08-06 14:28:40 +08:00
|
|
|
ExplodedGraph &GRBugReporter::getGraph() { return Eng.getGraph(); }
|
2009-04-01 07:00:32 +08:00
|
|
|
|
|
|
|
GRStateManager&
|
|
|
|
GRBugReporter::getStateManager() { return Eng.getStateManager(); }
|
|
|
|
|
|
|
|
BugReporter::~BugReporter() { FlushReports(); }
|
|
|
|
|
|
|
|
void BugReporter::FlushReports() {
|
|
|
|
if (BugTypes.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
// First flush the warnings for each BugType. This may end up creating new
|
2011-02-23 08:16:01 +08:00
|
|
|
// warnings and new BugTypes.
|
|
|
|
// FIXME: Only NSErrorChecker needs BugType's FlushReports.
|
|
|
|
// Turn NSErrorChecker into a proper checker and remove this.
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<const BugType*, 16> bugTypes;
|
2009-04-01 07:00:32 +08:00
|
|
|
for (BugTypesTy::iterator I=BugTypes.begin(), E=BugTypes.end(); I!=E; ++I)
|
2011-02-23 08:16:01 +08:00
|
|
|
bugTypes.push_back(*I);
|
2011-07-23 18:55:15 +08:00
|
|
|
for (SmallVector<const BugType*, 16>::iterator
|
2011-02-23 08:16:01 +08:00
|
|
|
I = bugTypes.begin(), E = bugTypes.end(); I != E; ++I)
|
2009-04-01 07:00:32 +08:00
|
|
|
const_cast<BugType*>(*I)->FlushReports(*this);
|
|
|
|
|
2011-02-23 08:16:01 +08:00
|
|
|
typedef llvm::FoldingSet<BugReportEquivClass> SetTy;
|
|
|
|
for (SetTy::iterator EI=EQClasses.begin(), EE=EQClasses.end(); EI!=EE;++EI){
|
|
|
|
BugReportEquivClass& EQ = *EI;
|
|
|
|
FlushReport(EQ);
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
|
|
|
|
2011-02-23 08:16:01 +08:00
|
|
|
// BugReporter owns and deletes only BugTypes created implicitly through
|
|
|
|
// EmitBasicReport.
|
|
|
|
// FIXME: There are leaks from checkers that assume that the BugTypes they
|
|
|
|
// create will be destroyed by the BugReporter.
|
|
|
|
for (llvm::StringMap<BugType*>::iterator
|
|
|
|
I = StrBugTypes.begin(), E = StrBugTypes.end(); I != E; ++I)
|
|
|
|
delete I->second;
|
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Remove all references to the BugType objects.
|
2010-11-24 08:54:37 +08:00
|
|
|
BugTypes = F.getEmptySet();
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// PathDiagnostics generation.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2009-08-06 14:28:40 +08:00
|
|
|
static std::pair<std::pair<ExplodedGraph*, NodeBackMap*>,
|
2009-08-06 09:32:16 +08:00
|
|
|
std::pair<ExplodedNode*, unsigned> >
|
2009-08-06 14:28:40 +08:00
|
|
|
MakeReportGraph(const ExplodedGraph* G,
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVectorImpl<const ExplodedNode*> &nodes) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Create the trimmed graph. It will contain the shortest paths from the
|
2009-09-09 23:08:12 +08:00
|
|
|
// error nodes to the root. In the new graph we should only have one
|
2009-04-01 07:00:32 +08:00
|
|
|
// error node unless there are two or more error nodes with the same minimum
|
|
|
|
// path length.
|
2009-08-06 14:28:40 +08:00
|
|
|
ExplodedGraph* GTrim;
|
2009-08-06 09:32:16 +08:00
|
|
|
InterExplodedGraphMap* NMap;
|
2009-04-01 07:00:32 +08:00
|
|
|
|
|
|
|
llvm::DenseMap<const void*, const void*> InverseMap;
|
2010-12-03 14:52:30 +08:00
|
|
|
llvm::tie(GTrim, NMap) = G->Trim(nodes.data(), nodes.data() + nodes.size(),
|
|
|
|
&InverseMap);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Create owning pointers for GTrim and NMap just to ensure that they are
|
|
|
|
// released when this function exists.
|
2009-08-06 14:28:40 +08:00
|
|
|
llvm::OwningPtr<ExplodedGraph> AutoReleaseGTrim(GTrim);
|
2009-08-06 09:32:16 +08:00
|
|
|
llvm::OwningPtr<InterExplodedGraphMap> AutoReleaseNMap(NMap);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Find the (first) error node in the trimmed graph. We just need to consult
|
|
|
|
// the node map (NMap) which maps from nodes in the original graph to nodes
|
|
|
|
// in the new graph.
|
2009-05-16 09:11:58 +08:00
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
std::queue<const ExplodedNode*> WS;
|
2009-08-06 14:28:40 +08:00
|
|
|
typedef llvm::DenseMap<const ExplodedNode*, unsigned> IndexMapTy;
|
2009-05-16 09:11:58 +08:00
|
|
|
IndexMapTy IndexMap;
|
2009-04-01 07:00:32 +08:00
|
|
|
|
2010-12-03 14:52:30 +08:00
|
|
|
for (unsigned nodeIndex = 0 ; nodeIndex < nodes.size(); ++nodeIndex) {
|
|
|
|
const ExplodedNode *originalNode = nodes[nodeIndex];
|
|
|
|
if (const ExplodedNode *N = NMap->getMappedNode(originalNode)) {
|
2009-05-16 09:11:58 +08:00
|
|
|
WS.push(N);
|
2010-12-03 14:52:30 +08:00
|
|
|
IndexMap[originalNode] = nodeIndex;
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
2010-12-03 14:52:30 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-16 09:11:58 +08:00
|
|
|
assert(!WS.empty() && "No error node found in the trimmed graph.");
|
2009-04-01 07:00:32 +08:00
|
|
|
|
|
|
|
// Create a new (third!) graph with a single path. This is the graph
|
|
|
|
// that will be returned to the caller.
|
2010-07-01 15:10:59 +08:00
|
|
|
ExplodedGraph *GNew = new ExplodedGraph();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Sometimes the trimmed graph can contain a cycle. Perform a reverse BFS
|
|
|
|
// to the root node, and then construct a new graph that contains only
|
|
|
|
// a single path.
|
|
|
|
llvm::DenseMap<const void*,unsigned> Visited;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
unsigned cnt = 0;
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode* Root = 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
while (!WS.empty()) {
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode* Node = WS.front();
|
2009-04-01 07:00:32 +08:00
|
|
|
WS.pop();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (Visited.find(Node) != Visited.end())
|
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
Visited[Node] = cnt++;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (Node->pred_empty()) {
|
|
|
|
Root = Node;
|
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
for (ExplodedNode::const_pred_iterator I=Node->pred_begin(),
|
2009-04-01 07:00:32 +08:00
|
|
|
E=Node->pred_end(); I!=E; ++I)
|
|
|
|
WS.push(*I);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-16 09:11:58 +08:00
|
|
|
assert(Root);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Now walk from the root down the BFS path, always taking the successor
|
|
|
|
// with the lowest number.
|
2009-09-09 23:08:12 +08:00
|
|
|
ExplodedNode *Last = 0, *First = 0;
|
2009-04-01 07:00:32 +08:00
|
|
|
NodeBackMap *BM = new NodeBackMap();
|
2009-05-16 09:11:58 +08:00
|
|
|
unsigned NodeIndex = 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-08-06 09:32:16 +08:00
|
|
|
for ( const ExplodedNode *N = Root ;;) {
|
2009-04-01 07:00:32 +08:00
|
|
|
// Lookup the number associated with the current node.
|
|
|
|
llvm::DenseMap<const void*,unsigned>::iterator I = Visited.find(N);
|
2009-05-16 09:11:58 +08:00
|
|
|
assert(I != Visited.end());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Create the equivalent node in the new graph with the same state
|
|
|
|
// and location.
|
2009-08-06 14:28:40 +08:00
|
|
|
ExplodedNode* NewN = GNew->getNode(N->getLocation(), N->getState());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Store the mapping to the original node.
|
|
|
|
llvm::DenseMap<const void*, const void*>::iterator IMitr=InverseMap.find(N);
|
|
|
|
assert(IMitr != InverseMap.end() && "No mapping to original node.");
|
2009-08-06 09:32:16 +08:00
|
|
|
(*BM)[NewN] = (const ExplodedNode*) IMitr->second;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Link up the new node with the previous node.
|
|
|
|
if (Last)
|
2009-10-07 08:42:52 +08:00
|
|
|
NewN->addPredecessor(Last, *GNew);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
Last = NewN;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Are we at the final node?
|
2009-05-16 09:11:58 +08:00
|
|
|
IndexMapTy::iterator IMI =
|
2009-08-06 09:32:16 +08:00
|
|
|
IndexMap.find((const ExplodedNode*)(IMitr->second));
|
2009-05-16 09:11:58 +08:00
|
|
|
if (IMI != IndexMap.end()) {
|
2009-04-01 07:00:32 +08:00
|
|
|
First = NewN;
|
2009-05-16 09:11:58 +08:00
|
|
|
NodeIndex = IMI->second;
|
2009-04-01 07:00:32 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Find the next successor node. We choose the node that is marked
|
|
|
|
// with the lowest DFS number.
|
2009-08-06 09:32:16 +08:00
|
|
|
ExplodedNode::const_succ_iterator SI = N->succ_begin();
|
|
|
|
ExplodedNode::const_succ_iterator SE = N->succ_end();
|
2009-04-01 07:00:32 +08:00
|
|
|
N = 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
for (unsigned MinVal = 0; SI != SE; ++SI) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
I = Visited.find(*SI);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (I == Visited.end())
|
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!N || I->second < MinVal) {
|
|
|
|
N = *SI;
|
|
|
|
MinVal = I->second;
|
|
|
|
}
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-16 09:11:58 +08:00
|
|
|
assert(N);
|
2009-04-01 07:00:32 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-16 09:11:58 +08:00
|
|
|
assert(First);
|
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
return std::make_pair(std::make_pair(GNew, BM),
|
|
|
|
std::make_pair(First, NodeIndex));
|
|
|
|
}
|
2009-03-28 05:16:25 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
/// CompactPathDiagnostic - This function postprocesses a PathDiagnostic object
|
|
|
|
/// and collapses PathDiagosticPieces that are expanded by macros.
|
|
|
|
static void CompactPathDiagnostic(PathDiagnostic &PD, const SourceManager& SM) {
|
|
|
|
typedef std::vector<std::pair<PathDiagnosticMacroPiece*, SourceLocation> >
|
|
|
|
MacroStackTy;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
typedef std::vector<PathDiagnosticPiece*>
|
|
|
|
PiecesTy;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
MacroStackTy MacroStack;
|
|
|
|
PiecesTy Pieces;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
for (PathDiagnostic::iterator I = PD.begin(), E = PD.end(); I!=E; ++I) {
|
|
|
|
// Get the location of the PathDiagnosticPiece.
|
2009-09-09 23:08:12 +08:00
|
|
|
const FullSourceLoc Loc = I->getLocation().asLocation();
|
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Determine the instantiation location, which is the location we group
|
|
|
|
// related PathDiagnosticPieces.
|
2009-09-09 23:08:12 +08:00
|
|
|
SourceLocation InstantiationLoc = Loc.isMacroID() ?
|
2011-07-26 00:49:02 +08:00
|
|
|
SM.getExpansionLoc(Loc) :
|
2009-04-01 07:00:32 +08:00
|
|
|
SourceLocation();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (Loc.isFileID()) {
|
|
|
|
MacroStack.clear();
|
|
|
|
Pieces.push_back(&*I);
|
|
|
|
continue;
|
|
|
|
}
|
2008-04-08 07:35:17 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
assert(Loc.isMacroID());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Is the PathDiagnosticPiece within the same macro group?
|
|
|
|
if (!MacroStack.empty() && InstantiationLoc == MacroStack.back().second) {
|
|
|
|
MacroStack.back().first->push_back(&*I);
|
|
|
|
continue;
|
|
|
|
}
|
2009-03-28 05:16:25 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// We aren't in the same group. Are we descending into a new macro
|
|
|
|
// or are part of an old one?
|
|
|
|
PathDiagnosticMacroPiece *MacroGroup = 0;
|
2009-03-28 05:16:25 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
SourceLocation ParentInstantiationLoc = InstantiationLoc.isMacroID() ?
|
2011-07-26 00:49:02 +08:00
|
|
|
SM.getExpansionLoc(Loc) :
|
2009-04-01 07:00:32 +08:00
|
|
|
SourceLocation();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Walk the entire macro stack.
|
|
|
|
while (!MacroStack.empty()) {
|
|
|
|
if (InstantiationLoc == MacroStack.back().second) {
|
|
|
|
MacroGroup = MacroStack.back().first;
|
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (ParentInstantiationLoc == MacroStack.back().second) {
|
|
|
|
MacroGroup = MacroStack.back().first;
|
|
|
|
break;
|
2008-04-03 12:42:52 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
MacroStack.pop_back();
|
2008-04-09 08:20:43 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (!MacroGroup || ParentInstantiationLoc == MacroStack.back().second) {
|
|
|
|
// Create a new macro group and add it to the stack.
|
|
|
|
PathDiagnosticMacroPiece *NewGroup = new PathDiagnosticMacroPiece(Loc);
|
2008-04-24 07:35:07 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
if (MacroGroup)
|
|
|
|
MacroGroup->push_back(NewGroup);
|
|
|
|
else {
|
|
|
|
assert(InstantiationLoc.isFileID());
|
|
|
|
Pieces.push_back(NewGroup);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
MacroGroup = NewGroup;
|
|
|
|
MacroStack.push_back(std::make_pair(MacroGroup, InstantiationLoc));
|
2009-04-01 04:22:36 +08:00
|
|
|
}
|
2009-04-01 07:00:32 +08:00
|
|
|
|
|
|
|
// Finally, add the PathDiagnosticPiece to the group.
|
|
|
|
MacroGroup->push_back(&*I);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Now take the pieces and construct a new PathDiagnostic.
|
|
|
|
PD.resetPath(false);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
for (PiecesTy::iterator I=Pieces.begin(), E=Pieces.end(); I!=E; ++I) {
|
|
|
|
if (PathDiagnosticMacroPiece *MP=dyn_cast<PathDiagnosticMacroPiece>(*I))
|
|
|
|
if (!MP->containsEvent()) {
|
|
|
|
delete MP;
|
|
|
|
continue;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
PD.push_back(*I);
|
2008-04-03 12:42:52 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
void GRBugReporter::GeneratePathDiagnostic(PathDiagnostic& PD,
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVectorImpl<BugReport *> &bugReports) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-12-03 14:52:30 +08:00
|
|
|
assert(!bugReports.empty());
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<const ExplodedNode *, 10> errorNodes;
|
|
|
|
for (SmallVectorImpl<BugReport*>::iterator I = bugReports.begin(),
|
2010-12-03 14:52:30 +08:00
|
|
|
E = bugReports.end(); I != E; ++I) {
|
|
|
|
errorNodes.push_back((*I)->getErrorNode());
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Construct a new graph that contains only a single path from the error
|
2009-09-09 23:08:12 +08:00
|
|
|
// node to a root.
|
2009-08-06 14:28:40 +08:00
|
|
|
const std::pair<std::pair<ExplodedGraph*, NodeBackMap*>,
|
2009-08-06 09:32:16 +08:00
|
|
|
std::pair<ExplodedNode*, unsigned> >&
|
2010-12-03 14:52:30 +08:00
|
|
|
GPair = MakeReportGraph(&getGraph(), errorNodes);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
// Find the BugReport with the original location.
|
2010-12-03 14:52:30 +08:00
|
|
|
assert(GPair.second.second < bugReports.size());
|
|
|
|
BugReport *R = bugReports[GPair.second.second];
|
2009-04-01 07:00:32 +08:00
|
|
|
assert(R && "No original report found for sliced graph.");
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-08-06 14:28:40 +08:00
|
|
|
llvm::OwningPtr<ExplodedGraph> ReportGraph(GPair.first.first);
|
2009-04-01 07:00:32 +08:00
|
|
|
llvm::OwningPtr<NodeBackMap> BackMap(GPair.first.second);
|
2009-08-06 09:32:16 +08:00
|
|
|
const ExplodedNode *N = GPair.second.first;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
// Start building the path diagnostic...
|
2009-05-07 05:39:49 +08:00
|
|
|
PathDiagnosticBuilder PDB(*this, R, BackMap.get(), getPathDiagnosticClient());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-07 05:39:49 +08:00
|
|
|
if (PathDiagnosticPiece* Piece = R->getEndPath(PDB, N))
|
2009-04-01 07:00:32 +08:00
|
|
|
PD.push_back(Piece);
|
|
|
|
else
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-03-21 02:02:01 +08:00
|
|
|
// Register node visitors.
|
2009-05-07 08:45:33 +08:00
|
|
|
R->registerInitialVisitors(PDB, N);
|
2010-03-21 02:02:01 +08:00
|
|
|
bugreporter::registerNilReceiverVisitor(PDB);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-01 07:00:32 +08:00
|
|
|
switch (PDB.getGenerationScheme()) {
|
|
|
|
case PathDiagnosticClient::Extensive:
|
2009-05-07 05:39:49 +08:00
|
|
|
GenerateExtensivePathDiagnostic(PD, PDB, N);
|
2009-04-01 14:13:56 +08:00
|
|
|
break;
|
2009-04-01 07:00:32 +08:00
|
|
|
case PathDiagnosticClient::Minimal:
|
|
|
|
GenerateMinimalPathDiagnostic(PD, PDB, N);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2008-05-23 07:45:19 +08:00
|
|
|
|
2009-02-05 07:49:09 +08:00
|
|
|
void BugReporter::Register(BugType *BT) {
|
2010-11-24 08:54:37 +08:00
|
|
|
BugTypes = F.add(BugTypes, BT);
|
2008-05-17 02:33:14 +08:00
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
void BugReporter::EmitReport(BugReport* R) {
|
2009-02-05 07:49:09 +08:00
|
|
|
// Compute the bug report's hash to determine its equivalence class.
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
R->Profile(ID);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
// Lookup the equivance class. If there isn't one, create it.
|
2009-02-05 07:49:09 +08:00
|
|
|
BugType& BT = R->getBugType();
|
|
|
|
Register(&BT);
|
|
|
|
void *InsertPos;
|
2011-02-23 08:16:01 +08:00
|
|
|
BugReportEquivClass* EQ = EQClasses.FindNodeOrInsertPos(ID, InsertPos);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-05 07:49:09 +08:00
|
|
|
if (!EQ) {
|
|
|
|
EQ = new BugReportEquivClass(R);
|
2011-02-23 08:16:01 +08:00
|
|
|
EQClasses.InsertNode(EQ, InsertPos);
|
2009-02-05 07:49:09 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
EQ->AddReport(R);
|
2008-04-03 12:42:52 +08:00
|
|
|
}
|
|
|
|
|
2009-09-15 06:01:32 +08:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Emitting reports in equivalence classes.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
namespace {
|
2009-11-28 14:07:30 +08:00
|
|
|
struct FRIEC_WLItem {
|
2009-09-15 06:01:32 +08:00
|
|
|
const ExplodedNode *N;
|
|
|
|
ExplodedNode::const_succ_iterator I, E;
|
|
|
|
|
|
|
|
FRIEC_WLItem(const ExplodedNode *n)
|
|
|
|
: N(n), I(N->succ_begin()), E(N->succ_end()) {}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2010-09-10 03:05:34 +08:00
|
|
|
static BugReport *
|
|
|
|
FindReportInEquivalenceClass(BugReportEquivClass& EQ,
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVectorImpl<BugReport*> &bugReports) {
|
2010-09-10 03:05:34 +08:00
|
|
|
|
2009-09-15 06:01:32 +08:00
|
|
|
BugReportEquivClass::iterator I = EQ.begin(), E = EQ.end();
|
|
|
|
assert(I != E);
|
|
|
|
BugReport *R = *I;
|
|
|
|
BugType& BT = R->getBugType();
|
2010-09-10 03:05:34 +08:00
|
|
|
|
2010-12-03 14:52:30 +08:00
|
|
|
// If we don't need to suppress any of the nodes because they are
|
|
|
|
// post-dominated by a sink, simply add all the nodes in the equivalence class
|
|
|
|
// to 'Nodes'. Any of the reports will serve as a "representative" report.
|
2010-09-10 03:05:34 +08:00
|
|
|
if (!BT.isSuppressOnSink()) {
|
|
|
|
for (BugReportEquivClass::iterator I=EQ.begin(), E=EQ.end(); I!=E; ++I) {
|
2010-09-16 11:50:38 +08:00
|
|
|
const ExplodedNode* N = I->getErrorNode();
|
2010-09-10 03:05:34 +08:00
|
|
|
if (N) {
|
|
|
|
R = *I;
|
2010-12-03 14:52:30 +08:00
|
|
|
bugReports.push_back(R);
|
2010-09-10 03:05:34 +08:00
|
|
|
}
|
|
|
|
}
|
2009-09-15 06:01:32 +08:00
|
|
|
return R;
|
2010-09-10 03:05:34 +08:00
|
|
|
}
|
|
|
|
|
2009-09-15 06:01:32 +08:00
|
|
|
// For bug reports that should be suppressed when all paths are post-dominated
|
|
|
|
// by a sink node, iterate through the reports in the equivalence class
|
|
|
|
// until we find one that isn't post-dominated (if one exists). We use a
|
|
|
|
// DFS traversal of the ExplodedGraph to find a non-sink node. We could write
|
|
|
|
// this as a recursive function, but we don't want to risk blowing out the
|
|
|
|
// stack for very long paths.
|
2010-12-03 14:52:30 +08:00
|
|
|
BugReport *exampleReport = 0;
|
2010-09-10 03:05:34 +08:00
|
|
|
|
2009-09-15 06:01:32 +08:00
|
|
|
for (; I != E; ++I) {
|
|
|
|
R = *I;
|
2010-12-03 14:52:30 +08:00
|
|
|
const ExplodedNode *errorNode = R->getErrorNode();
|
2009-09-15 06:01:32 +08:00
|
|
|
|
2010-12-03 14:52:30 +08:00
|
|
|
if (!errorNode)
|
2009-09-15 06:01:32 +08:00
|
|
|
continue;
|
2010-12-03 14:52:30 +08:00
|
|
|
if (errorNode->isSink()) {
|
2009-09-15 06:01:32 +08:00
|
|
|
assert(false &&
|
|
|
|
"BugType::isSuppressSink() should not be 'true' for sink end nodes");
|
2010-09-10 03:05:34 +08:00
|
|
|
return 0;
|
2009-09-15 06:01:32 +08:00
|
|
|
}
|
2010-09-10 03:05:34 +08:00
|
|
|
// No successors? By definition this nodes isn't post-dominated by a sink.
|
2010-12-03 14:52:30 +08:00
|
|
|
if (errorNode->succ_empty()) {
|
|
|
|
bugReports.push_back(R);
|
|
|
|
if (!exampleReport)
|
|
|
|
exampleReport = R;
|
2010-09-10 03:05:34 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2009-09-15 06:01:32 +08:00
|
|
|
// At this point we know that 'N' is not a sink and it has at least one
|
|
|
|
// successor. Use a DFS worklist to find a non-sink end-of-path node.
|
|
|
|
typedef FRIEC_WLItem WLItem;
|
2011-07-23 18:55:15 +08:00
|
|
|
typedef SmallVector<WLItem, 10> DFSWorkList;
|
2009-09-15 06:01:32 +08:00
|
|
|
llvm::DenseMap<const ExplodedNode *, unsigned> Visited;
|
|
|
|
|
|
|
|
DFSWorkList WL;
|
2010-12-03 14:52:30 +08:00
|
|
|
WL.push_back(errorNode);
|
|
|
|
Visited[errorNode] = 1;
|
2009-09-15 06:01:32 +08:00
|
|
|
|
|
|
|
while (!WL.empty()) {
|
|
|
|
WLItem &WI = WL.back();
|
|
|
|
assert(!WI.N->succ_empty());
|
|
|
|
|
|
|
|
for (; WI.I != WI.E; ++WI.I) {
|
|
|
|
const ExplodedNode *Succ = *WI.I;
|
|
|
|
// End-of-path node?
|
|
|
|
if (Succ->succ_empty()) {
|
2010-09-10 03:05:34 +08:00
|
|
|
// If we found an end-of-path node that is not a sink.
|
|
|
|
if (!Succ->isSink()) {
|
2010-12-03 14:52:30 +08:00
|
|
|
bugReports.push_back(R);
|
|
|
|
if (!exampleReport)
|
|
|
|
exampleReport = R;
|
2010-09-10 03:05:34 +08:00
|
|
|
WL.clear();
|
|
|
|
break;
|
|
|
|
}
|
2009-09-15 06:01:32 +08:00
|
|
|
// Found a sink? Continue on to the next successor.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// Mark the successor as visited. If it hasn't been explored,
|
|
|
|
// enqueue it to the DFS worklist.
|
|
|
|
unsigned &mark = Visited[Succ];
|
|
|
|
if (!mark) {
|
|
|
|
mark = 1;
|
|
|
|
WL.push_back(Succ);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2010-09-10 03:05:34 +08:00
|
|
|
|
|
|
|
// The worklist may have been cleared at this point. First
|
|
|
|
// check if it is empty before checking the last item.
|
|
|
|
if (!WL.empty() && &WL.back() == &WI)
|
2009-09-15 06:01:32 +08:00
|
|
|
WL.pop_back();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-10 03:05:34 +08:00
|
|
|
// ExampleReport will be NULL if all the nodes in the equivalence class
|
|
|
|
// were post-dominated by sinks.
|
2010-12-03 14:52:30 +08:00
|
|
|
return exampleReport;
|
2010-09-10 03:05:34 +08:00
|
|
|
}
|
2009-09-19 06:37:37 +08:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// DiagnosticCache. This is a hack to cache analyzer diagnostics. It
|
|
|
|
// uses global state, which eventually should go elsewhere.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
namespace {
|
2009-11-28 14:07:30 +08:00
|
|
|
class DiagCacheItem : public llvm::FoldingSetNode {
|
2009-09-19 06:37:37 +08:00
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
public:
|
|
|
|
DiagCacheItem(BugReport *R, PathDiagnostic *PD) {
|
|
|
|
ID.AddString(R->getBugType().getName());
|
|
|
|
ID.AddString(R->getBugType().getCategory());
|
|
|
|
ID.AddString(R->getDescription());
|
|
|
|
ID.AddInteger(R->getLocation().getRawEncoding());
|
|
|
|
PD->Profile(ID);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Profile(llvm::FoldingSetNodeID &id) {
|
|
|
|
id = ID;
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::FoldingSetNodeID &getID() { return ID; }
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool IsCachedDiagnostic(BugReport *R, PathDiagnostic *PD) {
|
|
|
|
// FIXME: Eventually this diagnostic cache should reside in something
|
|
|
|
// like AnalysisManager instead of being a static variable. This is
|
|
|
|
// really unsafe in the long term.
|
|
|
|
typedef llvm::FoldingSet<DiagCacheItem> DiagnosticCache;
|
|
|
|
static DiagnosticCache DC;
|
|
|
|
|
|
|
|
void *InsertPos;
|
|
|
|
DiagCacheItem *Item = new DiagCacheItem(R, PD);
|
|
|
|
|
|
|
|
if (DC.FindNodeOrInsertPos(Item->getID(), InsertPos)) {
|
|
|
|
delete Item;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
DC.InsertNode(Item, InsertPos);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-02-05 07:49:09 +08:00
|
|
|
void BugReporter::FlushReport(BugReportEquivClass& EQ) {
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<BugReport*, 10> bugReports;
|
2010-12-03 14:52:30 +08:00
|
|
|
BugReport *exampleReport = FindReportInEquivalenceClass(EQ, bugReports);
|
|
|
|
if (!exampleReport)
|
2009-09-15 06:01:32 +08:00
|
|
|
return;
|
|
|
|
|
2009-04-30 05:58:13 +08:00
|
|
|
PathDiagnosticClient* PD = getPathDiagnosticClient();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-05 07:49:09 +08:00
|
|
|
// FIXME: Make sure we use the 'R' for the path that was actually used.
|
2009-09-09 23:08:12 +08:00
|
|
|
// Probably doesn't make a difference in practice.
|
2010-12-03 14:52:30 +08:00
|
|
|
BugType& BT = exampleReport->getBugType();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-30 05:58:13 +08:00
|
|
|
llvm::OwningPtr<PathDiagnostic>
|
2010-12-03 14:52:30 +08:00
|
|
|
D(new PathDiagnostic(exampleReport->getBugType().getName(),
|
2009-04-30 06:05:03 +08:00
|
|
|
!PD || PD->useVerboseDescription()
|
2010-12-03 14:52:30 +08:00
|
|
|
? exampleReport->getDescription()
|
|
|
|
: exampleReport->getShortDescription(),
|
2009-04-30 05:58:13 +08:00
|
|
|
BT.getCategory()));
|
|
|
|
|
2010-12-03 14:52:30 +08:00
|
|
|
if (!bugReports.empty())
|
|
|
|
GeneratePathDiagnostic(*D.get(), bugReports);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-12-03 14:52:30 +08:00
|
|
|
if (IsCachedDiagnostic(exampleReport, D.get()))
|
2009-09-19 06:37:37 +08:00
|
|
|
return;
|
|
|
|
|
2008-05-01 07:47:44 +08:00
|
|
|
// Get the meta data.
|
2010-12-03 14:52:30 +08:00
|
|
|
std::pair<const char**, const char**> Meta =
|
|
|
|
exampleReport->getExtraDescriptiveText();
|
2009-09-15 06:01:32 +08:00
|
|
|
for (const char** s = Meta.first; s != Meta.second; ++s)
|
|
|
|
D->addMeta(*s);
|
2008-04-18 09:56:37 +08:00
|
|
|
|
2009-01-24 08:55:43 +08:00
|
|
|
// Emit a summary diagnostic to the regular Diagnostics engine.
|
2010-12-04 09:12:15 +08:00
|
|
|
BugReport::ranges_iterator Beg, End;
|
|
|
|
llvm::tie(Beg, End) = exampleReport->getRanges();
|
2010-12-03 14:52:30 +08:00
|
|
|
Diagnostic &Diag = getDiagnostic();
|
|
|
|
FullSourceLoc L(exampleReport->getLocation(), getSourceManager());
|
2010-01-15 15:56:51 +08:00
|
|
|
|
|
|
|
// Search the description for '%', as that will be interpretted as a
|
|
|
|
// format character by FormatDiagnostics.
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef desc = exampleReport->getShortDescription();
|
2010-01-15 15:56:51 +08:00
|
|
|
unsigned ErrorDiag;
|
|
|
|
{
|
|
|
|
llvm::SmallString<512> TmpStr;
|
|
|
|
llvm::raw_svector_ostream Out(TmpStr);
|
2011-07-23 18:55:15 +08:00
|
|
|
for (StringRef::iterator I=desc.begin(), E=desc.end(); I!=E; ++I)
|
2010-01-15 15:56:51 +08:00
|
|
|
if (*I == '%')
|
|
|
|
Out << "%%";
|
|
|
|
else
|
|
|
|
Out << *I;
|
|
|
|
|
|
|
|
Out.flush();
|
|
|
|
ErrorDiag = Diag.getCustomDiagID(Diagnostic::Warning, TmpStr);
|
|
|
|
}
|
This reworks some of the Diagnostic interfaces a bit to change how diagnostics
are formed. In particular, a diagnostic with all its strings and ranges is now
packaged up and sent to DiagnosticClients as a DiagnosticInfo instead of as a
ton of random stuff. This has the benefit of simplifying the interface, making
it more extensible, and allowing us to do more checking for things like access
past the end of the various arrays passed in.
In addition to introducing DiagnosticInfo, this also substantially changes how
Diagnostic::Report works. Instead of being passed in all of the info required
to issue a diagnostic, Report now takes only the required info (a location and
ID) and returns a fresh DiagnosticInfo *by value*. The caller is then free to
stuff strings and ranges into the DiagnosticInfo with the << operator. When
the dtor runs on the DiagnosticInfo object (which should happen at the end of
the statement), the diagnostic is actually emitted with all of the accumulated
information. This is a somewhat tricky dance, but it means that the
accumulated DiagnosticInfo is allowed to keep pointers to other expression
temporaries without those pointers getting invalidated.
This is just the minimal change to get this stuff working, but this will allow
us to eliminate the zillions of variant "Diag" methods scattered throughout
(e.g.) sema. For example, instead of calling:
Diag(BuiltinLoc, diag::err_overload_no_match, typeNames,
SourceRange(BuiltinLoc, RParenLoc));
We will soon be able to just do:
Diag(BuiltinLoc, diag::err_overload_no_match)
<< typeNames << SourceRange(BuiltinLoc, RParenLoc));
This scales better to support arbitrary types being passed in (not just
strings) in a type-safe way. Go operator overloading?!
llvm-svn: 59502
2008-11-18 15:04:44 +08:00
|
|
|
|
2010-12-03 08:58:10 +08:00
|
|
|
{
|
|
|
|
DiagnosticBuilder diagBuilder = Diag.Report(L, ErrorDiag);
|
2010-12-04 09:12:15 +08:00
|
|
|
for (BugReport::ranges_iterator I = Beg; I != End; ++I)
|
2010-12-03 08:58:10 +08:00
|
|
|
diagBuilder << *I;
|
2008-04-19 06:56:53 +08:00
|
|
|
}
|
2009-01-24 08:55:43 +08:00
|
|
|
|
|
|
|
// Emit a full diagnostic for the path if we have a PathDiagnosticClient.
|
|
|
|
if (!PD)
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
if (D->empty()) {
|
2009-03-07 07:58:11 +08:00
|
|
|
PathDiagnosticPiece* piece =
|
2010-12-03 14:52:30 +08:00
|
|
|
new PathDiagnosticEventPiece(L, exampleReport->getDescription());
|
2009-03-07 07:58:11 +08:00
|
|
|
|
2009-01-24 08:55:43 +08:00
|
|
|
for ( ; Beg != End; ++Beg) piece->addRange(*Beg);
|
|
|
|
D->push_back(piece);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-01-24 08:55:43 +08:00
|
|
|
PD->HandlePathDiagnostic(D.take());
|
2008-04-03 12:42:52 +08:00
|
|
|
}
|
2008-07-15 01:40:50 +08:00
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
void BugReporter::EmitBasicReport(StringRef name, StringRef str,
|
2008-09-20 12:23:38 +08:00
|
|
|
SourceLocation Loc,
|
|
|
|
SourceRange* RBeg, unsigned NumRanges) {
|
|
|
|
EmitBasicReport(name, "", str, Loc, RBeg, NumRanges);
|
|
|
|
}
|
2009-02-05 07:49:09 +08:00
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
void BugReporter::EmitBasicReport(StringRef name,
|
|
|
|
StringRef category,
|
|
|
|
StringRef str, SourceLocation Loc,
|
2008-09-20 12:23:38 +08:00
|
|
|
SourceRange* RBeg, unsigned NumRanges) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-02-23 08:16:01 +08:00
|
|
|
// 'BT' is owned by BugReporter.
|
|
|
|
BugType *BT = getBugTypeForName(name, category);
|
This reworks some of the Diagnostic interfaces a bit to change how diagnostics
are formed. In particular, a diagnostic with all its strings and ranges is now
packaged up and sent to DiagnosticClients as a DiagnosticInfo instead of as a
ton of random stuff. This has the benefit of simplifying the interface, making
it more extensible, and allowing us to do more checking for things like access
past the end of the various arrays passed in.
In addition to introducing DiagnosticInfo, this also substantially changes how
Diagnostic::Report works. Instead of being passed in all of the info required
to issue a diagnostic, Report now takes only the required info (a location and
ID) and returns a fresh DiagnosticInfo *by value*. The caller is then free to
stuff strings and ranges into the DiagnosticInfo with the << operator. When
the dtor runs on the DiagnosticInfo object (which should happen at the end of
the statement), the diagnostic is actually emitted with all of the accumulated
information. This is a somewhat tricky dance, but it means that the
accumulated DiagnosticInfo is allowed to keep pointers to other expression
temporaries without those pointers getting invalidated.
This is just the minimal change to get this stuff working, but this will allow
us to eliminate the zillions of variant "Diag" methods scattered throughout
(e.g.) sema. For example, instead of calling:
Diag(BuiltinLoc, diag::err_overload_no_match, typeNames,
SourceRange(BuiltinLoc, RParenLoc));
We will soon be able to just do:
Diag(BuiltinLoc, diag::err_overload_no_match)
<< typeNames << SourceRange(BuiltinLoc, RParenLoc));
This scales better to support arbitrary types being passed in (not just
strings) in a type-safe way. Go operator overloading?!
llvm-svn: 59502
2008-11-18 15:04:44 +08:00
|
|
|
FullSourceLoc L = getContext().getFullLoc(Loc);
|
2009-02-05 07:49:09 +08:00
|
|
|
RangedBugReport *R = new DiagBugReport(*BT, str, L);
|
|
|
|
for ( ; NumRanges > 0 ; --NumRanges, ++RBeg) R->addRange(*RBeg);
|
|
|
|
EmitReport(R);
|
2009-01-24 04:28:53 +08:00
|
|
|
}
|
2011-02-23 08:16:01 +08:00
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
BugType *BugReporter::getBugTypeForName(StringRef name,
|
|
|
|
StringRef category) {
|
2011-02-23 08:16:01 +08:00
|
|
|
llvm::SmallString<136> fullDesc;
|
|
|
|
llvm::raw_svector_ostream(fullDesc) << name << ":" << category;
|
|
|
|
llvm::StringMapEntry<BugType *> &
|
|
|
|
entry = StrBugTypes.GetOrCreateValue(fullDesc);
|
|
|
|
BugType *BT = entry.getValue();
|
|
|
|
if (!BT) {
|
|
|
|
BT = new BugType(name, category);
|
|
|
|
entry.setValue(BT);
|
|
|
|
}
|
|
|
|
return BT;
|
|
|
|
}
|