| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --function-signature |
| ; RUN: opt < %s -passes=globalopt -S | FileCheck %s |
| |
| declare token @llvm.call.preallocated.setup(i32) |
| declare ptr @llvm.call.preallocated.arg(token, i32) |
| declare i32 @__CxxFrameHandler3(...) |
| |
| ; Don't touch functions with any musttail calls |
| define internal i32 @preallocated_musttail(ptr preallocated(i32) %p) { |
| ; CHECK-LABEL: define {{[^@]+}}@preallocated_musttail |
| ; CHECK-SAME: (ptr preallocated(i32) [[P:%.*]]) unnamed_addr { |
| ; CHECK-NEXT: [[RV:%.*]] = load i32, ptr [[P]], align 4 |
| ; CHECK-NEXT: ret i32 [[RV]] |
| ; |
| %rv = load i32, ptr %p |
| ret i32 %rv |
| } |
| |
| define i32 @call_preallocated_musttail(ptr preallocated(i32) %a) { |
| ; CHECK-LABEL: define {{[^@]+}}@call_preallocated_musttail |
| ; CHECK-SAME: (ptr preallocated(i32) [[A:%.*]]) local_unnamed_addr { |
| ; CHECK-NEXT: [[R:%.*]] = musttail call i32 @preallocated_musttail(ptr preallocated(i32) [[A]]) |
| ; CHECK-NEXT: ret i32 [[R]] |
| ; |
| %r = musttail call i32 @preallocated_musttail(ptr preallocated(i32) %a) |
| ret i32 %r |
| } |
| |
| define i32 @call_preallocated_musttail_without_musttail() { |
| ; CHECK-LABEL: define {{[^@]+}}@call_preallocated_musttail_without_musttail() local_unnamed_addr { |
| ; CHECK-NEXT: [[C:%.*]] = call token @llvm.call.preallocated.setup(i32 1) |
| ; CHECK-NEXT: [[N:%.*]] = call ptr @llvm.call.preallocated.arg(token [[C]], i32 0) #[[ATTR1:[0-9]+]] |
| ; CHECK-NEXT: [[R:%.*]] = call i32 @preallocated_musttail(ptr preallocated(i32) [[N]]) [ "preallocated"(token [[C]]) ] |
| ; CHECK-NEXT: ret i32 [[R]] |
| ; |
| %c = call token @llvm.call.preallocated.setup(i32 1) |
| %N = call ptr @llvm.call.preallocated.arg(token %c, i32 0) preallocated(i32) |
| %r = call i32 @preallocated_musttail(ptr preallocated(i32) %N) ["preallocated"(token %c)] |
| ret i32 %r |
| } |
| |
| ; Check that only one alloca per preallocated arg |
| define internal i32 @preallocated(ptr preallocated(i32) %a) { |
| ; CHECK-LABEL: define {{[^@]+}}@preallocated |
| ; CHECK-SAME: (ptr [[A:%.*]]) unnamed_addr { |
| ; CHECK-NEXT: [[RV:%.*]] = load i32, ptr [[A]], align 4 |
| ; CHECK-NEXT: ret i32 [[RV]] |
| ; |
| %rv = load i32, ptr %a |
| ret i32 %rv |
| } |
| |
| declare void @foo(ptr) |
| |
| define i32 @call_preallocated_multiple_args() { |
| ; CHECK-LABEL: define {{[^@]+}}@call_preallocated_multiple_args() local_unnamed_addr { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call ptr @llvm.stacksave.p0() |
| ; CHECK-NEXT: [[PAARG:%.*]] = alloca i32, align 4 |
| ; CHECK-NEXT: call void @foo(ptr [[PAARG]]) |
| ; CHECK-NEXT: call void @foo(ptr [[PAARG]]) |
| ; CHECK-NEXT: call void @foo(ptr [[PAARG]]) |
| ; CHECK-NEXT: [[R:%.*]] = call fastcc i32 @preallocated(ptr [[PAARG]]) |
| ; CHECK-NEXT: call void @llvm.stackrestore.p0(ptr [[TMP1]]) |
| ; CHECK-NEXT: ret i32 [[R]] |
| ; |
| %c = call token @llvm.call.preallocated.setup(i32 1) |
| %a1 = call ptr @llvm.call.preallocated.arg(token %c, i32 0) preallocated(i32) |
| call void @foo(ptr %a1) |
| %a2 = call ptr @llvm.call.preallocated.arg(token %c, i32 0) preallocated(i32) |
| call void @foo(ptr %a2) |
| %a3 = call ptr @llvm.call.preallocated.arg(token %c, i32 0) preallocated(i32) |
| call void @foo(ptr %a3) |
| %r = call i32 @preallocated(ptr preallocated(i32) %a3) ["preallocated"(token %c)] |
| ret i32 %r |
| } |
| |
| ; Don't touch functions with any invokes |
| define internal i32 @preallocated_invoke(ptr preallocated(i32) %p) { |
| ; CHECK-LABEL: define {{[^@]+}}@preallocated_invoke |
| ; CHECK-SAME: (ptr preallocated(i32) [[P:%.*]]) unnamed_addr { |
| ; CHECK-NEXT: [[RV:%.*]] = load i32, ptr [[P]], align 4 |
| ; CHECK-NEXT: ret i32 [[RV]] |
| ; |
| %rv = load i32, ptr %p |
| ret i32 %rv |
| } |
| |
| define i32 @call_preallocated_invoke() personality ptr @__CxxFrameHandler3 { |
| ; CHECK-LABEL: define {{[^@]+}}@call_preallocated_invoke() local_unnamed_addr personality ptr @__CxxFrameHandler3 { |
| ; CHECK-NEXT: [[C:%.*]] = call token @llvm.call.preallocated.setup(i32 1) |
| ; CHECK-NEXT: [[A:%.*]] = call ptr @llvm.call.preallocated.arg(token [[C]], i32 0) #[[ATTR1]] |
| ; CHECK-NEXT: [[R:%.*]] = invoke i32 @preallocated_invoke(ptr preallocated(i32) [[A]]) [ "preallocated"(token [[C]]) ] |
| ; CHECK-NEXT: to label [[CONTA:%.*]] unwind label [[CONTB:%.*]] |
| ; CHECK: conta: |
| ; CHECK-NEXT: ret i32 [[R]] |
| ; CHECK: contb: |
| ; CHECK-NEXT: [[S:%.*]] = catchswitch within none [label %catch] unwind to caller |
| ; CHECK: catch: |
| ; CHECK-NEXT: [[P:%.*]] = catchpad within [[S]] [] |
| ; CHECK-NEXT: catchret from [[P]] to label [[CONT:%.*]] |
| ; CHECK: cont: |
| ; CHECK-NEXT: ret i32 42 |
| ; |
| %c = call token @llvm.call.preallocated.setup(i32 1) |
| %a = call ptr @llvm.call.preallocated.arg(token %c, i32 0) preallocated(i32) |
| %r = invoke i32 @preallocated_invoke(ptr preallocated(i32) %a) ["preallocated"(token %c)] |
| to label %conta unwind label %contb |
| conta: |
| ret i32 %r |
| contb: |
| %s = catchswitch within none [label %catch] unwind to caller |
| catch: |
| %p = catchpad within %s [] |
| catchret from %p to label %cont |
| cont: |
| ret i32 42 |
| } |