14#ifndef LLVM_SUPPORT_GENERICLOOPINFOIMPL_H
15#define LLVM_SUPPORT_GENERICLOOPINFOIMPL_H
32template <
class BlockT,
class LoopT>
36 for (
const auto BB :
blocks())
47template <
class BlockT,
class LoopT>
50 auto notInLoop = [&](BlockT *BB) {
return !
contains(BB); };
51 auto isExitBlock = [&](BlockT *BB,
bool AllowRepeats) -> BlockT * {
52 assert(!AllowRepeats &&
"Unexpected parameter value.");
63template <
class BlockT,
class LoopT>
67 for (
const auto BB :
blocks())
76template <
class BlockT,
class LoopT>
79 assert(!L->isInvalid() &&
"Loop not in a valid state!");
80 auto notInLoop = [&](BlockT *BB,
81 bool AllowRepeats) -> std::pair<BlockT *, bool> {
82 assert(AllowRepeats == Unique &&
"Unexpected parameter value.");
83 return {!L->contains(BB) ? BB :
nullptr,
false};
85 auto singleExitBlock = [&](BlockT *BB,
86 bool AllowRepeats) -> std::pair<BlockT *, bool> {
87 assert(AllowRepeats == Unique &&
"Unexpected parameter value.");
94template <
class BlockT,
class LoopT>
106template <
class BlockT,
class LoopT>
111template <
class BlockT,
class LoopT>
117 for (BlockT *EB : UniqueExitBlocks)
127template <
class BlockT,
class LoopT,
typename PredicateT>
131 assert(!L->isInvalid() &&
"Loop not in a valid state!");
134 for (BlockT *BB : Filtered)
141template <
class BlockT,
class LoopT>
145 [](
const BlockT *BB) {
return true; });
148template <
class BlockT,
class LoopT>
152 assert(Latch &&
"Latch block must exists");
154 [Latch](
const BlockT *BB) {
return BB != Latch; });
157template <
class BlockT,
class LoopT>
162template <
class BlockT,
class LoopT>
165 assert(Latch &&
"Latch block must exists");
166 auto IsExitBlock = [
this](BlockT *BB,
bool AllowRepeats) -> BlockT * {
167 assert(!AllowRepeats &&
"Unexpected parameter value.");
168 return !
contains(BB) ? BB :
nullptr;
174template <
class BlockT,
class LoopT>
178 for (
const auto BB :
blocks())
186template <
class BlockT>
189template <
class BlockT>
196 return Block->isLegalToHoistInto();
209template <
class BlockT,
class LoopT>
234template <
class BlockT,
class LoopT>
238 BlockT *Out =
nullptr;
244 if (Out && Out != Pred)
255template <
class BlockT,
class LoopT>
259 BlockT *Latch =
nullptr;
281template <
class BlockT,
class LoopT>
286 if (!Blocks.empty()) {
289 "Incorrect LI specified for this loop!");
292 assert(NewBB &&
"Cannot add a null basic block to the loop!");
293 assert(!LIB[NewBB] &&
"BasicBlock already in the loop!");
295 LoopT *L =
static_cast<LoopT *
>(
this);
298 LIB.BBMap[NewBB] = L;
302 L->addBlockEntry(NewBB);
303 L = L->getParentLoop();
311template <
class BlockT,
class LoopT>
315 assert(OldChild->ParentLoop ==
this &&
"This loop is already broken!");
316 assert(!NewChild->ParentLoop &&
"NewChild already has a parent!");
317 typename std::vector<LoopT *>::iterator
I =
find(SubLoops, OldChild);
318 assert(
I != SubLoops.end() &&
"OldChild not in loop!");
320 OldChild->ParentLoop =
nullptr;
321 NewChild->ParentLoop =
static_cast<LoopT *
>(
this);
325template <
class BlockT,
class LoopT>
329 assert(!Blocks.empty() &&
"Loop header is missing");
344 "Loop block has no in-loop successors!");
348 "Loop block has no in-loop predecessors!");
356 assert(!OutsideLoopPreds.empty() &&
"Loop is unreachable!");
357 }
else if (!OutsideLoopPreds.empty()) {
361 BlockT *EntryBB = &BB->getParent()->front();
363 for (
unsigned i = 0, e = OutsideLoopPreds.size(); i != e; ++i)
364 assert(CB != OutsideLoopPreds[i] &&
365 "Loop has multiple entry points!");
368 "Loop contains function entry block!");
374 dbgs() <<
"The following blocks are unreachable in the loop: ";
375 for (
auto *BB : Blocks) {
376 if (!VisitedBBs.
count(BB)) {
377 dbgs() << *BB <<
"\n";
380 assert(
false &&
"Unreachable block in loop");
386 for (block_iterator BI = (*I)->block_begin(), BE = (*I)->block_end();
389 "Loop does not contain all the blocks of a subloop!");
395 "Loop is not a subloop of its parent!");
401template <
class BlockT,
class LoopT>
405 Loops->insert(
static_cast<const LoopT *
>(
this));
410 (*I)->verifyLoopNest(
Loops);
413template <
class BlockT,
class LoopT>
415 bool PrintNested,
unsigned Depth)
const {
419 OS <<
"Loop at depth " <<
getLoopDepth() <<
" containing: ";
422 for (
unsigned i = 0; i <
getBlocks().size(); ++i) {
427 BB->printAsOperand(OS,
false);
446 (*I)->print(OS,
false, PrintNested,
Depth + 2);
458template <
class BlockT,
class LoopT>
464 unsigned NumBlocks = 0;
465 unsigned NumSubloops = 0;
468 std::vector<BlockT *> ReverseCFGWorklist(Backedges.
begin(), Backedges.
end());
469 while (!ReverseCFGWorklist.empty()) {
470 BlockT *PredBB = ReverseCFGWorklist.back();
471 ReverseCFGWorklist.pop_back();
481 if (PredBB == L->getHeader())
484 ReverseCFGWorklist.insert(ReverseCFGWorklist.end(),
485 InvBlockTraits::child_begin(PredBB),
486 InvBlockTraits::child_end(PredBB));
489 Subloop = Subloop->getOutermostLoop();
496 Subloop->setParentLoop(L);
498 NumBlocks += Subloop->getBlocksVector().capacity();
499 PredBB = Subloop->getHeader();
506 ReverseCFGWorklist.push_back(Pred);
510 L->getSubLoopsVector().reserve(NumSubloops);
511 L->reserveBlocks(NumBlocks);
517 typedef typename BlockTraits::ChildIteratorType SuccIterTy;
531template <
class BlockT,
class LoopT>
540template <
class BlockT,
class LoopT>
542 LoopT *Subloop = LI->getLoopFor(
Block);
543 if (Subloop &&
Block == Subloop->getHeader()) {
546 if (!Subloop->isOutermost())
547 Subloop->getParentLoop()->getSubLoopsVector().push_back(Subloop);
549 LI->addTopLevelLoop(Subloop);
553 Subloop->reverseBlock(1);
554 std::reverse(Subloop->getSubLoopsVector().begin(),
555 Subloop->getSubLoopsVector().end());
557 Subloop = Subloop->getParentLoop();
559 for (; Subloop; Subloop = Subloop->getParentLoop())
560 Subloop->addBlockEntry(
Block);
577template <
class BlockT,
class LoopT>
583 BlockT *Header = DomNode->getBlock();
590 if (BackedgeNode && DomTree.
dominates(DomNode, BackedgeNode))
594 if (!Backedges.
empty()) {
595 LoopT *L = AllocateLoop(Header);
605template <
class BlockT,
class LoopT>
614 for (LoopT *RootL :
reverse(*
this)) {
615 auto PreOrderLoopsInRootL = RootL->getLoopsInPreorder();
616 PreOrderLoops.
append(PreOrderLoopsInRootL.begin(),
617 PreOrderLoopsInRootL.end());
620 return PreOrderLoops;
623template <
class BlockT,
class LoopT>
632 for (LoopT *RootL : *
this) {
634 "Must start with an empty preorder walk worklist.");
640 PreOrderWorklist.
append(L->begin(), L->end());
642 }
while (!PreOrderWorklist.
empty());
645 return PreOrderLoops;
649template <
class BlockT,
class LoopT>
651 for (
unsigned i = 0; i < TopLevelLoops.size(); ++i)
652 TopLevelLoops[i]->
print(OS);
655 E = BBMap.end();
I !=
E; ++
I)
656 OS <<
"BB '" <<
I->first->getName() <<
"' level = "
657 <<
I->second->getLoopDepth() <<
"\n";
668template <
class BlockT,
class LoopT>
672 LoopHeaders[L.getHeader()] = &L;
678template <
class BlockT,
class LoopT>
681 BlockT *
H = L->getHeader();
682 BlockT *OtherH = OtherL->getHeader();
684 "Mismatched headers even though found in the same map entry!");
686 assert(L->getLoopDepth() == OtherL->getLoopDepth() &&
687 "Mismatched loop depth!");
688 const LoopT *ParentL = L, *OtherParentL = OtherL;
690 assert(ParentL->getHeader() == OtherParentL->getHeader() &&
691 "Mismatched parent loop headers!");
692 ParentL = ParentL->getParentLoop();
693 OtherParentL = OtherParentL->getParentLoop();
696 for (
const LoopT *SubL : *L) {
697 BlockT *SubH = SubL->getHeader();
698 const LoopT *OtherSubL = OtherLoopHeaders.
lookup(SubH);
699 assert(OtherSubL &&
"Inner loop is missing in computed loop info!");
700 OtherLoopHeaders.
erase(SubH);
704 std::vector<BlockT *> BBs = L->getBlocks();
705 std::vector<BlockT *> OtherBBs = OtherL->getBlocks();
707 "Mismatched basic blocks in the loops!");
711 OtherL->getBlocksSet();
714 "Mismatched basic blocks in BlocksSets!");
718template <
class BlockT,
class LoopT>
723 assert((*I)->isOutermost() &&
"Top-level loop has a parent!");
724 (*I)->verifyLoopNest(&
Loops);
729 for (
auto &Entry : BBMap) {
730 const BlockT *BB = Entry.first;
731 LoopT *L = Entry.second;
733 assert(L->contains(BB) &&
"orphaned block");
734 for (LoopT *ChildLoop : *L)
735 assert(!ChildLoop->contains(BB) &&
736 "BBMap should point to the innermost loop containing BB");
740 LoopInfoBase<BlockT, LoopT> OtherLI;
747 for (LoopT *L : OtherLI)
753 for (LoopT *L : *
this) {
754 BlockT *Header = L->getHeader();
755 const LoopT *OtherL = OtherLoopHeaders.
lookup(Header);
756 assert(OtherL &&
"Top level loop is missing in computed loop info!");
758 OtherLoopHeaders.
erase(Header);
765 if (!OtherLoopHeaders.
empty()) {
766 for (
const auto &HeaderAndLoop : OtherLoopHeaders)
767 dbgs() <<
"Found new loop: " << *HeaderAndLoop.second <<
"\n";
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static const Function * getParent(const Value *V)
bbsections Prepares for basic block by splitting functions into clusters of basic blocks
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
This file builds on the ADT/GraphTraits.h file to build generic depth first graph iterator.
static bool isExitBlock(BasicBlock *BB, const SmallVectorImpl< BasicBlock * > &ExitBlocks)
Return true if the specified block is in the list.
This file builds on the ADT/GraphTraits.h file to build a generic graph post order iterator.
static bool contains(SmallPtrSetImpl< ConstantExpr * > &Cache, ConstantExpr *Expr, Constant *C)
This file defines generic set operations that may be used on set's of different types,...
SmallPtrSet< BasicBlock *, 8 > BlocksSet
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
ValueT lookup(const_arg_type_t< KeyT > Val) const
lookup - Return the entry for the specified key, or a default constructed value if no such entry exis...
bool erase(const KeyT &Val)
DenseMapIterator< KeyT, ValueT, KeyInfoT, BucketT, true > const_iterator
Implements a dense probed hash-table based set.
Base class for the actual dominator tree node.
DomTreeNodeBase< NodeT > * getRootNode()
getRootNode - This returns the entry node for the CFG of the function.
bool dominates(const DomTreeNodeBase< NodeT > *A, const DomTreeNodeBase< NodeT > *B) const
dominates - Returns true iff A dominates B.
bool isReachableFromEntry(const NodeT *A) const
isReachableFromEntry - Return true if A is dominated by the entry block of the function containing it...
DomTreeNodeBase< NodeT > * getNode(const NodeT *BB) const
getNode - return the (Post)DominatorTree node for the specified basic block.
Instances of this class are used to represent loops that are detected in the flow graph.
bool isAnnotatedParallel() const
Returns true if the loop is annotated parallel.
BlockT * getLoopLatch() const
If there is a single latch block for this loop, return it.
void getExitBlocks(SmallVectorImpl< BlockT * > &ExitBlocks) const
Return all of the successor blocks of this loop.
unsigned getNumBlocks() const
void verifyLoop() const
Verify loop structure.
void verifyLoopNest(DenseSet< const LoopT * > *Loops) const
Verify loop structure of this loop and all nested loops.
BlockT * getUniqueLatchExitBlock() const
Return the unique exit block for the latch, or null if there are multiple different exit blocks or th...
void getExitingBlocks(SmallVectorImpl< BlockT * > &ExitingBlocks) const
Return all blocks inside the loop that have successors outside of the loop.
BlockT * getHeader() const
unsigned getLoopDepth() const
Return the nesting level of this loop.
void print(raw_ostream &OS, bool Verbose=false, bool PrintNested=true, unsigned Depth=0) const
Print loop with all the BBs inside it.
void addBasicBlockToLoop(BlockT *NewBB, LoopInfoBase< BlockT, LoopT > &LI)
This method is used by other analyses to update loop information.
std::vector< LoopT * >::const_iterator iterator
bool isInvalid() const
Return true if this loop is no longer valid.
BlockT * getLoopPredecessor() const
If the given loop's header has exactly one unique predecessor outside the loop, return it.
bool isLoopLatch(const BlockT *BB) const
void getExitEdges(SmallVectorImpl< Edge > &ExitEdges) const
Return all pairs of (inside_block,outside_block).
BlockT * getExitBlock() const
If getExitBlocks would return exactly one block, return that block.
bool hasNoExitBlocks() const
Return true if this loop does not have any exit blocks.
void replaceChildLoopWith(LoopT *OldChild, LoopT *NewChild)
This is used when splitting loops up.
BlockT * getLoopPreheader() const
If there is a preheader for this loop, return it.
ArrayRef< BlockT * > getBlocks() const
Get a list of the basic blocks which make up this loop.
BlockT * getExitingBlock() const
If getExitingBlocks would return exactly one block, return that block.
void getUniqueExitBlocks(SmallVectorImpl< BlockT * > &ExitBlocks) const
Return all unique successor blocks of this loop.
bool hasDedicatedExits() const
Return true if no exit block for the loop has a predecessor that is outside the loop.
void getUniqueNonLatchExitBlocks(SmallVectorImpl< BlockT * > &ExitBlocks) const
Return all unique successor blocks of this loop except successors from Latch block are not considered...
bool isLoopExiting(const BlockT *BB) const
True if terminator in the block can branch to another block that is outside of the current loop.
BlockT * getUniqueExitBlock() const
If getUniqueExitBlocks would return exactly one block, return that block.
This class builds and contains all of the top-level loop structures in the specified function.
void verify(const DominatorTreeBase< BlockT, false > &DomTree) const
void analyze(const DominatorTreeBase< BlockT, false > &DomTree)
Create the loop forest using a stable algorithm.
SmallVector< LoopT *, 4 > getLoopsInReverseSiblingPreorder() const
Return all of the loops in the function in preorder across the loop nests, with siblings in reverse p...
void print(raw_ostream &OS) 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...
void changeLoopFor(BlockT *BB, LoopT *L)
Change the top-level loop that contains BB to the specified loop.
LoopT * getLoopFor(const BlockT *BB) const
Return the inner most loop that BB lives in.
std::vector< LoopT * >::const_iterator iterator
iterator/begin/end - The interface to the top-level loops in the current function.
Populate all loop data in a stable order during a single forward DFS.
void traverse(BlockT *EntryBlock)
Top-level driver for the forward DFS within the loop.
PopulateLoopsDFS(LoopInfoBase< BlockT, LoopT > *li)
void insertIntoLoop(BlockT *Block)
Add a single Block to its ancestor loops in PostOrder.
A templated base class for SmallPtrSet which provides the typesafe interface that is common across al...
size_type count(ConstPtrType Ptr) const
count - Return 1 if the specified pointer is in the set, 0 otherwise.
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
reference emplace_back(ArgTypes &&... Args)
void append(ItTy in_start, ItTy in_end)
Add the specified range to the end of the SmallVector.
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
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.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
decltype(&BlockT::isLegalToHoistInto) has_hoist_check
llvm::is_detected< has_hoist_check, BlockT > detect_has_hoist_check
bool isLegalToHoistInto(BlockT *Block)
SFINAE functions that dispatch to the isLegalToHoistInto member function or return false,...
This is an optimization pass for GlobalISel generic memory operations.
iterator_range< df_ext_iterator< T, SetTy > > depth_first_ext(const T &G, SetTy &S)
auto find(R &&Range, const T &Val)
Provide wrappers to std::find which take ranges instead of having to pass begin/end explicitly.
Printable print(const GCNRegPressure &RP, const GCNSubtarget *ST=nullptr, unsigned DynamicVGPRBlockSize=0)
static void compareLoops(const LoopT *L, const LoopT *OtherL, DenseMap< BlockT *, const LoopT * > &OtherLoopHeaders)
bool set_is_subset(const S1Ty &S1, const S2Ty &S2)
set_is_subset(A, B) - Return true iff A in B
iterator_range< po_iterator< T > > post_order(const T &G)
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
auto reverse(ContainerTy &&C)
void sort(IteratorTy Start, IteratorTy End)
DominatorTreeBase< T, false > DomTreeBase
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
bool hasSingleElement(ContainerTy &&C)
Returns true if the given container only contains a single element.
iterator_range< filter_iterator< detail::IterOfRange< RangeT >, PredicateT > > make_filter_range(RangeT &&Range, PredicateT Pred)
Convenience function that takes a range of elements and a predicate, and return a new filter_iterator...
std::pair< BlockT *, bool > getExitBlockHelper(const LoopBase< BlockT, LoopT > *L, bool Unique)
getExitBlock - If getExitBlocks would return exactly one block, return that block.
std::pair< T *, bool > find_singleton_nested(R &&Range, Predicate P, bool AllowRepeats=false)
Return a pair consisting of the single value in Range that satisfies P(<member of Range> ,...
T * find_singleton(R &&Range, Predicate P, bool AllowRepeats=false)
Return the single value in Range that satisfies P(<member of Range> *, AllowRepeats)->T * returning n...
iterator_range< typename GraphTraits< Inverse< GraphType > >::ChildIteratorType > inverse_children(const typename GraphTraits< GraphType >::NodeRef &G)
void addInnerLoopsToHeadersMap(DenseMap< BlockT *, const LoopT * > &LoopHeaders, const LoopInfoBase< BlockT, LoopT > &LI, const LoopT &L)
void getUniqueExitBlocksHelper(const LoopT *L, SmallVectorImpl< BlockT * > &ExitBlocks, PredicateT Pred)
typename detail::detector< void, Op, Args... >::value_t is_detected
Detects if a given trait holds for some set of arguments 'Args'.
iterator_range< typename GraphTraits< GraphType >::ChildIteratorType > children(const typename GraphTraits< GraphType >::NodeRef &G)
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
bool compareVectors(std::vector< T > &BB1, std::vector< T > &BB2)
iterator_range< df_iterator< T > > depth_first(const T &G)
static void discoverAndMapSubloop(LoopT *L, ArrayRef< BlockT * > Backedges, LoopInfoBase< BlockT, LoopT > *LI, const DomTreeBase< BlockT > &DomTree)
Stable LoopInfo Analysis - Build a loop tree using stable iterators so the result does / not depend o...
std::pair< iterator, bool > insert(NodeRef N)