forked from OSchip/llvm-project
160 lines
5.8 KiB
C++
160 lines
5.8 KiB
C++
//===- AnalysisOrderChecker - Print callbacks called ------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This checker prints callbacks that are called during analysis.
|
|
// This is required to ensure that callbacks are fired in order
|
|
// and do not duplicate or get lost.
|
|
// Feel free to extend this checker with any callback you need to check.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "ClangSACheckers.h"
|
|
#include "clang/AST/ExprCXX.h"
|
|
#include "clang/StaticAnalyzer/Core/Checker.h"
|
|
#include "clang/StaticAnalyzer/Core/CheckerManager.h"
|
|
#include "clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h"
|
|
#include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h"
|
|
|
|
using namespace clang;
|
|
using namespace ento;
|
|
|
|
namespace {
|
|
|
|
class AnalysisOrderChecker
|
|
: public Checker<check::PreStmt<CastExpr>,
|
|
check::PostStmt<CastExpr>,
|
|
check::PreStmt<ArraySubscriptExpr>,
|
|
check::PostStmt<ArraySubscriptExpr>,
|
|
check::PreStmt<CXXNewExpr>,
|
|
check::PostStmt<CXXNewExpr>,
|
|
check::PreStmt<OffsetOfExpr>,
|
|
check::PostStmt<OffsetOfExpr>,
|
|
check::PreCall,
|
|
check::PostCall,
|
|
check::NewAllocator,
|
|
check::Bind,
|
|
check::RegionChanges,
|
|
check::LiveSymbols> {
|
|
|
|
bool isCallbackEnabled(AnalyzerOptions &Opts, StringRef CallbackName) const {
|
|
return Opts.getBooleanOption("*", false, this) ||
|
|
Opts.getBooleanOption(CallbackName, false, this);
|
|
}
|
|
|
|
bool isCallbackEnabled(CheckerContext &C, StringRef CallbackName) const {
|
|
AnalyzerOptions &Opts = C.getAnalysisManager().getAnalyzerOptions();
|
|
return isCallbackEnabled(Opts, CallbackName);
|
|
}
|
|
|
|
bool isCallbackEnabled(ProgramStateRef State, StringRef CallbackName) const {
|
|
AnalyzerOptions &Opts = State->getStateManager().getOwningEngine()
|
|
->getAnalysisManager().getAnalyzerOptions();
|
|
return isCallbackEnabled(Opts, CallbackName);
|
|
}
|
|
|
|
public:
|
|
void checkPreStmt(const CastExpr *CE, CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "PreStmtCastExpr"))
|
|
llvm::errs() << "PreStmt<CastExpr> (Kind : " << CE->getCastKindName()
|
|
<< ")\n";
|
|
}
|
|
|
|
void checkPostStmt(const CastExpr *CE, CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "PostStmtCastExpr"))
|
|
llvm::errs() << "PostStmt<CastExpr> (Kind : " << CE->getCastKindName()
|
|
<< ")\n";
|
|
}
|
|
|
|
void checkPreStmt(const ArraySubscriptExpr *SubExpr,
|
|
CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "PreStmtArraySubscriptExpr"))
|
|
llvm::errs() << "PreStmt<ArraySubscriptExpr>\n";
|
|
}
|
|
|
|
void checkPostStmt(const ArraySubscriptExpr *SubExpr,
|
|
CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "PostStmtArraySubscriptExpr"))
|
|
llvm::errs() << "PostStmt<ArraySubscriptExpr>\n";
|
|
}
|
|
|
|
void checkPreStmt(const CXXNewExpr *NE, CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "PreStmtCXXNewExpr"))
|
|
llvm::errs() << "PreStmt<CXXNewExpr>\n";
|
|
}
|
|
|
|
void checkPostStmt(const CXXNewExpr *NE, CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "PostStmtCXXNewExpr"))
|
|
llvm::errs() << "PostStmt<CXXNewExpr>\n";
|
|
}
|
|
|
|
void checkPreStmt(const OffsetOfExpr *OOE, CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "PreStmtOffsetOfExpr"))
|
|
llvm::errs() << "PreStmt<OffsetOfExpr>\n";
|
|
}
|
|
|
|
void checkPostStmt(const OffsetOfExpr *OOE, CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "PostStmtOffsetOfExpr"))
|
|
llvm::errs() << "PostStmt<OffsetOfExpr>\n";
|
|
}
|
|
|
|
void checkPreCall(const CallEvent &Call, CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "PreCall")) {
|
|
llvm::errs() << "PreCall";
|
|
if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))
|
|
llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';
|
|
llvm::errs() << '\n';
|
|
}
|
|
}
|
|
|
|
void checkPostCall(const CallEvent &Call, CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "PostCall")) {
|
|
llvm::errs() << "PostCall";
|
|
if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))
|
|
llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';
|
|
llvm::errs() << '\n';
|
|
}
|
|
}
|
|
|
|
void checkNewAllocator(const CXXNewExpr *CNE, SVal Target,
|
|
CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "NewAllocator"))
|
|
llvm::errs() << "NewAllocator\n";
|
|
}
|
|
|
|
void checkBind(SVal Loc, SVal Val, const Stmt *S, CheckerContext &C) const {
|
|
if (isCallbackEnabled(C, "Bind"))
|
|
llvm::errs() << "Bind\n";
|
|
}
|
|
|
|
void checkLiveSymbols(ProgramStateRef State, SymbolReaper &SymReaper) const {
|
|
if (isCallbackEnabled(State, "LiveSymbols"))
|
|
llvm::errs() << "LiveSymbols\n";
|
|
}
|
|
|
|
ProgramStateRef
|
|
checkRegionChanges(ProgramStateRef State,
|
|
const InvalidatedSymbols *Invalidated,
|
|
ArrayRef<const MemRegion *> ExplicitRegions,
|
|
ArrayRef<const MemRegion *> Regions,
|
|
const LocationContext *LCtx, const CallEvent *Call) const {
|
|
if (isCallbackEnabled(State, "RegionChanges"))
|
|
llvm::errs() << "RegionChanges\n";
|
|
return State;
|
|
}
|
|
};
|
|
} // end anonymous namespace
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Registration.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
void ento::registerAnalysisOrderChecker(CheckerManager &mgr) {
|
|
mgr.registerChecker<AnalysisOrderChecker>();
|
|
}
|