142#define DEBUG_TYPE "fix-irreducible"
165char FixIrreducible::ID = 0;
170 "Convert irreducible control-flow into natural loops",
175 "Convert irreducible control-flow into natural loops",
182 auto &CandidateLoops = ParentLoop ? ParentLoop->getSubLoopsVector()
183 : LI.getTopLevelLoopsVector();
187 return NewLoop == L || !NewLoop->contains(L->getHeader());
190 CandidateLoops.
erase(FirstChild, CandidateLoops.end());
192 for (
Loop *Child : ChildLoops) {
193 LLVM_DEBUG(
dbgs() <<
"child loop: " << Child->getHeader()->getName()
197 if (Child->getHeader() == OldHeader) {
198 for (
auto *BB : Child->blocks()) {
199 if (LI.getLoopFor(BB) != Child)
201 LI.changeLoopFor(BB, NewLoop);
205 std::vector<Loop *> GrandChildLoops;
206 std::swap(GrandChildLoops, Child->getSubLoopsVector());
207 for (
auto *GrandChildLoop : GrandChildLoops) {
208 GrandChildLoop->setParentLoop(
nullptr);
209 NewLoop->addChildLoop(GrandChildLoop);
216 Child->setParentLoop(
nullptr);
217 NewLoop->addChildLoop(Child);
229 if (ParentLoop && ParentLoop->
getHeader() == CycleHeader)
245 for (
auto *
G : GuardBlocks) {
246 LLVM_DEBUG(
dbgs() <<
"added guard block to loop: " <<
G->getName() <<
"\n");
247 NewLoop->addBasicBlockToLoop(
G, LI);
250 for (
auto *BB :
C.blocks()) {
251 NewLoop->addBlockEntry(BB);
257 LLVM_DEBUG(
dbgs() <<
"added block from child: " << BB->getName() <<
"\n");
261 << NewLoop->getHeader()->getName() <<
"\n");
266 NewLoop->verifyLoop();
295 assert(
P->getTerminator()->getSuccessor(0) == Header);
301 BasicBlock *Succ0 = Branch->getSuccessor(0) == Header ? Header :
nullptr;
302 BasicBlock *Succ1 = Branch->getSuccessor(1) == Header ? Header :
nullptr;
311 for (
unsigned I = 0;
I < CallBr->getNumSuccessors(); ++
I) {
323 "only supports br and callbr instructions");
328 Predecessors.
clear();
339 Succ0 =
C.contains(Succ0) ? Succ0 :
nullptr;
346 Succ0 =
C.contains(Succ0) ? Succ0 :
nullptr;
348 Succ1 =
C.contains(Succ1) ? Succ1 :
nullptr;
356 for (
unsigned I = 0;
I < CallBr->getNumSuccessors(); ++
I) {
358 if (!
C.contains(Succ))
368 "only supports br and callbr instructions");
383 Entries.
insert(
C.entry_rbegin(),
C.entry_rend());
385 CHub.
finalize(&DTU, GuardBlocks,
"irr");
386#if defined(EXPENSIVE_CHECKS)
387 assert(DT.
verify(DominatorTree::VerificationLevel::Full));
389 assert(DT.
verify(DominatorTree::VerificationLevel::Fast));
397 for (
auto *
G : GuardBlocks) {
402 C.setSingleEntry(GuardBlocks[0]);
405 if (
Cycle *Parent =
C.getParentCycle())
406 Parent->verifyCycle();
414 LLVM_DEBUG(
dbgs() <<
"===== Fix irreducible control-flow in function: "
415 <<
F.getName() <<
"\n");
427#if defined(EXPENSIVE_CHECKS)
437bool FixIrreducible::runOnFunction(
Function &
F) {
438 auto *LIWP = getAnalysisIfAvailable<LoopInfoWrapperPass>();
439 LoopInfo *LI = LIWP ? &LIWP->getLoopInfo() :
nullptr;
440 auto &CI = getAnalysis<CycleInfoWrapperPass>().getResult();
441 auto &DT = getAnalysis<DominatorTreeWrapperPass>().getDomTree();
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")
This file declares an analysis pass that computes CycleInfo for LLVM IR, specialized from GenericCycl...
static bool runOnFunction(Function &F, bool PostInlining)
fix Convert irreducible control flow into natural static false void reconnectChildLoops(LoopInfo &LI, Loop *ParentLoop, Loop *NewLoop, BasicBlock *OldHeader)
static void updateLoopInfo(LoopInfo &LI, Cycle &C, ArrayRef< BasicBlock * > GuardBlocks)
static bool FixIrreducibleImpl(Function &F, CycleInfo &CI, DominatorTree &DT, LoopInfo *LI)
static bool fixIrreducible(Cycle &C, CycleInfo &CI, DominatorTree &DT, LoopInfo *LI)
#define INITIALIZE_PASS_DEPENDENCY(depName)
#define INITIALIZE_PASS_END(passName, arg, name, cfg, analysis)
#define INITIALIZE_PASS_BEGIN(passName, arg, name, cfg, analysis)
PassT::Result * getCachedResult(IRUnitT &IR) const
Get the cached result of an analysis pass for a given IR unit.
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
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.
Represent a constant reference to an array (0 or more elements consecutively in memory),...
LLVM Basic Block Representation.
CallBr instruction, tracking function calls that may not return control but instead transfer it to a ...
Conditional Branch instruction.
Analysis pass which computes a CycleInfo.
Legacy analysis pass which computes a CycleInfo.
Analysis pass which computes a DominatorTree.
bool verify(VerificationLevel VL=VerificationLevel::Full) const
verify - checks if the tree is correct.
Legacy analysis pass which computes a DominatorTree.
Concrete subclass of DominatorTreeBase that is used to compute a normal dominator tree.
FunctionPass class - This class is used to implement most global optimizations.
void verify() const
Verify that the entire cycle tree well-formed.
void addBlockToCycle(BlockT *Block, CycleT *Cycle)
Assumes that Cycle is the innermost cycle containing Block.
iterator_range< const_toplevel_iterator > toplevel_cycles() const
void print(raw_ostream &Out) const
Print the cycle info.
Analysis pass that exposes the LoopInfo for a function.
void verifyLoop() const
Verify loop structure.
BlockT * getHeader() const
void print(raw_ostream &OS, bool Verbose=false, bool PrintNested=true, unsigned Depth=0) const
Print loop with all the BBs inside it.
void addChildLoop(LoopT *NewChild)
Add the specified loop to be a child of this loop.
LoopT * getParentLoop() const
Return the parent loop if it exists or nullptr for top level loops.
void verify(const DominatorTreeBase< BlockT, false > &DomTree) const
void addTopLevelLoop(LoopT *New)
This adds the specified loop to the collection of top-level loops.
LoopT * AllocateLoop(ArgsTy &&...Args)
LoopT * getLoopFor(const BlockT *BB) const
Return the inner most loop that BB lives in.
void changeLoopFor(const BlockT *BB, LoopT *L)
Change the top-level loop that contains BB to the specified loop.
The legacy pass manager's analysis pass to compute loop information.
Represents a single loop in the control flow graph.
static LLVM_ABI PassRegistry * getPassRegistry()
getPassRegistry - Access the global registry object, which is automatically initialized at applicatio...
A set of analyses that are preserved following a run of a transformation pass.
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
PreservedAnalyses & preserve()
Mark an analysis as preserved.
A vector that has set insertion semantics.
void clear()
Completely clear the SetVector.
bool insert(const value_type &X)
Insert a new element into the SetVector.
iterator erase(const_iterator CI)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Unconditional Branch instruction.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
@ C
The default llvm calling convention, compatible with C.
This is an optimization pass for GlobalISel generic memory operations.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
LLVM_ABI BasicBlock * SplitCallBrEdge(BasicBlock *CallBrBlock, BasicBlock *Succ, unsigned SuccIdx, DomTreeUpdater *DTU=nullptr, CycleInfo *CI=nullptr, LoopInfo *LI=nullptr, bool *UpdatedLI=nullptr)
Create a new intermediate target block for a callbr edge.
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
LLVM_ABI FunctionPass * createFixIrreduciblePass()
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
auto partition(R &&Range, UnaryPredicate P)
Provide wrappers to std::partition which take ranges instead of having to pass begin/end explicitly.
LLVM_ABI Printable printBasicBlock(const BasicBlock *BB)
Print BasicBlock BB as an operand or print "<nullptr>" if BB is a nullptr.
auto predecessors(const MachineBasicBlock *BB)
iterator_range< df_iterator< T > > depth_first(const T &G)
AnalysisManager< Function > FunctionAnalysisManager
Convenience typedef for the Function analysis manager.
LLVM_ABI void initializeFixIrreduciblePass(PassRegistry &)
LLVM_ABI void reportFatalUsageError(Error Err)
Report a fatal error that does not indicate a bug in LLVM.
void swap(llvm::BitVector &LHS, llvm::BitVector &RHS)
Implement std::swap in terms of BitVector swap.
Given a set of branch descriptors [BB, Succ0, Succ1], create a "hub" such that the control flow from ...
void addBranch(BasicBlock *BB, BasicBlock *Succ0, BasicBlock *Succ1=nullptr)
LLVM_ABI std::pair< BasicBlock *, bool > finalize(DomTreeUpdater *DTU, SmallVectorImpl< BasicBlock * > &GuardBlocks, const StringRef Prefix, std::optional< unsigned > MaxControlFlowBooleans=std::nullopt)
Return the unified loop exit block and a flag indicating if the CFG was changed at all.
LLVM_ABI PreservedAnalyses run(Function &F, FunctionAnalysisManager &AM)