blob: 2ec576a2efa89dc6831f191ee3e15efc74ad9996 [file] [log] [blame]
; 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 @nested(i1 %PredB3, i1 %PredB4, i1 %PredA4, i1 %PredA3, i32 %X, i32 %Y, i32 %Z) {
; CHECK-LABEL: @nested(
; CHECK-NEXT: entry:
; CHECK-NEXT: br label [[A1:%.*]]
; CHECK: A1:
; CHECK-NEXT: br label [[B1:%.*]]
; CHECK: B1:
; CHECK-NEXT: br label [[B2:%.*]]
; CHECK: B2:
; CHECK-NEXT: [[X_INC:%.*]] = add i32 [[X:%.*]], 1
; CHECK-NEXT: br label [[B3:%.*]]
; CHECK: B3:
; CHECK-NEXT: br i1 [[PREDB3:%.*]], label [[B4:%.*]], label [[LOOP_EXIT_GUARD1:%.*]]
; CHECK: B4:
; CHECK-NEXT: br i1 [[PREDB4:%.*]], label [[B1]], label [[LOOP_EXIT_GUARD1]]
; CHECK: A2:
; CHECK-NEXT: br label [[A4:%.*]]
; CHECK: A3:
; CHECK-NEXT: br label [[A4]]
; CHECK: A4:
; CHECK-NEXT: [[A4_PHI:%.*]] = phi i32 [ [[Y:%.*]], [[A3:%.*]] ], [ [[X_INC_MOVED:%.*]], [[A2:%.*]] ]
; CHECK-NEXT: br i1 [[PREDA4:%.*]], label [[LOOP_EXIT_GUARD:%.*]], label [[A5:%.*]]
; CHECK: A5:
; CHECK-NEXT: br i1 [[PREDA3:%.*]], label [[LOOP_EXIT_GUARD]], label [[A1]]
; CHECK: C:
; CHECK-NEXT: br label [[EXIT:%.*]]
; CHECK: exit:
; CHECK-NEXT: [[EXIT_PHI:%.*]] = phi i32 [ [[Z:%.*]], [[C:%.*]] ], [ [[EXIT_PHI_MOVED:%.*]], [[LOOP_EXIT_GUARD]] ]
; CHECK-NEXT: ret void
; CHECK: loop.exit.guard:
; CHECK-NEXT: [[EXIT_PHI_MOVED]] = phi i32 [ poison, [[A4]] ], [ [[A4_PHI]], [[A5]] ]
; CHECK-NEXT: [[GUARD_C:%.*]] = phi i1 [ true, [[A4]] ], [ false, [[A5]] ]
; CHECK-NEXT: br i1 [[GUARD_C]], label [[C]], label [[EXIT]]
; CHECK: loop.exit.guard1:
; CHECK-NEXT: [[X_INC_MOVED]] = phi i32 [ [[X_INC]], [[B3]] ], [ [[X_INC]], [[B4]] ]
; CHECK-NEXT: [[GUARD_A3:%.*]] = phi i1 [ true, [[B3]] ], [ false, [[B4]] ]
; CHECK-NEXT: br i1 [[GUARD_A3]], label [[A3]], label [[A2]]
;
entry:
br label %A1
A1:
br label %B1
B1:
br label %B2
B2:
%X.inc = add i32 %X, 1
br label %B3
B3:
br i1 %PredB3, label %B4, label %A3
B4:
br i1 %PredB4, label %B1, label %A2
A2:
br label %A4
A3:
br label %A4
A4:
%A4.phi = phi i32 [%Y, %A3], [%X.inc, %A2]
br i1 %PredA4, label %C, label %A5
A5:
br i1 %PredA3, label %exit, label %A1
C:
br label %exit
exit:
%exit.phi = phi i32 [%A4.phi, %A5], [%Z, %C]
ret void
}
define void @nested_callbr(i1 %PredB3, i1 %PredB4, i1 %PredA4, i1 %PredA3, i32 %X, i32 %Y, i32 %Z) {
; CHECK-LABEL: @nested_callbr(
; CHECK-NEXT: entry:
; CHECK-NEXT: br label [[A1:%.*]]
; CHECK: A1:
; CHECK-NEXT: br label [[B1:%.*]]
; CHECK: B1:
; CHECK-NEXT: br label [[B2:%.*]]
; CHECK: B2:
; CHECK-NEXT: [[X_INC:%.*]] = add i32 [[X:%.*]], 1
; CHECK-NEXT: br label [[B3:%.*]]
; CHECK: B3:
; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDB3:%.*]])
; CHECK-NEXT: to label [[B4:%.*]] [label %B3.target.A3]
; CHECK: B4:
; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDB4:%.*]])
; CHECK-NEXT: to label [[B1]] [label %B4.target.A2]
; CHECK: A2:
; CHECK-NEXT: br label [[A4:%.*]]
; CHECK: A3:
; CHECK-NEXT: br label [[A4]]
; CHECK: A4:
; CHECK-NEXT: [[A4_PHI:%.*]] = phi i32 [ [[Y:%.*]], [[A3:%.*]] ], [ [[X_INC_MOVED:%.*]], [[A2:%.*]] ]
; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDA4:%.*]])
; CHECK-NEXT: to label [[A4_TARGET_C:%.*]] [label %A5]
; CHECK: A5:
; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDA3:%.*]])
; CHECK-NEXT: to label [[A5_TARGET_EXIT:%.*]] [label %A1]
; CHECK: C:
; CHECK-NEXT: br label [[EXIT:%.*]]
; CHECK: exit:
; CHECK-NEXT: [[EXIT_PHI:%.*]] = phi i32 [ [[Z:%.*]], [[C:%.*]] ], [ [[EXIT_PHI_MOVED:%.*]], [[LOOP_EXIT_GUARD:%.*]] ]
; CHECK-NEXT: ret void
; CHECK: A4.target.C:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]]
; CHECK: A5.target.exit:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]]
; CHECK: loop.exit.guard:
; CHECK-NEXT: [[EXIT_PHI_MOVED]] = phi i32 [ poison, [[A4_TARGET_C]] ], [ [[A4_PHI]], [[A5_TARGET_EXIT]] ]
; CHECK-NEXT: [[GUARD_C:%.*]] = phi i1 [ true, [[A4_TARGET_C]] ], [ false, [[A5_TARGET_EXIT]] ]
; CHECK-NEXT: br i1 [[GUARD_C]], label [[C]], label [[EXIT]]
; CHECK: B3.target.A3:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD1:%.*]]
; CHECK: B4.target.A2:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD1]]
; CHECK: loop.exit.guard1:
; CHECK-NEXT: [[X_INC_MOVED]] = phi i32 [ [[X_INC]], [[B3_TARGET_A3:%.*]] ], [ [[X_INC]], [[B4_TARGET_A2:%.*]] ]
; CHECK-NEXT: [[GUARD_A3:%.*]] = phi i1 [ true, [[B3_TARGET_A3]] ], [ false, [[B4_TARGET_A2]] ]
; CHECK-NEXT: br i1 [[GUARD_A3]], label [[A3]], label [[A2]]
;
entry:
br label %A1
A1:
br label %B1
B1:
br label %B2
B2:
%X.inc = add i32 %X, 1
br label %B3
B3:
callbr void asm "", "r,!i"(i1 %PredB3) to label %B4 [label %A3]
B4:
callbr void asm "", "r,!i"(i1 %PredB4) to label %B1 [label %A2]
A2:
br label %A4
A3:
br label %A4
A4:
%A4.phi = phi i32 [%Y, %A3], [%X.inc, %A2]
callbr void asm "", "r,!i"(i1 %PredA4) to label %C [label %A5]
A5:
callbr void asm "", "r,!i"(i1 %PredA3) to label %exit [label %A1]
C:
br label %exit
exit:
%exit.phi = phi i32 [%A4.phi, %A5], [%Z, %C]
ret void
}
; Here, the newly created target loop that connects b to r1 needs to be part of
; the parent loop (the outer loop b participates in). Otherwise, it will be
; regarded as an additional loop entry point to this outer loop.
define void @nested_callbr_multiple_exits() {
; CHECK-LABEL: @nested_callbr_multiple_exits(
; CHECK-NEXT: br label [[A:%.*]]
; CHECK: a:
; CHECK-NEXT: callbr void asm "", ""()
; CHECK-NEXT: to label [[B:%.*]] []
; CHECK: b:
; CHECK-NEXT: callbr void asm "", "!i"()
; CHECK-NEXT: to label [[C:%.*]] [label %b.target.b.target.r1]
; CHECK: c:
; CHECK-NEXT: callbr void asm "", "!i"()
; CHECK-NEXT: to label [[C_TARGET_E:%.*]] [label %b]
; CHECK: e:
; CHECK-NEXT: callbr void asm "", "!i"()
; CHECK-NEXT: to label [[A]] [label %e.target.r2]
; CHECK: r1:
; CHECK-NEXT: ret void
; CHECK: r2:
; CHECK-NEXT: ret void
; CHECK: b.target.r1:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD:%.*]]
; CHECK: e.target.r2:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]]
; CHECK: loop.exit.guard:
; CHECK-NEXT: [[GUARD_R1:%.*]] = phi i1 [ true, [[B_TARGET_R1:%.*]] ], [ false, [[E_TARGET_R2:%.*]] ]
; CHECK-NEXT: br i1 [[GUARD_R1]], label [[R1:%.*]], label [[R2:%.*]]
; CHECK: b.target.b.target.r1:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD1:%.*]]
; CHECK: c.target.e:
; CHECK-NEXT: br label [[LOOP_EXIT_GUARD1]]
; CHECK: loop.exit.guard1:
; CHECK-NEXT: [[GUARD_B_TARGET_R1:%.*]] = phi i1 [ true, [[B_TARGET_B_TARGET_R1:%.*]] ], [ false, [[C_TARGET_E]] ]
; CHECK-NEXT: br i1 [[GUARD_B_TARGET_R1]], label [[B_TARGET_R1]], label [[E:%.*]]
;
br label %a
a:
callbr void asm "", ""() to label %b []
b:
callbr void asm "", "!i"() to label %c [label %r1]
c:
callbr void asm "", "!i"() to label %e [label %b]
e:
callbr void asm "", "!i"() to label %a [label %r2]
r1:
ret void
r2:
ret void
}