17330f729Sjoerg //===- AnalysisOrderChecker - Print callbacks called ------------*- C++ -*-===//
27330f729Sjoerg //
37330f729Sjoerg // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
47330f729Sjoerg // See https://llvm.org/LICENSE.txt for license information.
57330f729Sjoerg // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
67330f729Sjoerg //
77330f729Sjoerg //===----------------------------------------------------------------------===//
87330f729Sjoerg //
97330f729Sjoerg // This checker prints callbacks that are called during analysis.
107330f729Sjoerg // This is required to ensure that callbacks are fired in order
117330f729Sjoerg // and do not duplicate or get lost.
127330f729Sjoerg // Feel free to extend this checker with any callback you need to check.
137330f729Sjoerg //
147330f729Sjoerg //===----------------------------------------------------------------------===//
157330f729Sjoerg
167330f729Sjoerg #include "clang/AST/ExprCXX.h"
177330f729Sjoerg #include "clang/Analysis/CFGStmtMap.h"
18*e038c9c4Sjoerg #include "clang/StaticAnalyzer/Checkers/BuiltinCheckerRegistration.h"
197330f729Sjoerg #include "clang/StaticAnalyzer/Core/Checker.h"
207330f729Sjoerg #include "clang/StaticAnalyzer/Core/CheckerManager.h"
217330f729Sjoerg #include "clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h"
227330f729Sjoerg #include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h"
23*e038c9c4Sjoerg #include "llvm/Support/ErrorHandling.h"
247330f729Sjoerg
257330f729Sjoerg using namespace clang;
267330f729Sjoerg using namespace ento;
277330f729Sjoerg
287330f729Sjoerg namespace {
297330f729Sjoerg
307330f729Sjoerg class AnalysisOrderChecker
31*e038c9c4Sjoerg : public Checker<
32*e038c9c4Sjoerg check::PreStmt<CastExpr>, check::PostStmt<CastExpr>,
337330f729Sjoerg check::PreStmt<ArraySubscriptExpr>,
34*e038c9c4Sjoerg check::PostStmt<ArraySubscriptExpr>, check::PreStmt<CXXNewExpr>,
35*e038c9c4Sjoerg check::PostStmt<CXXNewExpr>, check::PreStmt<CXXDeleteExpr>,
36*e038c9c4Sjoerg check::PostStmt<CXXDeleteExpr>, check::PreStmt<CXXConstructExpr>,
37*e038c9c4Sjoerg check::PostStmt<CXXConstructExpr>, check::PreStmt<OffsetOfExpr>,
38*e038c9c4Sjoerg check::PostStmt<OffsetOfExpr>, check::PreCall, check::PostCall,
39*e038c9c4Sjoerg check::EndFunction, check::EndAnalysis, check::NewAllocator,
40*e038c9c4Sjoerg check::Bind, check::PointerEscape, check::RegionChanges,
41*e038c9c4Sjoerg check::LiveSymbols, eval::Call> {
427330f729Sjoerg
isCallbackEnabled(const AnalyzerOptions & Opts,StringRef CallbackName) const43*e038c9c4Sjoerg bool isCallbackEnabled(const AnalyzerOptions &Opts,
44*e038c9c4Sjoerg StringRef CallbackName) const {
457330f729Sjoerg return Opts.getCheckerBooleanOption(this, "*") ||
467330f729Sjoerg Opts.getCheckerBooleanOption(this, CallbackName);
477330f729Sjoerg }
487330f729Sjoerg
isCallbackEnabled(CheckerContext & C,StringRef CallbackName) const497330f729Sjoerg bool isCallbackEnabled(CheckerContext &C, StringRef CallbackName) const {
507330f729Sjoerg AnalyzerOptions &Opts = C.getAnalysisManager().getAnalyzerOptions();
517330f729Sjoerg return isCallbackEnabled(Opts, CallbackName);
527330f729Sjoerg }
537330f729Sjoerg
isCallbackEnabled(ProgramStateRef State,StringRef CallbackName) const547330f729Sjoerg bool isCallbackEnabled(ProgramStateRef State, StringRef CallbackName) const {
557330f729Sjoerg AnalyzerOptions &Opts = State->getStateManager().getOwningEngine()
567330f729Sjoerg .getAnalysisManager().getAnalyzerOptions();
577330f729Sjoerg return isCallbackEnabled(Opts, CallbackName);
587330f729Sjoerg }
597330f729Sjoerg
607330f729Sjoerg public:
checkPreStmt(const CastExpr * CE,CheckerContext & C) const617330f729Sjoerg void checkPreStmt(const CastExpr *CE, CheckerContext &C) const {
627330f729Sjoerg if (isCallbackEnabled(C, "PreStmtCastExpr"))
637330f729Sjoerg llvm::errs() << "PreStmt<CastExpr> (Kind : " << CE->getCastKindName()
647330f729Sjoerg << ")\n";
657330f729Sjoerg }
667330f729Sjoerg
checkPostStmt(const CastExpr * CE,CheckerContext & C) const677330f729Sjoerg void checkPostStmt(const CastExpr *CE, CheckerContext &C) const {
687330f729Sjoerg if (isCallbackEnabled(C, "PostStmtCastExpr"))
697330f729Sjoerg llvm::errs() << "PostStmt<CastExpr> (Kind : " << CE->getCastKindName()
707330f729Sjoerg << ")\n";
717330f729Sjoerg }
727330f729Sjoerg
checkPreStmt(const ArraySubscriptExpr * SubExpr,CheckerContext & C) const737330f729Sjoerg void checkPreStmt(const ArraySubscriptExpr *SubExpr,
747330f729Sjoerg CheckerContext &C) const {
757330f729Sjoerg if (isCallbackEnabled(C, "PreStmtArraySubscriptExpr"))
767330f729Sjoerg llvm::errs() << "PreStmt<ArraySubscriptExpr>\n";
777330f729Sjoerg }
787330f729Sjoerg
checkPostStmt(const ArraySubscriptExpr * SubExpr,CheckerContext & C) const797330f729Sjoerg void checkPostStmt(const ArraySubscriptExpr *SubExpr,
807330f729Sjoerg CheckerContext &C) const {
817330f729Sjoerg if (isCallbackEnabled(C, "PostStmtArraySubscriptExpr"))
827330f729Sjoerg llvm::errs() << "PostStmt<ArraySubscriptExpr>\n";
837330f729Sjoerg }
847330f729Sjoerg
checkPreStmt(const CXXNewExpr * NE,CheckerContext & C) const857330f729Sjoerg void checkPreStmt(const CXXNewExpr *NE, CheckerContext &C) const {
867330f729Sjoerg if (isCallbackEnabled(C, "PreStmtCXXNewExpr"))
877330f729Sjoerg llvm::errs() << "PreStmt<CXXNewExpr>\n";
887330f729Sjoerg }
897330f729Sjoerg
checkPostStmt(const CXXNewExpr * NE,CheckerContext & C) const907330f729Sjoerg void checkPostStmt(const CXXNewExpr *NE, CheckerContext &C) const {
917330f729Sjoerg if (isCallbackEnabled(C, "PostStmtCXXNewExpr"))
927330f729Sjoerg llvm::errs() << "PostStmt<CXXNewExpr>\n";
937330f729Sjoerg }
947330f729Sjoerg
checkPreStmt(const CXXDeleteExpr * NE,CheckerContext & C) const95*e038c9c4Sjoerg void checkPreStmt(const CXXDeleteExpr *NE, CheckerContext &C) const {
96*e038c9c4Sjoerg if (isCallbackEnabled(C, "PreStmtCXXDeleteExpr"))
97*e038c9c4Sjoerg llvm::errs() << "PreStmt<CXXDeleteExpr>\n";
98*e038c9c4Sjoerg }
99*e038c9c4Sjoerg
checkPostStmt(const CXXDeleteExpr * NE,CheckerContext & C) const100*e038c9c4Sjoerg void checkPostStmt(const CXXDeleteExpr *NE, CheckerContext &C) const {
101*e038c9c4Sjoerg if (isCallbackEnabled(C, "PostStmtCXXDeleteExpr"))
102*e038c9c4Sjoerg llvm::errs() << "PostStmt<CXXDeleteExpr>\n";
103*e038c9c4Sjoerg }
104*e038c9c4Sjoerg
checkPreStmt(const CXXConstructExpr * NE,CheckerContext & C) const105*e038c9c4Sjoerg void checkPreStmt(const CXXConstructExpr *NE, CheckerContext &C) const {
106*e038c9c4Sjoerg if (isCallbackEnabled(C, "PreStmtCXXConstructExpr"))
107*e038c9c4Sjoerg llvm::errs() << "PreStmt<CXXConstructExpr>\n";
108*e038c9c4Sjoerg }
109*e038c9c4Sjoerg
checkPostStmt(const CXXConstructExpr * NE,CheckerContext & C) const110*e038c9c4Sjoerg void checkPostStmt(const CXXConstructExpr *NE, CheckerContext &C) const {
111*e038c9c4Sjoerg if (isCallbackEnabled(C, "PostStmtCXXConstructExpr"))
112*e038c9c4Sjoerg llvm::errs() << "PostStmt<CXXConstructExpr>\n";
113*e038c9c4Sjoerg }
114*e038c9c4Sjoerg
checkPreStmt(const OffsetOfExpr * OOE,CheckerContext & C) const1157330f729Sjoerg void checkPreStmt(const OffsetOfExpr *OOE, CheckerContext &C) const {
1167330f729Sjoerg if (isCallbackEnabled(C, "PreStmtOffsetOfExpr"))
1177330f729Sjoerg llvm::errs() << "PreStmt<OffsetOfExpr>\n";
1187330f729Sjoerg }
1197330f729Sjoerg
checkPostStmt(const OffsetOfExpr * OOE,CheckerContext & C) const1207330f729Sjoerg void checkPostStmt(const OffsetOfExpr *OOE, CheckerContext &C) const {
1217330f729Sjoerg if (isCallbackEnabled(C, "PostStmtOffsetOfExpr"))
1227330f729Sjoerg llvm::errs() << "PostStmt<OffsetOfExpr>\n";
1237330f729Sjoerg }
1247330f729Sjoerg
evalCall(const CallEvent & Call,CheckerContext & C) const125*e038c9c4Sjoerg bool evalCall(const CallEvent &Call, CheckerContext &C) const {
126*e038c9c4Sjoerg if (isCallbackEnabled(C, "EvalCall")) {
127*e038c9c4Sjoerg llvm::errs() << "EvalCall";
128*e038c9c4Sjoerg if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))
129*e038c9c4Sjoerg llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';
130*e038c9c4Sjoerg llvm::errs() << " {argno: " << Call.getNumArgs() << '}';
131*e038c9c4Sjoerg llvm::errs() << " [" << Call.getKindAsString() << ']';
132*e038c9c4Sjoerg llvm::errs() << '\n';
133*e038c9c4Sjoerg return true;
134*e038c9c4Sjoerg }
135*e038c9c4Sjoerg return false;
136*e038c9c4Sjoerg }
137*e038c9c4Sjoerg
checkPreCall(const CallEvent & Call,CheckerContext & C) const1387330f729Sjoerg void checkPreCall(const CallEvent &Call, CheckerContext &C) const {
1397330f729Sjoerg if (isCallbackEnabled(C, "PreCall")) {
1407330f729Sjoerg llvm::errs() << "PreCall";
1417330f729Sjoerg if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))
1427330f729Sjoerg llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';
143*e038c9c4Sjoerg llvm::errs() << " [" << Call.getKindAsString() << ']';
1447330f729Sjoerg llvm::errs() << '\n';
1457330f729Sjoerg }
1467330f729Sjoerg }
1477330f729Sjoerg
checkPostCall(const CallEvent & Call,CheckerContext & C) const1487330f729Sjoerg void checkPostCall(const CallEvent &Call, CheckerContext &C) const {
1497330f729Sjoerg if (isCallbackEnabled(C, "PostCall")) {
1507330f729Sjoerg llvm::errs() << "PostCall";
1517330f729Sjoerg if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))
1527330f729Sjoerg llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';
153*e038c9c4Sjoerg llvm::errs() << " [" << Call.getKindAsString() << ']';
1547330f729Sjoerg llvm::errs() << '\n';
1557330f729Sjoerg }
1567330f729Sjoerg }
1577330f729Sjoerg
checkEndFunction(const ReturnStmt * S,CheckerContext & C) const1587330f729Sjoerg void checkEndFunction(const ReturnStmt *S, CheckerContext &C) const {
1597330f729Sjoerg if (isCallbackEnabled(C, "EndFunction")) {
1607330f729Sjoerg llvm::errs() << "EndFunction\nReturnStmt: " << (S ? "yes" : "no") << "\n";
1617330f729Sjoerg if (!S)
1627330f729Sjoerg return;
1637330f729Sjoerg
1647330f729Sjoerg llvm::errs() << "CFGElement: ";
1657330f729Sjoerg CFGStmtMap *Map = C.getCurrentAnalysisDeclContext()->getCFGStmtMap();
1667330f729Sjoerg CFGElement LastElement = Map->getBlock(S)->back();
1677330f729Sjoerg
1687330f729Sjoerg if (LastElement.getAs<CFGStmt>())
1697330f729Sjoerg llvm::errs() << "CFGStmt\n";
1707330f729Sjoerg else if (LastElement.getAs<CFGAutomaticObjDtor>())
1717330f729Sjoerg llvm::errs() << "CFGAutomaticObjDtor\n";
1727330f729Sjoerg }
1737330f729Sjoerg }
1747330f729Sjoerg
checkEndAnalysis(ExplodedGraph & G,BugReporter & BR,ExprEngine & Eng) const175*e038c9c4Sjoerg void checkEndAnalysis(ExplodedGraph &G, BugReporter &BR,
176*e038c9c4Sjoerg ExprEngine &Eng) const {
177*e038c9c4Sjoerg if (isCallbackEnabled(BR.getAnalyzerOptions(), "EndAnalysis"))
178*e038c9c4Sjoerg llvm::errs() << "EndAnalysis\n";
179*e038c9c4Sjoerg }
180*e038c9c4Sjoerg
checkNewAllocator(const CXXAllocatorCall & Call,CheckerContext & C) const181*e038c9c4Sjoerg void checkNewAllocator(const CXXAllocatorCall &Call,
1827330f729Sjoerg CheckerContext &C) const {
1837330f729Sjoerg if (isCallbackEnabled(C, "NewAllocator"))
1847330f729Sjoerg llvm::errs() << "NewAllocator\n";
1857330f729Sjoerg }
1867330f729Sjoerg
checkBind(SVal Loc,SVal Val,const Stmt * S,CheckerContext & C) const1877330f729Sjoerg void checkBind(SVal Loc, SVal Val, const Stmt *S, CheckerContext &C) const {
1887330f729Sjoerg if (isCallbackEnabled(C, "Bind"))
1897330f729Sjoerg llvm::errs() << "Bind\n";
1907330f729Sjoerg }
1917330f729Sjoerg
checkLiveSymbols(ProgramStateRef State,SymbolReaper & SymReaper) const1927330f729Sjoerg void checkLiveSymbols(ProgramStateRef State, SymbolReaper &SymReaper) const {
1937330f729Sjoerg if (isCallbackEnabled(State, "LiveSymbols"))
1947330f729Sjoerg llvm::errs() << "LiveSymbols\n";
1957330f729Sjoerg }
1967330f729Sjoerg
1977330f729Sjoerg ProgramStateRef
checkRegionChanges(ProgramStateRef State,const InvalidatedSymbols * Invalidated,ArrayRef<const MemRegion * > ExplicitRegions,ArrayRef<const MemRegion * > Regions,const LocationContext * LCtx,const CallEvent * Call) const1987330f729Sjoerg checkRegionChanges(ProgramStateRef State,
1997330f729Sjoerg const InvalidatedSymbols *Invalidated,
2007330f729Sjoerg ArrayRef<const MemRegion *> ExplicitRegions,
2017330f729Sjoerg ArrayRef<const MemRegion *> Regions,
2027330f729Sjoerg const LocationContext *LCtx, const CallEvent *Call) const {
2037330f729Sjoerg if (isCallbackEnabled(State, "RegionChanges"))
2047330f729Sjoerg llvm::errs() << "RegionChanges\n";
2057330f729Sjoerg return State;
2067330f729Sjoerg }
207*e038c9c4Sjoerg
checkPointerEscape(ProgramStateRef State,const InvalidatedSymbols & Escaped,const CallEvent * Call,PointerEscapeKind Kind) const208*e038c9c4Sjoerg ProgramStateRef checkPointerEscape(ProgramStateRef State,
209*e038c9c4Sjoerg const InvalidatedSymbols &Escaped,
210*e038c9c4Sjoerg const CallEvent *Call,
211*e038c9c4Sjoerg PointerEscapeKind Kind) const {
212*e038c9c4Sjoerg if (isCallbackEnabled(State, "PointerEscape"))
213*e038c9c4Sjoerg llvm::errs() << "PointerEscape\n";
214*e038c9c4Sjoerg return State;
215*e038c9c4Sjoerg }
2167330f729Sjoerg };
2177330f729Sjoerg } // end anonymous namespace
2187330f729Sjoerg
2197330f729Sjoerg //===----------------------------------------------------------------------===//
2207330f729Sjoerg // Registration.
2217330f729Sjoerg //===----------------------------------------------------------------------===//
2227330f729Sjoerg
registerAnalysisOrderChecker(CheckerManager & mgr)2237330f729Sjoerg void ento::registerAnalysisOrderChecker(CheckerManager &mgr) {
2247330f729Sjoerg mgr.registerChecker<AnalysisOrderChecker>();
2257330f729Sjoerg }
2267330f729Sjoerg
shouldRegisterAnalysisOrderChecker(const CheckerManager & mgr)227*e038c9c4Sjoerg bool ento::shouldRegisterAnalysisOrderChecker(const CheckerManager &mgr) {
2287330f729Sjoerg return true;
2297330f729Sjoerg }
230