diff options
Diffstat (limited to 'llvm/lib/Transforms/Scalar/LICM.cpp')
-rw-r--r-- | llvm/lib/Transforms/Scalar/LICM.cpp | 77 |
1 files changed, 70 insertions, 7 deletions
diff --git a/llvm/lib/Transforms/Scalar/LICM.cpp b/llvm/lib/Transforms/Scalar/LICM.cpp index 46d5ca3dcad..e62daa63cd0 100644 --- a/llvm/lib/Transforms/Scalar/LICM.cpp +++ b/llvm/lib/Transforms/Scalar/LICM.cpp @@ -54,6 +54,7 @@ #include "llvm/Support/raw_ostream.h" #include "llvm/Target/TargetLibraryInfo.h" #include "llvm/Transforms/Utils/Local.h" +#include "llvm/Transforms/Utils/LoopUtils.h" #include "llvm/Transforms/Utils/SSAUpdater.h" #include <algorithm> using namespace llvm; @@ -85,10 +86,12 @@ namespace { AU.addRequired<DominatorTreeWrapperPass>(); AU.addRequired<LoopInfo>(); AU.addRequiredID(LoopSimplifyID); + AU.addPreservedID(LoopSimplifyID); + AU.addRequiredID(LCSSAID); + AU.addPreservedID(LCSSAID); AU.addRequired<AliasAnalysis>(); AU.addPreserved<AliasAnalysis>(); AU.addPreserved<ScalarEvolution>(); - AU.addPreservedID(LoopSimplifyID); AU.addRequired<TargetLibraryInfo>(); } @@ -193,6 +196,8 @@ INITIALIZE_PASS_BEGIN(LICM, "licm", "Loop Invariant Code Motion", false, false) INITIALIZE_PASS_DEPENDENCY(DominatorTreeWrapperPass) INITIALIZE_PASS_DEPENDENCY(LoopInfo) INITIALIZE_PASS_DEPENDENCY(LoopSimplify) +INITIALIZE_PASS_DEPENDENCY(LCSSA) +INITIALIZE_PASS_DEPENDENCY(ScalarEvolution) INITIALIZE_PASS_DEPENDENCY(TargetLibraryInfo) INITIALIZE_AG_DEPENDENCY(AliasAnalysis) INITIALIZE_PASS_END(LICM, "licm", "Loop Invariant Code Motion", false, false) @@ -283,6 +288,14 @@ bool LICM::runOnLoop(Loop *L, LPPassManager &LPM) { PromoteAliasSet(*I, ExitBlocks, InsertPts); } + // If we have successfully changed the loop, re-form LCSSA and also re-form + // LCSSA in the parent loop as hoisting or sinking may have broken it. + if (Changed) { + formLCSSA(*L, *DT, getAnalysisIfAvailable<ScalarEvolution>()); + if (Loop *ParentL = L->getParentLoop()) + formLCSSA(*ParentL, *DT, getAnalysisIfAvailable<ScalarEvolution>()); + } + // Clear out loops state information for the next iteration CurLoop = 0; Preheader = 0; @@ -451,6 +464,19 @@ bool LICM::canSinkOrHoistInst(Instruction &I) { return isSafeToExecuteUnconditionally(I); } +/// \brief Returns true if a PHINode is a trivially replaceable with an +/// Instruction. +/// +/// This is true when all incoming values are that instruction. This pattern +/// occurs most often with LCSSA PHI nodes. +static bool isTriviallyReplacablePHI(PHINode &PN, Instruction &I) { + for (unsigned i = 0, e = PN.getNumIncomingValues(); i != e; ++i) + if (PN.getIncomingValue(i) != &I) + return false; + + return true; +} + /// isNotUsedInLoop - Return true if the only users of this instruction are /// outside of the loop. If this is true, we can sink the instruction to the /// exit blocks of the loop. @@ -459,18 +485,48 @@ bool LICM::isNotUsedInLoop(Instruction &I) { for (Value::use_iterator UI = I.use_begin(), E = I.use_end(); UI != E; ++UI) { Instruction *User = cast<Instruction>(*UI); if (PHINode *PN = dyn_cast<PHINode>(User)) { - // PHI node uses occur in predecessor blocks! + // A PHI node where all of the incoming values are this instruction are + // special -- they can just be RAUW'ed with the instruction and thus + // don't require a use in the predecessor. This is a particular important + // special case because it is the pattern found in LCSSA form. + if (isTriviallyReplacablePHI(*PN, I)) { + if (CurLoop->contains(PN)) + return false; + else + continue; + } + + // Otherwise, PHI node uses occur in predecessor blocks if the incoming + // values. Check for such a use being inside the loop. for (unsigned i = 0, e = PN->getNumIncomingValues(); i != e; ++i) if (PN->getIncomingValue(i) == &I) if (CurLoop->contains(PN->getIncomingBlock(i))) return false; - } else if (CurLoop->contains(User)) { - return false; + + continue; } + + if (CurLoop->contains(User)) + return false; } return true; } +static BasicBlock::iterator +replaceTrivialPHIsAndGetInsertionPt(BasicBlock &BB, Instruction &I) { + BasicBlock::iterator II = BB.begin(); + while (PHINode *PN = dyn_cast<PHINode>(II)) { + ++II; + if (isTriviallyReplacablePHI(*PN, I)) { + PN->replaceAllUsesWith(&I); + PN->eraseFromParent(); + } + } + if (isa<LandingPadInst>(II)) + ++II; + + return II; +} /// sink - When an instruction is found to only be used outside of the loop, /// this function moves it to the exit blocks and patches up SSA form as needed. @@ -502,9 +558,14 @@ void LICM::sink(Instruction &I) { I.replaceAllUsesWith(UndefValue::get(I.getType())); I.eraseFromParent(); } else { + // Look for any LCSSA PHI nodes for this instruction in the exit blocks + // and replace them. + BasicBlock::iterator II = + replaceTrivialPHIsAndGetInsertionPt(*ExitBlocks[0], I); + // Move the instruction to the start of the exit block, after any PHI // nodes in it. - I.moveBefore(ExitBlocks[0]->getFirstInsertionPt()); + I.moveBefore(II); // This instruction is no longer in the AST for the current loop, because // we just sunk it out of the loop. If we just sunk it into an outer @@ -546,8 +607,10 @@ void LICM::sink(Instruction &I) { if (!DT->dominates(InstOrigBB, ExitBlock)) continue; - // Insert the code after the last PHI node. - BasicBlock::iterator InsertPt = ExitBlock->getFirstInsertionPt(); + // Look for any LCSSA PHI nodes for this instruction in the exit blocks + // and replace them. Then get the insertion point after the last PHI. + BasicBlock::iterator InsertPt = + replaceTrivialPHIsAndGetInsertionPt(*ExitBlock, I); // If this is the first exit block processed, just move the original // instruction, otherwise clone the original instruction and insert |