blob: 37d6fcc58d26da6ac30c903c9835332af6c13f4b [file] [edit]
; RUN: llc -mtriple thumbv7-windows-msvc -o - %s | FileCheck %s
; struct S { int x; };
; void foo(int n);
; void foo(struct S o);
; void simple_seh() {
; struct S o;
;
; __try { foo(o.x); }
; __finally { foo(o.x); }
; }
; void stack_realign() {
; struct S __declspec(align(32)) o;
;
; __try { foo(o.x); }
; __finally { foo(o.x); }
; }
; void vla_present(int n) {
; int vla[n];
;
; __try { foo(n); }
; __finally { foo(n); }
; }
; void vla_and_realign(int n) {
; struct S __declspec(align(32)) o;
; int vla[n];
;
; __try { foo(o.x); }
; __finally { foo(o.x); }
; }
%struct.S = type { i32 }
; Test simple SEH (__try/__finally).
define arm_aapcs_vfpcc void @simple_seh() #0 personality ptr @__C_specific_handler {
entry:
; CHECK-LABEL: simple_seh:
; CHECK: mov r6, sp
; CHECK: $Msimple_seh$frame_escape_0 = 4
; CHECK: ldr r0, [r6, #4]
; CHECK: bl foo
%o = alloca %struct.S, align 4
call void (...) @llvm.localescape(ptr %o)
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%0 = load i32, ptr %x, align 4
invoke arm_aapcs_vfpcc void @foo(i32 noundef %0) #5
to label %invoke.cont unwind label %ehcleanup
invoke.cont: ; preds = %entry
; CHECK: movs r0, #0
; CHECK: mov r1, r6
; CHECK: bl "?fin$0@0@simple_seh@@"
%1 = call ptr @llvm.localaddress()
call arm_aapcs_vfpcc void @"?fin$0@0@simple_seh@@"(i8 noundef zeroext 0, ptr noundef %1)
ret void
ehcleanup: ; preds = %entry
; CHECK-LABEL: "?dtor$2@?0?simple_seh@4HA":
; CHECK: movs r0, #1
; CHECK: mov r1, r6
; CHECK: bl "?fin$0@0@simple_seh@@"
%2 = cleanuppad within none []
%3 = call ptr @llvm.localaddress()
call arm_aapcs_vfpcc void @"?fin$0@0@simple_seh@@"(i8 noundef zeroext 1, ptr noundef %3) [ "funclet"(token %2) ]
cleanupret from %2 unwind to caller
}
define arm_aapcs_vfpcc void @"?fin$0@0@simple_seh@@"(i8 noundef zeroext %abnormal_termination, ptr noundef %frame_pointer) #1 {
entry:
; CHECK-LABEL: "?fin$0@0@simple_seh@@":
; CHECK: movw r0, :lower16:$Msimple_seh$frame_escape_0
; CHECK: movt r0, :upper16:$Msimple_seh$frame_escape_0
; CHECK: ldr r0, [r1, r0]
; CHECK: bl foo
%o = call ptr @llvm.localrecover(ptr @simple_seh, ptr %frame_pointer, i32 0)
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%0 = load i32, ptr %x, align 4
call arm_aapcs_vfpcc void @foo(i32 noundef %0)
ret void
}
; Test SEH when stack realignment is needed in case highly aligned stack objects are present.
define arm_aapcs_vfpcc void @stack_realign() #0 personality ptr @__C_specific_handler {
entry:
; CHECK-LABEL: stack_realign:
; CHECK: bfc r4, #0, #5
; CHECK: mov sp, r4
; CHECK: mov r6, sp
; CHECK: $Mstack_realign$frame_escape_0 = 0
; CHECK: ldr r0, [sp]
; CHECK: bl foo
%o = alloca %struct.S, align 32
call void (...) @llvm.localescape(ptr %o)
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%0 = load i32, ptr %x, align 32
invoke arm_aapcs_vfpcc void @foo(i32 noundef %0) #5
to label %invoke.cont unwind label %ehcleanup
invoke.cont: ; preds = %entry
; CHECK: movs r0, #0
; CHECK: mov r1, r6
; CHECK: bl "?fin$0@0@stack_realign@@"
%1 = call ptr @llvm.localaddress()
call arm_aapcs_vfpcc void @"?fin$0@0@stack_realign@@"(i8 noundef zeroext 0, ptr noundef %1)
ret void
ehcleanup: ; preds = %entry
; CHECK-LABEL: "?dtor$2@?0?stack_realign@4HA":
; CHECK: movs r0, #1
; CHECK: mov r1, r6
; CHECK: bl "?fin$0@0@stack_realign@@"
%2 = cleanuppad within none []
%3 = call ptr @llvm.localaddress()
call arm_aapcs_vfpcc void @"?fin$0@0@stack_realign@@"(i8 noundef zeroext 1, ptr noundef %3) [ "funclet"(token %2) ]
cleanupret from %2 unwind to caller
}
define arm_aapcs_vfpcc void @"?fin$0@0@stack_realign@@"(i8 noundef zeroext %abnormal_termination, ptr noundef %frame_pointer) #1 {
entry:
; CHECK-LABEL: "?fin$0@0@stack_realign@@":
; CHECK: movw r0, :lower16:$Mstack_realign$frame_escape_0
; CHECK: movt r0, :upper16:$Mstack_realign$frame_escape_0
; CHECK: ldr r0, [r1, r0]
; CHECK: bl foo
%o = call ptr @llvm.localrecover(ptr @stack_realign, ptr %frame_pointer, i32 0)
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%0 = load i32, ptr %x, align 32
call arm_aapcs_vfpcc void @foo(i32 noundef %0)
ret void
}
; Test SEH when variable size objects are present on the stack. Note: Escaped vla's are current not supported by SEH.
define arm_aapcs_vfpcc void @vla_present(i32 noundef %n) #0 personality ptr @__C_specific_handler {
entry:
; CHECK-LABEL: vla_present:
; CHECK: mov r6, sp
; CHECK: $Mvla_present$frame_escape_0 = 12
; CHECK: bl foo
%n.addr = alloca i32, align 4
%saved_stack = alloca ptr, align 4
%__vla_expr0 = alloca i32, align 4
call void (...) @llvm.localescape(ptr %n.addr)
store i32 %n, ptr %n.addr, align 4
%0 = load i32, ptr %n.addr, align 4
%1 = call ptr @llvm.stacksave.p0()
store ptr %1, ptr %saved_stack, align 4
%vla = alloca i32, i32 %0, align 4
store i32 %0, ptr %__vla_expr0, align 4
%2 = load i32, ptr %n.addr, align 4
invoke arm_aapcs_vfpcc void @foo(i32 noundef %2) #5
to label %invoke.cont unwind label %ehcleanup
invoke.cont: ; preds = %entry
; CHECK: movs r0, #0
; CHECK: mov r1, r6
; CHECK: bl "?fin$0@0@vla_present@@"
%3 = call ptr @llvm.localaddress()
call arm_aapcs_vfpcc void @"?fin$0@0@vla_present@@"(i8 noundef zeroext 0, ptr noundef %3)
%4 = load ptr, ptr %saved_stack, align 4
call void @llvm.stackrestore.p0(ptr %4)
ret void
ehcleanup: ; preds = %entry
; CHECK-LABEL: "?dtor$2@?0?vla_present@4HA":
; CHECK: movs r0, #1
; CHECK: mov r1, r6
; CHECK: bl "?fin$0@0@vla_present@@"
%5 = cleanuppad within none []
%6 = call ptr @llvm.localaddress()
call arm_aapcs_vfpcc void @"?fin$0@0@vla_present@@"(i8 noundef zeroext 1, ptr noundef %6) [ "funclet"(token %5) ]
cleanupret from %5 unwind to caller
}
define arm_aapcs_vfpcc void @"?fin$0@0@vla_present@@"(i8 noundef zeroext %abnormal_termination, ptr noundef %frame_pointer) #1 {
entry:
; CHECK-LABEL: "?fin$0@0@vla_present@@":
; CHECK: movw r0, :lower16:$Mvla_present$frame_escape_0
; CHECK: movt r0, :upper16:$Mvla_present$frame_escape_0
; CHECK: ldr r0, [r1, r0]
; CHECK: bl foo
%n.addr = call ptr @llvm.localrecover(ptr @vla_present, ptr %frame_pointer, i32 0)
%0 = load i32, ptr %n.addr, align 4
call arm_aapcs_vfpcc void @foo(i32 noundef %0)
ret void
}
; Test when both vla's and highly aligned objects are present on stack.
define arm_aapcs_vfpcc void @vla_and_realign(i32 noundef %n) #0 personality ptr @__C_specific_handler {
entry:
; CHECK-LABEL: vla_and_realign:
; CHECK: bfc r4, #0, #5
; CHECK: mov sp, r4
; CHECK: mov r6, sp
; CHECK: $Mvla_and_realign$frame_escape_0 = 0
; CHECK: bl foo
%o = alloca %struct.S, align 32
call void (...) @llvm.localescape(ptr %o)
%0 = call ptr @llvm.stacksave.p0()
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%1 = load i32, ptr %x, align 32
invoke arm_aapcs_vfpcc void @foo(i32 noundef %1) #5
to label %invoke.cont unwind label %ehcleanup
invoke.cont: ; preds = %entry
; CHECK: movs r0, #0
; CHECK: mov r1, r6
; CHECK: bl "?fin$0@0@vla_and_realign@@"
%2 = call ptr @llvm.localaddress()
call arm_aapcs_vfpcc void @"?fin$0@0@vla_and_realign@@"(i8 noundef zeroext 0, ptr noundef %2)
call void @llvm.stackrestore.p0(ptr %0)
ret void
ehcleanup: ; preds = %entry
; CHECK-LABEL: "?dtor$2@?0?vla_and_realign@4HA":
; CHECK: movs r0, #1
; CHECK: mov r1, r6
; CHECK: bl "?fin$0@0@vla_and_realign@@"
%3 = cleanuppad within none []
%4 = call ptr @llvm.localaddress()
call arm_aapcs_vfpcc void @"?fin$0@0@vla_and_realign@@"(i8 noundef zeroext 1, ptr noundef %4) [ "funclet"(token %3) ]
cleanupret from %3 unwind to caller
}
define arm_aapcs_vfpcc void @"?fin$0@0@vla_and_realign@@"(i8 noundef zeroext %abnormal_termination, ptr noundef %frame_pointer) #1 {
entry:
; CHECK-LABEL: "?fin$0@0@vla_and_realign@@":
; CHECK: movw r0, :lower16:$Mvla_and_realign$frame_escape_0
; CHECK: movt r0, :upper16:$Mvla_and_realign$frame_escape_0
; CHECK: ldr r0, [r1, r0]
; CHECK: bl foo
%o = call ptr @llvm.localrecover(ptr @vla_and_realign, ptr %frame_pointer, i32 0)
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%0 = load i32, ptr %x, align 32
call arm_aapcs_vfpcc void @foo(i32 noundef %0)
ret void
}
declare arm_aapcs_vfpcc void @foo(i32 noundef)
declare void @llvm.stackrestore.p0(ptr)
declare ptr @llvm.stacksave.p0()
declare ptr @llvm.localrecover(ptr, ptr, i32 immarg)
declare ptr @llvm.localaddress()
declare void @llvm.localescape(...)
declare i32 @__C_specific_handler(...)
attributes #0 = { noinline optnone }
attributes #1 = { noinline }