; To test that safestack does not break the musttail call contract.
;
; RUN: opt < %s --safe-stack -S | FileCheck %s
; RUN: opt < %s -passes=safe-stack -S | FileCheck %s
target triple = "x86_64-unknown-linux-gnu"
declare i32 @foo(ptr %p)
declare void @alloca_test_use(ptr)
define i32 @call_foo(ptr %a) safestack {
; CHECK-LABEL: @call_foo(
; CHECK-NEXT: [[UNSAFE_STACK_PTR:%.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr, align 8
; CHECK-NEXT: [[UNSAFE_STACK_STATIC_TOP:%.*]] = getelementptr i8, ptr [[UNSAFE_STACK_PTR]], i32 -16
; CHECK-NEXT: store ptr [[UNSAFE_STACK_STATIC_TOP]], ptr @__safestack_unsafe_stack_ptr, align 8
; CHECK-NEXT: [[TMP1:%.*]] = getelementptr i8, ptr [[UNSAFE_STACK_PTR]], i32 -10
; CHECK-NEXT: call void @alloca_test_use(ptr [[TMP1]])
; CHECK-NEXT: store ptr [[UNSAFE_STACK_PTR]], ptr @__safestack_unsafe_stack_ptr, align 8
; CHECK-NEXT: [[R:%.*]] = musttail call i32 @foo(ptr [[A:%.*]])
; CHECK-NEXT: ret i32 [[R]]
;
%x = alloca [10 x i8], align 1
call void @alloca_test_use(ptr %x)
%r = musttail call i32 @foo(ptr %a)
ret i32 %r
}
define i32 @call_foo_cast(ptr %a) safestack {
; CHECK-LABEL: @call_foo_cast(
; CHECK-NEXT: [[UNSAFE_STACK_PTR:%.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr, align 8
; CHECK-NEXT: [[UNSAFE_STACK_STATIC_TOP:%.*]] = getelementptr i8, ptr [[UNSAFE_STACK_PTR]], i32 -16
; CHECK-NEXT: store ptr [[UNSAFE_STACK_STATIC_TOP]], ptr @__safestack_unsafe_stack_ptr, align 8
; CHECK-NEXT: [[TMP1:%.*]] = getelementptr i8, ptr [[UNSAFE_STACK_PTR]], i32 -10
; CHECK-NEXT: call void @alloca_test_use(ptr [[TMP1]])
; CHECK-NEXT: store ptr [[UNSAFE_STACK_PTR]], ptr @__safestack_unsafe_stack_ptr, align 8
; CHECK-NEXT: [[R:%.*]] = musttail call i32 @foo(ptr [[A:%.*]])
; CHECK-NEXT: [[T:%.*]] = bitcast i32 [[R]] to i32
; CHECK-NEXT: ret i32 [[T]]
;
%x = alloca [10 x i8], align 1
call void @alloca_test_use(ptr %x)
%r = musttail call i32 @foo(ptr %a)
%t = bitcast i32 %r to i32
ret i32 %t
}