From 8e7e76729d5f39856bd829f36a7f5a42217ac90e Mon Sep 17 00:00:00 2001 From: Michael Zolotukhin Date: Wed, 8 Jun 2016 23:13:21 +0000 Subject: [PATCH] [LoopSimplify] Preserve LCSSA when merging exit blocks. Summary: This fixes PR26682. Also add LCSSA as a preserved pass to LoopSimplify, that looks correct to me and allows to write a test for the issue. Reviewers: chandlerc, bogner, sanjoy Subscribers: llvm-commits Differential Revision: http://reviews.llvm.org/D21112 llvm-svn: 272224 --- llvm/lib/Transforms/Utils/LoopSimplify.cpp | 23 ++++++++++++-- llvm/test/Transforms/LoopSimplify/pr26682.ll | 32 ++++++++++++++++++++ 2 files changed, 53 insertions(+), 2 deletions(-) create mode 100644 llvm/test/Transforms/LoopSimplify/pr26682.ll diff --git a/llvm/lib/Transforms/Utils/LoopSimplify.cpp b/llvm/lib/Transforms/Utils/LoopSimplify.cpp index 5ef7c4aaa2ae..96ba9b4d1d67 100644 --- a/llvm/lib/Transforms/Utils/LoopSimplify.cpp +++ b/llvm/lib/Transforms/Utils/LoopSimplify.cpp @@ -686,8 +686,10 @@ ReprocessLoop: } DT->eraseNode(ExitingBlock); - BI->getSuccessor(0)->removePredecessor(ExitingBlock); - BI->getSuccessor(1)->removePredecessor(ExitingBlock); + BI->getSuccessor(0)->removePredecessor( + ExitingBlock, /* DontDeleteUselessPHIs */ PreserveLCSSA); + BI->getSuccessor(1)->removePredecessor( + ExitingBlock, /* DontDeleteUselessPHIs */ PreserveLCSSA); ExitingBlock->eraseFromParent(); } } @@ -748,6 +750,7 @@ namespace { AU.addPreserved(); AU.addPreserved(); AU.addPreserved(); + AU.addPreservedID(LCSSAID); AU.addPreserved(); AU.addPreservedID(BreakCriticalEdgesID); // No critical edges added. } @@ -781,11 +784,27 @@ bool LoopSimplify::runOnFunction(Function &F) { SE = SEWP ? &SEWP->getSE() : nullptr; AC = &getAnalysis().getAssumptionCache(F); bool PreserveLCSSA = mustPreserveAnalysisID(LCSSAID); +#ifndef NDEBUG + if (PreserveLCSSA) { + assert(DT && "DT not available."); + assert(LI && "LI not available."); + bool InLCSSA = + all_of(*LI, [&](Loop *L) { return L->isRecursivelyLCSSAForm(*DT); }); + assert(InLCSSA && "Requested to preserve LCSSA, but it's already broken."); + } +#endif // Simplify each loop nest in the function. for (LoopInfo::iterator I = LI->begin(), E = LI->end(); I != E; ++I) Changed |= simplifyLoop(*I, DT, LI, SE, AC, PreserveLCSSA); +#ifndef NDEBUG + if (PreserveLCSSA) { + bool InLCSSA = + all_of(*LI, [&](Loop *L) { return L->isRecursivelyLCSSAForm(*DT); }); + assert(InLCSSA && "LCSSA is broken after loop-simplify."); + } +#endif return Changed; } diff --git a/llvm/test/Transforms/LoopSimplify/pr26682.ll b/llvm/test/Transforms/LoopSimplify/pr26682.ll new file mode 100644 index 000000000000..092c0c3f0b04 --- /dev/null +++ b/llvm/test/Transforms/LoopSimplify/pr26682.ll @@ -0,0 +1,32 @@ +; RUN: opt < %s -lcssa -loop-simplify -indvars -S | FileCheck %s +target datalayout = "e-m:o-i64:64-f80:128-n8:16:32:64-S128" +target triple = "x86_64-unknown-unknown" + +@a = external global i32, align 4 + +; Check that loop-simplify merges two loop exits, but preserves LCSSA form. +; CHECK-LABEL: @foo +; CHECK: for: +; CHECK: %or.cond = and i1 %cmp1, %cmp2 +; CHECK-NOT: for.cond: +; CHECK: for.end: +; CHECK: %a.lcssa = phi i32 [ %a, %for ] +define i32 @foo(i32 %x) { +entry: + br label %for + +for: + %iv = phi i32 [ 0, %entry ], [ %iv.next, %for.cond ] + %cmp1 = icmp eq i32 %x, 0 + %iv.next = add nuw nsw i32 %iv, 1 + %a = load i32, i32* @a + br i1 %cmp1, label %for.cond, label %for.end + +for.cond: + %cmp2 = icmp slt i32 %iv.next, 4 + br i1 %cmp2, label %for, label %for.end + +for.end: + %a.lcssa = phi i32 [ %a, %for ], [ %a, %for.cond ] + ret i32 %a.lcssa +}