From 529b28da455a703d226a31a03400e6662ff569fe Mon Sep 17 00:00:00 2001 From: Chris Lattner Date: Tue, 13 Apr 2004 05:05:33 +0000 Subject: This patch addresses PR35: Loop simplify should reconstruct nested loops. This is fairly straight-forward, but was a real nightmare to get just perfect. aarg. :) git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@12884 91177308-0d34-0410-b5e6-96231b3b80d8 --- lib/Transforms/Utils/LoopSimplify.cpp | 202 +++++++++++++++++++++++++++++++++- 1 file changed, 196 insertions(+), 6 deletions(-) (limited to 'lib/Transforms') diff --git a/lib/Transforms/Utils/LoopSimplify.cpp b/lib/Transforms/Utils/LoopSimplify.cpp index f6e1374..5acece8 100644 --- a/lib/Transforms/Utils/LoopSimplify.cpp +++ b/lib/Transforms/Utils/LoopSimplify.cpp @@ -41,6 +41,7 @@ #include "llvm/Analysis/Dominators.h" #include "llvm/Analysis/LoopInfo.h" #include "llvm/Support/CFG.h" +#include "llvm/Transforms/Utils/Local.h" #include "Support/SetOperations.h" #include "Support/Statistic.h" #include "Support/DepthFirstIterator.h" @@ -49,6 +50,8 @@ using namespace llvm; namespace { Statistic<> NumInserted("loopsimplify", "Number of pre-header or exit blocks inserted"); + Statistic<> + NumNested("loopsimplify", "Number of nested loops split out"); struct LoopSimplify : public FunctionPass { virtual bool runOnFunction(Function &F); @@ -72,6 +75,7 @@ namespace { const std::vector &Preds); void RewriteLoopExitBlock(Loop *L, BasicBlock *Exit); void InsertPreheaderForLoop(Loop *L); + Loop *SeparateNestedLoop(Loop *L); void InsertUniqueBackedgeBlock(Loop *L); void UpdateDomInfoForRevectoredPreds(BasicBlock *NewBB, @@ -159,10 +163,17 @@ bool LoopSimplify::ProcessLoop(Loop *L) { } } - // The preheader may have more than two predecessors at this point (from the - // preheader and from the backedges). To simplify the loop more, insert an - // extra back-edge block in the loop so that there is exactly one backedge. + // If the header has more than two predecessors at this point (from the + // preheader and from multiple backedges), we must adjust the loop. if (L->getNumBackEdges() != 1) { + // If this is really a nested loop, rip it out into a child loop. + if (Loop *NL = SeparateNestedLoop(L)) { + ++NumNested; + // This is a big restructuring change, reprocess the whole loop. + ProcessLoop(NL); + return true; + } + InsertUniqueBackedgeBlock(L); NumInserted++; Changed = true; @@ -198,8 +209,9 @@ BasicBlock *LoopSimplify::SplitBlockPredecessors(BasicBlock *BB, // Check to see if the values being merged into the new block need PHI // nodes. If so, insert them. for (BasicBlock::iterator I = BB->begin(); - PHINode *PN = dyn_cast(I); ++I) { - + PHINode *PN = dyn_cast(I); ) { + ++I; + // Check to see if all of the values coming in are the same. If so, we // don't need to create a new PHI node. Value *InVal = PN->getIncomingValueForBlock(Preds[0]); @@ -216,7 +228,7 @@ BasicBlock *LoopSimplify::SplitBlockPredecessors(BasicBlock *BB, // Move all of the edges from blocks outside the loop to the new PHI for (unsigned i = 0, e = Preds.size(); i != e; ++i) { - Value *V = PN->removeIncomingValue(Preds[i]); + Value *V = PN->removeIncomingValue(Preds[i], false); NewPHI->addIncoming(V, Preds[i]); } InVal = NewPHI; @@ -230,6 +242,12 @@ BasicBlock *LoopSimplify::SplitBlockPredecessors(BasicBlock *BB, // Add an incoming value to the PHI node in the loop for the preheader // edge. PN->addIncoming(InVal, NewBB); + + // Can we eliminate this phi node now? + if (Value *V = hasConstantValue(PN)) { + PN->replaceAllUsesWith(V); + BB->getInstList().erase(PN); + } } // Now that the PHI nodes are updated, actually move the edges from @@ -382,6 +400,9 @@ void LoopSimplify::InsertPreheaderForLoop(Loop *L) { } } +/// RewriteLoopExitBlock - Ensure that the loop preheader dominates all exit +/// blocks. This method is used to split exit blocks that have predecessors +/// outside of the loop. void LoopSimplify::RewriteLoopExitBlock(Loop *L, BasicBlock *Exit) { DominatorSet &DS = getAnalysis(); assert(std::find(L->getExitBlocks().begin(), L->getExitBlocks().end(), Exit) @@ -409,6 +430,175 @@ void LoopSimplify::RewriteLoopExitBlock(Loop *L, BasicBlock *Exit) { UpdateDomInfoForRevectoredPreds(NewBB, LoopBlocks); } +/// AddBlockAndPredsToSet - Add the specified block, and all of its +/// predecessors, to the specified set, if it's not already in there. Stop +/// predecessor traversal when we reach StopBlock. +static void AddBlockAndPredsToSet(BasicBlock *BB, BasicBlock *StopBlock, + std::set &Blocks) { + if (!Blocks.insert(BB).second) return; // already processed. + if (BB == StopBlock) return; // Stop here! + + for (pred_iterator I = pred_begin(BB), E = pred_end(BB); I != E; ++I) + AddBlockAndPredsToSet(*I, StopBlock, Blocks); +} + +static void ReplaceExitBlocksOfLoopAndParents(Loop *L, BasicBlock *Old, + BasicBlock *New) { + if (!L->hasExitBlock(Old)) return; + L->changeExitBlock(Old, New); + ReplaceExitBlocksOfLoopAndParents(L->getParentLoop(), Old, New); +} + +/// VerifyExitBlocks - This is a function which can be useful for hacking on the +/// LoopSimplify Code. +static void VerifyExitBlocks(Loop *L) { + std::vector ExitBlocks; + for (unsigned i = 0, e = L->getBlocks().size(); i != e; ++i) { + BasicBlock *BB = L->getBlocks()[i]; + for (succ_iterator SI = succ_begin(BB), E = succ_end(BB); SI != E; ++SI) + if (!L->contains(*SI)) + ExitBlocks.push_back(*SI); + } + + std::vector EB = L->getExitBlocks(); + std::sort(EB.begin(), EB.end()); + std::sort(ExitBlocks.begin(), ExitBlocks.end()); + assert(EB == ExitBlocks && "Exit blocks were incorrectly updated!"); + + for (Loop::iterator I = L->begin(), E = L->end(); I != E; ++I) + VerifyExitBlocks(*I); +} + +/// SeparateNestedLoop - If this loop has multiple backedges, try to pull one of +/// them out into a nested loop. This is important for code that looks like +/// this: +/// +/// Loop: +/// ... +/// br cond, Loop, Next +/// ... +/// br cond2, Loop, Out +/// +/// To identify this common case, we look at the PHI nodes in the header of the +/// loop. PHI nodes with unchanging values on one backedge correspond to values +/// that change in the "outer" loop, but not in the "inner" loop. +/// +/// If we are able to separate out a loop, return the new outer loop that was +/// created. +/// +Loop *LoopSimplify::SeparateNestedLoop(Loop *L) { + BasicBlock *Header = L->getHeader(); + + std::vector OuterLoopPreds; + for (BasicBlock::iterator I = Header->begin(); + PHINode *PN = dyn_cast(I); ) { + ++I; + if (Value *V = hasConstantValue(PN)) { + // This is a degenerate PHI already, don't modify it! + PN->replaceAllUsesWith(V); + Header->getInstList().erase(PN); + continue; + } + + // Scan this PHI node looking for a use of the PHI node by itself. + for (unsigned i = 0, e = PN->getNumIncomingValues(); i != e; ++i) + if (PN->getIncomingValue(i) == PN && + L->contains(PN->getIncomingBlock(i))) { + // Wow, we found something tasty to remove. Pull out all predecessors + // that have varying values in the loop. This handles the case when a + // PHI node has multiple instances of itself as arguments. + for (unsigned i = 0, e = PN->getNumIncomingValues(); i != e; ++i) + if (PN->getIncomingValue(i) != PN || + !L->contains(PN->getIncomingBlock(i))) + OuterLoopPreds.push_back(PN->getIncomingBlock(i)); + goto FoundExtraction; + } + } + + return 0; // Nothing looks appetizing to separate out + +FoundExtraction: + BasicBlock *NewBB = SplitBlockPredecessors(Header, ".outer", OuterLoopPreds); + + // Update dominator information (set, immdom, domtree, and domfrontier) + UpdateDomInfoForRevectoredPreds(NewBB, OuterLoopPreds); + + // Create the new outer loop. + Loop *NewOuter = new Loop(); + + LoopInfo &LI = getAnalysis(); + + // Change the parent loop to use the outer loop as its child now. + if (Loop *Parent = L->getParentLoop()) + Parent->replaceChildLoopWith(L, NewOuter); + else + LI.changeTopLevelLoop(L, NewOuter); + + // This block is going to be our new header block: add it to this loop and all + // parent loops. + NewOuter->addBasicBlockToLoop(NewBB, getAnalysis()); + + // L is now a subloop of our outer loop. + NewOuter->addChildLoop(L); + + // Add all of L's exit blocks to the outer loop. + for (unsigned i = 0, e = L->getExitBlocks().size(); i != e; ++i) + NewOuter->addExitBlock(L->getExitBlocks()[i]); + + // Add temporary exit block entries for NewBB. Add one for each edge in L + // that goes to NewBB. + for (pred_iterator PI = pred_begin(NewBB), E = pred_end(NewBB); PI != E; ++PI) + if (L->contains(*PI)) + L->addExitBlock(NewBB); + + for (unsigned i = 0, e = L->getBlocks().size(); i != e; ++i) + NewOuter->addBlockEntry(L->getBlocks()[i]); + + // Determine which blocks should stay in L and which should be moved out to + // the Outer loop now. + DominatorSet &DS = getAnalysis(); + std::set BlocksInL; + for (pred_iterator PI = pred_begin(Header), E = pred_end(Header); PI!=E; ++PI) + if (DS.dominates(Header, *PI)) + AddBlockAndPredsToSet(*PI, Header, BlocksInL); + + + // Scan all of the loop children of L, moving them to OuterLoop if they are + // not part of the inner loop. + for (Loop::iterator I = L->begin(); I != L->end(); ) + if (BlocksInL.count((*I)->getHeader())) + ++I; // Loop remains in L + else + NewOuter->addChildLoop(L->removeChildLoop(I)); + + // Now that we know which blocks are in L and which need to be moved to + // OuterLoop, move any blocks that need it. + for (unsigned i = 0; i != L->getBlocks().size(); ++i) { + BasicBlock *BB = L->getBlocks()[i]; + if (!BlocksInL.count(BB)) { + // Move this block to the parent, updating the exit blocks sets + L->removeBlockFromLoop(BB); + if (LI[BB] == L) + LI.changeLoopFor(BB, NewOuter); + --i; + } + } + + // Check all subloops of this loop, replacing any exit blocks that got + // revectored with the new basic block. + for (pred_iterator I = pred_begin(NewBB), E = pred_end(NewBB); I != E; ++I) + if (NewOuter->contains(*I)) { + // Change any exit blocks that used to go to Header to go to NewBB + // instead. + ReplaceExitBlocksOfLoopAndParents((Loop*)LI[*I], Header, NewBB); + } + + //VerifyExitBlocks(NewOuter); + return NewOuter; +} + + + /// InsertUniqueBackedgeBlock - This method is called when the specified loop /// has more than one backedge in it. If this occurs, revector all of these /// backedges to target a new basic block and have that block branch to the loop -- cgit v1.1