; RUN: opt -passes=inline -S < %s | FileCheck %s
; RUN: opt -passes='cgscc(inline)' -S < %s | FileCheck %s
; PR10162
; Make sure doit is not inlined since the blockaddress is taken
; which could be unsafe
; CHECK: store ptr blockaddress(@doit, %here), ptr %pptr, align 8
@i = global i32 1, align 4
@ptr1 = common global ptr null, align 8
define void @doit(ptr nocapture %pptr, i32 %cond) nounwind uwtable {
entry:
%tobool = icmp eq i32 %cond, 0
br i1 %tobool, label %if.end, label %here
here:
store ptr blockaddress(@doit, %here), ptr %pptr, align 8
br label %if.end
if.end:
ret void
}
define void @f(i32 %cond) nounwind uwtable {
entry:
call void @doit(ptr @ptr1, i32 %cond)
ret void
}
; PR27233: We can inline @run into @init. Don't crash on it.
;
; CHECK-LABEL: define void @init
; CHECK: store ptr blockaddress(@run, %bb)
; CHECK-SAME: @run.bb
define void @init() {
entry:
call void @run()
ret void
}
define void @run() {
entry:
store ptr blockaddress(@run, %bb), ptr @run.bb, align 8
ret void
bb:
unreachable
}
@run.bb = global [1 x ptr] zeroinitializer
; Check that a function referenced by a global blockaddress wont be inlined,
; even if it contains a callbr. We might be able to relax this in the future
; as long as the global blockaddress is updated correctly.
@ba = internal global ptr blockaddress(@foo, %7), align 8
define internal i32 @foo(i32) {
%2 = alloca i32, align 4
%3 = alloca i32, align 4
store i32 %0, ptr %3, align 4
%4 = load i32, ptr %3, align 4
callbr void asm sideeffect "testl $0, $0; jne ${1:l};", "r,!i,!i,~{dirflag},~{fpsr},~{flags}"(i32 %4) #1
to label %5 [label %7, label %6]
; <label>:5: ; preds = %1
store i32 0, ptr %2, align 4
br label %8
; <label>:6: ; preds = %1
store i32 1, ptr %2, align 4
br label %8
; <label>:7: ; preds = %1
store i32 2, ptr %2, align 4
br label %8
; <label>:8: ; preds = %7, %6, %5
%9 = load i32, ptr %2, align 4
ret i32 %9
}
define dso_local i32 @bar() {
%1 = call i32 @foo(i32 0)
ret i32 %1
}
; CHECK: define dso_local i32 @bar() {
; CHECK: %1 = call i32 @foo(i32 0)
; CHECK: ret i32 %1
; CHECK: }
; Triple check that even with a global aggregate whose member is a blockaddress,
; we still don't inline referred to functions.
%struct.foo = type { ptr }
@my_foo = dso_local global %struct.foo { ptr blockaddress(@baz, %7) }
define internal i32 @baz(i32) {
%2 = alloca i32, align 4
%3 = alloca i32, align 4
store i32 %0, ptr %3, align 4
%4 = load i32, ptr %3, align 4
callbr void asm sideeffect "testl $0, $0; jne ${1:l};", "r,!i,!i,~{dirflag},~{fpsr},~{flags}"(i32 %4) #1
to label %5 [label %7, label %6]
; <label>:5: ; preds = %1
store i32 0, ptr %2, align 4
br label %8
; <label>:6: ; preds = %1
store i32 1, ptr %2, align 4
br label %8
; <label>:7: ; preds = %1
store i32 2, ptr %2, align 4
br label %8
; <label>:8: ; preds = %7, %6, %5
%9 = load i32, ptr %2, align 4
ret i32 %9
}
define dso_local i32 @quux() {
%1 = call i32 @baz(i32 0)
ret i32 %1
}
; CHECK: define dso_local i32 @quux() {
; CHECK: %1 = call i32 @baz(i32 0)
; CHECK: ret i32 %1
; CHECK: }