44#define DEBUG_TYPE "cgscc-passmgr"
51STATISTIC(MaxSCCIterations,
"Maximum CGSCCPassMgr iterations on one SCC");
68 bool runOnModule(
Module &M)
override;
70 using ModulePass::doInitialization;
71 using ModulePass::doFinalization;
73 bool doInitialization(CallGraph &CG);
74 bool doFinalization(CallGraph &CG);
77 void getAnalysisUsage(AnalysisUsage &Info)
const override {
79 Info.addRequired<CallGraphWrapperPass>();
80 Info.setPreservesAll();
83 StringRef getPassName()
const override {
return "CallGraph Pass Manager"; }
85 PMDataManager *getAsPMDataManager()
override {
return this; }
86 Pass *getAsPass()
override {
return this; }
89 void dumpPassStructure(
unsigned Offset)
override {
91 for (
unsigned Index = 0;
Index < getNumContainedPasses(); ++
Index) {
92 Pass *
P = getContainedPass(Index);
93 P->dumpPassStructure(
Offset + 1);
98 Pass *getContainedPass(
unsigned N) {
99 assert(
N < PassVector.size() &&
"Pass number out of range!");
100 return static_cast<Pass *
>(PassVector[
N]);
108 bool RunAllPassesOnSCC(CallGraphSCC &CurSCC, CallGraph &CG,
109 bool &DevirtualizedCall);
111 bool RunPassOnSCC(
Pass *
P, CallGraphSCC &CurSCC,
112 CallGraph &CG,
bool &CallGraphUpToDate,
113 bool &DevirtualizedCall);
114 bool RefreshCallGraph(
const CallGraphSCC &CurSCC, CallGraph &CG,
115 bool IsCheckingMode);
120char CGPassManager::ID = 0;
124 bool &DevirtualizedCall) {
126 PMDataManager *PM =
P->getAsPMDataManager();
130 CallGraphSCCPass *CGSP = (CallGraphSCCPass *)
P;
131 if (!CallGraphUpToDate) {
132 DevirtualizedCall |= RefreshCallGraph(CurSCC, CG,
false);
133 CallGraphUpToDate =
true;
138 StringMap<std::pair<unsigned, unsigned>> FunctionToInstrCount;
139 bool EmitICRemark =
M.shouldEmitInstrCountChangedRemark();
142 InstrCount = initSizeRemarkInfo(M, FunctionToInstrCount);
147 SCCCount =
M.getInstructionCount();
152 static_cast<int64_t
>(SCCCount) -
static_cast<int64_t
>(
InstrCount);
153 emitInstrCountChangedRemark(
P, M, Delta,
InstrCount,
154 FunctionToInstrCount);
164 RefreshCallGraph(CurSCC, CG,
true);
171 "Invalid CGPassManager member");
172 FPPassManager *FPP = (FPPassManager*)
P;
175 for (CallGraphNode *CGN : CurSCC) {
176 if (Function *
F = CGN->getFunction()) {
182 F->getContext().yield();
188 if (
Changed && CallGraphUpToDate) {
189 LLVM_DEBUG(
dbgs() <<
"CGSCCPASSMGR: Pass Dirtied SCC: " <<
P->getPassName()
191 CallGraphUpToDate =
false;
205bool CGPassManager::RefreshCallGraph(
const CallGraphSCC &CurSCC, CallGraph &CG,
207 DenseMap<Value *, CallGraphNode *> Calls;
211 for (CallGraphNode *CGN
212 : CurSCC) CGN->dump(););
214 bool MadeChange =
false;
215 bool DevirtualizedCall =
false;
218 unsigned FunctionNo = 0;
220 SCCIdx !=
E; ++SCCIdx, ++FunctionNo) {
221 CallGraphNode *CGN = *SCCIdx;
223 if (!
F ||
F->isDeclaration())
continue;
230 unsigned NumDirectRemoved = 0, NumIndirectRemoved = 0;
237 bool WasLast =
I + 1 == CGNEnd;
259 if (RemoveAndCheckForDone(
I))
273 "CallGraphSCCPass did not update the CallGraph correctly!");
276 if (!
I->second->getFunction())
277 ++NumIndirectRemoved;
281 if (RemoveAndCheckForDone(
I))
286 assert(!Calls.
count(
Call) &&
"Call site occurs in node multiple times");
291 if (!Callee || !(
Callee->isIntrinsic()))
299 unsigned NumDirectAdded = 0, NumIndirectAdded = 0;
301 for (BasicBlock &BB : *
F)
302 for (Instruction &
I : BB) {
307 if (Callee &&
Callee->isIntrinsic())
321 DenseMap<Value *, CallGraphNode *>::iterator ExistingIt =
323 if (ExistingIt != Calls.
end()) {
324 CallGraphNode *ExistingNode = ExistingIt->second;
327 Calls.
erase(ExistingIt);
343 "CallGraphSCCPass did not update the CallGraph correctly!");
347 CallGraphNode *CalleeNode;
353 DevirtualizedCall =
true;
355 <<
Callee->getName() <<
"'\n");
368 "CallGraphSCCPass did not update the CallGraph correctly!");
371 CallGraphNode *CalleeNode;
394 if (NumIndirectRemoved > NumIndirectAdded &&
395 NumDirectRemoved < NumDirectAdded)
396 DevirtualizedCall =
true;
402 assert(Calls.
empty() &&
"Dangling pointers found in call sites map");
406 if ((FunctionNo & 15) == 15)
411 dbgs() <<
"CGSCCPASSMGR: Refreshed SCC is now:\n";
412 for (CallGraphNode *CGN : CurSCC)
414 if (DevirtualizedCall)
415 dbgs() <<
"CGSCCPASSMGR: Refresh devirtualized a call!\n";
417 dbgs() <<
"CGSCCPASSMGR: SCC Refresh didn't change call graph.\n";
421 return DevirtualizedCall;
427bool CGPassManager::RunAllPassesOnSCC(CallGraphSCC &CurSCC, CallGraph &CG,
428 bool &DevirtualizedCall) {
438 bool CallGraphUpToDate =
true;
441 for (
unsigned PassNo = 0, e = getNumContainedPasses();
442 PassNo !=
e; ++PassNo) {
443 Pass *
P = getContainedPass(PassNo);
447 if (isPassDebuggingExecutionsOrMore()) {
448 std::string Functions;
450 raw_string_ostream OS(Functions);
452 for (
const CallGraphNode *CGN : CurSCC) {
461 initializeAnalysisImpl(
P);
463#ifdef EXPENSIVE_CHECKS
464 uint64_t RefHash =
P->structuralHash(CG.
getModule());
469 RunPassOnSCC(
P, CurSCC, CG, CallGraphUpToDate, DevirtualizedCall);
473#ifdef EXPENSIVE_CHECKS
474 if (!LocalChanged && (RefHash !=
P->structuralHash(CG.
getModule()))) {
475 llvm::errs() <<
"Pass modifies its input and doesn't report it: "
476 <<
P->getPassName() <<
"\n";
484 verifyPreservedAnalysis(
P);
486 removeNotPreservedAnalysis(
P);
487 recordAvailableAnalysis(
P);
493 if (!CallGraphUpToDate)
494 DevirtualizedCall |= RefreshCallGraph(CurSCC, CG,
false);
500bool CGPassManager::runOnModule(
Module &M) {
501 CallGraph &CG = getAnalysis<CallGraphWrapperPass>().getCallGraph();
502 bool Changed = doInitialization(CG);
505 scc_iterator<CallGraph*> CGI =
scc_begin(&CG);
507 CallGraphSCC CurSCC(CG, &CGI);
511 const std::vector<CallGraphNode *> &NodeVec = *CGI;
512 CurSCC.initialize(NodeVec);
527 unsigned Iteration = 0;
528 bool DevirtualizedCall =
false;
531 <<
" SCCPASSMGR: Re-visiting SCC, iteration #" << Iteration
533 DevirtualizedCall =
false;
534 Changed |= RunAllPassesOnSCC(CurSCC, CG, DevirtualizedCall);
537 if (DevirtualizedCall)
540 <<
" times, due to -max-devirt-iterations\n");
542 MaxSCCIterations.updateMax(Iteration);
549bool CGPassManager::doInitialization(CallGraph &CG) {
551 for (
unsigned i = 0, e = getNumContainedPasses(); i !=
e; ++i) {
552 if (PMDataManager *PM = getContainedPass(i)->getAsPMDataManager()) {
554 "Invalid CGPassManager member");
557 Changed |= ((CallGraphSCCPass*)getContainedPass(i))->doInitialization(CG);
564bool CGPassManager::doFinalization(CallGraph &CG) {
566 for (
unsigned i = 0, e = getNumContainedPasses(); i !=
e; ++i) {
567 if (PMDataManager *PM = getContainedPass(i)->getAsPMDataManager()) {
569 "Invalid CGPassManager member");
572 Changed |= ((CallGraphSCCPass*)getContainedPass(i))->doFinalization(CG);
585 assert(Old != New &&
"Should not replace node with self");
586 for (
unsigned i = 0; ; ++i) {
587 assert(i != Nodes.size() &&
"Node not in SCC");
588 if (Nodes[i] != Old)
continue;
592 Nodes.erase(Nodes.begin() + i);
614 while (!PMS.
empty() &&
618 assert(!PMS.
empty() &&
"Unable to handle Call Graph Pass");
622 CGP = (CGPassManager*)PMS.
top();
625 assert(!PMS.
empty() &&
"Unable to create Call Graph Pass Manager");
629 CGP =
new CGPassManager();
670 PrintCallGraphPass(
const std::string &
B,
raw_ostream &OS)
677 bool runOnSCC(CallGraphSCC &SCC)
override {
678 bool BannerPrinted =
false;
679 auto PrintBannerOnce = [&]() {
683 BannerPrinted =
true;
690 SCC.getCallGraph().getModule().print(OS,
nullptr);
693 bool FoundFunction =
false;
694 for (CallGraphNode *CGN : SCC) {
697 FoundFunction =
true;
705 OS <<
"\nPrinting <null> Function\n";
708 if (NeedModule && FoundFunction) {
711 SCC.getCallGraph().getModule().print(OS,
nullptr);
716 StringRef getPassName()
const override {
return "Print CallGraph IR"; }
721char PrintCallGraphPass::ID = 0;
724 const std::string &Banner)
const {
725 return new PrintCallGraphPass(Banner, OS);
for(const MachineOperand &MO :llvm::drop_begin(OldMI.operands(), Desc.getNumOperands()))
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
This file provides interfaces used to build and manipulate a call graph, which is a very useful tool ...
static unsigned InstrCount
This file defines the DenseMap class.
Module.h This file contains the declarations for the Module class.
print mir2vec MIR2Vec Vocabulary Printer Pass
Machine Check Debug Module
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
This header defines classes/functions to handle pass execution timing information with interfaces for...
This builds on the llvm/ADT/GraphTraits.h file to find the strongly connected components (SCCs) of a ...
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
Represent the analysis usage information of a pass.
AnalysisUsage & addRequired()
AnalysisUsage & addPreserved()
Add the specified Pass class to the set of analyses preserved by this pass.
void setPreservesAll()
Set by analyses that do not transform their input at all.
Function * getCalledFunction() const
Returns the function called, or null if this is an indirect function invocation or the function signa...
A node in the call graph for a module.
LLVM_ABI void print(raw_ostream &OS) const
void addCalledFunction(CallBase *Call, CallGraphNode *M)
Adds a function to the list of functions called by this one.
LLVM_ABI void replaceCallEdge(CallBase &Call, CallBase &NewCall, CallGraphNode *NewNode)
Replaces the edge in the node for the specified call site with a new one.
LLVM_ABI void dump() const
Print out this call graph node.
Function * getFunction() const
Returns the function that this call graph node represents.
std::vector< CallRecord >::iterator iterator
void removeCallEdge(iterator I)
Pass * createPrinterPass(raw_ostream &OS, const std::string &Banner) const override
createPrinterPass - Get a pass that prints the Module corresponding to a CallGraph.
virtual bool runOnSCC(CallGraphSCC &SCC)=0
runOnSCC - This method should be implemented by the subclass to perform whatever action is necessary ...
void getAnalysisUsage(AnalysisUsage &Info) const override
getAnalysisUsage - For this class, we declare that we require and preserve the call graph.
void assignPassManager(PMStack &PMS, PassManagerType PMT) override
Assign pass manager to manager this pass.
CallGraphSCC - This is a single SCC that a CallGraphSCCPass is run on.
std::vector< CallGraphNode * >::const_iterator iterator
LLVM_ABI void ReplaceNode(CallGraphNode *Old, CallGraphNode *New)
ReplaceNode - This informs the SCC and the pass manager that the specified Old node has been deleted,...
LLVM_ABI void DeleteNode(CallGraphNode *Old)
DeleteNode - This informs the SCC and the pass manager that the specified Old node has been deleted.
The ModulePass which wraps up a CallGraph and the logic to build it.
The basic data container for the call graph of a Module of IR.
LLVM_ABI CallGraphNode * getOrInsertFunction(const Function *F)
Similar to operator[], but this will insert a new CallGraphNode for F if one does not already exist.
CallGraphNode * getCallsExternalNode() const
Module & getModule() const
Returns the module the call graph corresponds to.
iterator find(const_arg_type_t< KeyT > Val)
bool erase(const KeyT &Val)
size_type count(const_arg_type_t< KeyT > Val) const
Return 1 if the specified key is in the map, 0 otherwise.
std::pair< iterator, bool > insert(const std::pair< KeyT, ValueT > &KV)
This pass is required by interprocedural register allocation.
bool runOnFunction(Function &F)
run - Execute all of the passes scheduled for execution.
ModulePass class - This class is used to implement unstructured interprocedural optimizations and ana...
PMDataManager provides the common place to manage the analysis data used by pass managers.
PMTopLevelManager * getTopLevelManager()
virtual PassManagerType getPassManagerType() const
PMStack - This class implements a stack data structure of PMDataManager pointers.
PMDataManager * top() const
LLVM_ABI void push(PMDataManager *PM)
PMTopLevelManager manages LastUser info and collects common APIs used by top level pass managers.
void addIndirectPassManager(PMDataManager *Manager)
void schedulePass(Pass *P)
Schedule pass P for execution.
Pass interface - Implemented by all 'passes'.
Pass(PassKind K, char &pid)
This class implements an extremely fast bulk output stream that can only output to a stream.
raw_ostream & indent(unsigned NumSpaces)
indent - Insert 'NumSpaces' spaces.
Enumerate the SCCs of a directed graph in reverse topological order of the SCC DAG.
bool isAtEnd() const
Direct loop termination test which is more efficient than comparison with end().
void ReplaceNode(NodeRef Old, NodeRef New)
This informs the scc_iterator that the specified Old node has been deleted, and New is to be used in ...
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
initializer< Ty > init(const Ty &Val)
This is an optimization pass for GlobalISel generic memory operations.
bool forcePrintModuleIR()
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
PassManagerType
Different types of internal pass managers.
@ PMT_CallGraphPassManager
CGPassManager.
@ PMT_FunctionPassManager
FPPassManager.
scc_iterator< T > scc_begin(const T &G)
Construct the begin iterator for a deduced graph type T.
auto dyn_cast_or_null(const Y &Val)
LLVM_ABI Timer * getPassTimer(Pass *)
Request the timer for this legacy-pass-manager's pass instance.
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
bool isFunctionInPrintList(StringRef FunctionName)
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
void forEachCallbackFunction(const CallBase &CB, UnaryFunction Func)
Apply function Func to each CB's callback function.
cl::opt< unsigned > MaxDevirtIterations("max-devirt-iterations", cl::ReallyHidden, cl::init(4))