205 lines
4.3 KiB
LLVM
205 lines
4.3 KiB
LLVM
; RUN: opt -inline < %s -S -o - -inline-threshold=3 | FileCheck %s
|
|
|
|
target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128"
|
|
target triple = "x86_64-unknown-linux-gnu"
|
|
|
|
declare void @pad() readnone
|
|
|
|
define void @outer1(i32* %a) {
|
|
; CHECK-LABEL: @outer1(
|
|
; CHECK-NOT: call void @inner1
|
|
%b = alloca i32
|
|
call void @inner1(i32* %a, i32* %b)
|
|
ret void
|
|
}
|
|
|
|
define void @inner1(i32* %a, i32* %b) {
|
|
%1 = load i32, i32* %a
|
|
store i32 %1, i32 * %b ; This store does not clobber the first load.
|
|
%2 = load i32, i32* %a
|
|
call void @pad()
|
|
%3 = load i32, i32* %a
|
|
ret void
|
|
}
|
|
|
|
|
|
define void @outer2(i32* %a, i32* %b) {
|
|
; CHECK-LABEL: @outer2(
|
|
; CHECK: call void @inner2
|
|
call void @inner2(i32* %a, i32* %b)
|
|
ret void
|
|
}
|
|
|
|
define void @inner2(i32* %a, i32* %b) {
|
|
%1 = load i32, i32* %a
|
|
store i32 %1, i32 * %b ; This store clobbers the first load.
|
|
%2 = load i32, i32* %a
|
|
call void @pad()
|
|
ret void
|
|
}
|
|
|
|
|
|
define void @outer3(i32* %a) {
|
|
; CHECK-LABEL: @outer3(
|
|
; CHECK: call void @inner3
|
|
call void @inner3(i32* %a)
|
|
ret void
|
|
}
|
|
|
|
declare void @ext()
|
|
|
|
define void @inner3(i32* %a) {
|
|
%1 = load i32, i32* %a
|
|
call void @ext() ; This call clobbers the first load.
|
|
%2 = load i32, i32* %a
|
|
ret void
|
|
}
|
|
|
|
|
|
define void @outer4(i32* %a, i32* %b, i32* %c) {
|
|
; CHECK-LABEL: @outer4(
|
|
; CHECK-NOT: call void @inner4
|
|
call void @inner4(i32* %a, i32* %b, i1 false)
|
|
ret void
|
|
}
|
|
|
|
define void @inner4(i32* %a, i32* %b, i1 %pred) {
|
|
%1 = load i32, i32* %a
|
|
br i1 %pred, label %cond_true, label %cond_false
|
|
|
|
cond_true:
|
|
store i32 %1, i32 * %b ; This store does not clobber the first load.
|
|
br label %cond_false
|
|
|
|
cond_false:
|
|
%2 = load i32, i32* %a
|
|
call void @pad()
|
|
%3 = load i32, i32* %a
|
|
%4 = load i32, i32* %a
|
|
ret void
|
|
}
|
|
|
|
|
|
define void @outer5(i32* %a, double %b) {
|
|
; CHECK-LABEL: @outer5(
|
|
; CHECK-NOT: call void @inner5
|
|
call void @inner5(i32* %a, double %b)
|
|
ret void
|
|
}
|
|
|
|
declare double @llvm.fabs.f64(double) nounwind readnone
|
|
|
|
define void @inner5(i32* %a, double %b) {
|
|
%1 = load i32, i32* %a
|
|
%2 = call double @llvm.fabs.f64(double %b) ; This intrinsic does not clobber the first load.
|
|
%3 = load i32, i32* %a
|
|
call void @pad()
|
|
ret void
|
|
}
|
|
|
|
define void @outer6(i32* %a, i8* %ptr) {
|
|
; CHECK-LABEL: @outer6(
|
|
; CHECK-NOT: call void @inner6
|
|
call void @inner6(i32* %a, i8* %ptr)
|
|
ret void
|
|
}
|
|
|
|
declare void @llvm.lifetime.start.p0i8(i64, i8* nocapture) argmemonly nounwind
|
|
|
|
define void @inner6(i32* %a, i8* %ptr) {
|
|
%1 = load i32, i32* %a
|
|
call void @llvm.lifetime.start.p0i8(i64 32, i8* %ptr) ; This intrinsic does not clobber the first load.
|
|
%2 = load i32, i32* %a
|
|
call void @pad()
|
|
%3 = load i32, i32* %a
|
|
ret void
|
|
}
|
|
|
|
define void @outer7(i32* %a) {
|
|
; CHECK-LABEL: @outer7(
|
|
; CHECK-NOT: call void @inner7
|
|
call void @inner7(i32* %a)
|
|
ret void
|
|
}
|
|
|
|
declare void @ext2() readnone
|
|
|
|
define void @inner7(i32* %a) {
|
|
%1 = load i32, i32* %a
|
|
call void @ext2() ; This call does not clobber the first load.
|
|
%2 = load i32, i32* %a
|
|
ret void
|
|
}
|
|
|
|
|
|
define void @outer8(i32* %a) {
|
|
; CHECK-LABEL: @outer8(
|
|
; CHECK-NOT: call void @inner8
|
|
call void @inner8(i32* %a, void ()* @ext2)
|
|
ret void
|
|
}
|
|
|
|
define void @inner8(i32* %a, void ()* %f) {
|
|
%1 = load i32, i32* %a
|
|
call void %f() ; This indirect call does not clobber the first load.
|
|
%2 = load i32, i32* %a
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
ret void
|
|
}
|
|
|
|
|
|
define void @outer9(i32* %a) {
|
|
; CHECK-LABEL: @outer9(
|
|
; CHECK: call void @inner9
|
|
call void @inner9(i32* %a, void ()* @ext)
|
|
ret void
|
|
}
|
|
|
|
define void @inner9(i32* %a, void ()* %f) {
|
|
%1 = load i32, i32* %a
|
|
call void %f() ; This indirect call clobbers the first load.
|
|
%2 = load i32, i32* %a
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
call void @pad()
|
|
ret void
|
|
}
|
|
|
|
|
|
define void @outer10(i32* %a) {
|
|
; CHECK-LABEL: @outer10(
|
|
; CHECK: call void @inner10
|
|
%b = alloca i32
|
|
call void @inner10(i32* %a, i32* %b)
|
|
ret void
|
|
}
|
|
|
|
define void @inner10(i32* %a, i32* %b) {
|
|
%1 = load i32, i32* %a
|
|
store i32 %1, i32 * %b
|
|
%2 = load volatile i32, i32* %a ; volatile load should be kept.
|
|
call void @pad()
|
|
%3 = load volatile i32, i32* %a ; Same as the above.
|
|
ret void
|
|
}
|