[analyzer] Obtain a ReturnStmt from a CFGAutomaticObjDtor.

The CoreEngine only gives us a ReturnStmt if the last element in the
CFGBlock is a CFGStmt, otherwise the ReturnStmt is nullptr.
This patch adds support for the case when the last element is a
CFGAutomaticObjDtor, by returning its TriggerStmt as a ReturnStmt.

Differential Revision: https://reviews.llvm.org/D49811

llvm-svn: 338777
This commit is contained in:
Reka Kovacs 2018-08-02 22:31:03 +00:00
parent 5a9baa330c
commit 38679fd630
3 changed files with 58 additions and 1 deletions

View File

@ -16,6 +16,7 @@
#include "ClangSACheckers.h"
#include "clang/AST/ExprCXX.h"
#include "clang/Analysis/CFGStmtMap.h"
#include "clang/StaticAnalyzer/Core/Checker.h"
#include "clang/StaticAnalyzer/Core/CheckerManager.h"
#include "clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h"
@ -37,6 +38,7 @@ class AnalysisOrderChecker
check::PostStmt<OffsetOfExpr>,
check::PreCall,
check::PostCall,
check::EndFunction,
check::NewAllocator,
check::Bind,
check::RegionChanges,
@ -121,6 +123,23 @@ public:
}
}
void checkEndFunction(const ReturnStmt *S, CheckerContext &C) const {
if (isCallbackEnabled(C, "EndFunction")) {
llvm::errs() << "EndFunction\nReturnStmt: " << (S ? "yes" : "no") << "\n";
if (!S)
return;
llvm::errs() << "CFGElement: ";
CFGStmtMap *Map = C.getCurrentAnalysisDeclContext()->getCFGStmtMap();
CFGElement LastElement = Map->getBlock(S)->back();
if (LastElement.getAs<CFGStmt>())
llvm::errs() << "CFGStmt\n";
else if (LastElement.getAs<CFGAutomaticObjDtor>())
llvm::errs() << "CFGAutomaticObjDtor\n";
}
}
void checkNewAllocator(const CXXNewExpr *CNE, SVal Target,
CheckerContext &C) const {
if (isCallbackEnabled(C, "NewAllocator"))

View File

@ -223,8 +223,12 @@ void CoreEngine::HandleBlockEdge(const BlockEdge &L, ExplodedNode *Pred) {
// Get return statement..
const ReturnStmt *RS = nullptr;
if (!L.getSrc()->empty()) {
if (Optional<CFGStmt> LastStmt = L.getSrc()->back().getAs<CFGStmt>()) {
CFGElement LastElement = L.getSrc()->back();
if (Optional<CFGStmt> LastStmt = LastElement.getAs<CFGStmt>()) {
RS = dyn_cast<ReturnStmt>(LastStmt->getStmt());
} else if (Optional<CFGAutomaticObjDtor> AutoDtor =
LastElement.getAs<CFGAutomaticObjDtor>()) {
RS = dyn_cast<ReturnStmt>(AutoDtor->getTriggerStmt());
}
}

View File

@ -0,0 +1,34 @@
//RUN: %clang_analyze_cc1 -analyzer-checker=debug.AnalysisOrder -analyzer-config debug.AnalysisOrder:EndFunction=true %s 2>&1 | FileCheck %s
// At the end of a function, we can only obtain a ReturnStmt if the last
// CFGElement in the CFGBlock is either a CFGStmt or a CFGAutomaticObjDtor.
void noReturnStmt() {}
struct S {
S();
~S();
};
int dtorAfterReturnStmt() {
S s;
return 0;
}
S endsWithReturnStmt() {
return S();
}
// endsWithReturnStmt()
// CHECK: EndFunction
// CHECK-NEXT: ReturnStmt: yes
// CHECK-NEXT: CFGElement: CFGStmt
// dtorAfterReturnStmt()
// CHECK: EndFunction
// CHECK-NEXT: ReturnStmt: yes
// CHECK-NEXT: CFGElement: CFGAutomaticObjDtor
// noReturnStmt()
// CHECK: EndFunction
// CHECK-NEXT: ReturnStmt: no