forked from OSchip/llvm-project
[SimpleLoopUnswitch] After unswitch delete dead blocks in parent loops
Summary: Assert from PR38737 happens on the dead block inside the parent loop after unswitching nontrivial switch in the inner loop. deleteDeadBlocksFromLoop now takes extra care to detect/remove dead blocks in all the parent loops in addition to the blocks from original loop being unswitched. Reviewers: asbirlea, chandlerc Reviewed By: asbirlea Subscribers: llvm-commits Differential Revision: https://reviews.llvm.org/D51415 llvm-svn: 340955
This commit is contained in:
parent
cf311cfc20
commit
7b49aa03af
|
@ -1380,13 +1380,21 @@ deleteDeadBlocksFromLoop(Loop &L,
|
||||||
DominatorTree &DT, LoopInfo &LI) {
|
DominatorTree &DT, LoopInfo &LI) {
|
||||||
// Find all the dead blocks, and remove them from their successors.
|
// Find all the dead blocks, and remove them from their successors.
|
||||||
SmallVector<BasicBlock *, 16> DeadBlocks;
|
SmallVector<BasicBlock *, 16> DeadBlocks;
|
||||||
for (BasicBlock *BB : llvm::concat<BasicBlock *const>(L.blocks(), ExitBlocks))
|
for (BasicBlock *BB : ExitBlocks)
|
||||||
if (!DT.isReachableFromEntry(BB)) {
|
if (!DT.isReachableFromEntry(BB)) {
|
||||||
for (BasicBlock *SuccBB : successors(BB))
|
for (BasicBlock *SuccBB : successors(BB))
|
||||||
SuccBB->removePredecessor(BB);
|
SuccBB->removePredecessor(BB);
|
||||||
DeadBlocks.push_back(BB);
|
DeadBlocks.push_back(BB);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (Loop *ParentL = &L; ParentL; ParentL = ParentL->getParentLoop())
|
||||||
|
for (BasicBlock *BB : ParentL->blocks())
|
||||||
|
if (!DT.isReachableFromEntry(BB)) {
|
||||||
|
for (BasicBlock *SuccBB : successors(BB))
|
||||||
|
SuccBB->removePredecessor(BB);
|
||||||
|
DeadBlocks.push_back(BB);
|
||||||
|
}
|
||||||
|
|
||||||
SmallPtrSet<BasicBlock *, 16> DeadBlockSet(DeadBlocks.begin(),
|
SmallPtrSet<BasicBlock *, 16> DeadBlockSet(DeadBlocks.begin(),
|
||||||
DeadBlocks.end());
|
DeadBlocks.end());
|
||||||
|
|
||||||
|
@ -1431,7 +1439,7 @@ deleteDeadBlocksFromLoop(Loop &L,
|
||||||
|
|
||||||
// Actually delete the blocks now that they've been fully unhooked from the
|
// Actually delete the blocks now that they've been fully unhooked from the
|
||||||
// IR.
|
// IR.
|
||||||
for (auto *BB : DeadBlocks)
|
for (auto *BB : DeadBlockSet)
|
||||||
BB->eraseFromParent();
|
BB->eraseFromParent();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,45 @@
|
||||||
|
; RUN: opt < %s -simple-loop-unswitch -enable-nontrivial-unswitch -S 2>&1 | FileCheck %s
|
||||||
|
; RUN: opt < %s -passes=unswitch -enable-nontrivial-unswitch -S 2>&1 | FileCheck %s
|
||||||
|
;
|
||||||
|
; Checking that (dead) blocks from inner loop are deleted after unswitch.
|
||||||
|
;
|
||||||
|
declare void @foo()
|
||||||
|
|
||||||
|
; CHECK-LABEL: @Test
|
||||||
|
define void @Test(i32) {
|
||||||
|
entry:
|
||||||
|
br label %outer
|
||||||
|
outer:
|
||||||
|
%oi = phi i32 [ 0, %entry ], [ %oinc, %outer_continue]
|
||||||
|
br label %inner
|
||||||
|
inner:
|
||||||
|
%ii = phi i32 [ 0, %outer ], [ %iinc, %continue]
|
||||||
|
call void @foo()
|
||||||
|
switch i32 %0, label %get_out2 [
|
||||||
|
i32 0, label %continue
|
||||||
|
i32 1, label %case1
|
||||||
|
i32 2, label %get_out
|
||||||
|
]
|
||||||
|
;
|
||||||
|
; since we unswitch on the above switch, %case1 and %continue blocks
|
||||||
|
; become dead in the original loop
|
||||||
|
;
|
||||||
|
; CHECK-NOT: case1:
|
||||||
|
case1:
|
||||||
|
br label %continue
|
||||||
|
; CHECK-NOT: {{^}}continue:
|
||||||
|
continue:
|
||||||
|
%iinc = add i32 %ii, 1
|
||||||
|
%icmp = icmp eq i32 %ii, 100
|
||||||
|
br i1 %icmp, label %inner, label %outer_continue
|
||||||
|
|
||||||
|
outer_continue:
|
||||||
|
%oinc = add i32 %oi, 1
|
||||||
|
%ocmp = icmp eq i32 %oi, 100
|
||||||
|
br i1 %ocmp, label %outer, label %get_out
|
||||||
|
|
||||||
|
get_out:
|
||||||
|
ret void
|
||||||
|
get_out2:
|
||||||
|
unreachable
|
||||||
|
}
|
Loading…
Reference in New Issue