Skip to content
Snippets Groups Projects
Commit 245adabd authored by Tom Care's avatar Tom Care
Browse files

Added psuedo-constant analysis and integrated it into the false positive...

Added psuedo-constant analysis and integrated it into the false positive reduction stage in IdempotentOperationChecker.
- Renamed IdempotentOperationChecker::isConstant to isConstantOrPseudoConstant to better reflect the function
- Changed IdempotentOperationChecker::PreVisitBinaryOperator to only run 'CanVary' once on undefined assumptions
- Created new PsuedoConstantAnalysis class and added it to AnalysisContext
- Changed IdempotentOperationChecker to exploit the new analysis
- Updated tests with psuedo-constants
- Added check to IdempotentOperationChecker to see if a Decl is const qualified

git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@111426 91177308-0d34-0410-b5e6-96231b3b80d8
parent 5e1e89b8
No related branches found
No related tags found
No related merge requests found
//== PsuedoConstantAnalysis.h - Find Psuedo-constants in the AST -*- C++ -*-==//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// This file tracks the usage of variables in a Decl body to see if they are
// never written to, implying that they constant. This is useful in static
// analysis to see if a developer might have intended a variable to be const.
//
//===----------------------------------------------------------------------===//
#ifndef LLVM_CLANG_ANALYSIS_PSUEDOCONSTANTANALYSIS
#define LLVM_CLANG_ANALYSIS_PSUEDOCONSTANTANALYSIS
#include "clang/AST/Stmt.h"
// The number of ValueDecls we want to keep track of by default (per-function)
#define VALUEDECL_SET_SIZE 256
namespace clang {
class PsuedoConstantAnalysis {
public:
PsuedoConstantAnalysis(const Stmt *DeclBody) :
DeclBody(DeclBody), Analyzed(false) {}
bool isPsuedoConstant(const ValueDecl *VD);
private:
void RunAnalysis();
// for storing the result of analyzed ValueDecls
llvm::SmallPtrSet<const ValueDecl*, VALUEDECL_SET_SIZE> NonConstants;
const Stmt *DeclBody;
bool Analyzed;
};
}
#endif
...@@ -30,6 +30,7 @@ class CFG; ...@@ -30,6 +30,7 @@ class CFG;
class CFGBlock; class CFGBlock;
class LiveVariables; class LiveVariables;
class ParentMap; class ParentMap;
class PsuedoConstantAnalysis;
class ImplicitParamDecl; class ImplicitParamDecl;
class LocationContextManager; class LocationContextManager;
class StackFrameContext; class StackFrameContext;
...@@ -49,6 +50,7 @@ class AnalysisContext { ...@@ -49,6 +50,7 @@ class AnalysisContext {
bool builtCFG, builtCompleteCFG; bool builtCFG, builtCompleteCFG;
LiveVariables *liveness; LiveVariables *liveness;
ParentMap *PM; ParentMap *PM;
PsuedoConstantAnalysis *PCA;
llvm::DenseMap<const BlockDecl*,void*> *ReferencedBlockVars; llvm::DenseMap<const BlockDecl*,void*> *ReferencedBlockVars;
llvm::BumpPtrAllocator A; llvm::BumpPtrAllocator A;
bool UseUnoptimizedCFG; bool UseUnoptimizedCFG;
...@@ -59,7 +61,7 @@ public: ...@@ -59,7 +61,7 @@ public:
bool addehedges = false) bool addehedges = false)
: D(d), TU(tu), cfg(0), completeCFG(0), : D(d), TU(tu), cfg(0), completeCFG(0),
builtCFG(false), builtCompleteCFG(false), builtCFG(false), builtCompleteCFG(false),
liveness(0), PM(0), liveness(0), PM(0), PCA(0),
ReferencedBlockVars(0), UseUnoptimizedCFG(useUnoptimizedCFG), ReferencedBlockVars(0), UseUnoptimizedCFG(useUnoptimizedCFG),
AddEHEdges(addehedges) {} AddEHEdges(addehedges) {}
...@@ -85,6 +87,7 @@ public: ...@@ -85,6 +87,7 @@ public:
CFG *getUnoptimizedCFG(); CFG *getUnoptimizedCFG();
ParentMap &getParentMap(); ParentMap &getParentMap();
PsuedoConstantAnalysis *getPsuedoConstantAnalysis();
LiveVariables *getLiveVariables(); LiveVariables *getLiveVariables();
typedef const VarDecl * const * referenced_decls_iterator; typedef const VarDecl * const * referenced_decls_iterator;
......
...@@ -18,6 +18,7 @@ ...@@ -18,6 +18,7 @@
#include "clang/AST/ParentMap.h" #include "clang/AST/ParentMap.h"
#include "clang/AST/StmtVisitor.h" #include "clang/AST/StmtVisitor.h"
#include "clang/Analysis/Analyses/LiveVariables.h" #include "clang/Analysis/Analyses/LiveVariables.h"
#include "clang/Analysis/Analyses/PsuedoConstantAnalysis.h"
#include "clang/Analysis/AnalysisContext.h" #include "clang/Analysis/AnalysisContext.h"
#include "clang/Analysis/CFG.h" #include "clang/Analysis/CFG.h"
#include "clang/Analysis/Support/BumpVector.h" #include "clang/Analysis/Support/BumpVector.h"
...@@ -83,6 +84,12 @@ ParentMap &AnalysisContext::getParentMap() { ...@@ -83,6 +84,12 @@ ParentMap &AnalysisContext::getParentMap() {
return *PM; return *PM;
} }
PsuedoConstantAnalysis *AnalysisContext::getPsuedoConstantAnalysis() {
if (!PCA)
PCA = new PsuedoConstantAnalysis(getBody());
return PCA;
}
LiveVariables *AnalysisContext::getLiveVariables() { LiveVariables *AnalysisContext::getLiveVariables() {
if (!liveness) { if (!liveness) {
CFG *c = getCFG(); CFG *c = getCFG();
...@@ -314,6 +321,7 @@ AnalysisContext::~AnalysisContext() { ...@@ -314,6 +321,7 @@ AnalysisContext::~AnalysisContext() {
delete completeCFG; delete completeCFG;
delete liveness; delete liveness;
delete PM; delete PM;
delete PCA;
delete ReferencedBlockVars; delete ReferencedBlockVars;
} }
......
...@@ -7,6 +7,7 @@ add_clang_library(clangAnalysis ...@@ -7,6 +7,7 @@ add_clang_library(clangAnalysis
FormatString.cpp FormatString.cpp
LiveVariables.cpp LiveVariables.cpp
PrintfFormatString.cpp PrintfFormatString.cpp
PsuedoConstantAnalysis.cpp
ReachableCode.cpp ReachableCode.cpp
ScanfFormatString.cpp ScanfFormatString.cpp
UninitializedValues.cpp UninitializedValues.cpp
......
//== PsuedoConstantAnalysis.cpp - Find Psuedoconstants in the AST-*- C++ -*-==//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// This file tracks the usage of variables in a Decl body to see if they are
// never written to, implying that they constant. This is useful in static
// analysis to see if a developer might have intended a variable to be const.
//
//===----------------------------------------------------------------------===//
#include "clang/Analysis/Analyses/PsuedoConstantAnalysis.h"
#include "clang/AST/Decl.h"
#include "clang/AST/Expr.h"
#include "clang/AST/Stmt.h"
#include <deque>
using namespace clang;
// Returns true if the given ValueDecl is never written to in the given DeclBody
bool PsuedoConstantAnalysis::isPsuedoConstant(const ValueDecl *VD) {
if (!Analyzed) {
RunAnalysis();
Analyzed = true;
}
return !NonConstants.count(VD);
}
void PsuedoConstantAnalysis::RunAnalysis() {
std::deque<const Stmt *> WorkList;
// Start with the top level statement of the function
WorkList.push_back(DeclBody);
while (!WorkList.empty()) {
const Stmt* Head = WorkList.front();
WorkList.pop_front();
switch (Head->getStmtClass()) {
// Case 1: Assignment operators modifying ValueDecl
case Stmt::BinaryOperatorClass: {
const BinaryOperator *BO = cast<BinaryOperator>(Head);
const Expr *LHS = BO->getLHS()->IgnoreParenImpCasts();
const DeclRefExpr *DR = dyn_cast<DeclRefExpr>(LHS);
// We only care about DeclRefExprs on the LHS
if (!DR)
break;
// We found a binary operator with a DeclRefExpr on the LHS. We now check
// for any of the assignment operators, implying that this Decl is being
// written to.
switch (BO->getOpcode()) {
case BinaryOperator::Assign:
case BinaryOperator::AddAssign:
case BinaryOperator::SubAssign:
case BinaryOperator::MulAssign:
case BinaryOperator::DivAssign:
case BinaryOperator::AndAssign:
case BinaryOperator::OrAssign:
case BinaryOperator::XorAssign:
case BinaryOperator::ShlAssign:
case BinaryOperator::ShrAssign:
// The DeclRefExpr is being assigned to - mark it as non-constant
NonConstants.insert(DR->getDecl());
continue; // Continue without looking at children
default:
break;
}
break;
}
// Case 2: Pre/post increment/decrement and address of
case Stmt::UnaryOperatorClass: {
const UnaryOperator *UO = cast<UnaryOperator>(Head);
const Expr *SubExpr = UO->getSubExpr()->IgnoreParenImpCasts();
const DeclRefExpr *DR = dyn_cast<DeclRefExpr>(SubExpr);
// We only care about DeclRefExprs in the subexpression
if (!DR)
break;
// We found a unary operator with a DeclRefExpr as a subexpression. We now
// check for any of the increment/decrement operators, as well as
// addressOf.
switch (UO->getOpcode()) {
case UnaryOperator::PostDec:
case UnaryOperator::PostInc:
case UnaryOperator::PreDec:
case UnaryOperator::PreInc:
// The DeclRefExpr is being changed - mark it as non-constant
case UnaryOperator::AddrOf:
// If we are taking the address of the DeclRefExpr, assume it is
// non-constant.
NonConstants.insert(DR->getDecl());
default:
break;
}
break;
}
default:
break;
} // switch (head->getStmtClass())
// Add all substatements to the worklist
for (Stmt::const_child_iterator I = Head->child_begin(),
E = Head->child_end(); I != E; ++I)
if (*I)
WorkList.push_back(*I);
} // while (!WorkList.empty())
}
...@@ -44,6 +44,7 @@ ...@@ -44,6 +44,7 @@
#include "GRExprEngineExperimentalChecks.h" #include "GRExprEngineExperimentalChecks.h"
#include "clang/Analysis/CFGStmtMap.h" #include "clang/Analysis/CFGStmtMap.h"
#include "clang/Analysis/Analyses/PsuedoConstantAnalysis.h"
#include "clang/Checker/BugReporter/BugType.h" #include "clang/Checker/BugReporter/BugType.h"
#include "clang/Checker/PathSensitive/CheckerHelpers.h" #include "clang/Checker/PathSensitive/CheckerHelpers.h"
#include "clang/Checker/PathSensitive/CheckerVisitor.h" #include "clang/Checker/PathSensitive/CheckerVisitor.h"
...@@ -72,16 +73,16 @@ class IdempotentOperationChecker ...@@ -72,16 +73,16 @@ class IdempotentOperationChecker
void UpdateAssumption(Assumption &A, const Assumption &New); void UpdateAssumption(Assumption &A, const Assumption &New);
/// contains* - Useful recursive methods to see if a statement contains an // False positive reduction methods
/// element somewhere. Used in static analysis to reduce false positives.
static bool isParameterSelfAssign(const Expr *LHS, const Expr *RHS); static bool isParameterSelfAssign(const Expr *LHS, const Expr *RHS);
static bool isTruncationExtensionAssignment(const Expr *LHS, static bool isTruncationExtensionAssignment(const Expr *LHS,
const Expr *RHS); const Expr *RHS);
static bool PathWasCompletelyAnalyzed(const CFG *C, static bool PathWasCompletelyAnalyzed(const CFG *C,
const CFGBlock *CB, const CFGBlock *CB,
const GRCoreEngine &CE); const GRCoreEngine &CE);
static bool CanVary(const Expr *Ex, ASTContext &Ctx); static bool CanVary(const Expr *Ex, AnalysisContext *AC);
static bool isPseudoConstant(const DeclRefExpr *D); static bool isConstantOrPseudoConstant(const DeclRefExpr *DR,
AnalysisContext *AC);
// Hash table // Hash table
typedef llvm::DenseMap<const BinaryOperator *, typedef llvm::DenseMap<const BinaryOperator *,
...@@ -108,7 +109,8 @@ void IdempotentOperationChecker::PreVisitBinaryOperator( ...@@ -108,7 +109,8 @@ void IdempotentOperationChecker::PreVisitBinaryOperator(
// 'Possible'. // 'Possible'.
std::pair<Assumption, AnalysisContext *> &Data = hash[B]; std::pair<Assumption, AnalysisContext *> &Data = hash[B];
Assumption &A = Data.first; Assumption &A = Data.first;
Data.second = C.getCurrentAnalysisContext(); AnalysisContext *AC = C.getCurrentAnalysisContext();
Data.second = AC;
// If we already have visited this node on a path that does not contain an // If we already have visited this node on a path that does not contain an
// idempotent operation, return immediately. // idempotent operation, return immediately.
...@@ -119,8 +121,14 @@ void IdempotentOperationChecker::PreVisitBinaryOperator( ...@@ -119,8 +121,14 @@ void IdempotentOperationChecker::PreVisitBinaryOperator(
// may mean this is a false positive. // may mean this is a false positive.
const Expr *LHS = B->getLHS(); const Expr *LHS = B->getLHS();
const Expr *RHS = B->getRHS(); const Expr *RHS = B->getRHS();
bool LHSCanVary = CanVary(LHS, C.getASTContext());
bool RHSCanVary = CanVary(RHS, C.getASTContext()); // Check if either side can vary. We only need to calculate this when we have
// no assumption.
bool LHSCanVary = true, RHSCanVary = true;
if (A == Possible) {
LHSCanVary = CanVary(LHS, AC);
RHSCanVary = CanVary(RHS, AC);
}
const GRState *state = C.getState(); const GRState *state = C.getState();
...@@ -486,7 +494,8 @@ bool IdempotentOperationChecker::PathWasCompletelyAnalyzed( ...@@ -486,7 +494,8 @@ bool IdempotentOperationChecker::PathWasCompletelyAnalyzed(
// that varies. This could be due to a compile-time constant like sizeof. An // that varies. This could be due to a compile-time constant like sizeof. An
// expression may also involve a variable that behaves like a constant. The // expression may also involve a variable that behaves like a constant. The
// function returns true if the expression varies, and false otherwise. // function returns true if the expression varies, and false otherwise.
bool IdempotentOperationChecker::CanVary(const Expr *Ex, ASTContext &Ctx) { bool IdempotentOperationChecker::CanVary(const Expr *Ex,
AnalysisContext *AC) {
// Parentheses and casts are irrelevant here // Parentheses and casts are irrelevant here
Ex = Ex->IgnoreParenCasts(); Ex = Ex->IgnoreParenCasts();
...@@ -531,12 +540,13 @@ bool IdempotentOperationChecker::CanVary(const Expr *Ex, ASTContext &Ctx) { ...@@ -531,12 +540,13 @@ bool IdempotentOperationChecker::CanVary(const Expr *Ex, ASTContext &Ctx) {
return SE->getTypeOfArgument()->isVariableArrayType(); return SE->getTypeOfArgument()->isVariableArrayType();
} }
case Stmt::DeclRefExprClass: case Stmt::DeclRefExprClass:
return !isPseudoConstant(cast<DeclRefExpr>(Ex)); return !isConstantOrPseudoConstant(cast<DeclRefExpr>(Ex), AC);
// The next cases require recursion for subexpressions // The next cases require recursion for subexpressions
case Stmt::BinaryOperatorClass: { case Stmt::BinaryOperatorClass: {
const BinaryOperator *B = cast<const BinaryOperator>(Ex); const BinaryOperator *B = cast<const BinaryOperator>(Ex);
return CanVary(B->getRHS(), Ctx) || CanVary(B->getLHS(), Ctx); return CanVary(B->getRHS(), AC)
|| CanVary(B->getLHS(), AC);
} }
case Stmt::UnaryOperatorClass: { case Stmt::UnaryOperatorClass: {
const UnaryOperator *U = cast<const UnaryOperator>(Ex); const UnaryOperator *U = cast<const UnaryOperator>(Ex);
...@@ -545,18 +555,25 @@ bool IdempotentOperationChecker::CanVary(const Expr *Ex, ASTContext &Ctx) { ...@@ -545,18 +555,25 @@ bool IdempotentOperationChecker::CanVary(const Expr *Ex, ASTContext &Ctx) {
case UnaryOperator::Extension: case UnaryOperator::Extension:
return false; return false;
default: default:
return CanVary(U->getSubExpr(), Ctx); return CanVary(U->getSubExpr(), AC);
} }
} }
case Stmt::ChooseExprClass: case Stmt::ChooseExprClass:
return CanVary(cast<const ChooseExpr>(Ex)->getChosenSubExpr(Ctx), Ctx); return CanVary(cast<const ChooseExpr>(Ex)->getChosenSubExpr(
AC->getASTContext()), AC);
case Stmt::ConditionalOperatorClass: case Stmt::ConditionalOperatorClass:
return CanVary(cast<const ConditionalOperator>(Ex)->getCond(), Ctx); return CanVary(cast<const ConditionalOperator>(Ex)->getCond(), AC);
} }
} }
// Returns true if a DeclRefExpr behaves like a constant. // Returns true if a DeclRefExpr is or behaves like a constant.
bool IdempotentOperationChecker::isPseudoConstant(const DeclRefExpr *DR) { bool IdempotentOperationChecker::isConstantOrPseudoConstant(
const DeclRefExpr *DR,
AnalysisContext *AC) {
// Check if the type of the Decl is const-qualified
if (DR->getType().isConstQualified())
return true;
// Check for an enum // Check for an enum
if (isa<EnumConstantDecl>(DR->getDecl())) if (isa<EnumConstantDecl>(DR->getDecl()))
return true; return true;
...@@ -567,5 +584,10 @@ bool IdempotentOperationChecker::isPseudoConstant(const DeclRefExpr *DR) { ...@@ -567,5 +584,10 @@ bool IdempotentOperationChecker::isPseudoConstant(const DeclRefExpr *DR) {
if (VD->isStaticLocal()) if (VD->isStaticLocal())
return true; return true;
// Check if the Decl behaves like a constant
PsuedoConstantAnalysis *PCA = AC->getPsuedoConstantAnalysis();
if (PCA->isPsuedoConstant(DR->getDecl()))
return true;
return false; return false;
} }
...@@ -18,10 +18,10 @@ void testComparisons (int a) { ...@@ -18,10 +18,10 @@ void testComparisons (int a) {
} }
void testSelfOperations (int a) { void testSelfOperations (int a) {
if ((a|a) != a) WARN; // expected-warning{{Both operands to '|' always have the same value}} expected-warning{{never executed}} if ((a|a) != a) WARN; // expected-warning{{never executed}}
if ((a&a) != a) WARN; // expected-warning{{Both operands to '&' always have the same value}} expected-warning{{never executed}} if ((a&a) != a) WARN; // expected-warning{{never executed}}
if ((a^a) != 0) WARN; // expected-warning{{Both operands to '^' always have the same value}} expected-warning{{never executed}} if ((a^a) != 0) WARN; // expected-warning{{never executed}}
if ((a-a) != 0) WARN; // expected-warning{{Both operands to '-' always have the same value}} expected-warning{{never executed}} if ((a-a) != 0) WARN; // expected-warning{{never executed}}
} }
void testIdempotent (int a) { void testIdempotent (int a) {
...@@ -68,5 +68,5 @@ void testLocations (char *a) { ...@@ -68,5 +68,5 @@ void testLocations (char *a) {
if (b!=a) WARN; // expected-warning{{never executed}} if (b!=a) WARN; // expected-warning{{never executed}}
if (b>a) WARN; // expected-warning{{never executed}} if (b>a) WARN; // expected-warning{{never executed}}
if (b<a) WARN; // expected-warning{{never executed}} if (b<a) WARN; // expected-warning{{never executed}}
if (b-a) WARN; // expected-warning{{Both operands to '-' always have the same value}} expected-warning{{never executed}} if (b-a) WARN; // expected-warning{{never executed}}
} }
...@@ -5,7 +5,7 @@ ...@@ -5,7 +5,7 @@
extern void test(int i); extern void test(int i);
extern void test_f(float f); extern void test_f(float f);
void basic() { unsigned basic() {
int x = 10, zero = 0, one = 1; int x = 10, zero = 0, one = 1;
// x op x // x op x
...@@ -50,6 +50,13 @@ void basic() { ...@@ -50,6 +50,13 @@ void basic() {
test(zero ^ x); // expected-warning {{The left operand to '^' is always 0}} test(zero ^ x); // expected-warning {{The left operand to '^' is always 0}}
test(zero << x); // expected-warning {{The left operand to '<<' is always 0}} test(zero << x); // expected-warning {{The left operand to '<<' is always 0}}
test(zero >> x); // expected-warning {{The left operand to '>>' is always 0}} test(zero >> x); // expected-warning {{The left operand to '>>' is always 0}}
// Overwrite the values so these aren't marked as psuedoconstants
x = 1;
zero = 2;
one = 3;
return x + zero + one;
} }
void floats(float x) { void floats(float x) {
...@@ -84,4 +91,23 @@ unsigned false2() { ...@@ -84,4 +91,23 @@ unsigned false2() {
return enum1 + a; // no-warning return enum1 + a; // no-warning
} }
extern unsigned foo(); // Self assignments of parameters are common false positives
unsigned false3(int param) {
param = param; // no-warning
unsigned nonparam = 5;
nonparam = nonparam; // expected-warning{{Assigned value is always the same as the existing value}}
return nonparam;
}
// Psuedo-constants (vars only read) and constants should not be reported
unsigned false4() {
// Trivial constant
const int height = 1; // no-warning
// Psuedo-constant (never changes after decl)
int width = height; // no-warning
return width * 10; // no-warning
}
...@@ -260,7 +260,7 @@ void f12(HF12ITEM i, char *q) { ...@@ -260,7 +260,7 @@ void f12(HF12ITEM i, char *q) {
// Test handling of translating between integer "pointers" and back. // Test handling of translating between integer "pointers" and back.
void f13() { void f13() {
int *x = 0; int *x = 0;
if (((((int) x) << 2) + 1) >> 1) *x = 1; // expected-warning{{The left operand to '<<' is always 0}} expected-warning{{The left operand to '+' is always 0}} if (((((int) x) << 2) + 1) >> 1) *x = 1;
} }
// PR 4759 - Attribute non-null checking by the analyzer was not correctly // PR 4759 - Attribute non-null checking by the analyzer was not correctly
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment