blob: a1c4814883c93be8dc7fc21e1e32924293f1cc20 [file] [log] [blame] [edit]
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt < %s -passes='lower-switch,unify-loop-exits' -S | FileCheck %s
define void @loop_1(i1 %PredEntry, i1 %PredB, i1 %PredC, i1 %PredD) {
; CHECK-LABEL: @loop_1(
; CHECK-NEXT: entry:
; CHECK-NEXT: br i1 [[PREDENTRY:%.*]], label [[A:%.*]], label [[G:%.*]]
; CHECK: A:
; CHECK-NEXT: br label [[B:%.*]]
; CHECK: B:
; CHECK-NEXT: br i1 [[PREDB:%.*]], label [[C:%.*]], label [[LOOP_EXIT_GUARD:%.*]]
; CHECK: C:
; CHECK-NEXT: br i1 [[PREDC:%.*]], label [[D:%.*]], label [[LOOP_EXIT_GUARD]]
; CHECK: D:
; CHECK-NEXT: br i1 [[PREDD:%.*]], label [[A]], label [[LOOP_EXIT_GUARD]]
; CHECK: E:
; CHECK-NEXT: br label [[EXIT:%.*]]
; CHECK: F:
; CHECK-NEXT: br label [[EXIT]]
; CHECK: G:
; CHECK-NEXT: br label [[Y:%.*]]
; CHECK: exit:
; CHECK-NEXT: ret void
; CHECK: loop.exit.guard:
; CHECK-NEXT: [[GUARD_X:%.*]] = phi i1 [ true, [[B]] ], [ false, [[C]] ], [ false, [[D]] ]
; CHECK-NEXT: br i1 [[GUARD_X]], label [[X:%.*]], label [[Y]]
;
entry:
br i1 %PredEntry, label %A, label %G
A:
br label %B
B:
br i1 %PredB, label %C, label %E
C:
br i1 %PredC, label %D, label %F
D:
br i1 %PredD, label %A, label %F
E:
br label %exit
F:
br label %exit
G:
br label %F
exit:
ret void
}
define void @loop_1_callbr(i1 %PredEntry, i1 %PredB, i1 %PredC, i1 %PredD) {
; CHECK-LABEL: @loop_1_callbr(
; CHECK-NEXT: entry:
; CHECK-NEXT: br i1 [[PREDENTRY:%.*]], label [[A:%.*]], label [[G:%.*]]
; CHECK: A:
; CHECK-NEXT: br label [[B:%.*]]
; CHECK: B:
; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDB:%.*]])
; CHECK-NEXT: to label [[C:%.*]] [label [[B_TARGET_E:%.*]]]
; CHECK: C:
; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDC:%.*]])
; CHECK-NEXT: to label [[D:%.*]] [label [[C_TARGET_F:%.*]]]
; CHECK: D:
; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDD:%.*]])
; CHECK-NEXT: to label [[A]] [label [[D_TARGET_F:%.*]]]
; CHECK: E:
; CHECK-NEXT: br label [[EXIT:%.*]]
; CHECK: F:
; CHECK-NEXT: br label [[EXIT]]
; CHECK: G:
; CHECK-NEXT: br label [[Y:%.*]]
; CHECK: exit:
; CHECK-NEXT: ret void
; CHECK: B.target.E:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD:%.*]]
; CHECK: C.target.F:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]]
; CHECK: D.target.F:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]]
; CHECK: loop.exit.guard:
; CHECK-NEXT: [[GUARD_X:%.*]] = phi i1 [ true, [[B_TARGET_E]] ], [ false, [[C_TARGET_F]] ], [ false, [[D_TARGET_F]] ]
; CHECK-NEXT: br i1 [[GUARD_X]], label [[X:%.*]], label [[Y]]
;
entry:
br i1 %PredEntry, label %A, label %G
A:
br label %B
B:
callbr void asm "", "r,!i"(i1 %PredB) to label %C [label %E]
C:
callbr void asm "", "r,!i"(i1 %PredC) to label %D [label %F]
D:
callbr void asm "", "r,!i"(i1 %PredD) to label %A [label %F]
E:
br label %exit
F:
br label %exit
G:
br label %F
exit:
ret void
}
define void @loop_2(i1 %PredA, i1 %PredB, i1 %PredC) {
; CHECK-LABEL: @loop_2(
; CHECK-NEXT: entry:
; CHECK-NEXT: br label [[A:%.*]]
; CHECK: A:
; CHECK-NEXT: br i1 [[PREDA:%.*]], label [[B:%.*]], label [[LOOP_EXIT_GUARD:%.*]]
; CHECK: B:
; CHECK-NEXT: br i1 [[PREDB:%.*]], label [[C:%.*]], label [[LOOP_EXIT_GUARD]]
; CHECK: C:
; CHECK-NEXT: br i1 [[PREDC:%.*]], label [[D:%.*]], label [[LOOP_EXIT_GUARD]]
; CHECK: D:
; CHECK-NEXT: br label [[A]]
; CHECK: X:
; CHECK-NEXT: br label [[EXIT:%.*]]
; CHECK: Y:
; CHECK-NEXT: br label [[EXIT]]
; CHECK: Z:
; CHECK-NEXT: br label [[EXIT]]
; CHECK: exit:
; CHECK-NEXT: ret void
; CHECK: loop.exit.guard:
; CHECK-NEXT: [[GUARD_X:%.*]] = phi i1 [ true, [[A]] ], [ false, [[B]] ], [ false, [[C]] ]
; CHECK-NEXT: [[GUARD_Y:%.*]] = phi i1 [ false, [[A]] ], [ true, [[B]] ], [ false, [[C]] ]
; CHECK-NEXT: br i1 [[GUARD_X]], label [[X:%.*]], label [[LOOP_EXIT_GUARD1:%.*]]
; CHECK: loop.exit.guard1:
; CHECK-NEXT: br i1 [[GUARD_Y]], label [[Y:%.*]], label [[Z:%.*]]
;
entry:
br label %A
A:
br i1 %PredA, label %B, label %X
B:
br i1 %PredB, label %C, label %Y
C:
br i1 %PredC, label %D, label %Z
D:
br label %A
X:
br label %exit
Y:
br label %exit
Z:
br label %exit
exit:
ret void
}
define void @loop_2_callbr(i1 %PredA, i1 %PredB, i1 %PredC) {
; CHECK-LABEL: @loop_2_callbr(
; CHECK-NEXT: entry:
; CHECK-NEXT: br label [[A:%.*]]
; CHECK: A:
; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDA:%.*]])
; CHECK-NEXT: to label [[B:%.*]] [label [[A_TARGET_X:%.*]]]
; CHECK: B:
; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDB:%.*]])
; CHECK-NEXT: to label [[C:%.*]] [label [[B_TARGET_Y:%.*]]]
; CHECK: C:
; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDC:%.*]])
; CHECK-NEXT: to label [[D:%.*]] [label [[C_TARGET_Z:%.*]]]
; CHECK: D:
; CHECK-NEXT: br label [[A]]
; CHECK: X:
; CHECK-NEXT: br label [[EXIT:%.*]]
; CHECK: Y:
; CHECK-NEXT: br label [[EXIT]]
; CHECK: Z:
; CHECK-NEXT: br label [[EXIT]]
; CHECK: exit:
; CHECK-NEXT: ret void
; CHECK: A.target.X:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD:%.*]]
; CHECK: B.target.Y:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]]
; CHECK: C.target.Z:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]]
; CHECK: loop.exit.guard:
; CHECK-NEXT: [[GUARD_X:%.*]] = phi i1 [ true, [[A_TARGET_X]] ], [ false, [[B_TARGET_Y]] ], [ false, [[C_TARGET_Z]] ]
; CHECK-NEXT: [[GUARD_Y:%.*]] = phi i1 [ false, [[A_TARGET_X]] ], [ true, [[B_TARGET_Y]] ], [ false, [[C_TARGET_Z]] ]
; CHECK-NEXT: br i1 [[GUARD_X]], label [[X:%.*]], label [[LOOP_EXIT_GUARD1:%.*]]
; CHECK: loop.exit.guard1:
; CHECK-NEXT: br i1 [[GUARD_Y]], label [[Y:%.*]], label [[Z:%.*]]
;
entry:
br label %A
A:
callbr void asm "", "r,!i"(i1 %PredA) to label %B [label %X]
B:
callbr void asm "", "r,!i"(i1 %PredB) to label %C [label %Y]
C:
callbr void asm "", "r,!i"(i1 %PredC) to label %D [label %Z]
D:
br label %A
X:
br label %exit
Y:
br label %exit
Z:
br label %exit
exit:
ret void
}
; Test that UnifyLoopExits handles callbr with duplicate successors correctly.
; The exit block appears twice as a successor of the callbr instruction.
define void @callbr_duplicate_successors() {
; CHECK-LABEL: @callbr_duplicate_successors(
; CHECK-NEXT: entry:
; CHECK-NEXT: br label [[LOOP:%.*]]
; CHECK: loop:
; CHECK-NEXT: callbr void asm sideeffect "", "!i,!i"()
; CHECK-NEXT: to label [[LOOP_TARGET_EXIT:%.*]] [label [[LOOP]], label [[LOOP_TARGET_EXIT1:%.*]]]
; CHECK: exit:
; CHECK-NEXT: ret void
; CHECK: loop.target.exit:
; CHECK-NEXT: br label [[EXIT:%.*]]
; CHECK: loop.target.exit1:
; CHECK-NEXT: br label [[EXIT]]
;
entry:
br label %loop
loop:
callbr void asm sideeffect "", "!i,!i"()
to label %exit [label %loop, label %exit]
exit:
ret void
}