forked from OSchip/llvm-project
[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:
parent
5a9baa330c
commit
38679fd630
|
@ -16,6 +16,7 @@
|
||||||
|
|
||||||
#include "ClangSACheckers.h"
|
#include "ClangSACheckers.h"
|
||||||
#include "clang/AST/ExprCXX.h"
|
#include "clang/AST/ExprCXX.h"
|
||||||
|
#include "clang/Analysis/CFGStmtMap.h"
|
||||||
#include "clang/StaticAnalyzer/Core/Checker.h"
|
#include "clang/StaticAnalyzer/Core/Checker.h"
|
||||||
#include "clang/StaticAnalyzer/Core/CheckerManager.h"
|
#include "clang/StaticAnalyzer/Core/CheckerManager.h"
|
||||||
#include "clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h"
|
#include "clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h"
|
||||||
|
@ -37,6 +38,7 @@ class AnalysisOrderChecker
|
||||||
check::PostStmt<OffsetOfExpr>,
|
check::PostStmt<OffsetOfExpr>,
|
||||||
check::PreCall,
|
check::PreCall,
|
||||||
check::PostCall,
|
check::PostCall,
|
||||||
|
check::EndFunction,
|
||||||
check::NewAllocator,
|
check::NewAllocator,
|
||||||
check::Bind,
|
check::Bind,
|
||||||
check::RegionChanges,
|
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,
|
void checkNewAllocator(const CXXNewExpr *CNE, SVal Target,
|
||||||
CheckerContext &C) const {
|
CheckerContext &C) const {
|
||||||
if (isCallbackEnabled(C, "NewAllocator"))
|
if (isCallbackEnabled(C, "NewAllocator"))
|
||||||
|
|
|
@ -223,8 +223,12 @@ void CoreEngine::HandleBlockEdge(const BlockEdge &L, ExplodedNode *Pred) {
|
||||||
// Get return statement..
|
// Get return statement..
|
||||||
const ReturnStmt *RS = nullptr;
|
const ReturnStmt *RS = nullptr;
|
||||||
if (!L.getSrc()->empty()) {
|
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());
|
RS = dyn_cast<ReturnStmt>(LastStmt->getStmt());
|
||||||
|
} else if (Optional<CFGAutomaticObjDtor> AutoDtor =
|
||||||
|
LastElement.getAs<CFGAutomaticObjDtor>()) {
|
||||||
|
RS = dyn_cast<ReturnStmt>(AutoDtor->getTriggerStmt());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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
|
Loading…
Reference in New Issue