forked from OSchip/llvm-project
191 lines
6.4 KiB
C++
191 lines
6.4 KiB
C++
//==- DeadStores.cpp - Check for stores to dead variables --------*- 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 a DeadStores, a flow-sensitive checker that looks for
|
|
// stores to variables that are no longer live.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "clang/Analysis/LocalCheckers.h"
|
|
#include "clang/Analysis/Analyses/LiveVariables.h"
|
|
#include "clang/Analysis/Visitors/CFGRecStmtVisitor.h"
|
|
#include "clang/Analysis/PathSensitive/BugReporter.h"
|
|
#include "clang/Analysis/PathSensitive/GRExprEngine.h"
|
|
#include "clang/Basic/Diagnostic.h"
|
|
#include "clang/AST/ASTContext.h"
|
|
#include "llvm/Support/Compiler.h"
|
|
|
|
using namespace clang;
|
|
|
|
namespace {
|
|
|
|
class VISIBILITY_HIDDEN DeadStoreObs : public LiveVariables::ObserverTy {
|
|
ASTContext &Ctx;
|
|
Diagnostic &Diags;
|
|
DiagnosticClient &Client;
|
|
public:
|
|
DeadStoreObs(ASTContext &ctx, Diagnostic &diags, DiagnosticClient &client)
|
|
: Ctx(ctx), Diags(diags), Client(client) {}
|
|
|
|
virtual ~DeadStoreObs() {}
|
|
|
|
virtual void ObserveStmt(Stmt* S,
|
|
const LiveVariables::AnalysisDataTy& AD,
|
|
const LiveVariables::ValTy& Live) {
|
|
|
|
// Skip statements in macros.
|
|
if (S->getLocStart().isMacroID())
|
|
return;
|
|
|
|
if (BinaryOperator* B = dyn_cast<BinaryOperator>(S)) {
|
|
if (!B->isAssignmentOp()) return; // Skip non-assignments.
|
|
|
|
if (DeclRefExpr* DR = dyn_cast<DeclRefExpr>(B->getLHS()))
|
|
if (VarDecl* VD = dyn_cast<VarDecl>(DR->getDecl()))
|
|
if (VD->hasLocalStorage() && !Live(VD,AD)) {
|
|
SourceRange R = B->getRHS()->getSourceRange();
|
|
Diags.Report(&Client,
|
|
Ctx.getFullLoc(DR->getSourceRange().getBegin()),
|
|
diag::warn_dead_store, 0, 0, &R, 1);
|
|
}
|
|
}
|
|
else if(DeclStmt* DS = dyn_cast<DeclStmt>(S))
|
|
// Iterate through the decls. Warn if any initializers are complex
|
|
// expressions that are not live (never used).
|
|
for (ScopedDecl* SD = DS->getDecl(); SD; SD = SD->getNextDeclarator()) {
|
|
|
|
VarDecl* V = dyn_cast<VarDecl>(SD);
|
|
if (!V) continue;
|
|
|
|
if (V->hasLocalStorage())
|
|
if (Expr* E = V->getInit()) {
|
|
if (!Live(V, AD)) {
|
|
// Special case: check for initializations with constants.
|
|
//
|
|
// e.g. : int x = 0;
|
|
//
|
|
// If x is EVER assigned a new value later, don't issue
|
|
// a warning. This is because such initialization can be
|
|
// due to defensive programming.
|
|
if (!E->isConstantExpr(Ctx,NULL)) {
|
|
// Flag a warning.
|
|
SourceRange R = E->getSourceRange();
|
|
Diags.Report(&Client,
|
|
Ctx.getFullLoc(V->getLocation()),
|
|
diag::warn_dead_store, 0, 0, &R, 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Driver function to invoke the Dead-Stores checker on a CFG.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
void clang::CheckDeadStores(CFG& cfg, ASTContext &Ctx, Diagnostic &Diags) {
|
|
LiveVariables L(cfg);
|
|
L.runOnCFG(cfg);
|
|
DeadStoreObs A(Ctx, Diags, Diags.getClient());
|
|
L.runOnAllBlocks(cfg, &A);
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// BugReporter-based invocation of the Dead-Stores checker.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
namespace {
|
|
|
|
class VISIBILITY_HIDDEN DiagBugReport : public RangedBugReport {
|
|
std::list<std::string> Strs;
|
|
FullSourceLoc L;
|
|
public:
|
|
DiagBugReport(const BugType& D, FullSourceLoc l) :
|
|
RangedBugReport(D, NULL), L(l) {}
|
|
|
|
virtual ~DiagBugReport() {}
|
|
virtual FullSourceLoc getLocation(SourceManager&) { return L; }
|
|
|
|
void addString(const std::string& s) { Strs.push_back(s); }
|
|
|
|
typedef std::list<std::string>::const_iterator str_iterator;
|
|
str_iterator str_begin() const { return Strs.begin(); }
|
|
str_iterator str_end() const { return Strs.end(); }
|
|
};
|
|
|
|
class VISIBILITY_HIDDEN DiagCollector : public DiagnosticClient {
|
|
std::list<DiagBugReport> Reports;
|
|
const BugType& D;
|
|
public:
|
|
DiagCollector(BugType& d) : D(d) {}
|
|
|
|
virtual ~DiagCollector() {}
|
|
|
|
virtual void HandleDiagnostic(Diagnostic &Diags,
|
|
Diagnostic::Level DiagLevel,
|
|
FullSourceLoc Pos,
|
|
diag::kind ID,
|
|
const std::string *Strs,
|
|
unsigned NumStrs,
|
|
const SourceRange *Ranges,
|
|
unsigned NumRanges) {
|
|
|
|
// FIXME: Use a map from diag::kind to BugType, instead of having just
|
|
// one BugType.
|
|
|
|
Reports.push_back(DiagBugReport(D, Pos));
|
|
DiagBugReport& R = Reports.back();
|
|
|
|
for ( ; NumRanges ; --NumRanges, ++Ranges)
|
|
R.addRange(*Ranges);
|
|
|
|
for ( ; NumStrs ; --NumStrs, ++Strs)
|
|
R.addString(*Strs);
|
|
}
|
|
|
|
// Iterators.
|
|
|
|
typedef std::list<DiagBugReport>::iterator iterator;
|
|
iterator begin() { return Reports.begin(); }
|
|
iterator end() { return Reports.end(); }
|
|
};
|
|
|
|
class VISIBILITY_HIDDEN DeadStoresChecker : public BugType {
|
|
public:
|
|
virtual const char* getName() const {
|
|
return "dead store";
|
|
}
|
|
|
|
virtual const char* getDescription() const {
|
|
return "Value stored to variable is never subsequently read.";
|
|
}
|
|
|
|
virtual void EmitWarnings(BugReporter& BR) {
|
|
|
|
// Run the dead store checker and collect the diagnostics.
|
|
DiagCollector C(*this);
|
|
DeadStoreObs A(BR.getContext(), BR.getDiagnostic(), C);
|
|
GRExprEngine& Eng = BR.getEngine();
|
|
Eng.getLiveness().runOnAllBlocks(Eng.getCFG(), &A);
|
|
|
|
// Emit the bug reports.
|
|
|
|
for (DiagCollector::iterator I = C.begin(), E = C.end(); I != E; ++I)
|
|
BR.EmitWarning(*I);
|
|
}
|
|
};
|
|
} // end anonymous namespace
|
|
|
|
BugType* clang::MakeDeadStoresChecker() {
|
|
return new DeadStoresChecker();
|
|
}
|