35#define DEBUG_TYPE "unify-loop-exits"
41 cl::desc(
"Set the maximum number of outgoing blocks for using a boolean "
42 "value to record the exiting block in the ControlFlowHub."));
51 void getAnalysisUsage(AnalysisUsage &AU)
const override {
62char UnifyLoopExitsLegacyPass::ID = 0;
65 return new UnifyLoopExitsLegacyPass();
69 "Fixup each natural loop to have a single exit block",
74 "Fixup each natural loop to have a single exit block",
99 for (
auto *BB : L->blocks()) {
100 for (auto &I : *BB) {
101 for (auto &U : I.uses()) {
102 auto UserInst = cast<Instruction>(U.getUser());
103 auto UserBlock = UserInst->getParent();
104 if (UserBlock == LoopExitBlock)
106 if (L->contains(UserBlock))
108 LLVM_DEBUG(dbgs() <<
"added ext use for " << I.getName() <<
"("
109 << BB->getName() <<
")"
110 <<
": " << UserInst->getName() <<
"("
111 << UserBlock->getName() <<
")"
113 ExternalUsers[&I].push_back(UserInst);
118 for (
const auto &
II : ExternalUsers) {
127 Def->getName() +
".moved", LoopExitBlock->begin());
130 if (
Def->getParent() == In || DT.dominates(Def, In)) {
132 NewPhi->addIncoming(Def, In);
140 for (
auto *U :
II.second) {
142 U->replaceUsesOfWith(Def, NewPhi);
155 L->getExitingBlocks(ExitingBlocks);
162 for (
unsigned I = 0;
I < ExitingBlocks.
size(); ++
I) {
166 Succ0 = L->contains(Succ0) ? nullptr : Succ0;
169 Branch->isUnconditional() ? nullptr : Branch->getSuccessor(1);
170 Succ1 = L->contains(Succ1) ? nullptr : Succ1;
178 for (
unsigned J = 0; J < CallBr->getNumSuccessors(); ++J) {
180 if (L->contains(Succ))
182 bool UpdatedLI =
false;
188 CallBrTargetBlocksToFix.
push_back(NewSucc);
194 ExitingBlocks[
I] = NewSucc;
208 std::tie(LoopExitBlock, ChangedCFG) = CHub.
finalize(
213 restoreSSA(DT, L, ExitingBlocks, LoopExitBlock);
215#if defined(EXPENSIVE_CHECKS)
216 assert(DT.
verify(DominatorTree::VerificationLevel::Full));
218 assert(DT.
verify(DominatorTree::VerificationLevel::Fast));
230 if (
auto *ParentLoop = L->getParentLoop()) {
231 for (
auto *
G : GuardBlocks) {
232 ParentLoop->addBasicBlockToLoop(
G, LI);
234 for (
auto *
C : CallBrTargetBlocksToFix) {
235 ParentLoop->addBasicBlockToLoop(
C, LI);
237 ParentLoop->verifyLoop();
240#if defined(EXPENSIVE_CHECKS)
251 for (
auto *L :
Loops) {
258bool UnifyLoopExitsLegacyPass::runOnFunction(
Function &
F) {
259 LLVM_DEBUG(
dbgs() <<
"===== Unifying loop exits in function " <<
F.getName()
261 auto &LI = getAnalysis<LoopInfoWrapperPass>().getLoopInfo();
262 auto &DT = getAnalysis<DominatorTreeWrapperPass>().getDomTree();
271 LLVM_DEBUG(
dbgs() <<
"===== Unifying loop exits in function " <<
F.getName()
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
This file contains the declarations for the subclasses of Constant, which represent the different fla...
static bool runOnFunction(Function &F, bool PostInlining)
static bool runImpl(Function &F, const TargetLowering &TLI, AssumptionCache *AC)
This file implements a map that provides insertion order iteration.
uint64_t IntrinsicInst * II
#define INITIALIZE_PASS_DEPENDENCY(depName)
#define INITIALIZE_PASS_END(passName, arg, name, cfg, analysis)
#define INITIALIZE_PASS_BEGIN(passName, arg, name, cfg, analysis)
static bool unifyLoopExits(DominatorTree &DT, LoopInfo &LI, Loop *L)
unify loop Fixup each natural loop to have a single exit static false void restoreSSA(const DominatorTree &DT, const Loop *L, SmallVectorImpl< BasicBlock * > &Incoming, BasicBlock *LoopExitBlock)
static cl::opt< unsigned > MaxBooleansInControlFlowHub("max-booleans-in-control-flow-hub", cl::init(32), cl::Hidden, cl::desc("Set the maximum number of outgoing blocks for using a boolean " "value to record the exiting block in the ControlFlowHub."))
static bool runImpl(LoopInfo &LI, DominatorTree &DT)
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
AnalysisUsage & addRequired()
AnalysisUsage & addPreserved()
Add the specified Pass class to the set of analyses preserved by this pass.
LLVM Basic Block Representation.
const Instruction * getTerminator() const LLVM_READONLY
Returns the terminator instruction if the block is well formed or null if the block is not well forme...
Conditional or Unconditional Branch instruction.
CallBr instruction, tracking function calls that may not return control but instead transfer it to a ...
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.
Analysis pass that exposes the LoopInfo for a function.
void verify(const DominatorTreeBase< BlockT, false > &DomTree) const
SmallVector< LoopT *, 4 > getLoopsInPreorder() const
Return all of the loops in the function in preorder across the loop nests, with siblings in forward p...
The legacy pass manager's analysis pass to compute loop information.
Represents a single loop in the control flow graph.
This class implements a map that also provides access to all stored values in a deterministic order.
static PHINode * Create(Type *Ty, unsigned NumReservedValues, const Twine &NameStr="", InsertPosition InsertBefore=nullptr)
Constructors - NumReservedValues is a hint for the number of incoming edges that this phi node will h...
static LLVM_ABI PassRegistry * getPassRegistry()
getPassRegistry - Access the global registry object, which is automatically initialized at applicatio...
static LLVM_ABI PoisonValue * get(Type *T)
Static factory methods - Return an 'poison' object of the specified type.
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.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
PreservedAnalyses run(Function &F, FunctionAnalysisManager &AM)
#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.
@ C
The default llvm calling convention, compatible with C.
initializer< Ty > init(const Ty &Val)
NodeAddr< DefNode * > Def
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 void initializeUnifyLoopExitsLegacyPassPass(PassRegistry &)
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 * createUnifyLoopExitsPass()
LLVM_ABI Printable printBasicBlock(const BasicBlock *BB)
Print BasicBlock BB as an operand or print "<nullptr>" if BB is a nullptr.
AnalysisManager< Function > FunctionAnalysisManager
Convenience typedef for the Function analysis manager.
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)
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.
Incoming for lane maks phi as machine instruction, incoming register Reg and incoming block Block are...