diff options
author | Chandler Carruth <chandlerc@gmail.com> | 2016-02-27 04:34:07 +0000 |
---|---|---|
committer | Chandler Carruth <chandlerc@gmail.com> | 2016-02-27 04:34:07 +0000 |
commit | ad8cb382fa6d0bcc0359d99a3ddec381c218ba89 (patch) | |
tree | 5b8f54fde3219ae0997260c6831ea247587b4baf /llvm/lib | |
parent | bf821db9322b57160886c79f022a73182d5ce280 (diff) | |
download | bcm5719-llvm-ad8cb382fa6d0bcc0359d99a3ddec381c218ba89.tar.gz bcm5719-llvm-ad8cb382fa6d0bcc0359d99a3ddec381c218ba89.zip |
[LICM] Teach LICM how to handle cases where the alias set tracker was
merged into a loop that was subsequently unrolled (or otherwise nuked).
In this case it can't merge in the ASTs for any remaining nested loops,
it needs to re-add their instructions dircetly.
The fix is very isolated, but I've pulled the code for merging blocks
into the AST into a single place in the process. The only behavior
change is in the case which would have crashed before.
This fixes a crash reported by Mikael Holmen on the list after r261316
restored much of the loop pass pipelining and allowed us to actually do
this kind of nested transformation sequenc. I've taken that test case
and further reduced it into the somewhat twisty maze of loops in the
included test case. This does in fact trigger the bug even in this
reduced form.
llvm-svn: 262108
Diffstat (limited to 'llvm/lib')
-rw-r--r-- | llvm/lib/Transforms/Scalar/LICM.cpp | 52 |
1 files changed, 32 insertions, 20 deletions
diff --git a/llvm/lib/Transforms/Scalar/LICM.cpp b/llvm/lib/Transforms/Scalar/LICM.cpp index 0e7f534c32a..0ac53ba92cb 100644 --- a/llvm/lib/Transforms/Scalar/LICM.cpp +++ b/llvm/lib/Transforms/Scalar/LICM.cpp @@ -154,10 +154,7 @@ namespace { /// Simple Analysis hook. Delete loop L from alias set map. void deleteAnalysisLoop(Loop *L) override; - /// Returns an owning pointer to an alias set which incorporates aliasing - /// info from all subloops of L, but does not include instructions in L - /// itself. - AliasSetTracker *collectAliasInfoFromSubLoops(Loop *L); + AliasSetTracker *collectAliasInfoForLoop(Loop *L); }; } @@ -188,22 +185,13 @@ bool LICM::runOnLoop(Loop *L, LPPassManager &LPM) { assert(L->isLCSSAForm(*DT) && "Loop is not in LCSSA form."); - CurAST = collectAliasInfoFromSubLoops(L); + CurAST = collectAliasInfoForLoop(L); CurLoop = L; // Get the preheader block to move instructions into... Preheader = L->getLoopPreheader(); - // Loop over the body of this loop, looking for calls, invokes, and stores. - // Because subloops have already been incorporated into AST, we skip blocks in - // subloops. - // - for (BasicBlock *BB : L->blocks()) { - if (LI->getLoopFor(BB) == L) // Ignore blocks in subloops. - CurAST->add(*BB); // Incorporate the specified basic block - } - // Compute loop safety information. LICMSafetyInfo SafetyInfo; computeLICMSafetyInfo(&SafetyInfo, CurLoop); @@ -1039,13 +1027,20 @@ bool llvm::promoteLoopAccessesToScalars(AliasSet &AS, } /// Returns an owning pointer to an alias set which incorporates aliasing info -/// from all subloops of L, but does not include instructions in L itself. -/// -AliasSetTracker *LICM::collectAliasInfoFromSubLoops(Loop *L) { +/// from L and all subloops of L. +AliasSetTracker *LICM::collectAliasInfoForLoop(Loop *L) { AliasSetTracker *CurAST = nullptr; + SmallVector<Loop *, 4> RecomputeLoops; for (Loop *InnerL : L->getSubLoops()) { - AliasSetTracker *InnerAST = LoopToAliasSetMap[InnerL]; - assert(InnerAST && "Where is my AST?"); + auto MapI = LoopToAliasSetMap.find(InnerL); + // If the AST for this inner loop is missing it may have been merged into + // some other loop's AST and then that loop unrolled, and so we need to + // recompute it. + if (MapI == LoopToAliasSetMap.end()) { + RecomputeLoops.push_back(InnerL); + continue; + } + AliasSetTracker *InnerAST = MapI->second; if (CurAST != nullptr) { // What if InnerLoop was modified by other passes ? @@ -1057,10 +1052,27 @@ AliasSetTracker *LICM::collectAliasInfoFromSubLoops(Loop *L) { } else { CurAST = InnerAST; } - LoopToAliasSetMap.erase(InnerL); + LoopToAliasSetMap.erase(MapI); } if (CurAST == nullptr) CurAST = new AliasSetTracker(*AA); + + auto mergeLoop = [&](Loop *L) { + // Loop over the body of this loop, looking for calls, invokes, and stores. + // Because subloops have already been incorporated into AST, we skip blocks + // in subloops. + for (BasicBlock *BB : L->blocks()) + if (LI->getLoopFor(BB) == L) // Ignore blocks in subloops. + CurAST->add(*BB); // Incorporate the specified basic block + }; + + // Add everything from the sub loops that are no longer directly available. + for (Loop *InnerL : RecomputeLoops) + mergeLoop(InnerL); + + // And merge in this loop. + mergeLoop(L); + return CurAST; } |