diff options
| author | Dehao Chen <dehao@google.com> | 2016-07-12 22:37:48 +0000 | 
|---|---|---|
| committer | Dehao Chen <dehao@google.com> | 2016-07-12 22:37:48 +0000 | 
| commit | 9cba1f4e7e2588cfed2ce66564a59c0e421e5ff1 (patch) | |
| tree | 9df27d17e800874696bcdd5194478eff5579a2c0 /llvm/lib/Transforms/Scalar/LICM.cpp | |
| parent | 72eebfa4b0f3ac876d5600558ffca08d3893c5f8 (diff) | |
| download | bcm5719-llvm-9cba1f4e7e2588cfed2ce66564a59c0e421e5ff1.tar.gz bcm5719-llvm-9cba1f4e7e2588cfed2ce66564a59c0e421e5ff1.zip | |
New pass manager for LICM.
Summary: Port LICM to the new pass manager.
Reviewers: davidxl, silvas
Subscribers: krasin, vitalybuka, silvas, davide, sanjoy, llvm-commits, mehdi_amini
Differential Revision: http://reviews.llvm.org/D21772
llvm-svn: 275222
Diffstat (limited to 'llvm/lib/Transforms/Scalar/LICM.cpp')
| -rw-r--r-- | llvm/lib/Transforms/Scalar/LICM.cpp | 164 | 
1 files changed, 99 insertions, 65 deletions
| diff --git a/llvm/lib/Transforms/Scalar/LICM.cpp b/llvm/lib/Transforms/Scalar/LICM.cpp index 96ed53a401e..2c0a70e44f5 100644 --- a/llvm/lib/Transforms/Scalar/LICM.cpp +++ b/llvm/lib/Transforms/Scalar/LICM.cpp @@ -30,6 +30,7 @@  //  //===----------------------------------------------------------------------===// +#include "llvm/Transforms/Scalar/LICM.h"  #include "llvm/ADT/Statistic.h"  #include "llvm/Analysis/AliasAnalysis.h"  #include "llvm/Analysis/AliasSetTracker.h" @@ -40,6 +41,7 @@  #include "llvm/Analysis/Loads.h"  #include "llvm/Analysis/LoopInfo.h"  #include "llvm/Analysis/LoopPass.h" +#include "llvm/Analysis/LoopPassManager.h"  #include "llvm/Analysis/MemoryBuiltins.h"  #include "llvm/Analysis/ScalarEvolution.h"  #include "llvm/Analysis/ScalarEvolutionAliasAnalysis.h" @@ -104,13 +106,39 @@ static bool canSinkOrHoistInst(Instruction &I, AliasAnalysis *AA,                                 LoopSafetyInfo *SafetyInfo);  namespace { -struct LICM : public LoopPass { +struct LoopInvariantCodeMotion { +  bool runOnLoop(Loop *L, AliasAnalysis *AA, LoopInfo *LI, DominatorTree *DT, +                 TargetLibraryInfo *TLI, ScalarEvolution *SE, bool DeleteAST); + +  DenseMap<Loop *, AliasSetTracker *> &getLoopToAliasSetMap() { +    return LoopToAliasSetMap; +  } + +private: +  DenseMap<Loop *, AliasSetTracker *> LoopToAliasSetMap; + +  AliasSetTracker *collectAliasInfoForLoop(Loop *L, LoopInfo *LI, +                                           AliasAnalysis *AA); +}; + +struct LegacyLICMPass : public LoopPass {    static char ID; // Pass identification, replacement for typeid -  LICM() : LoopPass(ID) { -    initializeLICMPass(*PassRegistry::getPassRegistry()); +  LegacyLICMPass() : LoopPass(ID) { +    initializeLegacyLICMPassPass(*PassRegistry::getPassRegistry());    } -  bool runOnLoop(Loop *L, LPPassManager &LPM) override; +  bool runOnLoop(Loop *L, LPPassManager &LPM) override { +    if (skipLoop(L)) +      return false; + +    auto *SE = getAnalysisIfAvailable<ScalarEvolutionWrapperPass>(); +    return LICM.runOnLoop(L, +                          &getAnalysis<AAResultsWrapperPass>().getAAResults(), +                          &getAnalysis<LoopInfoWrapperPass>().getLoopInfo(), +                          &getAnalysis<DominatorTreeWrapperPass>().getDomTree(), +                          &getAnalysis<TargetLibraryInfoWrapperPass>().getTLI(), +                          SE ? &SE->getSE() : nullptr, false); +  }    /// This transformation requires natural loop information & requires that    /// loop preheaders be inserted into the CFG... @@ -124,23 +152,13 @@ struct LICM : public LoopPass {    using llvm::Pass::doFinalization;    bool doFinalization() override { -    assert(LoopToAliasSetMap.empty() && "Didn't free loop alias sets"); +    assert(LICM.getLoopToAliasSetMap().empty() && +           "Didn't free loop alias sets");      return false;    }  private: -  AliasAnalysis *AA; // Current AliasAnalysis information -  LoopInfo *LI;      // Current LoopInfo -  DominatorTree *DT; // Dominator Tree for the current Loop. - -  TargetLibraryInfo *TLI; // TargetLibraryInfo for constant folding. - -  // State that is updated as we process loops. -  bool Changed;            // Set to true when we change anything. -  BasicBlock *Preheader;   // The preheader block of the current loop... -  Loop *CurLoop;           // The current loop we are working on... -  AliasSetTracker *CurAST; // AliasSet information for the current loop... -  DenseMap<Loop *, AliasSetTracker *> LoopToAliasSetMap; +  LoopInvariantCodeMotion LICM;    /// cloneBasicBlockAnalysis - Simple Analysis hook. Clone alias set info.    void cloneBasicBlockAnalysis(BasicBlock *From, BasicBlock *To, @@ -152,48 +170,63 @@ private:    /// Simple Analysis hook. Delete loop L from alias set map.    void deleteAnalysisLoop(Loop *L) override; - -  AliasSetTracker *collectAliasInfoForLoop(Loop *L);  };  } -char LICM::ID = 0; -INITIALIZE_PASS_BEGIN(LICM, "licm", "Loop Invariant Code Motion", false, false) +PreservedAnalyses LICMPass::run(Loop &L, AnalysisManager<Loop> &AM) { +  const auto &FAM = +      AM.getResult<FunctionAnalysisManagerLoopProxy>(L).getManager(); +  Function *F = L.getHeader()->getParent(); + +  auto *AA = FAM.getCachedResult<AAManager>(*F); +  auto *LI = FAM.getCachedResult<LoopAnalysis>(*F); +  auto *DT = FAM.getCachedResult<DominatorTreeAnalysis>(*F); +  auto *TLI = FAM.getCachedResult<TargetLibraryAnalysis>(*F); +  auto *SE = FAM.getCachedResult<ScalarEvolutionAnalysis>(*F); +  assert((AA && LI && DT && TLI && SE) && "Analyses for LICM not available"); + +  LoopInvariantCodeMotion LICM; + +  if (!LICM.runOnLoop(&L, AA, LI, DT, TLI, SE, true)) +    return PreservedAnalyses::all(); + +  // FIXME: There is no setPreservesCFG in the new PM. When that becomes +  // available, it should be used here. +  return getLoopPassPreservedAnalyses(); +} + +char LegacyLICMPass::ID = 0; +INITIALIZE_PASS_BEGIN(LegacyLICMPass, "licm", "Loop Invariant Code Motion", +                      false, false)  INITIALIZE_PASS_DEPENDENCY(LoopPass)  INITIALIZE_PASS_DEPENDENCY(TargetLibraryInfoWrapperPass) -INITIALIZE_PASS_END(LICM, "licm", "Loop Invariant Code Motion", false, false) +INITIALIZE_PASS_END(LegacyLICMPass, "licm", "Loop Invariant Code Motion", false, +                    false) -Pass *llvm::createLICMPass() { return new LICM(); } +Pass *llvm::createLICMPass() { return new LegacyLICMPass(); }  /// Hoist expressions out of the specified loop. Note, alias info for inner  /// loop is not preserved so it is not a good idea to run LICM multiple  /// times on one loop. +/// We should delete AST for inner loops in the new pass manager to avoid +/// memory leak.  /// -bool LICM::runOnLoop(Loop *L, LPPassManager &LPM) { -  if (skipLoop(L)) -    return false; - -  Changed = false; - -  // Get our Loop and Alias Analysis information... -  LI = &getAnalysis<LoopInfoWrapperPass>().getLoopInfo(); -  AA = &getAnalysis<AAResultsWrapperPass>().getAAResults(); -  DT = &getAnalysis<DominatorTreeWrapperPass>().getDomTree(); - -  TLI = &getAnalysis<TargetLibraryInfoWrapperPass>().getTLI(); +bool LoopInvariantCodeMotion::runOnLoop(Loop *L, AliasAnalysis *AA, +                                        LoopInfo *LI, DominatorTree *DT, +                                        TargetLibraryInfo *TLI, +                                        ScalarEvolution *SE, bool DeleteAST) { +  bool Changed = false;    assert(L->isLCSSAForm(*DT) && "Loop is not in LCSSA form."); -  CurAST = collectAliasInfoForLoop(L); - -  CurLoop = L; +  AliasSetTracker *CurAST = collectAliasInfoForLoop(L, LI, AA);    // Get the preheader block to move instructions into... -  Preheader = L->getLoopPreheader(); +  BasicBlock *Preheader = L->getLoopPreheader();    // Compute loop safety information.    LoopSafetyInfo SafetyInfo; -  computeLoopSafetyInfo(&SafetyInfo, CurLoop); +  computeLoopSafetyInfo(&SafetyInfo, L);    // We want to visit all of the instructions in this loop... that are not parts    // of our subloops (they have already had their invariants hoisted out of @@ -206,11 +239,11 @@ bool LICM::runOnLoop(Loop *L, LPPassManager &LPM) {    // instructions, we perform another pass to hoist them out of the loop.    //    if (L->hasDedicatedExits()) -    Changed |= sinkRegion(DT->getNode(L->getHeader()), AA, LI, DT, TLI, CurLoop, +    Changed |= sinkRegion(DT->getNode(L->getHeader()), AA, LI, DT, TLI, L,                            CurAST, &SafetyInfo);    if (Preheader) -    Changed |= hoistRegion(DT->getNode(L->getHeader()), AA, LI, DT, TLI, -                           CurLoop, CurAST, &SafetyInfo); +    Changed |= hoistRegion(DT->getNode(L->getHeader()), AA, LI, DT, TLI, L, +                           CurAST, &SafetyInfo);    // Now that all loop invariants have been removed from the loop, promote any    // memory references to scalars that we can. @@ -221,9 +254,8 @@ bool LICM::runOnLoop(Loop *L, LPPassManager &LPM) {      // Loop over all of the alias sets in the tracker object.      for (AliasSet &AS : *CurAST) -      Changed |= -          promoteLoopAccessesToScalars(AS, ExitBlocks, InsertPts, PIC, LI, DT, -                                       TLI, CurLoop, CurAST, &SafetyInfo); +      Changed |= promoteLoopAccessesToScalars( +          AS, ExitBlocks, InsertPts, PIC, LI, DT, TLI, L, CurAST, &SafetyInfo);      // Once we have promoted values across the loop body we have to recursively      // reform LCSSA as any nested loop may now have values defined within the @@ -232,8 +264,7 @@ bool LICM::runOnLoop(Loop *L, LPPassManager &LPM) {      // SSAUpdater strategy during promotion that was LCSSA aware and reformed      // it as it went.      if (Changed) { -      auto *SEWP = getAnalysisIfAvailable<ScalarEvolutionWrapperPass>(); -      formLCSSARecursively(*L, *DT, LI, SEWP ? &SEWP->getSE() : nullptr); +      formLCSSARecursively(*L, *DT, LI, SE);      }    } @@ -244,20 +275,15 @@ bool LICM::runOnLoop(Loop *L, LPPassManager &LPM) {    assert((!L->getParentLoop() || L->getParentLoop()->isLCSSAForm(*DT)) &&           "Parent loop not left in LCSSA form after LICM!"); -  // Clear out loops state information for the next iteration -  CurLoop = nullptr; -  Preheader = nullptr; -    // If this loop is nested inside of another one, save the alias information    // for when we process the outer loop. -  if (L->getParentLoop()) +  if (L->getParentLoop() && !DeleteAST)      LoopToAliasSetMap[L] = CurAST;    else      delete CurAST; -  if (Changed) -    if (auto *SEWP = getAnalysisIfAvailable<ScalarEvolutionWrapperPass>()) -      SEWP->getSE().forgetLoopDispositions(L); +  if (Changed && SE) +    SE->forgetLoopDispositions(L);    return Changed;  } @@ -387,7 +413,8 @@ void llvm::computeLoopSafetyInfo(LoopSafetyInfo *SafetyInfo, Loop *CurLoop) {    // Iterate over header and compute safety info.    for (BasicBlock::iterator I = Header->begin(), E = Header->end();         (I != E) && !SafetyInfo->HeaderMayThrow; ++I) -    SafetyInfo->HeaderMayThrow |= !isGuaranteedToTransferExecutionToSuccessor(&*I); +    SafetyInfo->HeaderMayThrow |= +        !isGuaranteedToTransferExecutionToSuccessor(&*I);    SafetyInfo->MayThrow = SafetyInfo->HeaderMayThrow;    // Iterate over loop instructions and compute safety info. @@ -1042,7 +1069,13 @@ bool llvm::promoteLoopAccessesToScalars(  /// Returns an owning pointer to an alias set which incorporates aliasing info  /// from L and all subloops of L. -AliasSetTracker *LICM::collectAliasInfoForLoop(Loop *L) { +/// FIXME: In new pass manager, there is no helper functions to handle loop +/// analysis such as cloneBasicBlockAnalysis. So the AST needs to be recompute +/// from scratch for every loop. Hook up with the helper functions when +/// available in the new pass manager to avoid redundant computation. +AliasSetTracker * +LoopInvariantCodeMotion::collectAliasInfoForLoop(Loop *L, LoopInfo *LI, +                                                 AliasAnalysis *AA) {    AliasSetTracker *CurAST = nullptr;    SmallVector<Loop *, 4> RecomputeLoops;    for (Loop *InnerL : L->getSubLoops()) { @@ -1092,8 +1125,9 @@ AliasSetTracker *LICM::collectAliasInfoForLoop(Loop *L) {  /// Simple analysis hook. Clone alias set info.  /// -void LICM::cloneBasicBlockAnalysis(BasicBlock *From, BasicBlock *To, Loop *L) { -  AliasSetTracker *AST = LoopToAliasSetMap.lookup(L); +void LegacyLICMPass::cloneBasicBlockAnalysis(BasicBlock *From, BasicBlock *To, +                                             Loop *L) { +  AliasSetTracker *AST = LICM.getLoopToAliasSetMap().lookup(L);    if (!AST)      return; @@ -1102,8 +1136,8 @@ void LICM::cloneBasicBlockAnalysis(BasicBlock *From, BasicBlock *To, Loop *L) {  /// Simple Analysis hook. Delete value V from alias set  /// -void LICM::deleteAnalysisValue(Value *V, Loop *L) { -  AliasSetTracker *AST = LoopToAliasSetMap.lookup(L); +void LegacyLICMPass::deleteAnalysisValue(Value *V, Loop *L) { +  AliasSetTracker *AST = LICM.getLoopToAliasSetMap().lookup(L);    if (!AST)      return; @@ -1112,13 +1146,13 @@ void LICM::deleteAnalysisValue(Value *V, Loop *L) {  /// Simple Analysis hook. Delete value L from alias set map.  /// -void LICM::deleteAnalysisLoop(Loop *L) { -  AliasSetTracker *AST = LoopToAliasSetMap.lookup(L); +void LegacyLICMPass::deleteAnalysisLoop(Loop *L) { +  AliasSetTracker *AST = LICM.getLoopToAliasSetMap().lookup(L);    if (!AST)      return;    delete AST; -  LoopToAliasSetMap.erase(L); +  LICM.getLoopToAliasSetMap().erase(L);  }  /// Return true if the body of this loop may store into the memory | 

