Path: blob/main/contrib/llvm-project/clang/lib/StaticAnalyzer/Checkers/AnalysisOrderChecker.cpp
35266 views
//===- AnalysisOrderChecker - Print callbacks called ------------*- C++ -*-===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//7//8// This checker prints callbacks that are called during analysis.9// This is required to ensure that callbacks are fired in order10// and do not duplicate or get lost.11// Feel free to extend this checker with any callback you need to check.12//13//===----------------------------------------------------------------------===//1415#include "clang/AST/ExprCXX.h"16#include "clang/Analysis/CFGStmtMap.h"17#include "clang/StaticAnalyzer/Checkers/BuiltinCheckerRegistration.h"18#include "clang/StaticAnalyzer/Core/Checker.h"19#include "clang/StaticAnalyzer/Core/CheckerManager.h"20#include "clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h"21#include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h"22#include "llvm/Support/ErrorHandling.h"2324using namespace clang;25using namespace ento;2627namespace {2829class AnalysisOrderChecker30: public Checker<31check::PreStmt<CastExpr>, check::PostStmt<CastExpr>,32check::PreStmt<ArraySubscriptExpr>,33check::PostStmt<ArraySubscriptExpr>, check::PreStmt<CXXNewExpr>,34check::PostStmt<CXXNewExpr>, check::PreStmt<CXXDeleteExpr>,35check::PostStmt<CXXDeleteExpr>, check::PreStmt<CXXConstructExpr>,36check::PostStmt<CXXConstructExpr>, check::PreStmt<OffsetOfExpr>,37check::PostStmt<OffsetOfExpr>, check::PreCall, check::PostCall,38check::EndFunction, check::EndAnalysis, check::NewAllocator,39check::Bind, check::PointerEscape, check::RegionChanges,40check::LiveSymbols, eval::Call> {4142bool isCallbackEnabled(const AnalyzerOptions &Opts,43StringRef CallbackName) const {44return Opts.getCheckerBooleanOption(this, "*") ||45Opts.getCheckerBooleanOption(this, CallbackName);46}4748bool isCallbackEnabled(CheckerContext &C, StringRef CallbackName) const {49AnalyzerOptions &Opts = C.getAnalysisManager().getAnalyzerOptions();50return isCallbackEnabled(Opts, CallbackName);51}5253bool isCallbackEnabled(ProgramStateRef State, StringRef CallbackName) const {54AnalyzerOptions &Opts = State->getStateManager().getOwningEngine()55.getAnalysisManager().getAnalyzerOptions();56return isCallbackEnabled(Opts, CallbackName);57}5859public:60void checkPreStmt(const CastExpr *CE, CheckerContext &C) const {61if (isCallbackEnabled(C, "PreStmtCastExpr"))62llvm::errs() << "PreStmt<CastExpr> (Kind : " << CE->getCastKindName()63<< ")\n";64}6566void checkPostStmt(const CastExpr *CE, CheckerContext &C) const {67if (isCallbackEnabled(C, "PostStmtCastExpr"))68llvm::errs() << "PostStmt<CastExpr> (Kind : " << CE->getCastKindName()69<< ")\n";70}7172void checkPreStmt(const ArraySubscriptExpr *SubExpr,73CheckerContext &C) const {74if (isCallbackEnabled(C, "PreStmtArraySubscriptExpr"))75llvm::errs() << "PreStmt<ArraySubscriptExpr>\n";76}7778void checkPostStmt(const ArraySubscriptExpr *SubExpr,79CheckerContext &C) const {80if (isCallbackEnabled(C, "PostStmtArraySubscriptExpr"))81llvm::errs() << "PostStmt<ArraySubscriptExpr>\n";82}8384void checkPreStmt(const CXXNewExpr *NE, CheckerContext &C) const {85if (isCallbackEnabled(C, "PreStmtCXXNewExpr"))86llvm::errs() << "PreStmt<CXXNewExpr>\n";87}8889void checkPostStmt(const CXXNewExpr *NE, CheckerContext &C) const {90if (isCallbackEnabled(C, "PostStmtCXXNewExpr"))91llvm::errs() << "PostStmt<CXXNewExpr>\n";92}9394void checkPreStmt(const CXXDeleteExpr *NE, CheckerContext &C) const {95if (isCallbackEnabled(C, "PreStmtCXXDeleteExpr"))96llvm::errs() << "PreStmt<CXXDeleteExpr>\n";97}9899void checkPostStmt(const CXXDeleteExpr *NE, CheckerContext &C) const {100if (isCallbackEnabled(C, "PostStmtCXXDeleteExpr"))101llvm::errs() << "PostStmt<CXXDeleteExpr>\n";102}103104void checkPreStmt(const CXXConstructExpr *NE, CheckerContext &C) const {105if (isCallbackEnabled(C, "PreStmtCXXConstructExpr"))106llvm::errs() << "PreStmt<CXXConstructExpr>\n";107}108109void checkPostStmt(const CXXConstructExpr *NE, CheckerContext &C) const {110if (isCallbackEnabled(C, "PostStmtCXXConstructExpr"))111llvm::errs() << "PostStmt<CXXConstructExpr>\n";112}113114void checkPreStmt(const OffsetOfExpr *OOE, CheckerContext &C) const {115if (isCallbackEnabled(C, "PreStmtOffsetOfExpr"))116llvm::errs() << "PreStmt<OffsetOfExpr>\n";117}118119void checkPostStmt(const OffsetOfExpr *OOE, CheckerContext &C) const {120if (isCallbackEnabled(C, "PostStmtOffsetOfExpr"))121llvm::errs() << "PostStmt<OffsetOfExpr>\n";122}123124bool evalCall(const CallEvent &Call, CheckerContext &C) const {125if (isCallbackEnabled(C, "EvalCall")) {126llvm::errs() << "EvalCall";127if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))128llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';129llvm::errs() << " {argno: " << Call.getNumArgs() << '}';130llvm::errs() << " [" << Call.getKindAsString() << ']';131llvm::errs() << '\n';132return true;133}134return false;135}136137void checkPreCall(const CallEvent &Call, CheckerContext &C) const {138if (isCallbackEnabled(C, "PreCall")) {139llvm::errs() << "PreCall";140if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))141llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';142llvm::errs() << " [" << Call.getKindAsString() << ']';143llvm::errs() << '\n';144}145}146147void checkPostCall(const CallEvent &Call, CheckerContext &C) const {148if (isCallbackEnabled(C, "PostCall")) {149llvm::errs() << "PostCall";150if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(Call.getDecl()))151llvm::errs() << " (" << ND->getQualifiedNameAsString() << ')';152llvm::errs() << " [" << Call.getKindAsString() << ']';153llvm::errs() << '\n';154}155}156157void checkEndFunction(const ReturnStmt *S, CheckerContext &C) const {158if (isCallbackEnabled(C, "EndFunction")) {159llvm::errs() << "EndFunction\nReturnStmt: " << (S ? "yes" : "no") << "\n";160if (!S)161return;162163llvm::errs() << "CFGElement: ";164CFGStmtMap *Map = C.getCurrentAnalysisDeclContext()->getCFGStmtMap();165CFGElement LastElement = Map->getBlock(S)->back();166167if (LastElement.getAs<CFGStmt>())168llvm::errs() << "CFGStmt\n";169else if (LastElement.getAs<CFGAutomaticObjDtor>())170llvm::errs() << "CFGAutomaticObjDtor\n";171}172}173174void checkEndAnalysis(ExplodedGraph &G, BugReporter &BR,175ExprEngine &Eng) const {176if (isCallbackEnabled(BR.getAnalyzerOptions(), "EndAnalysis"))177llvm::errs() << "EndAnalysis\n";178}179180void checkNewAllocator(const CXXAllocatorCall &Call,181CheckerContext &C) const {182if (isCallbackEnabled(C, "NewAllocator"))183llvm::errs() << "NewAllocator\n";184}185186void checkBind(SVal Loc, SVal Val, const Stmt *S, CheckerContext &C) const {187if (isCallbackEnabled(C, "Bind"))188llvm::errs() << "Bind\n";189}190191void checkLiveSymbols(ProgramStateRef State, SymbolReaper &SymReaper) const {192if (isCallbackEnabled(State, "LiveSymbols"))193llvm::errs() << "LiveSymbols\n";194}195196ProgramStateRef197checkRegionChanges(ProgramStateRef State,198const InvalidatedSymbols *Invalidated,199ArrayRef<const MemRegion *> ExplicitRegions,200ArrayRef<const MemRegion *> Regions,201const LocationContext *LCtx, const CallEvent *Call) const {202if (isCallbackEnabled(State, "RegionChanges"))203llvm::errs() << "RegionChanges\n";204return State;205}206207ProgramStateRef checkPointerEscape(ProgramStateRef State,208const InvalidatedSymbols &Escaped,209const CallEvent *Call,210PointerEscapeKind Kind) const {211if (isCallbackEnabled(State, "PointerEscape"))212llvm::errs() << "PointerEscape\n";213return State;214}215};216} // end anonymous namespace217218//===----------------------------------------------------------------------===//219// Registration.220//===----------------------------------------------------------------------===//221222void ento::registerAnalysisOrderChecker(CheckerManager &mgr) {223mgr.registerChecker<AnalysisOrderChecker>();224}225226bool ento::shouldRegisterAnalysisOrderChecker(const CheckerManager &mgr) {227return true;228}229230231