mirror of
https://github.com/RPCSX/llvm.git
synced 2025-04-03 00:31:49 +00:00
Modify df_iterator to support post-order actions
Summary: This makes a change to the state used to maintain visited information for depth first iterator. We know assume a method "completed(...)" which is called after all children of a node have been visited. In all existing cases, this method does nothing so this patch has no functional changes. It will however allow a client to distinguish back from cross edges in a DFS tree. Reviewers: nadav, mehdi_amini, dberlin Subscribers: MatzeB, mzolotukhin, twoh, freik, llvm-commits Differential Revision: https://reviews.llvm.org/D25191 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@283391 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
5ea3570b6a
commit
8be61a8c7e
@ -58,10 +58,25 @@ public:
|
||||
SetType &Visited;
|
||||
};
|
||||
|
||||
// The visited stated for the iteration is a simple set augmented with
|
||||
// one more method, completed, which is invoked when all children of a
|
||||
// node have been processed. It is intended to distinguish of back and
|
||||
// cross edges in the spanning tree but is not used in the common case.
|
||||
template <typename NodeRef, unsigned SmallSize=8>
|
||||
struct df_iterator_default_set : public llvm::SmallPtrSet<NodeRef, SmallSize> {
|
||||
typedef llvm::SmallPtrSet<NodeRef, SmallSize> BaseSet;
|
||||
typedef typename BaseSet::iterator iterator;
|
||||
std::pair<iterator,bool> insert(NodeRef N) { return BaseSet::insert(N) ; }
|
||||
template <typename IterT>
|
||||
void insert(IterT Begin, IterT End) { BaseSet::insert(Begin,End); }
|
||||
|
||||
void completed(NodeRef) { }
|
||||
};
|
||||
|
||||
// Generic Depth First Iterator
|
||||
template <class GraphT,
|
||||
class SetType =
|
||||
llvm::SmallPtrSet<typename GraphTraits<GraphT>::NodeRef, 8>,
|
||||
df_iterator_default_set<typename GraphTraits<GraphT>::NodeRef>,
|
||||
bool ExtStorage = false, class GT = GraphTraits<GraphT>>
|
||||
class df_iterator
|
||||
: public std::iterator<std::forward_iterator_tag, typename GT::NodeRef>,
|
||||
@ -89,10 +104,8 @@ private:
|
||||
}
|
||||
inline df_iterator(NodeRef Node, SetType &S)
|
||||
: df_iterator_storage<SetType, ExtStorage>(S) {
|
||||
if (!S.count(Node)) {
|
||||
if (this->Visited.insert(Node).second)
|
||||
VisitStack.push_back(StackElement(Node, None));
|
||||
this->Visited.insert(Node);
|
||||
}
|
||||
}
|
||||
inline df_iterator(SetType &S)
|
||||
: df_iterator_storage<SetType, ExtStorage>(S) {
|
||||
@ -119,7 +132,8 @@ private:
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
this->Visited.completed(Node);
|
||||
|
||||
// Oops, ran out of successors... go up a level on the stack.
|
||||
VisitStack.pop_back();
|
||||
} while (!VisitStack.empty());
|
||||
@ -235,7 +249,8 @@ iterator_range<df_ext_iterator<T, SetTy>> depth_first_ext(const T& G,
|
||||
|
||||
// Provide global definitions of inverse depth first iterators...
|
||||
template <class T,
|
||||
class SetTy = llvm::SmallPtrSet<typename GraphTraits<T>::NodeRef, 8>,
|
||||
class SetTy =
|
||||
df_iterator_default_set<typename GraphTraits<T>::NodeRef>,
|
||||
bool External = false>
|
||||
struct idf_iterator : public df_iterator<Inverse<T>, SetTy, External> {
|
||||
idf_iterator(const df_iterator<Inverse<T>, SetTy, External> &V)
|
||||
|
@ -228,9 +228,9 @@ void LoopBase<BlockT, LoopT>::verifyLoop() const {
|
||||
// Setup for using a depth-first iterator to visit every block in the loop.
|
||||
SmallVector<BlockT*, 8> ExitBBs;
|
||||
getExitBlocks(ExitBBs);
|
||||
llvm::SmallPtrSet<BlockT*, 8> VisitSet;
|
||||
df_iterator_default_set<BlockT*> VisitSet;
|
||||
VisitSet.insert(ExitBBs.begin(), ExitBBs.end());
|
||||
df_ext_iterator<BlockT*, llvm::SmallPtrSet<BlockT*, 8> >
|
||||
df_ext_iterator<BlockT*, df_iterator_default_set<BlockT*>>
|
||||
BI = df_ext_begin(getHeader(), VisitSet),
|
||||
BE = df_ext_end(getHeader(), VisitSet);
|
||||
|
||||
|
@ -626,12 +626,14 @@ public:
|
||||
/// are direct children of this Region. It does not iterate over any
|
||||
/// RegionNodes that are also element of a subregion of this Region.
|
||||
//@{
|
||||
typedef df_iterator<RegionNodeT *, SmallPtrSet<RegionNodeT *, 8>, false,
|
||||
GraphTraits<RegionNodeT *>> element_iterator;
|
||||
typedef df_iterator<RegionNodeT *, df_iterator_default_set<RegionNodeT *>,
|
||||
false, GraphTraits<RegionNodeT *>>
|
||||
element_iterator;
|
||||
|
||||
typedef df_iterator<const RegionNodeT *, SmallPtrSet<const RegionNodeT *, 8>,
|
||||
false,
|
||||
GraphTraits<const RegionNodeT *>> const_element_iterator;
|
||||
typedef df_iterator<const RegionNodeT *,
|
||||
df_iterator_default_set<const RegionNodeT *>, false,
|
||||
GraphTraits<const RegionNodeT *>>
|
||||
const_element_iterator;
|
||||
|
||||
element_iterator element_begin();
|
||||
element_iterator element_end();
|
||||
|
@ -294,7 +294,7 @@ inline RNSuccIterator<NodeRef, BlockT, RegionT> succ_end(NodeRef Node) {
|
||||
template <> \
|
||||
struct GraphTraits<FlatIt<RegionT *>> \
|
||||
: public GraphTraits<FlatIt<NodeT *>> { \
|
||||
typedef df_iterator<NodeRef, SmallPtrSet<NodeRef, 8>, false, \
|
||||
typedef df_iterator<NodeRef, df_iterator_default_set<NodeRef>, false, \
|
||||
GraphTraits<FlatIt<NodeRef>>> \
|
||||
nodes_iterator; \
|
||||
static NodeRef getEntryNode(RegionT *R) { \
|
||||
@ -316,7 +316,7 @@ RegionGraphTraits(const Region, const RegionNode);
|
||||
|
||||
template <> struct GraphTraits<RegionInfo*>
|
||||
: public GraphTraits<FlatIt<RegionNode*> > {
|
||||
typedef df_iterator<NodeRef, SmallPtrSet<NodeRef, 8>, false,
|
||||
typedef df_iterator<NodeRef, df_iterator_default_set<NodeRef>, false,
|
||||
GraphTraits<FlatIt<NodeRef>>>
|
||||
nodes_iterator;
|
||||
|
||||
@ -333,7 +333,7 @@ template <> struct GraphTraits<RegionInfo*>
|
||||
|
||||
template <> struct GraphTraits<RegionInfoPass*>
|
||||
: public GraphTraits<RegionInfo *> {
|
||||
typedef df_iterator<NodeRef, SmallPtrSet<NodeRef, 8>, false,
|
||||
typedef df_iterator<NodeRef, df_iterator_default_set<NodeRef>, false,
|
||||
GraphTraits<FlatIt<NodeRef>>>
|
||||
nodes_iterator;
|
||||
|
||||
|
@ -142,7 +142,7 @@ RegionGraphTraits(const MachineRegion, const MachineRegionNode);
|
||||
|
||||
template <> struct GraphTraits<MachineRegionInfo*>
|
||||
: public GraphTraits<FlatIt<MachineRegionNode*> > {
|
||||
typedef df_iterator<NodeRef, SmallPtrSet<NodeRef, 8>, false,
|
||||
typedef df_iterator<NodeRef, df_iterator_default_set<NodeRef>, false,
|
||||
GraphTraits<FlatIt<NodeRef>>>
|
||||
nodes_iterator;
|
||||
|
||||
@ -159,7 +159,7 @@ template <> struct GraphTraits<MachineRegionInfo*>
|
||||
|
||||
template <> struct GraphTraits<MachineRegionInfoPass*>
|
||||
: public GraphTraits<MachineRegionInfo *> {
|
||||
typedef df_iterator<NodeRef, SmallPtrSet<NodeRef, 8>, false,
|
||||
typedef df_iterator<NodeRef, df_iterator_default_set<NodeRef>, false,
|
||||
GraphTraits<FlatIt<NodeRef>>>
|
||||
nodes_iterator;
|
||||
|
||||
|
@ -157,7 +157,7 @@ public:
|
||||
template <class Node, class ChildIterator> struct DomTreeGraphTraitsBase {
|
||||
typedef Node *NodeRef;
|
||||
typedef ChildIterator ChildIteratorType;
|
||||
typedef df_iterator<Node *, SmallPtrSet<NodeRef, 8>> nodes_iterator;
|
||||
typedef df_iterator<Node *, df_iterator_default_set<Node*>> nodes_iterator;
|
||||
|
||||
static NodeRef getEntryNode(NodeRef N) { return N; }
|
||||
static ChildIteratorType child_begin(NodeRef N) { return N->begin(); }
|
||||
|
@ -599,7 +599,7 @@ void LiveIntervals::pruneValue(LiveRange &LR, SlotIndex Kill,
|
||||
// Find all blocks that are reachable from KillMBB without leaving VNI's live
|
||||
// range. It is possible that KillMBB itself is reachable, so start a DFS
|
||||
// from each successor.
|
||||
typedef SmallPtrSet<MachineBasicBlock*, 9> VisitedTy;
|
||||
typedef df_iterator_default_set<MachineBasicBlock*,9> VisitedTy;
|
||||
VisitedTy Visited;
|
||||
for (MachineBasicBlock::succ_iterator
|
||||
SuccI = KillMBB->succ_begin(), SuccE = KillMBB->succ_end();
|
||||
|
@ -643,7 +643,7 @@ bool LiveVariables::runOnMachineFunction(MachineFunction &mf) {
|
||||
// register before its uses due to dominance properties of SSA (except for PHI
|
||||
// nodes, which are treated as a special case).
|
||||
MachineBasicBlock *Entry = &MF->front();
|
||||
SmallPtrSet<MachineBasicBlock*,16> Visited;
|
||||
df_iterator_default_set<MachineBasicBlock*,16> Visited;
|
||||
|
||||
for (MachineBasicBlock *MBB : depth_first_ext(Entry, Visited)) {
|
||||
runOnBlock(MBB, NumRegs);
|
||||
|
@ -2014,11 +2014,11 @@ void MachineVerifier::verifyStackFrame() {
|
||||
|
||||
SmallVector<StackStateOfBB, 8> SPState;
|
||||
SPState.resize(MF->getNumBlockIDs());
|
||||
SmallPtrSet<const MachineBasicBlock*, 8> Reachable;
|
||||
df_iterator_default_set<const MachineBasicBlock*> Reachable;
|
||||
|
||||
// Visit the MBBs in DFS order.
|
||||
for (df_ext_iterator<const MachineFunction*,
|
||||
SmallPtrSet<const MachineBasicBlock*, 8> >
|
||||
df_iterator_default_set<const MachineBasicBlock*> >
|
||||
DFI = df_ext_begin(MF, Reachable), DFE = df_ext_end(MF, Reachable);
|
||||
DFI != DFE; ++DFI) {
|
||||
const MachineBasicBlock *MBB = *DFI;
|
||||
|
@ -1008,7 +1008,7 @@ void PEI::replaceFrameIndices(MachineFunction &Fn) {
|
||||
// Store SPAdj at exit of a basic block.
|
||||
SmallVector<int, 8> SPState;
|
||||
SPState.resize(Fn.getNumBlockIDs());
|
||||
SmallPtrSet<MachineBasicBlock*, 8> Reachable;
|
||||
df_iterator_default_set<MachineBasicBlock*> Reachable;
|
||||
|
||||
// Iterate over the reachable blocks in DFS order.
|
||||
for (auto DFI = df_ext_begin(&Fn, Reachable), DFE = df_ext_end(&Fn, Reachable);
|
||||
|
@ -40,7 +40,7 @@
|
||||
using namespace llvm;
|
||||
|
||||
static bool eliminateUnreachableBlock(Function &F) {
|
||||
SmallPtrSet<BasicBlock*, 8> Reachable;
|
||||
df_iterator_default_set<BasicBlock*> Reachable;
|
||||
|
||||
// Mark all reachable blocks.
|
||||
for (BasicBlock *BB : depth_first_ext(&F, Reachable))
|
||||
@ -130,7 +130,7 @@ void UnreachableMachineBlockElim::getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
}
|
||||
|
||||
bool UnreachableMachineBlockElim::runOnMachineFunction(MachineFunction &F) {
|
||||
SmallPtrSet<MachineBasicBlock*, 8> Reachable;
|
||||
df_iterator_default_set<MachineBasicBlock*> Reachable;
|
||||
bool ModifiedPHI = false;
|
||||
|
||||
MMI = getAnalysisIfAvailable<MachineModuleInfo>();
|
||||
|
@ -326,7 +326,7 @@ bool FPS::runOnMachineFunction(MachineFunction &MF) {
|
||||
|
||||
// Process the function in depth first order so that we process at least one
|
||||
// of the predecessors for every reachable block in the function.
|
||||
SmallPtrSet<MachineBasicBlock*, 8> Processed;
|
||||
df_iterator_default_set<MachineBasicBlock*> Processed;
|
||||
MachineBasicBlock *Entry = &MF.front();
|
||||
|
||||
bool Changed = false;
|
||||
|
@ -600,7 +600,7 @@ static bool isSafeToPromoteArgument(Argument *Arg, bool isByValOrInAlloca,
|
||||
|
||||
// Because there could be several/many load instructions, remember which
|
||||
// blocks we know to be transparent to the load.
|
||||
SmallPtrSet<BasicBlock*, 16> TranspBlocks;
|
||||
df_iterator_default_set<BasicBlock*, 16> TranspBlocks;
|
||||
|
||||
for (LoadInst *Load : Loads) {
|
||||
// Check to see if the load is invalidated from the start of the block to
|
||||
|
@ -27,6 +27,8 @@ template <typename T> struct CountedSet {
|
||||
}
|
||||
|
||||
size_t count(const T &Item) const { return S.count(Item); }
|
||||
|
||||
void completed(T) { }
|
||||
};
|
||||
|
||||
template <typename T> class df_iterator_storage<CountedSet<T>, true> {
|
||||
|
Loading…
x
Reference in New Issue
Block a user