summaryrefslogtreecommitdiffstats
path: root/clang/lib/Analysis
diff options
context:
space:
mode:
authorTed Kremenek <kremenek@apple.com>2008-04-02 05:15:22 +0000
committerTed Kremenek <kremenek@apple.com>2008-04-02 05:15:22 +0000
commit2c71d5151353887c74af9ecbf4aa83b66114252f (patch)
tree6b5568cd2208bfbac1158bf213264058af3ae94c /clang/lib/Analysis
parent52595d7551ba0d4f81deacf8505ef98b83eec65f (diff)
downloadbcm5719-llvm-2c71d5151353887c74af9ecbf4aa83b66114252f.tar.gz
bcm5719-llvm-2c71d5151353887c74af9ecbf4aa83b66114252f.zip
Added initial hacked support for display path diagnostics with
GRSimpleVals warnings. Cleaning up, but now we get multiple bubbles (branches are annotated). llvm-svn: 49077
Diffstat (limited to 'clang/lib/Analysis')
-rw-r--r--clang/lib/Analysis/GRSimpleVals.cpp110
1 files changed, 107 insertions, 3 deletions
diff --git a/clang/lib/Analysis/GRSimpleVals.cpp b/clang/lib/Analysis/GRSimpleVals.cpp
index 5944c12e589..4baad54e94e 100644
--- a/clang/lib/Analysis/GRSimpleVals.cpp
+++ b/clang/lib/Analysis/GRSimpleVals.cpp
@@ -24,12 +24,23 @@ using namespace clang;
namespace clang {
template <typename ITERATOR>
+static inline ExplodedNode<ValueState>* GetNode(ITERATOR I) {
+ return *I;
+}
+
+template <>
+static inline ExplodedNode<ValueState>*
+GetNode(GRExprEngine::undef_arg_iterator I) {
+ return I->first;
+}
+
+template <typename ITERATOR>
static inline ProgramPoint GetLocation(ITERATOR I) {
return (*I)->getLocation();
}
template <>
-inline ProgramPoint GetLocation(GRExprEngine::undef_arg_iterator I) {
+static inline ProgramPoint GetLocation(GRExprEngine::undef_arg_iterator I) {
return I->first->getLocation();
}
@@ -44,7 +55,93 @@ static inline Stmt* GetStmt(const ProgramPoint& P) {
assert (false && "Unsupported ProgramPoint.");
return NULL;
}
+
+
+//===----------------------------------------------------------------------===//
+// Path Warnings.
+//===----------------------------------------------------------------------===//
+
+static inline SourceLocation GetSourceLoc(ProgramPoint P) {
+
+ if (const PostStmt* PS = dyn_cast<PostStmt>(&P)) {
+ return PS->getStmt()->getLocStart();
+ }
+ else if (const BlockEdge* BE = dyn_cast<BlockEdge>(&P)) {
+ return BE->getSrc()->getTerminator()->getLocStart();
+ }
+
+ return SourceLocation();
+}
+
+static inline
+FullSourceLoc GetFullSourceLoc(SourceManager& SMgr, ProgramPoint P) {
+ return FullSourceLoc(GetSourceLoc(P), SMgr);
+}
+
+template <typename ITERATOR>
+static void EmitPathWarning(Diagnostic& Diag, PathDiagnosticClient* PD,
+ SourceManager& SrcMgr, const std::string& msg,
+ ITERATOR I) {
+
+
+ PathDiagnostic D;
+
+ { // Add the end message.
+
+ ProgramPoint P = GetLocation(I);
+ D.push_back(new PathDiagnosticPiece(GetFullSourceLoc(SrcMgr, P), msg));
+ }
+
+ // Walk up the path.
+
+ ExplodedNode<ValueState> *N = GetNode(I);
+
+ while (N) {
+
+ if (N->pred_empty())
+ break;
+
+ N = *(N->pred_begin()); // Grab the first predecessor.
+
+ ProgramPoint P = N->getLocation();
+
+ if (const BlockEdge* BE = dyn_cast<BlockEdge>(&P)) {
+
+ CFGBlock* Src = BE->getSrc();
+ CFGBlock* Dst = BE->getDst();
+
+ // FIXME: Better handling for switches.
+
+ if (Src->succ_size() == 2) {
+
+ Stmt* T = Src->getTerminator();
+
+ if (!Src)
+ continue;
+
+ if (*(Src->succ_begin()+1) == Dst) {
+ D.push_front(new PathDiagnosticPiece(FullSourceLoc(T->getLocStart(),
+ SrcMgr),
+ "Taking false branch."));
+
+ }
+ else {
+ D.push_front(new PathDiagnosticPiece(FullSourceLoc(T->getLocStart(),
+ SrcMgr),
+ "Taking true branch."));
+ }
+ }
+ }
+ }
+
+ // Emit the path.
+ PD->HandlePathDiagnostic(D);
+}
+//===----------------------------------------------------------------------===//
+// Pathless Warnings
+//===----------------------------------------------------------------------===//
+
template <typename ITERATOR>
static void EmitDiag(Diagnostic& Diag, PathDiagnosticClient* PD,
SourceManager& SrcMgr,
@@ -76,7 +173,7 @@ void EmitWarning(Diagnostic& Diag, PathDiagnosticClient* PD,
ITERATOR I, ITERATOR E, const char* msg) {
std::ostringstream Out;
- std::string Str;
+ std::string Str(msg);
if (!PD) {
Out << "[CHECKER] " << msg;
@@ -107,10 +204,17 @@ void EmitWarning(Diagnostic& Diag, PathDiagnosticClient* PD,
CachedErrors.insert(p);
}
- EmitDiag(Diag, PD, SrcMgr, ErrorDiag, I);
+ if (PD)
+ EmitPathWarning(Diag, PD, SrcMgr, Str, I);
+ else
+ EmitDiag(Diag, PD, SrcMgr, ErrorDiag, I);
}
}
+//===----------------------------------------------------------------------===//
+// Analysis Driver.
+//===----------------------------------------------------------------------===//
+
unsigned RunGRSimpleVals(CFG& cfg, Decl& CD, ASTContext& Ctx,
Diagnostic& Diag, PathDiagnosticClient* PD,
bool Visualize, bool TrimGraph) {
OpenPOWER on IntegriCloud