129afb193SAleksei Sidorin //===- AnalysisOrderChecker - Print callbacks called ------------*- C++ -*-===//
229afb193SAleksei Sidorin //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
629afb193SAleksei Sidorin //
729afb193SAleksei Sidorin //===----------------------------------------------------------------------===//
829afb193SAleksei Sidorin //
929afb193SAleksei Sidorin // This checker prints callbacks that are called during analysis.
1029afb193SAleksei Sidorin // This is required to ensure that callbacks are fired in order
1129afb193SAleksei Sidorin // and do not duplicate or get lost.
1229afb193SAleksei Sidorin // Feel free to extend this checker with any callback you need to check.
1329afb193SAleksei Sidorin //
1429afb193SAleksei Sidorin //===----------------------------------------------------------------------===//
1529afb193SAleksei Sidorin
16e941daefSArtem Dergachev #include "clang/AST/ExprCXX.h"
1738679fd6SReka Kovacs #include "clang/Analysis/CFGStmtMap.h"
18023c4d40SKirstóf Umann #include "clang/StaticAnalyzer/Checkers/BuiltinCheckerRegistration.h"
1929afb193SAleksei Sidorin #include "clang/StaticAnalyzer/Core/Checker.h"
2029afb193SAleksei Sidorin #include "clang/StaticAnalyzer/Core/CheckerManager.h"
21e941daefSArtem Dergachev #include "clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h"
2229afb193SAleksei Sidorin #include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h"
23023c4d40SKirstóf Umann #include "llvm/Support/ErrorHandling.h"
2429afb193SAleksei Sidorin
2529afb193SAleksei Sidorin using namespace clang;
2629afb193SAleksei Sidorin using namespace ento;
2729afb193SAleksei Sidorin
2829afb193SAleksei Sidorin namespace {
2929afb193SAleksei Sidorin
30647804a8SArtem Dergachev class AnalysisOrderChecker
31023c4d40SKirstóf Umann : public Checker<
32023c4d40SKirstóf Umann check::PreStmt<CastExpr>, check::PostStmt<CastExpr>,
33fb859a93SAnna Zaks check::PreStmt<ArraySubscriptExpr>,
34023c4d40SKirstóf Umann check::PostStmt<ArraySubscriptExpr>, check::PreStmt<CXXNewExpr>,
35023c4d40SKirstóf Umann check::PostStmt<CXXNewExpr>, check::PreStmt<CXXDeleteExpr>,
36023c4d40SKirstóf Umann check::PostStmt<CXXDeleteExpr>, check::PreStmt<CXXConstructExpr>,
37023c4d40SKirstóf Umann check::PostStmt<CXXConstructExpr>, check::PreStmt<OffsetOfExpr>,
38023c4d40SKirstóf Umann check::PostStmt<OffsetOfExpr>, check::PreCall, check::PostCall,
39023c4d40SKirstóf Umann check::EndFunction, check::EndAnalysis, check::NewAllocator,
40023c4d40SKirstóf Umann check::Bind, check::PointerEscape, check::RegionChanges,
41*37c1bf21SNithin Vadukkumchery Rajendrakumar check::LiveSymbols, eval::Call> {
42fb4acffbSGeorge Karpenkov
isCallbackEnabled(const AnalyzerOptions & Opts,StringRef CallbackName) const43023c4d40SKirstóf Umann bool isCallbackEnabled(const AnalyzerOptions &Opts,
44023c4d40SKirstóf Umann StringRef CallbackName) const {
4583cc1b35SKristof Umann return Opts.getCheckerBooleanOption(this, "*") ||
4683cc1b35SKristof Umann Opts.getCheckerBooleanOption(this, CallbackName);
4729afb193SAleksei Sidorin }
4829afb193SAleksei Sidorin
isCallbackEnabled(CheckerContext & C,StringRef CallbackName) const49647804a8SArtem Dergachev bool isCallbackEnabled(CheckerContext &C, StringRef CallbackName) const {
50647804a8SArtem Dergachev AnalyzerOptions &Opts = C.getAnalysisManager().getAnalyzerOptions();
51647804a8SArtem Dergachev return isCallbackEnabled(Opts, CallbackName);
52647804a8SArtem Dergachev }
53647804a8SArtem Dergachev
isCallbackEnabled(ProgramStateRef State,StringRef CallbackName) const54647804a8SArtem Dergachev bool isCallbackEnabled(ProgramStateRef State, StringRef CallbackName) const {
55647804a8SArtem Dergachev AnalyzerOptions &Opts = State->getStateManager().getOwningEngine()
5621aa8db6SGabor Horvath .getAnalysisManager().getAnalyzerOptions();
57647804a8SArtem Dergachev return isCallbackEnabled(Opts, CallbackName);
58647804a8SArtem Dergachev }
59647804a8SArtem Dergachev
6029afb193SAleksei Sidorin public:
checkPreStmt(const CastExpr * CE,CheckerContext & C) const6129afb193SAleksei Sidorin void checkPreStmt(const CastExpr *CE, CheckerContext &C) const {
6229afb193SAleksei Sidorin if (isCallbackEnabled(C, "PreStmtCastExpr"))
6329afb193SAleksei Sidorin llvm::errs() << "PreStmt<CastExpr> (Kind : " << CE->getCastKindName()
6429afb193SAleksei Sidorin << ")\n";
6529afb193SAleksei Sidorin }
6629afb193SAleksei Sidorin
checkPostStmt(const CastExpr * CE,CheckerContext & C) const6729afb193SAleksei Sidorin void checkPostStmt(const CastExpr *CE, CheckerContext &C) const {
6829afb193SAleksei Sidorin if (isCallbackEnabled(C, "PostStmtCastExpr"))
6929afb193SAleksei Sidorin llvm::errs() << "PostStmt<CastExpr> (Kind : " << CE->getCastKindName()
7029afb193SAleksei Sidorin << ")\n";
7129afb193SAleksei Sidorin }
72fb859a93SAnna Zaks
checkPreStmt(const ArraySubscriptExpr * SubExpr,CheckerContext & C) const73647804a8SArtem Dergachev void checkPreStmt(const ArraySubscriptExpr *SubExpr,
74647804a8SArtem Dergachev CheckerContext &C) const {
75fb859a93SAnna Zaks if (isCallbackEnabled(C, "PreStmtArraySubscriptExpr"))
76fb859a93SAnna Zaks llvm::errs() << "PreStmt<ArraySubscriptExpr>\n";
77fb859a93SAnna Zaks }
78fb859a93SAnna Zaks
checkPostStmt(const ArraySubscriptExpr * SubExpr,CheckerContext & C) const79647804a8SArtem Dergachev void checkPostStmt(const ArraySubscriptExpr *SubExpr,
80647804a8SArtem Dergachev CheckerContext &C) const {
81fb859a93SAnna Zaks if (isCallbackEnabled(C, "PostStmtArraySubscriptExpr"))
82fb859a93SAnna Zaks llvm::errs() << "PostStmt<ArraySubscriptExpr>\n";
83fb859a93SAnna Zaks }
84647804a8SArtem Dergachev
checkPreStmt(const CXXNewExpr * NE,CheckerContext & C) const85e941daefSArtem Dergachev void checkPreStmt(const CXXNewExpr *NE, CheckerContext &C) const {
86e941daefSArtem Dergachev if (isCallbackEnabled(C, "PreStmtCXXNewExpr"))
87e941daefSArtem Dergachev llvm::errs() << "PreStmt<CXXNewExpr>\n";
88e941daefSArtem Dergachev }
89e941daefSArtem Dergachev
checkPostStmt(const CXXNewExpr * NE,CheckerContext & C) const90e941daefSArtem Dergachev void checkPostStmt(const CXXNewExpr *NE, CheckerContext &C) const {
91e941daefSArtem Dergachev if (isCallbackEnabled(C, "PostStmtCXXNewExpr"))
92e941daefSArtem Dergachev llvm::errs() << "PostStmt<CXXNewExpr>\n";
93e941daefSArtem Dergachev }
94e941daefSArtem Dergachev
checkPreStmt(const CXXDeleteExpr * NE,CheckerContext & C) const95023c4d40SKirstóf Umann void checkPreStmt(const CXXDeleteExpr *NE, CheckerContext &C) const {
96023c4d40SKirstóf Umann if (isCallbackEnabled(C, "PreStmtCXXDeleteExpr"))
97023c4d40SKirstóf Umann llvm::errs() << "PreStmt<CXXDeleteExpr>\n";
98023c4d40SKirstóf Umann }
99023c4d40SKirstóf Umann
checkPostStmt(const CXXDeleteExpr * NE,CheckerContext & C) const100023c4d40SKirstóf Umann void checkPostStmt(const CXXDeleteExpr *NE, CheckerContext &C) const {
101023c4d40SKirstóf Umann if (isCallbackEnabled(C, "PostStmtCXXDeleteExpr"))
102023c4d40SKirstóf Umann llvm::errs() << "PostStmt<CXXDeleteExpr>\n";
103023c4d40SKirstóf Umann }
104023c4d40SKirstóf Umann
checkPreStmt(const CXXConstructExpr * NE,CheckerContext & C) const105023c4d40SKirstóf Umann void checkPreStmt(const CXXConstructExpr *NE, CheckerContext &C) const {
106023c4d40SKirstóf Umann if (isCallbackEnabled(C, "PreStmtCXXConstructExpr"))
107023c4d40SKirstóf Umann llvm::errs() << "PreStmt<CXXConstructExpr>\n";
108023c4d40SKirstóf Umann }
109023c4d40SKirstóf Umann
checkPostStmt(const CXXConstructExpr * NE,CheckerContext & C) const110023c4d40SKirstóf Umann void checkPostStmt(const CXXConstructExpr *NE, CheckerContext &C) const {
111023c4d40SKirstóf Umann if (isCallbackEnabled(C, "PostStmtCXXConstructExpr"))
112023c4d40SKirstóf Umann llvm::errs() << "PostStmt<CXXConstructExpr>\n";
113023c4d40SKirstóf Umann }
114023c4d40SKirstóf Umann
checkPreStmt(const OffsetOfExpr * OOE,CheckerContext & C) const1154b0d160aSArtem Dergachev void checkPreStmt(const OffsetOfExpr *OOE, CheckerContext &C) const {
1164b0d160aSArtem Dergachev if (isCallbackEnabled(C, "PreStmtOffsetOfExpr"))
1174b0d160aSArtem Dergachev llvm::errs() << "PreStmt<OffsetOfExpr>\n";
1184b0d160aSArtem Dergachev }
1194b0d160aSArtem Dergachev
checkPostStmt(const OffsetOfExpr * OOE,CheckerContext & C) const1204b0d160aSArtem Dergachev void checkPostStmt(const OffsetOfExpr *OOE, CheckerContext &C) const {
1214b0d160aSArtem Dergachev if (isCallbackEnabled(C, "PostStmtOffsetOfExpr"))
1224b0d160aSArtem Dergachev llvm::errs() << "PostStmt<OffsetOfExpr>\n";
1234b0d160aSArtem Dergachev }
1244b0d160aSArtem Dergachev
evalCall(const CallEvent & Call,CheckerContext & C) const125*37c1bf21SNithin Vadukkumchery Rajendrakumar bool evalCall(const CallEvent &Call, CheckerContext &C) const {
126*37c1bf21SNithin Vadukkumchery Rajendrakumar if (isCallbackEnabled(C, "EvalCall")) {
127*37c1bf21SNithin Vadukkumchery Rajendrakumar llvm::errs() << "EvalCall";
128*37c1bf21SNithin Vadukkumchery Rajendrakumar if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))
129*37c1bf21SNithin Vadukkumchery Rajendrakumar llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';
130*37c1bf21SNithin Vadukkumchery Rajendrakumar llvm::errs() << " {argno: " << Call.getNumArgs() << '}';
131*37c1bf21SNithin Vadukkumchery Rajendrakumar llvm::errs() << " [" << Call.getKindAsString() << ']';
132*37c1bf21SNithin Vadukkumchery Rajendrakumar llvm::errs() << '\n';
133*37c1bf21SNithin Vadukkumchery Rajendrakumar return true;
134*37c1bf21SNithin Vadukkumchery Rajendrakumar }
135*37c1bf21SNithin Vadukkumchery Rajendrakumar return false;
136*37c1bf21SNithin Vadukkumchery Rajendrakumar }
137*37c1bf21SNithin Vadukkumchery Rajendrakumar
checkPreCall(const CallEvent & Call,CheckerContext & C) const138e941daefSArtem Dergachev void checkPreCall(const CallEvent &Call, CheckerContext &C) const {
139e941daefSArtem Dergachev if (isCallbackEnabled(C, "PreCall")) {
140e941daefSArtem Dergachev llvm::errs() << "PreCall";
141e941daefSArtem Dergachev if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))
142e941daefSArtem Dergachev llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';
143023c4d40SKirstóf Umann llvm::errs() << " [" << Call.getKindAsString() << ']';
144e941daefSArtem Dergachev llvm::errs() << '\n';
145e941daefSArtem Dergachev }
146e941daefSArtem Dergachev }
147e941daefSArtem Dergachev
checkPostCall(const CallEvent & Call,CheckerContext & C) const148e941daefSArtem Dergachev void checkPostCall(const CallEvent &Call, CheckerContext &C) const {
149e941daefSArtem Dergachev if (isCallbackEnabled(C, "PostCall")) {
150e941daefSArtem Dergachev llvm::errs() << "PostCall";
151e941daefSArtem Dergachev if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))
152e941daefSArtem Dergachev llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';
153023c4d40SKirstóf Umann llvm::errs() << " [" << Call.getKindAsString() << ']';
154e941daefSArtem Dergachev llvm::errs() << '\n';
155e941daefSArtem Dergachev }
156e941daefSArtem Dergachev }
157e941daefSArtem Dergachev
checkEndFunction(const ReturnStmt * S,CheckerContext & C) const15838679fd6SReka Kovacs void checkEndFunction(const ReturnStmt *S, CheckerContext &C) const {
15938679fd6SReka Kovacs if (isCallbackEnabled(C, "EndFunction")) {
16038679fd6SReka Kovacs llvm::errs() << "EndFunction\nReturnStmt: " << (S ? "yes" : "no") << "\n";
16138679fd6SReka Kovacs if (!S)
16238679fd6SReka Kovacs return;
16338679fd6SReka Kovacs
16438679fd6SReka Kovacs llvm::errs() << "CFGElement: ";
16538679fd6SReka Kovacs CFGStmtMap *Map = C.getCurrentAnalysisDeclContext()->getCFGStmtMap();
16638679fd6SReka Kovacs CFGElement LastElement = Map->getBlock(S)->back();
16738679fd6SReka Kovacs
16838679fd6SReka Kovacs if (LastElement.getAs<CFGStmt>())
16938679fd6SReka Kovacs llvm::errs() << "CFGStmt\n";
17038679fd6SReka Kovacs else if (LastElement.getAs<CFGAutomaticObjDtor>())
17138679fd6SReka Kovacs llvm::errs() << "CFGAutomaticObjDtor\n";
17238679fd6SReka Kovacs }
17338679fd6SReka Kovacs }
17438679fd6SReka Kovacs
checkEndAnalysis(ExplodedGraph & G,BugReporter & BR,ExprEngine & Eng) const175023c4d40SKirstóf Umann void checkEndAnalysis(ExplodedGraph &G, BugReporter &BR,
176023c4d40SKirstóf Umann ExprEngine &Eng) const {
177023c4d40SKirstóf Umann if (isCallbackEnabled(BR.getAnalyzerOptions(), "EndAnalysis"))
178023c4d40SKirstóf Umann llvm::errs() << "EndAnalysis\n";
179023c4d40SKirstóf Umann }
180023c4d40SKirstóf Umann
checkNewAllocator(const CXXAllocatorCall & Call,CheckerContext & C) const181f2be30deSKirstóf Umann void checkNewAllocator(const CXXAllocatorCall &Call,
182e941daefSArtem Dergachev CheckerContext &C) const {
183e941daefSArtem Dergachev if (isCallbackEnabled(C, "NewAllocator"))
184e941daefSArtem Dergachev llvm::errs() << "NewAllocator\n";
185e941daefSArtem Dergachev }
186e941daefSArtem Dergachev
checkBind(SVal Loc,SVal Val,const Stmt * S,CheckerContext & C) const187647804a8SArtem Dergachev void checkBind(SVal Loc, SVal Val, const Stmt *S, CheckerContext &C) const {
188647804a8SArtem Dergachev if (isCallbackEnabled(C, "Bind"))
189647804a8SArtem Dergachev llvm::errs() << "Bind\n";
19029afb193SAleksei Sidorin }
19129afb193SAleksei Sidorin
checkLiveSymbols(ProgramStateRef State,SymbolReaper & SymReaper) const192fb4acffbSGeorge Karpenkov void checkLiveSymbols(ProgramStateRef State, SymbolReaper &SymReaper) const {
193fb4acffbSGeorge Karpenkov if (isCallbackEnabled(State, "LiveSymbols"))
194fb4acffbSGeorge Karpenkov llvm::errs() << "LiveSymbols\n";
195fb4acffbSGeorge Karpenkov }
196fb4acffbSGeorge Karpenkov
197647804a8SArtem Dergachev ProgramStateRef
checkRegionChanges(ProgramStateRef State,const InvalidatedSymbols * Invalidated,ArrayRef<const MemRegion * > ExplicitRegions,ArrayRef<const MemRegion * > Regions,const LocationContext * LCtx,const CallEvent * Call) const198647804a8SArtem Dergachev checkRegionChanges(ProgramStateRef State,
199647804a8SArtem Dergachev const InvalidatedSymbols *Invalidated,
200647804a8SArtem Dergachev ArrayRef<const MemRegion *> ExplicitRegions,
201647804a8SArtem Dergachev ArrayRef<const MemRegion *> Regions,
202647804a8SArtem Dergachev const LocationContext *LCtx, const CallEvent *Call) const {
203647804a8SArtem Dergachev if (isCallbackEnabled(State, "RegionChanges"))
204647804a8SArtem Dergachev llvm::errs() << "RegionChanges\n";
205647804a8SArtem Dergachev return State;
206647804a8SArtem Dergachev }
2075882e6f3SGabor Horvath
checkPointerEscape(ProgramStateRef State,const InvalidatedSymbols & Escaped,const CallEvent * Call,PointerEscapeKind Kind) const2085882e6f3SGabor Horvath ProgramStateRef checkPointerEscape(ProgramStateRef State,
2095882e6f3SGabor Horvath const InvalidatedSymbols &Escaped,
2105882e6f3SGabor Horvath const CallEvent *Call,
2115882e6f3SGabor Horvath PointerEscapeKind Kind) const {
2125882e6f3SGabor Horvath if (isCallbackEnabled(State, "PointerEscape"))
2135882e6f3SGabor Horvath llvm::errs() << "PointerEscape\n";
2145882e6f3SGabor Horvath return State;
2155882e6f3SGabor Horvath }
216647804a8SArtem Dergachev };
217647804a8SArtem Dergachev } // end anonymous namespace
218647804a8SArtem Dergachev
21929afb193SAleksei Sidorin //===----------------------------------------------------------------------===//
22029afb193SAleksei Sidorin // Registration.
22129afb193SAleksei Sidorin //===----------------------------------------------------------------------===//
22229afb193SAleksei Sidorin
registerAnalysisOrderChecker(CheckerManager & mgr)22329afb193SAleksei Sidorin void ento::registerAnalysisOrderChecker(CheckerManager &mgr) {
22429afb193SAleksei Sidorin mgr.registerChecker<AnalysisOrderChecker>();
22529afb193SAleksei Sidorin }
226058a7a45SKristof Umann
shouldRegisterAnalysisOrderChecker(const CheckerManager & mgr)227bda3dd0dSKirstóf Umann bool ento::shouldRegisterAnalysisOrderChecker(const CheckerManager &mgr) {
228058a7a45SKristof Umann return true;
229058a7a45SKristof Umann }
230