mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-14 11:32:34 +00:00
Make Loop and MachineLoop be subclasses of LoopBase, rather than typedefs,
using the Curiously Recurring Template Pattern with LoopBase. This will help further refactoring, and future functionality for Loop. Also, Headers can now foward-declare Loop, instead of pulling in LoopInfo.h or doing tricks. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@75519 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
92aa0bb4ca
commit
c8d76d5afb
@ -54,26 +54,27 @@ static void RemoveFromVector(std::vector<T*> &V, T *N) {
|
|||||||
|
|
||||||
class DominatorTree;
|
class DominatorTree;
|
||||||
class LoopInfo;
|
class LoopInfo;
|
||||||
template<class N> class LoopInfoBase;
|
class Loop;
|
||||||
template<class N> class LoopBase;
|
template<class N, class M> class LoopInfoBase;
|
||||||
|
template<class N, class M> class LoopBase;
|
||||||
typedef LoopBase<BasicBlock> Loop;
|
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
/// LoopBase class - Instances of this class are used to represent loops that
|
/// LoopBase class - Instances of this class are used to represent loops that
|
||||||
/// are detected in the flow graph
|
/// are detected in the flow graph
|
||||||
///
|
///
|
||||||
template<class BlockT>
|
template<class BlockT, class LoopT>
|
||||||
class LoopBase {
|
class LoopBase {
|
||||||
LoopBase<BlockT> *ParentLoop;
|
LoopT *ParentLoop;
|
||||||
// SubLoops - Loops contained entirely within this one.
|
// SubLoops - Loops contained entirely within this one.
|
||||||
std::vector<LoopBase<BlockT>*> SubLoops;
|
std::vector<LoopT *> SubLoops;
|
||||||
|
|
||||||
// Blocks - The list of blocks in this loop. First entry is the header node.
|
// Blocks - The list of blocks in this loop. First entry is the header node.
|
||||||
std::vector<BlockT*> Blocks;
|
std::vector<BlockT*> Blocks;
|
||||||
|
|
||||||
LoopBase(const LoopBase<BlockT> &); // DO NOT IMPLEMENT
|
// DO NOT IMPLEMENT
|
||||||
const LoopBase<BlockT>&operator=(const LoopBase<BlockT> &);// DO NOT IMPLEMENT
|
LoopBase(const LoopBase<BlockT, LoopT> &);
|
||||||
|
// DO NOT IMPLEMENT
|
||||||
|
const LoopBase<BlockT, LoopT>&operator=(const LoopBase<BlockT, LoopT> &);
|
||||||
public:
|
public:
|
||||||
/// Loop ctor - This creates an empty loop.
|
/// Loop ctor - This creates an empty loop.
|
||||||
LoopBase() : ParentLoop(0) {}
|
LoopBase() : ParentLoop(0) {}
|
||||||
@ -87,13 +88,13 @@ public:
|
|||||||
/// blocks, where depth 0 is used for blocks not inside any loops.
|
/// blocks, where depth 0 is used for blocks not inside any loops.
|
||||||
unsigned getLoopDepth() const {
|
unsigned getLoopDepth() const {
|
||||||
unsigned D = 1;
|
unsigned D = 1;
|
||||||
for (const LoopBase<BlockT> *CurLoop = ParentLoop; CurLoop;
|
for (const LoopT *CurLoop = ParentLoop; CurLoop;
|
||||||
CurLoop = CurLoop->ParentLoop)
|
CurLoop = CurLoop->ParentLoop)
|
||||||
++D;
|
++D;
|
||||||
return D;
|
return D;
|
||||||
}
|
}
|
||||||
BlockT *getHeader() const { return Blocks.front(); }
|
BlockT *getHeader() const { return Blocks.front(); }
|
||||||
LoopBase<BlockT> *getParentLoop() const { return ParentLoop; }
|
LoopT *getParentLoop() const { return ParentLoop; }
|
||||||
|
|
||||||
/// contains - Return true if the specified basic block is in this loop
|
/// contains - Return true if the specified basic block is in this loop
|
||||||
///
|
///
|
||||||
@ -103,8 +104,8 @@ public:
|
|||||||
|
|
||||||
/// iterator/begin/end - Return the loops contained entirely within this loop.
|
/// iterator/begin/end - Return the loops contained entirely within this loop.
|
||||||
///
|
///
|
||||||
const std::vector<LoopBase<BlockT>*> &getSubLoops() const { return SubLoops; }
|
const std::vector<LoopT *> &getSubLoops() const { return SubLoops; }
|
||||||
typedef typename std::vector<LoopBase<BlockT>*>::const_iterator iterator;
|
typedef typename std::vector<LoopT *>::const_iterator iterator;
|
||||||
iterator begin() const { return SubLoops.begin(); }
|
iterator begin() const { return SubLoops.begin(); }
|
||||||
iterator end() const { return SubLoops.end(); }
|
iterator end() const { return SubLoops.end(); }
|
||||||
bool empty() const { return SubLoops.empty(); }
|
bool empty() const { return SubLoops.empty(); }
|
||||||
@ -538,39 +539,39 @@ public:
|
|||||||
/// to the specified LoopInfo object as being in the current basic block. It
|
/// to the specified LoopInfo object as being in the current basic block. It
|
||||||
/// is not valid to replace the loop header with this method.
|
/// is not valid to replace the loop header with this method.
|
||||||
///
|
///
|
||||||
void addBasicBlockToLoop(BlockT *NewBB, LoopInfoBase<BlockT> &LI);
|
void addBasicBlockToLoop(BlockT *NewBB, LoopInfoBase<BlockT, LoopT> &LI);
|
||||||
|
|
||||||
/// replaceChildLoopWith - This is used when splitting loops up. It replaces
|
/// replaceChildLoopWith - This is used when splitting loops up. It replaces
|
||||||
/// the OldChild entry in our children list with NewChild, and updates the
|
/// the OldChild entry in our children list with NewChild, and updates the
|
||||||
/// parent pointer of OldChild to be null and the NewChild to be this loop.
|
/// parent pointer of OldChild to be null and the NewChild to be this loop.
|
||||||
/// This updates the loop depth of the new child.
|
/// This updates the loop depth of the new child.
|
||||||
void replaceChildLoopWith(LoopBase<BlockT> *OldChild,
|
void replaceChildLoopWith(LoopT *OldChild,
|
||||||
LoopBase<BlockT> *NewChild) {
|
LoopT *NewChild) {
|
||||||
assert(OldChild->ParentLoop == this && "This loop is already broken!");
|
assert(OldChild->ParentLoop == this && "This loop is already broken!");
|
||||||
assert(NewChild->ParentLoop == 0 && "NewChild already has a parent!");
|
assert(NewChild->ParentLoop == 0 && "NewChild already has a parent!");
|
||||||
typename std::vector<LoopBase<BlockT>*>::iterator I =
|
typename std::vector<LoopT *>::iterator I =
|
||||||
std::find(SubLoops.begin(), SubLoops.end(), OldChild);
|
std::find(SubLoops.begin(), SubLoops.end(), OldChild);
|
||||||
assert(I != SubLoops.end() && "OldChild not in loop!");
|
assert(I != SubLoops.end() && "OldChild not in loop!");
|
||||||
*I = NewChild;
|
*I = NewChild;
|
||||||
OldChild->ParentLoop = 0;
|
OldChild->ParentLoop = 0;
|
||||||
NewChild->ParentLoop = this;
|
NewChild->ParentLoop = static_cast<LoopT *>(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// addChildLoop - Add the specified loop to be a child of this loop. This
|
/// addChildLoop - Add the specified loop to be a child of this loop. This
|
||||||
/// updates the loop depth of the new child.
|
/// updates the loop depth of the new child.
|
||||||
///
|
///
|
||||||
void addChildLoop(LoopBase<BlockT> *NewChild) {
|
void addChildLoop(LoopT *NewChild) {
|
||||||
assert(NewChild->ParentLoop == 0 && "NewChild already has a parent!");
|
assert(NewChild->ParentLoop == 0 && "NewChild already has a parent!");
|
||||||
NewChild->ParentLoop = this;
|
NewChild->ParentLoop = static_cast<LoopT *>(this);
|
||||||
SubLoops.push_back(NewChild);
|
SubLoops.push_back(NewChild);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// removeChildLoop - This removes the specified child from being a subloop of
|
/// removeChildLoop - This removes the specified child from being a subloop of
|
||||||
/// this loop. The loop is not deleted, as it will presumably be inserted
|
/// this loop. The loop is not deleted, as it will presumably be inserted
|
||||||
/// into another loop.
|
/// into another loop.
|
||||||
LoopBase<BlockT> *removeChildLoop(iterator I) {
|
LoopT *removeChildLoop(iterator I) {
|
||||||
assert(I != SubLoops.end() && "Cannot remove end iterator!");
|
assert(I != SubLoops.end() && "Cannot remove end iterator!");
|
||||||
LoopBase<BlockT> *Child = *I;
|
LoopT *Child = *I;
|
||||||
assert(Child->ParentLoop == this && "Child is not a child of this loop!");
|
assert(Child->ParentLoop == this && "Child is not a child of this loop!");
|
||||||
SubLoops.erase(SubLoops.begin()+(I-begin()));
|
SubLoops.erase(SubLoops.begin()+(I-begin()));
|
||||||
Child->ParentLoop = 0;
|
Child->ParentLoop = 0;
|
||||||
@ -643,25 +644,32 @@ public:
|
|||||||
print(cerr);
|
print(cerr);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
protected:
|
||||||
friend class LoopInfoBase<BlockT>;
|
friend class LoopInfoBase<BlockT, LoopT>;
|
||||||
explicit LoopBase(BlockT *BB) : ParentLoop(0) {
|
explicit LoopBase(BlockT *BB) : ParentLoop(0) {
|
||||||
Blocks.push_back(BB);
|
Blocks.push_back(BB);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class Loop : public LoopBase<BasicBlock, Loop> {
|
||||||
|
public:
|
||||||
|
Loop() {}
|
||||||
|
private:
|
||||||
|
friend class LoopInfoBase<BasicBlock, Loop>;
|
||||||
|
explicit Loop(BasicBlock *BB) : LoopBase<BasicBlock, Loop>(BB) {}
|
||||||
|
};
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
/// LoopInfo - This class builds and contains all of the top level loop
|
/// LoopInfo - This class builds and contains all of the top level loop
|
||||||
/// structures in the specified function.
|
/// structures in the specified function.
|
||||||
///
|
///
|
||||||
|
|
||||||
template<class BlockT>
|
template<class BlockT, class LoopT>
|
||||||
class LoopInfoBase {
|
class LoopInfoBase {
|
||||||
// BBMap - Mapping of basic blocks to the inner most loop they occur in
|
// BBMap - Mapping of basic blocks to the inner most loop they occur in
|
||||||
std::map<BlockT*, LoopBase<BlockT>*> BBMap;
|
std::map<BlockT *, LoopT *> BBMap;
|
||||||
std::vector<LoopBase<BlockT>*> TopLevelLoops;
|
std::vector<LoopT *> TopLevelLoops;
|
||||||
friend class LoopBase<BlockT>;
|
friend class LoopBase<BlockT, LoopT>;
|
||||||
|
|
||||||
void operator=(const LoopInfoBase &); // do not implement
|
void operator=(const LoopInfoBase &); // do not implement
|
||||||
LoopInfoBase(const LoopInfo &); // do not implement
|
LoopInfoBase(const LoopInfo &); // do not implement
|
||||||
@ -670,7 +678,7 @@ public:
|
|||||||
~LoopInfoBase() { releaseMemory(); }
|
~LoopInfoBase() { releaseMemory(); }
|
||||||
|
|
||||||
void releaseMemory() {
|
void releaseMemory() {
|
||||||
for (typename std::vector<LoopBase<BlockT>* >::iterator I =
|
for (typename std::vector<LoopT *>::iterator I =
|
||||||
TopLevelLoops.begin(), E = TopLevelLoops.end(); I != E; ++I)
|
TopLevelLoops.begin(), E = TopLevelLoops.end(); I != E; ++I)
|
||||||
delete *I; // Delete all of the loops...
|
delete *I; // Delete all of the loops...
|
||||||
|
|
||||||
@ -681,7 +689,7 @@ public:
|
|||||||
/// iterator/begin/end - The interface to the top-level loops in the current
|
/// iterator/begin/end - The interface to the top-level loops in the current
|
||||||
/// function.
|
/// function.
|
||||||
///
|
///
|
||||||
typedef typename std::vector<LoopBase<BlockT>*>::const_iterator iterator;
|
typedef typename std::vector<LoopT *>::const_iterator iterator;
|
||||||
iterator begin() const { return TopLevelLoops.begin(); }
|
iterator begin() const { return TopLevelLoops.begin(); }
|
||||||
iterator end() const { return TopLevelLoops.end(); }
|
iterator end() const { return TopLevelLoops.end(); }
|
||||||
bool empty() const { return TopLevelLoops.empty(); }
|
bool empty() const { return TopLevelLoops.empty(); }
|
||||||
@ -689,15 +697,15 @@ public:
|
|||||||
/// getLoopFor - Return the inner most loop that BB lives in. If a basic
|
/// getLoopFor - Return the inner most loop that BB lives in. If a basic
|
||||||
/// block is in no loop (for example the entry node), null is returned.
|
/// block is in no loop (for example the entry node), null is returned.
|
||||||
///
|
///
|
||||||
LoopBase<BlockT> *getLoopFor(const BlockT *BB) const {
|
LoopT *getLoopFor(const BlockT *BB) const {
|
||||||
typename std::map<BlockT *, LoopBase<BlockT>*>::const_iterator I=
|
typename std::map<BlockT *, LoopT *>::const_iterator I=
|
||||||
BBMap.find(const_cast<BlockT*>(BB));
|
BBMap.find(const_cast<BlockT*>(BB));
|
||||||
return I != BBMap.end() ? I->second : 0;
|
return I != BBMap.end() ? I->second : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// operator[] - same as getLoopFor...
|
/// operator[] - same as getLoopFor...
|
||||||
///
|
///
|
||||||
const LoopBase<BlockT> *operator[](const BlockT *BB) const {
|
const LoopT *operator[](const BlockT *BB) const {
|
||||||
return getLoopFor(BB);
|
return getLoopFor(BB);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -705,22 +713,22 @@ public:
|
|||||||
/// depth of 0 means the block is not inside any loop.
|
/// depth of 0 means the block is not inside any loop.
|
||||||
///
|
///
|
||||||
unsigned getLoopDepth(const BlockT *BB) const {
|
unsigned getLoopDepth(const BlockT *BB) const {
|
||||||
const LoopBase<BlockT> *L = getLoopFor(BB);
|
const LoopT *L = getLoopFor(BB);
|
||||||
return L ? L->getLoopDepth() : 0;
|
return L ? L->getLoopDepth() : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// isLoopHeader - True if the block is a loop header node
|
// isLoopHeader - True if the block is a loop header node
|
||||||
bool isLoopHeader(BlockT *BB) const {
|
bool isLoopHeader(BlockT *BB) const {
|
||||||
const LoopBase<BlockT> *L = getLoopFor(BB);
|
const LoopT *L = getLoopFor(BB);
|
||||||
return L && L->getHeader() == BB;
|
return L && L->getHeader() == BB;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// removeLoop - This removes the specified top-level loop from this loop info
|
/// removeLoop - This removes the specified top-level loop from this loop info
|
||||||
/// object. The loop is not deleted, as it will presumably be inserted into
|
/// object. The loop is not deleted, as it will presumably be inserted into
|
||||||
/// another loop.
|
/// another loop.
|
||||||
LoopBase<BlockT> *removeLoop(iterator I) {
|
LoopT *removeLoop(iterator I) {
|
||||||
assert(I != end() && "Cannot remove end iterator!");
|
assert(I != end() && "Cannot remove end iterator!");
|
||||||
LoopBase<BlockT> *L = *I;
|
LoopT *L = *I;
|
||||||
assert(L->getParentLoop() == 0 && "Not a top-level loop!");
|
assert(L->getParentLoop() == 0 && "Not a top-level loop!");
|
||||||
TopLevelLoops.erase(TopLevelLoops.begin() + (I-begin()));
|
TopLevelLoops.erase(TopLevelLoops.begin() + (I-begin()));
|
||||||
return L;
|
return L;
|
||||||
@ -729,17 +737,17 @@ public:
|
|||||||
/// changeLoopFor - Change the top-level loop that contains BB to the
|
/// changeLoopFor - Change the top-level loop that contains BB to the
|
||||||
/// specified loop. This should be used by transformations that restructure
|
/// specified loop. This should be used by transformations that restructure
|
||||||
/// the loop hierarchy tree.
|
/// the loop hierarchy tree.
|
||||||
void changeLoopFor(BlockT *BB, LoopBase<BlockT> *L) {
|
void changeLoopFor(BlockT *BB, LoopT *L) {
|
||||||
LoopBase<BlockT> *&OldLoop = BBMap[BB];
|
LoopT *&OldLoop = BBMap[BB];
|
||||||
assert(OldLoop && "Block not in a loop yet!");
|
assert(OldLoop && "Block not in a loop yet!");
|
||||||
OldLoop = L;
|
OldLoop = L;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// changeTopLevelLoop - Replace the specified loop in the top-level loops
|
/// changeTopLevelLoop - Replace the specified loop in the top-level loops
|
||||||
/// list with the indicated loop.
|
/// list with the indicated loop.
|
||||||
void changeTopLevelLoop(LoopBase<BlockT> *OldLoop,
|
void changeTopLevelLoop(LoopT *OldLoop,
|
||||||
LoopBase<BlockT> *NewLoop) {
|
LoopT *NewLoop) {
|
||||||
typename std::vector<LoopBase<BlockT>*>::iterator I =
|
typename std::vector<LoopT *>::iterator I =
|
||||||
std::find(TopLevelLoops.begin(), TopLevelLoops.end(), OldLoop);
|
std::find(TopLevelLoops.begin(), TopLevelLoops.end(), OldLoop);
|
||||||
assert(I != TopLevelLoops.end() && "Old loop not at top level!");
|
assert(I != TopLevelLoops.end() && "Old loop not at top level!");
|
||||||
*I = NewLoop;
|
*I = NewLoop;
|
||||||
@ -749,7 +757,7 @@ public:
|
|||||||
|
|
||||||
/// addTopLevelLoop - This adds the specified loop to the collection of
|
/// addTopLevelLoop - This adds the specified loop to the collection of
|
||||||
/// top-level loops.
|
/// top-level loops.
|
||||||
void addTopLevelLoop(LoopBase<BlockT> *New) {
|
void addTopLevelLoop(LoopT *New) {
|
||||||
assert(New->getParentLoop() == 0 && "Loop already in subloop!");
|
assert(New->getParentLoop() == 0 && "Loop already in subloop!");
|
||||||
TopLevelLoops.push_back(New);
|
TopLevelLoops.push_back(New);
|
||||||
}
|
}
|
||||||
@ -758,9 +766,9 @@ public:
|
|||||||
/// including all of the Loop objects it is nested in and our mapping from
|
/// including all of the Loop objects it is nested in and our mapping from
|
||||||
/// BasicBlocks to loops.
|
/// BasicBlocks to loops.
|
||||||
void removeBlock(BlockT *BB) {
|
void removeBlock(BlockT *BB) {
|
||||||
typename std::map<BlockT *, LoopBase<BlockT>*>::iterator I = BBMap.find(BB);
|
typename std::map<BlockT *, LoopT *>::iterator I = BBMap.find(BB);
|
||||||
if (I != BBMap.end()) {
|
if (I != BBMap.end()) {
|
||||||
for (LoopBase<BlockT> *L = I->second; L; L = L->getParentLoop())
|
for (LoopT *L = I->second; L; L = L->getParentLoop())
|
||||||
L->removeBlockFromLoop(BB);
|
L->removeBlockFromLoop(BB);
|
||||||
|
|
||||||
BBMap.erase(I);
|
BBMap.erase(I);
|
||||||
@ -769,8 +777,8 @@ public:
|
|||||||
|
|
||||||
// Internals
|
// Internals
|
||||||
|
|
||||||
static bool isNotAlreadyContainedIn(const LoopBase<BlockT> *SubLoop,
|
static bool isNotAlreadyContainedIn(const LoopT *SubLoop,
|
||||||
const LoopBase<BlockT> *ParentLoop) {
|
const LoopT *ParentLoop) {
|
||||||
if (SubLoop == 0) return true;
|
if (SubLoop == 0) return true;
|
||||||
if (SubLoop == ParentLoop) return false;
|
if (SubLoop == ParentLoop) return false;
|
||||||
return isNotAlreadyContainedIn(SubLoop->getParentLoop(), ParentLoop);
|
return isNotAlreadyContainedIn(SubLoop->getParentLoop(), ParentLoop);
|
||||||
@ -781,11 +789,11 @@ public:
|
|||||||
|
|
||||||
for (df_iterator<BlockT*> NI = df_begin(RootNode),
|
for (df_iterator<BlockT*> NI = df_begin(RootNode),
|
||||||
NE = df_end(RootNode); NI != NE; ++NI)
|
NE = df_end(RootNode); NI != NE; ++NI)
|
||||||
if (LoopBase<BlockT> *L = ConsiderForLoop(*NI, DT))
|
if (LoopT *L = ConsiderForLoop(*NI, DT))
|
||||||
TopLevelLoops.push_back(L);
|
TopLevelLoops.push_back(L);
|
||||||
}
|
}
|
||||||
|
|
||||||
LoopBase<BlockT> *ConsiderForLoop(BlockT *BB, DominatorTreeBase<BlockT> &DT) {
|
LoopT *ConsiderForLoop(BlockT *BB, DominatorTreeBase<BlockT> &DT) {
|
||||||
if (BBMap.find(BB) != BBMap.end()) return 0;// Haven't processed this node?
|
if (BBMap.find(BB) != BBMap.end()) return 0;// Haven't processed this node?
|
||||||
|
|
||||||
std::vector<BlockT *> TodoStack;
|
std::vector<BlockT *> TodoStack;
|
||||||
@ -802,7 +810,7 @@ public:
|
|||||||
if (TodoStack.empty()) return 0; // No backedges to this block...
|
if (TodoStack.empty()) return 0; // No backedges to this block...
|
||||||
|
|
||||||
// Create a new loop to represent this basic block...
|
// Create a new loop to represent this basic block...
|
||||||
LoopBase<BlockT> *L = new LoopBase<BlockT>(BB);
|
LoopT *L = new LoopT(BB);
|
||||||
BBMap[BB] = L;
|
BBMap[BB] = L;
|
||||||
|
|
||||||
BlockT *EntryBlock = BB->getParent()->begin();
|
BlockT *EntryBlock = BB->getParent()->begin();
|
||||||
@ -819,13 +827,13 @@ public:
|
|||||||
// occurs, this child loop gets added to a part of the current loop,
|
// occurs, this child loop gets added to a part of the current loop,
|
||||||
// making it a sibling to the current loop. We have to reparent this
|
// making it a sibling to the current loop. We have to reparent this
|
||||||
// loop.
|
// loop.
|
||||||
if (LoopBase<BlockT> *SubLoop =
|
if (LoopT *SubLoop =
|
||||||
const_cast<LoopBase<BlockT>*>(getLoopFor(X)))
|
const_cast<LoopT *>(getLoopFor(X)))
|
||||||
if (SubLoop->getHeader() == X && isNotAlreadyContainedIn(SubLoop, L)){
|
if (SubLoop->getHeader() == X && isNotAlreadyContainedIn(SubLoop, L)){
|
||||||
// Remove the subloop from it's current parent...
|
// Remove the subloop from it's current parent...
|
||||||
assert(SubLoop->ParentLoop && SubLoop->ParentLoop != L);
|
assert(SubLoop->ParentLoop && SubLoop->ParentLoop != L);
|
||||||
LoopBase<BlockT> *SLP = SubLoop->ParentLoop; // SubLoopParent
|
LoopT *SLP = SubLoop->ParentLoop; // SubLoopParent
|
||||||
typename std::vector<LoopBase<BlockT>*>::iterator I =
|
typename std::vector<LoopT *>::iterator I =
|
||||||
std::find(SLP->SubLoops.begin(), SLP->SubLoops.end(), SubLoop);
|
std::find(SLP->SubLoops.begin(), SLP->SubLoops.end(), SubLoop);
|
||||||
assert(I != SLP->SubLoops.end() &&"SubLoop not a child of parent?");
|
assert(I != SLP->SubLoops.end() &&"SubLoop not a child of parent?");
|
||||||
SLP->SubLoops.erase(I); // Remove from parent...
|
SLP->SubLoops.erase(I); // Remove from parent...
|
||||||
@ -849,7 +857,7 @@ public:
|
|||||||
// If there are any loops nested within this loop, create them now!
|
// If there are any loops nested within this loop, create them now!
|
||||||
for (typename std::vector<BlockT*>::iterator I = L->Blocks.begin(),
|
for (typename std::vector<BlockT*>::iterator I = L->Blocks.begin(),
|
||||||
E = L->Blocks.end(); I != E; ++I)
|
E = L->Blocks.end(); I != E; ++I)
|
||||||
if (LoopBase<BlockT> *NewLoop = ConsiderForLoop(*I, DT)) {
|
if (LoopT *NewLoop = ConsiderForLoop(*I, DT)) {
|
||||||
L->SubLoops.push_back(NewLoop);
|
L->SubLoops.push_back(NewLoop);
|
||||||
NewLoop->ParentLoop = L;
|
NewLoop->ParentLoop = L;
|
||||||
}
|
}
|
||||||
@ -859,8 +867,7 @@ public:
|
|||||||
//
|
//
|
||||||
for (typename std::vector<BlockT*>::iterator I = L->Blocks.begin(),
|
for (typename std::vector<BlockT*>::iterator I = L->Blocks.begin(),
|
||||||
E = L->Blocks.end(); I != E; ++I) {
|
E = L->Blocks.end(); I != E; ++I) {
|
||||||
typename std::map<BlockT*, LoopBase<BlockT>*>::iterator BBMI =
|
typename std::map<BlockT*, LoopT *>::iterator BBMI = BBMap.find(*I);
|
||||||
BBMap.find(*I);
|
|
||||||
if (BBMI == BBMap.end()) // Not in map yet...
|
if (BBMI == BBMap.end()) // Not in map yet...
|
||||||
BBMap.insert(BBMI, std::make_pair(*I, L)); // Must be at this level
|
BBMap.insert(BBMI, std::make_pair(*I, L)); // Must be at this level
|
||||||
}
|
}
|
||||||
@ -870,13 +877,12 @@ public:
|
|||||||
// can accidentally pull loops our of their parents, so we must make sure to
|
// can accidentally pull loops our of their parents, so we must make sure to
|
||||||
// organize the loop nests correctly now.
|
// organize the loop nests correctly now.
|
||||||
{
|
{
|
||||||
std::map<BlockT*, LoopBase<BlockT>*> ContainingLoops;
|
std::map<BlockT *, LoopT *> ContainingLoops;
|
||||||
for (unsigned i = 0; i != L->SubLoops.size(); ++i) {
|
for (unsigned i = 0; i != L->SubLoops.size(); ++i) {
|
||||||
LoopBase<BlockT> *Child = L->SubLoops[i];
|
LoopT *Child = L->SubLoops[i];
|
||||||
assert(Child->getParentLoop() == L && "Not proper child loop?");
|
assert(Child->getParentLoop() == L && "Not proper child loop?");
|
||||||
|
|
||||||
if (LoopBase<BlockT> *ContainingLoop =
|
if (LoopT *ContainingLoop = ContainingLoops[Child->getHeader()]) {
|
||||||
ContainingLoops[Child->getHeader()]) {
|
|
||||||
// If there is already a loop which contains this loop, move this loop
|
// If there is already a loop which contains this loop, move this loop
|
||||||
// into the containing loop.
|
// into the containing loop.
|
||||||
MoveSiblingLoopInto(Child, ContainingLoop);
|
MoveSiblingLoopInto(Child, ContainingLoop);
|
||||||
@ -886,11 +892,11 @@ public:
|
|||||||
// if any of the contained blocks are loop headers for subloops we
|
// if any of the contained blocks are loop headers for subloops we
|
||||||
// have already processed.
|
// have already processed.
|
||||||
for (unsigned b = 0, e = Child->Blocks.size(); b != e; ++b) {
|
for (unsigned b = 0, e = Child->Blocks.size(); b != e; ++b) {
|
||||||
LoopBase<BlockT> *&BlockLoop = ContainingLoops[Child->Blocks[b]];
|
LoopT *&BlockLoop = ContainingLoops[Child->Blocks[b]];
|
||||||
if (BlockLoop == 0) { // Child block not processed yet...
|
if (BlockLoop == 0) { // Child block not processed yet...
|
||||||
BlockLoop = Child;
|
BlockLoop = Child;
|
||||||
} else if (BlockLoop != Child) {
|
} else if (BlockLoop != Child) {
|
||||||
LoopBase<BlockT> *SubLoop = BlockLoop;
|
LoopT *SubLoop = BlockLoop;
|
||||||
// Reparent all of the blocks which used to belong to BlockLoops
|
// Reparent all of the blocks which used to belong to BlockLoops
|
||||||
for (unsigned j = 0, e = SubLoop->Blocks.size(); j != e; ++j)
|
for (unsigned j = 0, e = SubLoop->Blocks.size(); j != e; ++j)
|
||||||
ContainingLoops[SubLoop->Blocks[j]] = Child;
|
ContainingLoops[SubLoop->Blocks[j]] = Child;
|
||||||
@ -911,14 +917,14 @@ public:
|
|||||||
|
|
||||||
/// MoveSiblingLoopInto - This method moves the NewChild loop to live inside
|
/// MoveSiblingLoopInto - This method moves the NewChild loop to live inside
|
||||||
/// of the NewParent Loop, instead of being a sibling of it.
|
/// of the NewParent Loop, instead of being a sibling of it.
|
||||||
void MoveSiblingLoopInto(LoopBase<BlockT> *NewChild,
|
void MoveSiblingLoopInto(LoopT *NewChild,
|
||||||
LoopBase<BlockT> *NewParent) {
|
LoopT *NewParent) {
|
||||||
LoopBase<BlockT> *OldParent = NewChild->getParentLoop();
|
LoopT *OldParent = NewChild->getParentLoop();
|
||||||
assert(OldParent && OldParent == NewParent->getParentLoop() &&
|
assert(OldParent && OldParent == NewParent->getParentLoop() &&
|
||||||
NewChild != NewParent && "Not sibling loops!");
|
NewChild != NewParent && "Not sibling loops!");
|
||||||
|
|
||||||
// Remove NewChild from being a child of OldParent
|
// Remove NewChild from being a child of OldParent
|
||||||
typename std::vector<LoopBase<BlockT>*>::iterator I =
|
typename std::vector<LoopT *>::iterator I =
|
||||||
std::find(OldParent->SubLoops.begin(), OldParent->SubLoops.end(),
|
std::find(OldParent->SubLoops.begin(), OldParent->SubLoops.end(),
|
||||||
NewChild);
|
NewChild);
|
||||||
assert(I != OldParent->SubLoops.end() && "Parent fields incorrect??");
|
assert(I != OldParent->SubLoops.end() && "Parent fields incorrect??");
|
||||||
@ -931,7 +937,7 @@ public:
|
|||||||
/// InsertLoopInto - This inserts loop L into the specified parent loop. If
|
/// InsertLoopInto - This inserts loop L into the specified parent loop. If
|
||||||
/// the parent loop contains a loop which should contain L, the loop gets
|
/// the parent loop contains a loop which should contain L, the loop gets
|
||||||
/// inserted into L instead.
|
/// inserted into L instead.
|
||||||
void InsertLoopInto(LoopBase<BlockT> *L, LoopBase<BlockT> *Parent) {
|
void InsertLoopInto(LoopT *L, LoopT *Parent) {
|
||||||
BlockT *LHeader = L->getHeader();
|
BlockT *LHeader = L->getHeader();
|
||||||
assert(Parent->contains(LHeader) &&
|
assert(Parent->contains(LHeader) &&
|
||||||
"This loop should not be inserted here!");
|
"This loop should not be inserted here!");
|
||||||
@ -955,7 +961,7 @@ public:
|
|||||||
for (unsigned i = 0; i < TopLevelLoops.size(); ++i)
|
for (unsigned i = 0; i < TopLevelLoops.size(); ++i)
|
||||||
TopLevelLoops[i]->print(OS);
|
TopLevelLoops[i]->print(OS);
|
||||||
#if 0
|
#if 0
|
||||||
for (std::map<BasicBlock*, Loop*>::const_iterator I = BBMap.begin(),
|
for (std::map<BasicBlock*, LoopT*>::const_iterator I = BBMap.begin(),
|
||||||
E = BBMap.end(); I != E; ++I)
|
E = BBMap.end(); I != E; ++I)
|
||||||
OS << "BB '" << I->first->getName() << "' level = "
|
OS << "BB '" << I->first->getName() << "' level = "
|
||||||
<< I->second->getLoopDepth() << "\n";
|
<< I->second->getLoopDepth() << "\n";
|
||||||
@ -964,8 +970,8 @@ public:
|
|||||||
};
|
};
|
||||||
|
|
||||||
class LoopInfo : public FunctionPass {
|
class LoopInfo : public FunctionPass {
|
||||||
LoopInfoBase<BasicBlock> LI;
|
LoopInfoBase<BasicBlock, Loop> LI;
|
||||||
friend class LoopBase<BasicBlock>;
|
friend class LoopBase<BasicBlock, Loop>;
|
||||||
|
|
||||||
void operator=(const LoopInfo &); // do not implement
|
void operator=(const LoopInfo &); // do not implement
|
||||||
LoopInfo(const LoopInfo &); // do not implement
|
LoopInfo(const LoopInfo &); // do not implement
|
||||||
@ -974,12 +980,12 @@ public:
|
|||||||
|
|
||||||
LoopInfo() : FunctionPass(&ID) {}
|
LoopInfo() : FunctionPass(&ID) {}
|
||||||
|
|
||||||
LoopInfoBase<BasicBlock>& getBase() { return LI; }
|
LoopInfoBase<BasicBlock, Loop>& getBase() { return LI; }
|
||||||
|
|
||||||
/// iterator/begin/end - The interface to the top-level loops in the current
|
/// iterator/begin/end - The interface to the top-level loops in the current
|
||||||
/// function.
|
/// function.
|
||||||
///
|
///
|
||||||
typedef LoopInfoBase<BasicBlock>::iterator iterator;
|
typedef LoopInfoBase<BasicBlock, Loop>::iterator iterator;
|
||||||
inline iterator begin() const { return LI.begin(); }
|
inline iterator begin() const { return LI.begin(); }
|
||||||
inline iterator end() const { return LI.end(); }
|
inline iterator end() const { return LI.end(); }
|
||||||
bool empty() const { return LI.empty(); }
|
bool empty() const { return LI.empty(); }
|
||||||
@ -1051,6 +1057,13 @@ public:
|
|||||||
void removeBlock(BasicBlock *BB) {
|
void removeBlock(BasicBlock *BB) {
|
||||||
LI.removeBlock(BB);
|
LI.removeBlock(BB);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool isNotAlreadyContainedIn(const Loop *SubLoop,
|
||||||
|
const Loop *ParentLoop) {
|
||||||
|
return
|
||||||
|
LoopInfoBase<BasicBlock, Loop>::isNotAlreadyContainedIn(SubLoop,
|
||||||
|
ParentLoop);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -1081,19 +1094,21 @@ template <> struct GraphTraits<Loop*> {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template<class BlockT>
|
template<class BlockT, class LoopT>
|
||||||
void LoopBase<BlockT>::addBasicBlockToLoop(BlockT *NewBB,
|
void
|
||||||
LoopInfoBase<BlockT> &LIB) {
|
LoopBase<BlockT, LoopT>::addBasicBlockToLoop(BlockT *NewBB,
|
||||||
|
LoopInfoBase<BlockT, LoopT> &LIB) {
|
||||||
assert((Blocks.empty() || LIB[getHeader()] == this) &&
|
assert((Blocks.empty() || LIB[getHeader()] == this) &&
|
||||||
"Incorrect LI specified for this loop!");
|
"Incorrect LI specified for this loop!");
|
||||||
assert(NewBB && "Cannot add a null basic block to the loop!");
|
assert(NewBB && "Cannot add a null basic block to the loop!");
|
||||||
assert(LIB[NewBB] == 0 && "BasicBlock already in the loop!");
|
assert(LIB[NewBB] == 0 && "BasicBlock already in the loop!");
|
||||||
|
|
||||||
|
LoopT *L = static_cast<LoopT *>(this);
|
||||||
|
|
||||||
// Add the loop mapping to the LoopInfo object...
|
// Add the loop mapping to the LoopInfo object...
|
||||||
LIB.BBMap[NewBB] = this;
|
LIB.BBMap[NewBB] = L;
|
||||||
|
|
||||||
// Add the basic block to this loop and all parent loops...
|
// Add the basic block to this loop and all parent loops...
|
||||||
LoopBase<BlockT> *L = this;
|
|
||||||
while (L) {
|
while (L) {
|
||||||
L->Blocks.push_back(NewBB);
|
L->Blocks.push_back(NewBB);
|
||||||
L = L->getParentLoop();
|
L = L->getParentLoop();
|
||||||
|
@ -35,48 +35,62 @@
|
|||||||
|
|
||||||
namespace llvm {
|
namespace llvm {
|
||||||
|
|
||||||
|
class MachineLoop;
|
||||||
|
|
||||||
// Provide overrides for Loop methods that don't make sense for machine loops.
|
// Provide overrides for Loop methods that don't make sense for machine loops.
|
||||||
template<> inline
|
template<> inline
|
||||||
PHINode *LoopBase<MachineBasicBlock>::getCanonicalInductionVariable() const {
|
PHINode *
|
||||||
|
LoopBase<MachineBasicBlock, MachineLoop>::getCanonicalInductionVariable() const {
|
||||||
assert(0 && "getCanonicalInductionVariable not supported for machine loops!");
|
assert(0 && "getCanonicalInductionVariable not supported for machine loops!");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<> inline Instruction*
|
template<> inline Instruction*
|
||||||
LoopBase<MachineBasicBlock>::getCanonicalInductionVariableIncrement() const {
|
LoopBase<MachineBasicBlock,
|
||||||
|
MachineLoop>::getCanonicalInductionVariableIncrement() const {
|
||||||
assert(0 &&
|
assert(0 &&
|
||||||
"getCanonicalInductionVariableIncrement not supported for machine loops!");
|
"getCanonicalInductionVariableIncrement not supported for machine loops!");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<>
|
template<>
|
||||||
inline bool LoopBase<MachineBasicBlock>::isLoopInvariant(Value *V) const {
|
inline bool
|
||||||
|
LoopBase<MachineBasicBlock, MachineLoop>::isLoopInvariant(Value *V) const {
|
||||||
assert(0 && "isLoopInvariant not supported for machine loops!");
|
assert(0 && "isLoopInvariant not supported for machine loops!");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<>
|
template<>
|
||||||
inline Value *LoopBase<MachineBasicBlock>::getTripCount() const {
|
inline Value *
|
||||||
|
LoopBase<MachineBasicBlock, MachineLoop>::getTripCount() const {
|
||||||
assert(0 && "getTripCount not supported for machine loops!");
|
assert(0 && "getTripCount not supported for machine loops!");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<>
|
template<>
|
||||||
inline bool LoopBase<MachineBasicBlock>::isLCSSAForm() const {
|
inline bool
|
||||||
|
LoopBase<MachineBasicBlock, MachineLoop>::isLCSSAForm() const {
|
||||||
assert(0 && "isLCSSAForm not supported for machine loops");
|
assert(0 && "isLCSSAForm not supported for machine loops");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
typedef LoopBase<MachineBasicBlock> MachineLoop;
|
class MachineLoop : public LoopBase<MachineBasicBlock, MachineLoop> {
|
||||||
|
public:
|
||||||
|
MachineLoop();
|
||||||
|
private:
|
||||||
|
friend class LoopInfoBase<MachineBasicBlock, MachineLoop>;
|
||||||
|
explicit MachineLoop(MachineBasicBlock *MBB)
|
||||||
|
: LoopBase<MachineBasicBlock, MachineLoop>(MBB) {}
|
||||||
|
};
|
||||||
|
|
||||||
class MachineLoopInfo : public MachineFunctionPass {
|
class MachineLoopInfo : public MachineFunctionPass {
|
||||||
LoopInfoBase<MachineBasicBlock> LI;
|
LoopInfoBase<MachineBasicBlock, MachineLoop> LI;
|
||||||
friend class LoopBase<MachineBasicBlock>;
|
friend class LoopBase<MachineBasicBlock, MachineLoop>;
|
||||||
|
|
||||||
void operator=(const MachineLoopInfo &); // do not implement
|
void operator=(const MachineLoopInfo &); // do not implement
|
||||||
MachineLoopInfo(const MachineLoopInfo &); // do not implement
|
MachineLoopInfo(const MachineLoopInfo &); // do not implement
|
||||||
|
|
||||||
LoopInfoBase<MachineBasicBlock>& getBase() { return LI; }
|
LoopInfoBase<MachineBasicBlock, MachineLoop>& getBase() { return LI; }
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static char ID; // Pass identification, replacement for typeid
|
static char ID; // Pass identification, replacement for typeid
|
||||||
@ -86,7 +100,7 @@ public:
|
|||||||
/// iterator/begin/end - The interface to the top-level loops in the current
|
/// iterator/begin/end - The interface to the top-level loops in the current
|
||||||
/// function.
|
/// function.
|
||||||
///
|
///
|
||||||
typedef LoopInfoBase<MachineBasicBlock>::iterator iterator;
|
typedef LoopInfoBase<MachineBasicBlock, MachineLoop>::iterator iterator;
|
||||||
inline iterator begin() const { return LI.begin(); }
|
inline iterator begin() const { return LI.begin(); }
|
||||||
inline iterator end() const { return LI.end(); }
|
inline iterator end() const { return LI.end(); }
|
||||||
bool empty() const { return LI.empty(); }
|
bool empty() const { return LI.empty(); }
|
||||||
|
@ -36,10 +36,9 @@ class CallSite;
|
|||||||
class Trace;
|
class Trace;
|
||||||
class CallGraph;
|
class CallGraph;
|
||||||
class TargetData;
|
class TargetData;
|
||||||
|
class Loop;
|
||||||
class LoopInfo;
|
class LoopInfo;
|
||||||
class LLVMContext;
|
class LLVMContext;
|
||||||
template<class N> class LoopBase;
|
|
||||||
typedef LoopBase<BasicBlock> Loop;
|
|
||||||
|
|
||||||
/// CloneModule - Return an exact copy of the specified module
|
/// CloneModule - Return an exact copy of the specified module
|
||||||
///
|
///
|
||||||
|
@ -54,7 +54,7 @@ static bool containsAddRecFromDifferentLoop(const SCEV *S, Loop *L) {
|
|||||||
if (newLoop == L)
|
if (newLoop == L)
|
||||||
return false;
|
return false;
|
||||||
// if newLoop is an outer loop of L, this is OK.
|
// if newLoop is an outer loop of L, this is OK.
|
||||||
if (!LoopInfoBase<BasicBlock>::isNotAlreadyContainedIn(L, newLoop))
|
if (!LoopInfo::isNotAlreadyContainedIn(L, newLoop))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
|
@ -19,8 +19,12 @@
|
|||||||
#include "llvm/CodeGen/Passes.h"
|
#include "llvm/CodeGen/Passes.h"
|
||||||
using namespace llvm;
|
using namespace llvm;
|
||||||
|
|
||||||
TEMPLATE_INSTANTIATION(class LoopBase<MachineBasicBlock>);
|
#define MLB class LoopBase<MachineBasicBlock, MachineLoop>
|
||||||
TEMPLATE_INSTANTIATION(class LoopInfoBase<MachineBasicBlock>);
|
TEMPLATE_INSTANTIATION(MLB);
|
||||||
|
#undef MLB
|
||||||
|
#define MLIB class LoopInfoBase<MachineBasicBlock, MachineLoop>
|
||||||
|
TEMPLATE_INSTANTIATION(MLIB);
|
||||||
|
#undef MLIB
|
||||||
|
|
||||||
char MachineLoopInfo::ID = 0;
|
char MachineLoopInfo::ID = 0;
|
||||||
static RegisterPass<MachineLoopInfo>
|
static RegisterPass<MachineLoopInfo>
|
||||||
|
@ -248,7 +248,7 @@ static bool containsAddRecFromDifferentLoop(const SCEV *S, Loop *L) {
|
|||||||
if (newLoop == L)
|
if (newLoop == L)
|
||||||
return false;
|
return false;
|
||||||
// if newLoop is an outer loop of L, this is OK.
|
// if newLoop is an outer loop of L, this is OK.
|
||||||
if (!LoopInfoBase<BasicBlock>::isNotAlreadyContainedIn(L, newLoop))
|
if (!LoopInfo::isNotAlreadyContainedIn(L, newLoop))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
|
Loading…
Reference in New Issue
Block a user