162 lines
4 KiB
LLVM
162 lines
4 KiB
LLVM
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
|
; RUN: opt -S -passes=ipsccp < %s | FileCheck %s
|
|
|
|
declare void @BB0_f()
|
|
declare void @BB1_f()
|
|
|
|
; Make sure we can eliminate what is in BB0 as we know that the indirectbr is going to BB1.
|
|
;
|
|
define void @indbrtest1() {
|
|
; CHECK-LABEL: @indbrtest1(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: br label [[BB1:%.*]]
|
|
; CHECK: BB1:
|
|
; CHECK-NEXT: call void @BB1_f()
|
|
; CHECK-NEXT: ret void
|
|
;
|
|
entry:
|
|
indirectbr ptr blockaddress(@indbrtest1, %BB1), [label %BB0, label %BB1]
|
|
BB0:
|
|
call void @BB0_f()
|
|
br label %BB1
|
|
BB1:
|
|
call void @BB1_f()
|
|
ret void
|
|
}
|
|
|
|
; Make sure we can eliminate what is in BB0 as we know that the indirectbr is going to BB1
|
|
; by looking through the casts. The casts should be folded away when they are visited
|
|
; before the indirectbr instruction.
|
|
;
|
|
define void @indbrtest2() {
|
|
; CHECK-LABEL: @indbrtest2(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: br label [[BB1:%.*]]
|
|
; CHECK: BB1:
|
|
; CHECK-NEXT: call void @BB1_f()
|
|
; CHECK-NEXT: ret void
|
|
;
|
|
entry:
|
|
%a = ptrtoint ptr blockaddress(@indbrtest2, %BB1) to i64
|
|
%b = inttoptr i64 %a to ptr
|
|
indirectbr ptr %b, [label %BB0, label %BB1]
|
|
BB0:
|
|
call void @BB0_f()
|
|
br label %BB1
|
|
BB1:
|
|
call void @BB1_f()
|
|
ret void
|
|
}
|
|
|
|
; Make sure we can not eliminate BB0 as we do not know the target of the indirectbr.
|
|
|
|
define void @indbrtest3(ptr %Q) {
|
|
; CHECK-LABEL: @indbrtest3(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: [[T:%.*]] = load ptr, ptr [[Q:%.*]], align 8
|
|
; CHECK-NEXT: indirectbr ptr [[T]], [label [[BB0:%.*]], label %BB1]
|
|
; CHECK: BB0:
|
|
; CHECK-NEXT: call void @BB0_f()
|
|
; CHECK-NEXT: br label [[BB1:%.*]]
|
|
; CHECK: BB1:
|
|
; CHECK-NEXT: call void @BB1_f()
|
|
; CHECK-NEXT: ret void
|
|
;
|
|
entry:
|
|
%t = load ptr, ptr %Q
|
|
indirectbr ptr %t, [label %BB0, label %BB1]
|
|
BB0:
|
|
call void @BB0_f()
|
|
br label %BB1
|
|
BB1:
|
|
call void @BB1_f()
|
|
ret void
|
|
}
|
|
|
|
; Branch on undef is UB, so we can convert the indirectbr to unreachable.
|
|
|
|
define void @indbrtest4(ptr %Q) {
|
|
; CHECK-LABEL: @indbrtest4(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: unreachable
|
|
;
|
|
entry:
|
|
indirectbr ptr undef, [label %BB0, label %BB1]
|
|
BB0:
|
|
call void @BB0_f()
|
|
ret void
|
|
BB1:
|
|
call void @BB1_f()
|
|
ret void
|
|
}
|
|
|
|
define internal i32 @indbrtest5(i1 %c) {
|
|
; CHECK-LABEL: @indbrtest5(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: br i1 [[C:%.*]], label [[BB1:%.*]], label [[BB2:%.*]]
|
|
; CHECK: bb1:
|
|
; CHECK-NEXT: br label [[BRANCH_BLOCK:%.*]]
|
|
; CHECK: bb2:
|
|
; CHECK-NEXT: br label [[BRANCH_BLOCK]]
|
|
; CHECK: branch.block:
|
|
; CHECK-NEXT: [[ADDR:%.*]] = phi ptr [ blockaddress(@indbrtest5, [[TARGET1:%.*]]), [[BB1]] ], [ blockaddress(@indbrtest5, [[TARGET2:%.*]]), [[BB2]] ]
|
|
; CHECK-NEXT: indirectbr ptr [[ADDR]], [label [[TARGET1]], label %target2]
|
|
; CHECK: target1:
|
|
; CHECK-NEXT: br label [[TARGET2]]
|
|
; CHECK: target2:
|
|
; CHECK-NEXT: ret i32 undef
|
|
;
|
|
entry:
|
|
br i1 %c, label %bb1, label %bb2
|
|
|
|
bb1:
|
|
br label %branch.block
|
|
|
|
|
|
bb2:
|
|
br label %branch.block
|
|
|
|
branch.block:
|
|
%addr = phi ptr [blockaddress(@indbrtest5, %target1), %bb1], [blockaddress(@indbrtest5, %target2), %bb2]
|
|
indirectbr ptr %addr, [label %target1, label %target2]
|
|
|
|
target1:
|
|
br label %target2
|
|
|
|
target2:
|
|
ret i32 10
|
|
}
|
|
|
|
|
|
define i32 @indbrtest5_callee(i1 %c) {
|
|
; CHECK-LABEL: @indbrtest5_callee(
|
|
; CHECK-NEXT: [[R:%.*]] = call i32 @indbrtest5(i1 [[C:%.*]])
|
|
; CHECK-NEXT: ret i32 10
|
|
;
|
|
%r = call i32 @indbrtest5(i1 %c)
|
|
ret i32 %r
|
|
}
|
|
|
|
define i32 @indbr_duplicate_successors_phi(i1 %c, i32 %x) {
|
|
; CHECK-LABEL: @indbr_duplicate_successors_phi(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: br i1 [[C:%.*]], label [[INDBR:%.*]], label [[BB0:%.*]]
|
|
; CHECK: indbr:
|
|
; CHECK-NEXT: br label [[BB0]]
|
|
; CHECK: BB0:
|
|
; CHECK-NEXT: [[PHI:%.*]] = phi i32 [ [[X:%.*]], [[ENTRY:%.*]] ], [ 0, [[INDBR]] ]
|
|
; CHECK-NEXT: ret i32 [[PHI]]
|
|
;
|
|
entry:
|
|
br i1 %c, label %indbr, label %BB0
|
|
|
|
indbr:
|
|
indirectbr ptr blockaddress(@indbr_duplicate_successors_phi, %BB0), [label %BB0, label %BB0, label %BB1]
|
|
|
|
BB0:
|
|
%phi = phi i32 [ %x, %entry ], [ 0, %indbr ], [ 0, %indbr ]
|
|
ret i32 %phi
|
|
|
|
BB1:
|
|
ret i32 0
|
|
}
|