diff options
| author | Johannes Doerfert <doerfert@cs.uni-saarland.de> | 2018-09-11 11:44:17 +0000 |
|---|---|---|
| committer | Johannes Doerfert <doerfert@cs.uni-saarland.de> | 2018-09-11 11:44:17 +0000 |
| commit | a58e9214ac6adc67e1a9229c7e8cea04819547a6 (patch) | |
| tree | 8d5bf7b5bab1e90fc6e961377e1f5b1b7721b03f /llvm | |
| parent | 4f3c0e27b6f5ea2d512689768ece7e23e48c79a6 (diff) | |
| download | bcm5719-llvm-a58e9214ac6adc67e1a9229c7e8cea04819547a6.tar.gz bcm5719-llvm-a58e9214ac6adc67e1a9229c7e8cea04819547a6.zip | |
[LoopInfo] Fix Loop::getLoopID() for loops with multiple latches
The previous implementation traversed all loop blocks and bailed if one
was not a latch block. Since we are only interested in latch blocks, we
should only traverse those.
llvm-svn: 341926
Diffstat (limited to 'llvm')
| -rw-r--r-- | llvm/lib/Analysis/LoopInfo.cpp | 17 | ||||
| -rw-r--r-- | llvm/unittests/Analysis/LoopInfoTest.cpp | 56 |
2 files changed, 61 insertions, 12 deletions
diff --git a/llvm/lib/Analysis/LoopInfo.cpp b/llvm/lib/Analysis/LoopInfo.cpp index 0ee59131356..4664c78b8c0 100644 --- a/llvm/lib/Analysis/LoopInfo.cpp +++ b/llvm/lib/Analysis/LoopInfo.cpp @@ -218,20 +218,13 @@ MDNode *Loop::getLoopID() const { } else { assert(!getLoopLatch() && "The loop should have no single latch at this point"); - // Go through each predecessor of the loop header and check the - // terminator for the metadata. - BasicBlock *H = getHeader(); - for (BasicBlock *BB : this->blocks()) { + // Go through the latch blocks and check the terminator for the metadata. + SmallVector<BasicBlock *, 4> LatchesBlocks; + getLoopLatches(LatchesBlocks); + for (BasicBlock *BB : LatchesBlocks) { TerminatorInst *TI = BB->getTerminator(); - MDNode *MD = nullptr; + MDNode *MD = TI->getMetadata(LLVMContext::MD_loop); - // Check if this terminator branches to the loop header. - for (BasicBlock *Successor : successors(TI)) { - if (Successor == H) { - MD = TI->getMetadata(LLVMContext::MD_loop); - break; - } - } if (!MD) return nullptr; diff --git a/llvm/unittests/Analysis/LoopInfoTest.cpp b/llvm/unittests/Analysis/LoopInfoTest.cpp index 647ce8a3c1b..240785f9eb9 100644 --- a/llvm/unittests/Analysis/LoopInfoTest.cpp +++ b/llvm/unittests/Analysis/LoopInfoTest.cpp @@ -82,6 +82,62 @@ TEST(LoopInfoTest, LoopWithSingleLatch) { }); } +// Test loop id handling for a loop with multiple latches. +TEST(LoopInfoTest, LoopWithMultipleLatches) { + const char *ModuleStr = + "target datalayout = \"e-m:o-i64:64-f80:128-n8:16:32:64-S128\"\n" + "define void @foo(i32 %n) {\n" + "entry:\n" + " br i1 undef, label %for.cond, label %for.end\n" + "for.cond:\n" + " %i.0 = phi i32 [ 0, %entry ], [ %inc, %latch.1 ], [ %inc, %latch.2 ]\n" + " %inc = add nsw i32 %i.0, 1\n" + " %cmp = icmp slt i32 %i.0, %n\n" + " br i1 %cmp, label %latch.1, label %for.end\n" + "latch.1:\n" + " br i1 undef, label %for.cond, label %latch.2, !llvm.loop !0\n" + "latch.2:\n" + " br label %for.cond, !llvm.loop !0\n" + "for.end:\n" + " ret void\n" + "}\n" + "!0 = distinct !{!0, !1}\n" + "!1 = !{!\"llvm.loop.distribute.enable\", i1 true}\n"; + + // Parse the module. + LLVMContext Context; + std::unique_ptr<Module> M = makeLLVMModule(Context, ModuleStr); + + runWithLoopInfo(*M, "foo", [&](Function &F, LoopInfo &LI) { + Function::iterator FI = F.begin(); + F.dump(); + // First basic block is entry - skip it. + BasicBlock *Header = &*(++FI); + assert(Header->getName() == "for.cond"); + Loop *L = LI.getLoopFor(Header); + EXPECT_NE(L, nullptr); + + // This loop is not in simplified form. + EXPECT_FALSE(L->isLoopSimplifyForm()); + + // Try to get and set the metadata id for the loop. + MDNode *OldLoopID = L->getLoopID(); + EXPECT_NE(OldLoopID, nullptr); + + MDNode *NewLoopID = MDNode::get(Context, {nullptr}); + // Set operand 0 to refer to the loop id itself. + NewLoopID->replaceOperandWith(0, NewLoopID); + + L->setLoopID(NewLoopID); + EXPECT_EQ(L->getLoopID(), NewLoopID); + EXPECT_NE(L->getLoopID(), OldLoopID); + + L->setLoopID(OldLoopID); + EXPECT_EQ(L->getLoopID(), OldLoopID); + EXPECT_NE(L->getLoopID(), NewLoopID); + }); +} + TEST(LoopInfoTest, PreorderTraversals) { const char *ModuleStr = "define void @f() {\n" "entry:\n" |

