forked from OSchip/llvm-project
[analyzer] Refactor the logic that determines if a functions should be
reanalyzed. The policy on what to reanalyze should be in AnalysisConsumer with the rest of visitation order logic. There is no reason why ExprEngine needs to pass the Visited set to CoreEngine, it can populate it itself. llvm-svn: 162957
This commit is contained in:
parent
cc61f87cf7
commit
a8017eca1a
|
@ -80,10 +80,6 @@ private:
|
|||
/// usually because it could not reason about something.
|
||||
BlocksAborted blocksAborted;
|
||||
|
||||
/// The functions which have been analyzed through inlining. This is owned by
|
||||
/// AnalysisConsumer. It can be null.
|
||||
SetOfConstDecls *AnalyzedCallees;
|
||||
|
||||
/// The information about functions shared by the whole translation unit.
|
||||
/// (This data is owned by AnalysisConsumer.)
|
||||
FunctionSummariesTy *FunctionSummaries;
|
||||
|
@ -108,12 +104,11 @@ private:
|
|||
|
||||
public:
|
||||
/// Construct a CoreEngine object to analyze the provided CFG.
|
||||
CoreEngine(SubEngine& subengine, SetOfConstDecls *VisitedCallees,
|
||||
CoreEngine(SubEngine& subengine,
|
||||
FunctionSummariesTy *FS)
|
||||
: SubEng(subengine), G(new ExplodedGraph()),
|
||||
WList(WorkList::makeDFS()),
|
||||
BCounterFactory(G->getAllocator()),
|
||||
AnalyzedCallees(VisitedCallees),
|
||||
FunctionSummaries(FS){}
|
||||
|
||||
/// getGraph - Returns the exploded graph.
|
||||
|
|
|
@ -90,9 +90,13 @@ class ExprEngine : public SubEngine {
|
|||
/// destructor is called before the rest of the ExprEngine is destroyed.
|
||||
GRBugReporter BR;
|
||||
|
||||
/// The functions which have been analyzed through inlining. This is owned by
|
||||
/// AnalysisConsumer. It can be null.
|
||||
SetOfConstDecls *VisitedCallees;
|
||||
|
||||
public:
|
||||
ExprEngine(AnalysisManager &mgr, bool gcEnabled,
|
||||
SetOfConstDecls *VisitedCallees,
|
||||
SetOfConstDecls *VisitedCalleesIn,
|
||||
FunctionSummariesTy *FS);
|
||||
|
||||
~ExprEngine();
|
||||
|
|
|
@ -243,11 +243,6 @@ void CoreEngine::dispatchWorkItem(ExplodedNode* Pred, ProgramPoint Loc,
|
|||
|
||||
case ProgramPoint::CallEnterKind: {
|
||||
CallEnter CEnter = cast<CallEnter>(Loc);
|
||||
if (AnalyzedCallees)
|
||||
if (const CallExpr* CE =
|
||||
dyn_cast_or_null<CallExpr>(CEnter.getCallExpr()))
|
||||
if (const Decl *CD = CE->getCalleeDecl())
|
||||
AnalyzedCallees->insert(CD);
|
||||
SubEng.processCallEnter(CEnter, Pred);
|
||||
break;
|
||||
}
|
||||
|
|
|
@ -55,11 +55,11 @@ STATISTIC(NumTimesRetriedWithoutInlining,
|
|||
//===----------------------------------------------------------------------===//
|
||||
|
||||
ExprEngine::ExprEngine(AnalysisManager &mgr, bool gcEnabled,
|
||||
SetOfConstDecls *VisitedCallees,
|
||||
SetOfConstDecls *VisitedCalleesIn,
|
||||
FunctionSummariesTy *FS)
|
||||
: AMgr(mgr),
|
||||
AnalysisDeclContexts(mgr.getAnalysisDeclContextManager()),
|
||||
Engine(*this, VisitedCallees, FS),
|
||||
Engine(*this, FS),
|
||||
G(Engine.getGraph()),
|
||||
StateMgr(getContext(), mgr.getStoreManagerCreator(),
|
||||
mgr.getConstraintManagerCreator(), G.getAllocator(),
|
||||
|
@ -70,7 +70,8 @@ ExprEngine::ExprEngine(AnalysisManager &mgr, bool gcEnabled,
|
|||
currStmt(NULL), currStmtIdx(0), currBldrCtx(0),
|
||||
NSExceptionII(NULL), NSExceptionInstanceRaiseSelectors(NULL),
|
||||
RaiseSel(GetNullarySelector("raise", getContext())),
|
||||
ObjCGCEnabled(gcEnabled), BR(mgr, *this)
|
||||
ObjCGCEnabled(gcEnabled), BR(mgr, *this),
|
||||
VisitedCallees(VisitedCalleesIn)
|
||||
{
|
||||
if (mgr.options.eagerlyTrimExplodedGraph) {
|
||||
// Enable eager node reclaimation when constructing the ExplodedGraph.
|
||||
|
|
|
@ -465,6 +465,10 @@ bool ExprEngine::inlineCall(const CallEvent &Call, const Decl *D,
|
|||
|
||||
NumInlinedCalls++;
|
||||
|
||||
// Mark the decl as visited.
|
||||
if (VisitedCallees)
|
||||
VisitedCallees->insert(D);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
|
|
@ -348,6 +348,22 @@ void AnalysisConsumer::storeTopLevelDecls(DeclGroupRef DG) {
|
|||
}
|
||||
}
|
||||
|
||||
static bool shouldSkipFunction(CallGraphNode *N,
|
||||
SmallPtrSet<CallGraphNode*,24> Visited) {
|
||||
// We want to re-analyse the functions as top level in several cases:
|
||||
// - The 'init' methods should be reanalyzed because
|
||||
// ObjCNonNilReturnValueChecker assumes that '[super init]' never returns
|
||||
// 'nil' and unless we analyze the 'init' functions as top level, we will not
|
||||
// catch errors within defensive code.
|
||||
// - We want to reanalyze all ObjC methods as top level to report Retain
|
||||
// Count naming convention errors more aggressively.
|
||||
if (isa<ObjCMethodDecl>(N->getDecl()))
|
||||
return false;
|
||||
|
||||
// Otherwise, if we visited the function before, do not reanalyze it.
|
||||
return Visited.count(N);
|
||||
}
|
||||
|
||||
void AnalysisConsumer::HandleDeclsGallGraph(const unsigned LocalTUDeclsSize) {
|
||||
// Otherwise, use the Callgraph to derive the order.
|
||||
// Build the Call Graph.
|
||||
|
@ -397,13 +413,13 @@ void AnalysisConsumer::HandleDeclsGallGraph(const unsigned LocalTUDeclsSize) {
|
|||
// Push the children into the queue.
|
||||
for (CallGraphNode::const_iterator CI = N->begin(),
|
||||
CE = N->end(); CI != CE; ++CI) {
|
||||
if (!Visited.count(*CI))
|
||||
if (!shouldSkipFunction(*CI, Visited))
|
||||
BFSQueue.push_back(*CI);
|
||||
}
|
||||
|
||||
// Skip the functions which have been processed already or previously
|
||||
// inlined.
|
||||
if (Visited.count(N))
|
||||
if (shouldSkipFunction(N, Visited))
|
||||
continue;
|
||||
|
||||
// Analyze the function.
|
||||
|
|
Loading…
Reference in New Issue