mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-26 04:32:44 +01:00
[LCSSA] Post-process PHI-nodes created by SSAUpdate when constructing LCSSA form.
Summary: SSAUpdate might insert PHI-nodes inside loops, which can break LCSSA form unless we fix it up. This fixes PR28424. Reviewers: sanjoy, chandlerc, hfinkel Subscribers: uabelho, llvm-commits Differential Revision: http://reviews.llvm.org/D21997 llvm-svn: 275883
This commit is contained in:
parent
f284f67d10
commit
12b86f8410
@ -115,7 +115,8 @@ bool llvm::formLCSSAForInstructions(SmallVectorImpl<Instruction *> &Worklist,
|
|||||||
SmallVector<PHINode *, 16> AddedPHIs;
|
SmallVector<PHINode *, 16> AddedPHIs;
|
||||||
SmallVector<PHINode *, 8> PostProcessPHIs;
|
SmallVector<PHINode *, 8> PostProcessPHIs;
|
||||||
|
|
||||||
SSAUpdater SSAUpdate;
|
SmallVector<PHINode *, 4> InsertedPHIs;
|
||||||
|
SSAUpdater SSAUpdate(&InsertedPHIs);
|
||||||
SSAUpdate.Initialize(I->getType(), I->getName());
|
SSAUpdate.Initialize(I->getType(), I->getName());
|
||||||
|
|
||||||
// Insert the LCSSA phi's into all of the exit blocks dominated by the
|
// Insert the LCSSA phi's into all of the exit blocks dominated by the
|
||||||
@ -184,6 +185,14 @@ bool llvm::formLCSSAForInstructions(SmallVectorImpl<Instruction *> &Worklist,
|
|||||||
|
|
||||||
// Otherwise, do full PHI insertion.
|
// Otherwise, do full PHI insertion.
|
||||||
SSAUpdate.RewriteUse(*UseToRewrite);
|
SSAUpdate.RewriteUse(*UseToRewrite);
|
||||||
|
|
||||||
|
// SSAUpdater might have inserted phi-nodes inside other loops. We'll need
|
||||||
|
// to post-process them to keep LCSSA form.
|
||||||
|
for (PHINode *InsertedPN : InsertedPHIs) {
|
||||||
|
if (auto *OtherLoop = LI.getLoopFor(InsertedPN->getParent()))
|
||||||
|
if (!L->contains(OtherLoop))
|
||||||
|
PostProcessPHIs.push_back(InsertedPN);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Post process PHI instructions that were inserted into another disjoint
|
// Post process PHI instructions that were inserted into another disjoint
|
||||||
|
87
test/Transforms/LCSSA/pr28424.ll
Normal file
87
test/Transforms/LCSSA/pr28424.ll
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
; RUN: opt < %s -lcssa -S -o - | FileCheck %s
|
||||||
|
target triple = "x86_64-unknown-linux-gnu"
|
||||||
|
|
||||||
|
; PR28424
|
||||||
|
; Here LCSSA adds phi-nodes for %x into the loop exits. Then, SSAUpdater needs
|
||||||
|
; to insert phi-nodes to merge these values. That creates a new def, which in
|
||||||
|
; its turn needs another LCCSA phi-node, and this test ensures that we insert
|
||||||
|
; it.
|
||||||
|
|
||||||
|
; CHECK-LABEL: @foo1
|
||||||
|
define internal i32 @foo1() {
|
||||||
|
entry:
|
||||||
|
br label %header
|
||||||
|
|
||||||
|
header:
|
||||||
|
%x = add i32 0, 1
|
||||||
|
br i1 undef, label %if, label %loopexit1
|
||||||
|
|
||||||
|
if:
|
||||||
|
br i1 undef, label %latch, label %loopexit2
|
||||||
|
|
||||||
|
latch:
|
||||||
|
br i1 undef, label %header, label %loopexit3
|
||||||
|
|
||||||
|
; CHECK: loopexit1:
|
||||||
|
; CHECK: %x.lcssa = phi i32 [ %x, %header ]
|
||||||
|
loopexit1:
|
||||||
|
br label %loop_with_insert_point
|
||||||
|
|
||||||
|
; CHECK: loopexit2:
|
||||||
|
; CHECK: %x.lcssa1 = phi i32 [ %x, %if ]
|
||||||
|
loopexit2:
|
||||||
|
br label %exit
|
||||||
|
|
||||||
|
; CHECK: loopexit3:
|
||||||
|
; CHECK: %x.lcssa2 = phi i32 [ %x, %latch ]
|
||||||
|
loopexit3:
|
||||||
|
br label %loop_with_insert_point
|
||||||
|
|
||||||
|
; CHECK: loop_with_insert_point:
|
||||||
|
; CHECK: %x4 = phi i32 [ %x4, %loop_with_insert_point ], [ %x.lcssa2, %loopexit3 ], [ %x.lcssa, %loopexit1 ]
|
||||||
|
loop_with_insert_point:
|
||||||
|
br i1 undef, label %loop_with_insert_point, label %bb
|
||||||
|
|
||||||
|
; CHECK: bb:
|
||||||
|
; CHECK: %x4.lcssa = phi i32 [ %x4, %loop_with_insert_point ]
|
||||||
|
bb:
|
||||||
|
br label %exit
|
||||||
|
|
||||||
|
; CHECK: exit:
|
||||||
|
; CHECK: %x3 = phi i32 [ %x4.lcssa, %bb ], [ %x.lcssa1, %loopexit2 ]
|
||||||
|
exit:
|
||||||
|
ret i32 %x
|
||||||
|
}
|
||||||
|
|
||||||
|
; CHECK-LABEL: @foo2
|
||||||
|
define internal i32 @foo2() {
|
||||||
|
entry:
|
||||||
|
br label %header
|
||||||
|
|
||||||
|
header:
|
||||||
|
%x = add i32 0, 1
|
||||||
|
br i1 undef, label %latch, label %loopexit1
|
||||||
|
|
||||||
|
latch:
|
||||||
|
br i1 undef, label %header, label %loopexit2
|
||||||
|
|
||||||
|
; CHECK: loopexit1:
|
||||||
|
; CHECK: %x.lcssa = phi i32 [ %x, %header ]
|
||||||
|
loopexit1:
|
||||||
|
br label %loop_with_insert_point
|
||||||
|
|
||||||
|
; CHECK: loopexit2:
|
||||||
|
; CHECK: %x.lcssa1 = phi i32 [ %x, %latch ]
|
||||||
|
loopexit2:
|
||||||
|
br label %loop_with_insert_point
|
||||||
|
|
||||||
|
; CHECK: loop_with_insert_point:
|
||||||
|
; CHECK: %x2 = phi i32 [ %x2, %loop_with_insert_point ], [ %x.lcssa1, %loopexit2 ], [ %x.lcssa, %loopexit1 ]
|
||||||
|
loop_with_insert_point:
|
||||||
|
br i1 undef, label %loop_with_insert_point, label %exit
|
||||||
|
|
||||||
|
; CHECK: exit:
|
||||||
|
; CHECK: %x2.lcssa = phi i32 [ %x2, %loop_with_insert_point ]
|
||||||
|
exit:
|
||||||
|
ret i32 %x
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user