; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt %s -mtriple=x86_64-- -codegenprepare -S | FileCheck %s
@g = global i32 0
@effect = global i32 0
define void @switch_phi_const(i32 %x) {
; CHECK-LABEL: @switch_phi_const(
; CHECK-NEXT: bb0:
; CHECK-NEXT: switch i32 [[X:%.*]], label [[DEFAULT:%.*]] [
; CHECK-NEXT: i32 13, label [[CASE_13:%.*]]
; CHECK-NEXT: i32 42, label [[CASE_42:%.*]]
; CHECK-NEXT: i32 50, label [[CASE_50_51:%.*]]
; CHECK-NEXT: i32 51, label [[CASE_50_51]]
; CHECK-NEXT: i32 55, label [[CASE_55:%.*]]
; CHECK-NEXT: i32 7, label [[CASE_7:%.*]]
; CHECK-NEXT: ]
; CHECK: case_13:
; CHECK-NEXT: [[X0:%.*]] = phi i32 [ [[X]], [[BB0:%.*]] ], [ [[X_LOOPBACK:%.*]], [[CASE_7]] ]
; CHECK-NEXT: store i32 13, ptr @effect, align 4
; CHECK-NEXT: br label [[CASE_42]]
; CHECK: case_42:
; CHECK-NEXT: [[X1:%.*]] = phi i32 [ [[X]], [[BB0]] ], [ [[X0]], [[CASE_13]] ]
; CHECK-NEXT: store i32 [[X1]], ptr @effect, align 4
; CHECK-NEXT: br label [[CASE_50_51]]
; CHECK: case_50_51:
; CHECK-NEXT: [[X2:%.*]] = phi i32 [ 50, [[BB0]] ], [ 50, [[BB0]] ], [ [[X1]], [[CASE_42]] ]
; CHECK-NEXT: [[X2_2:%.*]] = phi i32 [ 51, [[BB0]] ], [ 51, [[BB0]] ], [ [[X1]], [[CASE_42]] ]
; CHECK-NEXT: store i32 [[X2]], ptr @effect, align 4
; CHECK-NEXT: store i32 [[X2_2]], ptr @effect, align 4
; CHECK-NEXT: br label [[CASE_55]]
; CHECK: case_55:
; CHECK-NEXT: [[X3:%.*]] = phi i32 [ 42, [[BB0]] ], [ 55, [[CASE_50_51]] ]
; CHECK-NEXT: store i32 [[X3]], ptr @effect, align 4
; CHECK-NEXT: br label [[DEFAULT]]
; CHECK: case_7:
; CHECK-NEXT: [[X_LOOPBACK]] = load i32, ptr @g, align 4
; CHECK-NEXT: store i32 7, ptr @effect, align 4
; CHECK-NEXT: br label [[CASE_13]]
; CHECK: default:
; CHECK-NEXT: ret void
;
bb0:
switch i32 %x, label %default [
i32 13, label %case_13
i32 42, label %case_42
i32 50, label %case_50_51
i32 51, label %case_50_51
i32 55, label %case_55
i32 7, label %case_7
]
case_13:
; We should replace 13 with %x
%x0 = phi i32 [ 13, %bb0 ], [ %x_loopback, %case_7 ]
store i32 13, ptr @effect, align 4
br label %case_42
case_42:
; We should replace 42 with %x
%x1 = phi i32 [ 42, %bb0 ], [ %x0, %case_13 ]
store i32 %x1, ptr @effect, align 4
br label %case_50_51
case_50_51:
; Must not replace the PHI argument: Case values 50 and 51 jump here.
%x2 = phi i32 [ 50, %bb0 ], [ 50, %bb0 ], [ %x1, %case_42 ]
%x2.2 = phi i32 [ 51, %bb0 ], [ 51, %bb0 ], [ %x1, %case_42 ]
store i32 %x2, ptr @effect, align 4
store i32 %x2.2, ptr @effect, align 4
br label %case_55
case_55:
; We must not replace any of the PHI arguments:
; - 42 is the wrong constant
; - %case_42 is not the switch predecessor block.
%x3 = phi i32 [ 42, %bb0 ], [ 55, %case_50_51 ]
store i32 %x3, ptr @effect, align 4
br label %default
case_7:
%x_loopback = load i32, ptr @g, align 4
store i32 7, ptr @effect, align 4
br label %case_13
default:
ret void
}
define void @switch_phi_const_multiple_phis(i32 %x, i1 %c) {
; CHECK-LABEL: @switch_phi_const_multiple_phis(
; CHECK-NEXT: bb0:
; CHECK-NEXT: br i1 [[C:%.*]], label [[BB1:%.*]], label [[CASE_13:%.*]]
; CHECK: bb1:
; CHECK-NEXT: [[COND:%.*]] = icmp eq i32 [[X:%.*]], 13
; CHECK-NEXT: br i1 [[COND]], label [[CASE_13]], label [[DEFAULT:%.*]]
; CHECK: case_13:
; CHECK-NEXT: [[X0:%.*]] = phi i32 [ [[X]], [[BB1]] ], [ 1, [[BB0:%.*]] ]
; CHECK-NEXT: [[N0:%.*]] = phi i32 [ 14, [[BB1]] ], [ 1, [[BB0]] ]
; CHECK-NEXT: [[X1:%.*]] = phi i32 [ 27, [[BB0]] ], [ [[X]], [[BB1]] ]
; CHECK-NEXT: store volatile i32 [[X0]], ptr @effect, align 4
; CHECK-NEXT: store volatile i32 [[N0]], ptr @effect, align 4
; CHECK-NEXT: store volatile i32 [[X1]], ptr @effect, align 4
; CHECK-NEXT: ret void
; CHECK: default:
; CHECK-NEXT: ret void
;
bb0:
br i1 %c, label %bb1, label %case_13
bb1:
switch i32 %x, label %default [
i32 13, label %case_13
]
case_13:
; Check that replacement works for multiple PHIs.
; Should perform replacement for %x0, %x1 but not %n0
%x0 = phi i32 [13, %bb1], [1, %bb0]
%n0 = phi i32 [14, %bb1], [1, %bb0]
%x1 = phi i32 [27, %bb0], [13, %bb1]
store volatile i32 %x0, ptr @effect, align 4
store volatile i32 %n0, ptr @effect, align 4
store volatile i32 %x1, ptr @effect, align 4
ret void
default:
ret void
}
define void @switch_phi_const_degenerate(i1 %c) {
; CHECK-LABEL: @switch_phi_const_degenerate(
; CHECK-NEXT: bb0:
; CHECK-NEXT: br i1 [[C:%.*]], label [[CASE_42:%.*]], label [[BB1:%.*]]
; CHECK: bb1:
; CHECK-NEXT: br label [[CASE_42]]
; CHECK: case_42:
; CHECK-NEXT: [[X:%.*]] = phi i32 [ 0, [[BB0:%.*]] ], [ 42, [[BB1]] ]
; CHECK-NEXT: store volatile i32 [[X]], ptr @effect, align 4
; CHECK-NEXT: ret void
;
bb0:
br i1 %c, label %case_42, label %bb1
bb1:
switch i32 42, label %unreachable [
i32 42, label %case_42
]
case_42:
; We should not end up in an endless loop 42 with the switch condition 42.
%x = phi i32 [0, %bb0], [42, %bb1]
store volatile i32 %x, ptr @effect, align 4
ret void
unreachable:
unreachable
}
@g64 = global i64 0
@effect64 = global i64 0
define void @switch_trunc_phi_const(i32 %x) {
; CHECK-LABEL: @switch_trunc_phi_const(
; CHECK-NEXT: bb0:
; CHECK-NEXT: [[TMP0:%.*]] = zext i32 [[X:%.*]] to i64
; CHECK-NEXT: [[TMP1:%.*]] = zext i32 [[X]] to i64
; CHECK-NEXT: switch i32 [[X]], label [[DEFAULT:%.*]] [
; CHECK-NEXT: i32 13, label [[CASE_13:%.*]]
; CHECK-NEXT: i32 42, label [[CASE_42:%.*]]
; CHECK-NEXT: i32 55, label [[CASE_55:%.*]]
; CHECK-NEXT: i32 7, label [[CASE_7:%.*]]
; CHECK-NEXT: ]
; CHECK: case_13:
; CHECK-NEXT: [[X0:%.*]] = phi i64 [ [[TMP0]], [[BB0:%.*]] ], [ [[X7:%.*]], [[CASE_7]] ]
; CHECK-NEXT: store i64 13, ptr @effect64, align 4
; CHECK-NEXT: br label [[CASE_42]]
; CHECK: case_42:
; CHECK-NEXT: [[X1:%.*]] = phi i64 [ [[TMP1]], [[BB0]] ], [ [[X0]], [[CASE_13]] ]
; CHECK-NEXT: store i64 [[X1]], ptr @effect64, align 4
; CHECK-NEXT: br label [[CASE_55]]
; CHECK: case_55:
; CHECK-NEXT: [[X2:%.*]] = phi i64 [ 3895, [[BB0]] ], [ 55, [[CASE_42]] ]
; CHECK-NEXT: store i64 [[X2]], ptr @effect64, align 4
; CHECK-NEXT: br label [[DEFAULT]]
; CHECK: case_7:
; CHECK-NEXT: [[X7]] = load i64, ptr @g64, align 4
; CHECK-NEXT: store i64 7, ptr @effect64, align 4
; CHECK-NEXT: br label [[CASE_13]]
; CHECK: default:
; CHECK-NEXT: ret void
;
bb0:
switch i32 %x, label %default [
i32 13, label %case_13
i32 42, label %case_42
i32 55, label %case_55
i32 7, label %case_7
]
case_13:
; We should replace 13 with zext %x to i64
%x0 = phi i64 [ 13, %bb0 ], [ %x7, %case_7 ]
store i64 13, ptr @effect64, align 4
br label %case_42
case_42:
; We should replace 42 with zext i32 %x to i64
%x1 = phi i64 [ 42, %bb0 ], [ %x0, %case_13 ]
store i64 %x1, ptr @effect64, align 4
br label %case_55
case_55:
; We must not replace any of the PHI arguments! (3898 == 0xf00 + 55)
%x2 = phi i64 [ 3895, %bb0 ], [ 55, %case_42 ]
store i64 %x2, ptr @effect64, align 4
br label %default
case_7:
%x7 = load i64, ptr @g64, align 4
store i64 7, ptr @effect64, align 4
br label %case_13
default:
ret void
}