blob: 0bcfa13b7a89474132a3d950e08718090c8ed026 [file] [edit]
; RUN: llc -mtriple=thumbv7-windows-msvc -o - %s | FileCheck %s
; struct S { int x; };
; void foo(int n);
;
; void simple_except() {
; struct S o;
; __try {
; foo(o.x);
; } __except((foo(o.x), 1)) {
; foo(o.x);
; }
; }
;
; void stack_realign() {
; struct S __declspec(align(32)) o;
; __try {
; foo(o.x);
; } __except((foo(o.x), 1)) {
; foo(o.x);
; }
; }
;
; void vla_present(int n) {
; int vla[n];
; __try {
; foo(n);
; } __except((foo(n), 1)) {
; foo(n);
; }
; }
;
; void vla_and_realign(int n) {
; struct S __declspec(align(32)) o;
; int vla[n];
; __try {
; foo(o.x);
; } __except((foo(o.x), 1)) {
; foo(o.x);
; }
; }
;
; void stack_realign_filter() {
; struct S o;
; __try {
; foo(o.x);
; } __except(([] [[msvc::forceinline]] () {
; struct S __declspec(align(32)) p;
; foo(p.x);
; }(), foo(o.x), 1)) {
; foo(o.x);
; }
; }
%struct.S = type { i32 }
; Function Attrs: nounwind
define dso_local arm_aapcs_vfpcc void @simple_except() #0 personality ptr @__C_specific_handler {
; CHECK-LABEL: simple_except:
; CHECK: .seh_proc simple_except
; CHECK: .seh_handler __C_specific_handler, %unwind, %except
; CHECK: push {r6, lr}
; CHECK: .seh_save_regs {r6, lr}
; CHECK: sub sp, #8
; CHECK: .seh_stackalloc 8
; CHECK: .seh_endprologue
; CHECK: mov r6, sp
; CHECK: $Msimple_except$frame_escape_0 = 4
; CHECK: ldr r0, [r6, #4]
; CHECK: bl foo
entry:
%o = alloca %struct.S, align 4
%__exception_code = alloca i32, align 4
call void (...) @llvm.localescape(ptr %o)
call void @llvm.lifetime.start.p0(ptr %o) #6
%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) #7
to label %invoke.cont unwind label %catch.dispatch
catch.dispatch: ; preds = %entry
%1 = catchswitch within none [label %__except.ret] unwind label %ehcleanup
__except.ret: ; preds = %catch.dispatch
%2 = catchpad within %1 [ptr @"?filt$0@0@simple_except@@"]
catchret from %2 to label %__except
__except: ; preds = %__except.ret
; CHECK: ldr r0, [r6, #4]
; CHECK: bl foo
%3 = call i32 @llvm.eh.exceptioncode(token %2)
store i32 %3, ptr %__exception_code, align 4
%x1 = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%4 = load i32, ptr %x1, align 4
call arm_aapcs_vfpcc void @foo(i32 noundef %4)
br label %__try.cont
__try.cont: ; preds = %__except, %invoke.cont
call void @llvm.lifetime.end.p0(ptr %o) #6
ret void
invoke.cont: ; preds = %entry
br label %__try.cont
ehcleanup: ; preds = %catch.dispatch
%5 = cleanuppad within none []
call void @llvm.lifetime.end.p0(ptr %o) #6
cleanupret from %5 unwind to caller
}
; Function Attrs: nocallback nofree nosync nounwind willreturn memory(argmem: readwrite)
declare void @llvm.lifetime.start.p0(ptr captures(none))
; Function Attrs: nounwind
define internal arm_aapcs_vfpcc i32 @"?filt$0@0@simple_except@@"(ptr noundef %exception_pointers, ptr noundef %frame_pointer) #0 {
; CHECK-LABEL: "?filt$0@0@simple_except@@":
; CHECK: push.w {r11, lr}
; CHECK: sub sp, #16
; CHECK: movw r[[OFFSET:[0-9]+]], :lower16:{{.*}}frame_escape_0
; CHECK-NEXT: movt r[[OFFSET]], :upper16:{{.*}}frame_escape_0
; CHECK: ldr r0, [r6, r[[OFFSET]]]
; CHECK: bl foo
entry:
%frame_pointer.addr = alloca ptr, align 4
%exception_pointers.addr = alloca ptr, align 4
%0 = call ptr @llvm.eh.recoverfp(ptr @simple_except, ptr %frame_pointer)
%o = call ptr @llvm.localrecover(ptr @simple_except, ptr %0, i32 0)
%__exception_code = alloca i32, align 4
store ptr %frame_pointer, ptr %frame_pointer.addr, align 4
store ptr %exception_pointers, ptr %exception_pointers.addr, align 4
%1 = getelementptr inbounds nuw { ptr, ptr }, ptr %exception_pointers, i32 0, i32 0
%2 = load ptr, ptr %1, align 4
%3 = load i32, ptr %2, align 4
store i32 %3, ptr %__exception_code, align 4
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%4 = load i32, ptr %x, align 4
call arm_aapcs_vfpcc void @foo(i32 noundef %4)
ret i32 1
}
declare ptr @llvm.eh.recoverfp(ptr, ptr)
declare ptr @llvm.localrecover(ptr, ptr, i32 immarg)
declare dso_local arm_aapcs_vfpcc void @foo(i32 noundef)
declare dso_local arm_aapcs_vfpcc i32 @__C_specific_handler(...)
; Function Attrs: nounwind memory(none)
declare i32 @llvm.eh.exceptioncode(token)
; Function Attrs: nocallback nofree nosync nounwind willreturn memory(argmem: readwrite)
declare void @llvm.lifetime.end.p0(ptr captures(none))
; Function Attrs: nocallback nofree nosync nounwind willreturn
declare void @llvm.localescape(...)
; Function Attrs: nounwind
define dso_local arm_aapcs_vfpcc void @stack_realign() #0 personality ptr @__C_specific_handler {
; CHECK-LABEL: stack_realign:
; CHECK: push {r4, r6}
; CHECK: push.w {r11, lr}
; CHECK: mov r11, sp
; CHECK: sub sp, #48
; CHECK: bfc {{r[0-9]+}}, #0, #5
; CHECK: mov r6, sp
; CHECK: $Mstack_realign$frame_escape_0 = 32
; CHECK: ldr r0, [sp, #32]
; CHECK: bl foo
entry:
%o = alloca %struct.S, align 32
%__exception_code = alloca i32, align 4
call void (...) @llvm.localescape(ptr %o)
call void @llvm.lifetime.start.p0(ptr %o) #6
%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) #7
to label %invoke.cont unwind label %catch.dispatch
catch.dispatch: ; preds = %entry
%1 = catchswitch within none [label %__except.ret] unwind label %ehcleanup
__except.ret: ; preds = %catch.dispatch
%2 = catchpad within %1 [ptr @"?filt$0@0@stack_realign@@"]
catchret from %2 to label %__except
__except: ; preds = %__except.ret
; CHECK: ldr r0, [sp, #32]
; CHECK: bl foo
%3 = call i32 @llvm.eh.exceptioncode(token %2)
store i32 %3, ptr %__exception_code, align 4
%x1 = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%4 = load i32, ptr %x1, align 32
call arm_aapcs_vfpcc void @foo(i32 noundef %4)
br label %__try.cont
__try.cont: ; preds = %__except, %invoke.cont
call void @llvm.lifetime.end.p0(ptr %o) #6
ret void
invoke.cont: ; preds = %entry
br label %__try.cont
ehcleanup: ; preds = %catch.dispatch
%5 = cleanuppad within none []
call void @llvm.lifetime.end.p0(ptr %o) #6
cleanupret from %5 unwind to caller
}
; Function Attrs: nounwind
define internal arm_aapcs_vfpcc i32 @"?filt$0@0@stack_realign@@"(ptr noundef %exception_pointers, ptr noundef %frame_pointer) #0 {
; CHECK-LABEL: "?filt$0@0@stack_realign@@":
; CHECK: movw r[[OFFSET:[0-9]+]], :lower16:{{.*}}frame_escape_0
; CHECK-NEXT: movt r[[OFFSET]], :upper16:{{.*}}frame_escape_0
; CHECK: ldr r0, [r6, r[[OFFSET]]]
; CHECK: bl foo
entry:
%frame_pointer.addr = alloca ptr, align 4
%exception_pointers.addr = alloca ptr, align 4
%0 = call ptr @llvm.eh.recoverfp(ptr @stack_realign, ptr %frame_pointer)
%o = call ptr @llvm.localrecover(ptr @stack_realign, ptr %0, i32 0)
%__exception_code = alloca i32, align 4
store ptr %frame_pointer, ptr %frame_pointer.addr, align 4
store ptr %exception_pointers, ptr %exception_pointers.addr, align 4
%1 = getelementptr inbounds nuw { ptr, ptr }, ptr %exception_pointers, i32 0, i32 0
%2 = load ptr, ptr %1, align 4
%3 = load i32, ptr %2, align 4
store i32 %3, ptr %__exception_code, align 4
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%4 = load i32, ptr %x, align 32
call arm_aapcs_vfpcc void @foo(i32 noundef %4)
ret i32 1
}
; Function Attrs: nounwind
define dso_local arm_aapcs_vfpcc void @vla_present(i32 noundef %n) #0 personality ptr @__C_specific_handler {
; CHECK-LABEL: vla_present:
; CHECK: push {r6, lr}
; CHECK: sub sp, #16
; CHECK: mov r6, sp
; CHECK: $Mvla_present$frame_escape_0 = 12
; CHECK: bl foo
entry:
%n.addr = alloca i32, align 4
%saved_stack = alloca ptr, align 4
%__vla_expr0 = alloca i32, align 4
%__exception_code = 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) #7
to label %invoke.cont unwind label %catch.dispatch
catch.dispatch: ; preds = %entry
%3 = catchswitch within none [label %__except.ret] unwind to caller
__except.ret: ; preds = %catch.dispatch
%4 = catchpad within %3 [ptr @"?filt$0@0@vla_present@@"]
catchret from %4 to label %__except
__except: ; preds = %__except.ret
; CHECK: ldr r0, [r6, #12]
; CHECK: bl foo
%5 = call i32 @llvm.eh.exceptioncode(token %4)
store i32 %5, ptr %__exception_code, align 4
%6 = load i32, ptr %n.addr, align 4
call arm_aapcs_vfpcc void @foo(i32 noundef %6)
br label %__try.cont
__try.cont: ; preds = %__except, %invoke.cont
%7 = load ptr, ptr %saved_stack, align 4
call void @llvm.stackrestore.p0(ptr %7)
ret void
invoke.cont: ; preds = %entry
br label %__try.cont
}
declare ptr @llvm.stacksave.p0() #5
; Function Attrs: nounwind
define internal arm_aapcs_vfpcc i32 @"?filt$0@0@vla_present@@"(ptr noundef %exception_pointers, ptr noundef %frame_pointer) #0 {
; CHECK-LABEL: "?filt$0@0@vla_present@@":
; CHECK: movw r[[OFFSET:[0-9]+]], :lower16:{{.*}}frame_escape_0
; CHECK-NEXT: movt r[[OFFSET]], :upper16:{{.*}}frame_escape_0
; CHECK: ldr r0, [r6, r[[OFFSET]]]
; CHECK: bl foo
entry:
%frame_pointer.addr = alloca ptr, align 4
%exception_pointers.addr = alloca ptr, align 4
%0 = call ptr @llvm.eh.recoverfp(ptr @vla_present, ptr %frame_pointer)
%n.addr = call ptr @llvm.localrecover(ptr @vla_present, ptr %0, i32 0)
%__exception_code = alloca i32, align 4
store ptr %frame_pointer, ptr %frame_pointer.addr, align 4
store ptr %exception_pointers, ptr %exception_pointers.addr, align 4
%1 = getelementptr inbounds nuw { ptr, ptr }, ptr %exception_pointers, i32 0, i32 0
%2 = load ptr, ptr %1, align 4
%3 = load i32, ptr %2, align 4
store i32 %3, ptr %__exception_code, align 4
%4 = load i32, ptr %n.addr, align 4
call arm_aapcs_vfpcc void @foo(i32 noundef %4)
ret i32 1
}
declare void @llvm.stackrestore.p0(ptr) #5
; Function Attrs: nounwind
define dso_local arm_aapcs_vfpcc void @vla_and_realign(i32 noundef %n) #0 personality ptr @__C_specific_handler {
; CHECK-LABEL: vla_and_realign:
; CHECK: push {r4, r6}
; CHECK: push.w {r11, lr}
; CHECK: mov r11, sp
; CHECK: sub sp, #48
; CHECK: bfc {{r[0-9]+}}, #0, #5
; CHECK: mov r6, sp
; CHECK: $Mvla_and_realign$frame_escape_0 = 32
; CHECK: ldr r0, [sp, #32]
; CHECK: bl foo
entry:
%n.addr = alloca i32, align 4
%o = alloca %struct.S, align 32
%saved_stack = alloca ptr, align 4
%__vla_expr0 = alloca i32, align 4
%__exception_code = alloca i32, align 4
call void (...) @llvm.localescape(ptr %o)
store i32 %n, ptr %n.addr, align 4
call void @llvm.lifetime.start.p0(ptr %o) #6
%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
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%2 = load i32, ptr %x, align 32
invoke arm_aapcs_vfpcc void @foo(i32 noundef %2) #7
to label %invoke.cont unwind label %catch.dispatch
catch.dispatch: ; preds = %entry
%3 = catchswitch within none [label %__except.ret] unwind label %ehcleanup
__except.ret: ; preds = %catch.dispatch
%4 = catchpad within %3 [ptr @"?filt$0@0@vla_and_realign@@"]
catchret from %4 to label %__except
__except: ; preds = %__except.ret
; CHECK: ldr r0, [sp, #32]
; CHECK: bl foo
%5 = call i32 @llvm.eh.exceptioncode(token %4)
store i32 %5, ptr %__exception_code, align 4
%x1 = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%6 = load i32, ptr %x1, align 32
call arm_aapcs_vfpcc void @foo(i32 noundef %6)
br label %__try.cont
__try.cont: ; preds = %__except, %invoke.cont
%7 = load ptr, ptr %saved_stack, align 4
call void @llvm.stackrestore.p0(ptr %7)
call void @llvm.lifetime.end.p0(ptr %o) #6
ret void
invoke.cont: ; preds = %entry
br label %__try.cont
ehcleanup: ; preds = %catch.dispatch
%8 = cleanuppad within none []
call void @llvm.lifetime.end.p0(ptr %o) #6
cleanupret from %8 unwind to caller
}
; Function Attrs: nounwind
define internal arm_aapcs_vfpcc i32 @"?filt$0@0@vla_and_realign@@"(ptr noundef %exception_pointers, ptr noundef %frame_pointer) #0 {
; CHECK-LABEL: "?filt$0@0@vla_and_realign@@":
; CHECK: movw r[[OFFSET:[0-9]+]], :lower16:{{.*}}frame_escape_0
; CHECK-NEXT: movt r[[OFFSET]], :upper16:{{.*}}frame_escape_0
; CHECK: ldr r0, [r6, r[[OFFSET]]]
; CHECK: bl foo
entry:
%frame_pointer.addr = alloca ptr, align 4
%exception_pointers.addr = alloca ptr, align 4
%0 = call ptr @llvm.eh.recoverfp(ptr @vla_and_realign, ptr %frame_pointer)
%o = call ptr @llvm.localrecover(ptr @vla_and_realign, ptr %0, i32 0)
%__exception_code = alloca i32, align 4
store ptr %frame_pointer, ptr %frame_pointer.addr, align 4
store ptr %exception_pointers, ptr %exception_pointers.addr, align 4
%1 = getelementptr inbounds nuw { ptr, ptr }, ptr %exception_pointers, i32 0, i32 0
%2 = load ptr, ptr %1, align 4
%3 = load i32, ptr %2, align 4
store i32 %3, ptr %__exception_code, align 4
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%4 = load i32, ptr %x, align 32
call arm_aapcs_vfpcc void @foo(i32 noundef %4)
ret i32 1
}
%class.anon = type { i8 }
define void @stack_realign_filter() #0 personality ptr @__C_specific_handler {
; CHECK-LABEL: stack_realign_filter:
; CHECK: push {r6, lr}
; CHECK: mov r6, sp
; CHECK: $Mstack_realign_filter$frame_escape_0 = 4
; CHECK: bl foo
entry:
%o = alloca %struct.S, align 4
%__exception_code = alloca i32, 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 void @foo(i32 noundef %0)
to label %invoke.cont unwind label %catch.dispatch
catch.dispatch: ; preds = %entry
%1 = catchswitch within none [label %__except.ret] unwind to caller
__except.ret: ; preds = %catch.dispatch
%2 = catchpad within %1 [ptr @"?filt$0@0@stack_realign_filter@@"]
catchret from %2 to label %__except
__except: ; preds = %__except.ret
; CHECK: ldr r0, [r6, #4]
; CHECK: bl foo
%3 = call i32 @llvm.eh.exceptioncode(token %2)
store i32 %3, ptr %__exception_code, align 4
%x1 = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%4 = load i32, ptr %x1, align 4
call void @foo(i32 noundef %4)
br label %__try.cont
__try.cont: ; preds = %__except, %invoke.cont
ret void
invoke.cont: ; preds = %entry
br label %__try.cont
}
define internal arm_aapcs_vfpcc i32 @"?filt$0@0@stack_realign_filter@@"(ptr noundef %exception_pointers, ptr noundef %frame_pointer) #0 {
; CHECK-LABEL: "?filt$0@0@stack_realign_filter@@":
; CHECK: push.w {r11, lr}
; CHECK: mov r11, sp
; CHECK: bfc r4, #0, #5
; CHECK-NOT: mov r6, sp
; CHECK: ldr r0, [sp, #{{[0-9]+}}]
; CHECK-NEXT: bl foo
; CHECK: ldr r0, [r6, r{{.*}}]
; CHECK-NEXT: bl foo
entry:
%this.addr.i = alloca ptr, align 4
%p.i = alloca %struct.S, align 32
%frame_pointer.addr = alloca ptr, align 4
%exception_pointers.addr = alloca ptr, align 4
%0 = call ptr @llvm.eh.recoverfp(ptr @stack_realign_filter, ptr %frame_pointer)
%o = call ptr @llvm.localrecover(ptr @stack_realign_filter, ptr %0, i32 0)
%__exception_code = alloca i32, align 4
%ref.tmp = alloca %class.anon, align 1
store ptr %frame_pointer, ptr %frame_pointer.addr, align 4
store ptr %exception_pointers, ptr %exception_pointers.addr, align 4
%1 = getelementptr inbounds nuw { ptr, ptr }, ptr %exception_pointers, i32 0, i32 0
%2 = load ptr, ptr %1, align 4
%3 = load i32, ptr %2, align 4
store i32 %3, ptr %__exception_code, align 4
store ptr %ref.tmp, ptr %this.addr.i, align 4
%this1.i = load ptr, ptr %this.addr.i, align 4
%4 = load i32, ptr %p.i, align 32
call void @foo(i32 noundef %4)
%x = getelementptr inbounds nuw %struct.S, ptr %o, i32 0, i32 0
%5 = load i32, ptr %x, align 4
call void @foo(i32 noundef %5)
ret i32 1
}
attributes #0 = { noinline optnone }