diff options
| author | Mandeep Singh Grang <mgrang@quicinc.org> | 2019-02-01 21:41:33 +0000 |
|---|---|---|
| committer | Mandeep Singh Grang <mgrang@quicinc.org> | 2019-02-01 21:41:33 +0000 |
| commit | 70d484d94e3ec1f6c563b3f2e85f88becb977a41 (patch) | |
| tree | 22c4744f728a5db40148af9e0a6a474f6dfc4eff /llvm/test | |
| parent | e95550f508b850c5144b7d1a63bad5680fe83f6d (diff) | |
| download | bcm5719-llvm-70d484d94e3ec1f6c563b3f2e85f88becb977a41.tar.gz bcm5719-llvm-70d484d94e3ec1f6c563b3f2e85f88becb977a41.zip | |
[COFF, ARM64] Fix localaddress to handle stack realignment and variable size objects
Summary: This fixes using the correct stack registers for SEH when stack realignment is needed or when variable size objects are present.
Reviewers: rnk, efriedma, ssijaric, TomTan
Reviewed By: rnk, efriedma
Subscribers: javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D57183
llvm-svn: 352923
Diffstat (limited to 'llvm/test')
| -rw-r--r-- | llvm/test/CodeGen/AArch64/seh-finally.ll | 308 | ||||
| -rw-r--r-- | llvm/test/CodeGen/AArch64/seh-localescape.ll | 30 |
2 files changed, 262 insertions, 76 deletions
diff --git a/llvm/test/CodeGen/AArch64/seh-finally.ll b/llvm/test/CodeGen/AArch64/seh-finally.ll index 3cbbd03385c..2a7b2a925f3 100644 --- a/llvm/test/CodeGen/AArch64/seh-finally.ll +++ b/llvm/test/CodeGen/AArch64/seh-finally.ll @@ -1,67 +1,283 @@ ; RUN: llc -mtriple arm64-windows -o - %s | FileCheck %s -; Function Attrs: noinline optnone uwtable -define dso_local i32 @foo() { +; 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 void @simple_seh() #0 personality i8* bitcast (i32 (...)* @__C_specific_handler to i8*) { entry: -; CHECK-LABEL: foo -; CHECK: orr w8, wzr, #0x1 -; CHECK: mov w0, wzr -; CHECK: mov x1, x29 -; CHECK: .set .Lfoo$frame_escape_0, -4 -; CHECK: stur w8, [x29, #-4] -; CHECK: bl "?fin$0@0@foo@@" -; CHECK: ldur w0, [x29, #-4] +; CHECK-LABEL: simple_seh +; CHECK: add x29, sp, #16 +; CHECK: orr x1, xzr, #0xfffffffffffffffe +; CHECK: stur x1, [x29, #-16] +; CHECK: .set .Lsimple_seh$frame_escape_0, -8 +; CHECK: ldur w0, [x29, #-8] +; CHECK: bl foo + + %o = alloca %struct.S, align 4 + call void (...) @llvm.localescape(%struct.S* %o) + %x = getelementptr inbounds %struct.S, %struct.S* %o, i32 0, i32 0 + %0 = load i32, i32* %x, align 4 + invoke void @foo(i32 %0) #5 + to label %invoke.cont unwind label %ehcleanup - %count = alloca i32, align 4 - call void (...) @llvm.localescape(i32* %count) - store i32 0, i32* %count, align 4 - %0 = load i32, i32* %count, align 4 - %add = add nsw i32 %0, 1 - store i32 %add, i32* %count, align 4 +invoke.cont: ; preds = %entry %1 = call i8* @llvm.localaddress() - call void @"?fin$0@0@foo@@"(i8 0, i8* %1) - %2 = load i32, i32* %count, align 4 - ret i32 %2 + call void @fin_simple_seh(i8 0, i8* %1) + ret void + +ehcleanup: ; preds = %entry + %2 = cleanuppad within none [] + %3 = call i8* @llvm.localaddress() + call void @fin_simple_seh(i8 1, i8* %3) [ "funclet"(token %2) ] + cleanupret from %2 unwind to caller } -define internal void @"?fin$0@0@foo@@"(i8 %abnormal_termination, i8* %frame_pointer) { +define void @fin_simple_seh(i8 %abnormal_termination, i8* %frame_pointer) { entry: -; CHECK-LABEL: @"?fin$0@0@foo@@" -; CHECK: sub sp, sp, #16 -; CHECK: str x1, [sp, #8] -; CHECK: strb w0, [sp, #7] -; CHECK: movz x8, #:abs_g1_s:.Lfoo$frame_escape_0 -; CHECK: movk x8, #:abs_g0_nc:.Lfoo$frame_escape_0 -; CHECK: add x8, x1, x8 -; CHECK: ldr w9, [x8] -; CHECK: add w9, w9, #1 -; CHECK: str w9, [x8] +; CHECK-LABEL: fin_simple_seh +; CHECK: movz x8, #:abs_g1_s:.Lsimple_seh$frame_escape_0 +; CHECK: movk x8, #:abs_g0_nc:.Lsimple_seh$frame_escape_0 +; CHECK: strb w0, [sp, #15] +; CHECK: ldr w0, [x1, x8] +; CHECK: bl foo %frame_pointer.addr = alloca i8*, align 8 %abnormal_termination.addr = alloca i8, align 1 - %0 = call i8* @llvm.localrecover(i8* bitcast (i32 ()* @foo to i8*), i8* %frame_pointer, i32 0) - %count = bitcast i8* %0 to i32* + %0 = call i8* @llvm.localrecover(i8* bitcast (void ()* @simple_seh to i8*), i8* %frame_pointer, i32 0) + %o = bitcast i8* %0 to %struct.S* store i8* %frame_pointer, i8** %frame_pointer.addr, align 8 store i8 %abnormal_termination, i8* %abnormal_termination.addr, align 1 - %1 = zext i8 %abnormal_termination to i32 - %cmp = icmp eq i32 %1, 0 - br i1 %cmp, label %if.then, label %if.end + %x = getelementptr inbounds %struct.S, %struct.S* %o, i32 0, i32 0 + %1 = load i32, i32* %x, align 4 + call void @foo(i32 %1) + ret void +} -if.then: ; preds = %entry - %2 = load i32, i32* %count, align 4 - %add = add nsw i32 %2, 1 - store i32 %add, i32* %count, align 4 - br label %if.end +; Test SEH when stack realignment is needed in case highly aligned stack objects are present. +define void @stack_realign() #0 personality i8* bitcast (i32 (...)* @__C_specific_handler to i8*) { +entry: +; CHECK-LABEL: stack_realign +; CHECK: add x29, sp, #16 +; CHECK: sub x9, sp, #64 +; CHECK: and sp, x9, #0xffffffffffffffe0 +; CHECK: mov x19, sp +; CHECK: orr x1, xzr, #0xfffffffffffffffe +; CHECK: stur x1, [x19, #16] +; CHECK: .set .Lstack_realign$frame_escape_0, 32 +; CHECK: ldr w0, [x19, #32] +; CHECK: bl foo + + %o = alloca %struct.S, align 32 + call void (...) @llvm.localescape(%struct.S* %o) + %x = getelementptr inbounds %struct.S, %struct.S* %o, i32 0, i32 0 + %0 = load i32, i32* %x, align 32 + invoke void @foo(i32 %0) #5 + to label %invoke.cont unwind label %ehcleanup -if.end: ; preds = %if.then, %entry +invoke.cont: ; preds = %entry + %1 = call i8* @llvm.localaddress() + call void @fin_stack_realign(i8 0, i8* %1) ret void + +ehcleanup: ; preds = %entry + %2 = cleanuppad within none [] + %3 = call i8* @llvm.localaddress() + call void @fin_stack_realign(i8 1, i8* %3) [ "funclet"(token %2) ] + cleanupret from %2 unwind to caller } -; Function Attrs: nounwind readnone -declare i8* @llvm.localrecover(i8*, i8*, i32) +define void @fin_stack_realign(i8 %abnormal_termination, i8* %frame_pointer) { +entry: +; CHECK-LABEL: fin_stack_realign +; CHECK: movz x8, #:abs_g1_s:.Lstack_realign$frame_escape_0 +; CHECK: movk x8, #:abs_g0_nc:.Lstack_realign$frame_escape_0 +; CHECK: strb w0, [sp, #15] +; CHECK: ldr w0, [x1, x8] +; CHECK: bl foo -; Function Attrs: nounwind readnone -declare i8* @llvm.localaddress() + %frame_pointer.addr = alloca i8*, align 8 + %abnormal_termination.addr = alloca i8, align 1 + %0 = call i8* @llvm.localrecover(i8* bitcast (void ()* @stack_realign to i8*), i8* %frame_pointer, i32 0) + %o = bitcast i8* %0 to %struct.S* + store i8* %frame_pointer, i8** %frame_pointer.addr, align 8 + store i8 %abnormal_termination, i8* %abnormal_termination.addr, align 1 + %x = getelementptr inbounds %struct.S, %struct.S* %o, i32 0, i32 0 + %1 = load i32, i32* %x, align 32 + call void @foo(i32 %1) + ret void +} + +; Test SEH when variable size objects are present on the stack. Note: Escaped vla's are current not supported by SEH. +define void @vla_present(i32 %n) #0 personality i8* bitcast (i32 (...)* @__C_specific_handler to i8*) { +entry: +; CHECK-LABEL: vla_present +; CHECK: add x29, sp, #32 +; CHECK: orr x1, xzr, #0xfffffffffffffffe +; CHECK: stur x1, [x29, #-32] +; CHECK: .set .Lvla_present$frame_escape_0, -4 +; CHECK: stur w0, [x29, #-4] +; CHECK: ldur w8, [x29, #-4] +; CHECK: mov x9, sp +; CHECK: stur x9, [x29, #-16] +; CHECK: stur x8, [x29, #-24] +; CHECK: ldur w0, [x29, #-4] +; CHECK: bl foo + + %n.addr = alloca i32, align 4 + %saved_stack = alloca i8*, align 8 + %__vla_expr0 = alloca i64, align 8 + call void (...) @llvm.localescape(i32* %n.addr) + store i32 %n, i32* %n.addr, align 4 + %0 = load i32, i32* %n.addr, align 4 + %1 = zext i32 %0 to i64 + %2 = call i8* @llvm.stacksave() + store i8* %2, i8** %saved_stack, align 8 + %vla = alloca i32, i64 %1, align 4 + store i64 %1, i64* %__vla_expr0, align 8 + %3 = load i32, i32* %n.addr, align 4 + invoke void @foo(i32 %3) #5 + to label %invoke.cont unwind label %ehcleanup + +invoke.cont: ; preds = %entry + %4 = call i8* @llvm.localaddress() + call void @fin_vla_present(i8 0, i8* %4) + %5 = load i8*, i8** %saved_stack, align 8 + call void @llvm.stackrestore(i8* %5) + ret void + +ehcleanup: ; preds = %entry + %6 = cleanuppad within none [] + %7 = call i8* @llvm.localaddress() + call void @fin_vla_present(i8 1, i8* %7) [ "funclet"(token %6) ] + cleanupret from %6 unwind to caller +} -; Function Attrs: nounwind +define void @fin_vla_present(i8 %abnormal_termination, i8* %frame_pointer) { +entry: +; CHECK-LABEL: fin_vla_present +; CHECK: movz x8, #:abs_g1_s:.Lvla_present$frame_escape_0 +; CHECK: movk x8, #:abs_g0_nc:.Lvla_present$frame_escape_0 +; CHECK: strb w0, [sp, #15] +; CHECK: ldr w0, [x1, x8] +; CHECK: bl foo + + %frame_pointer.addr = alloca i8*, align 8 + %abnormal_termination.addr = alloca i8, align 1 + %0 = call i8* @llvm.localrecover(i8* bitcast (void (i32)* @vla_present to i8*), i8* %frame_pointer, i32 0) + %n.addr = bitcast i8* %0 to i32* + store i8* %frame_pointer, i8** %frame_pointer.addr, align 8 + store i8 %abnormal_termination, i8* %abnormal_termination.addr, align 1 + %1 = load i32, i32* %n.addr, align 4 + call void @foo(i32 %1) + ret void +} + +; Test when both vla's and highly aligned objects are present on stack. +define void @vla_and_realign(i32 %n) #0 personality i8* bitcast (i32 (...)* @__C_specific_handler to i8*) { +entry: +; CHECK-LABEL: vla_and_realign +; CHECK: add x29, sp, #16 +; CHECK: sub x9, sp, #64 +; CHECK: and sp, x9, #0xffffffffffffffe0 +; CHECK: mov x19, sp +; CHECK: orr x1, xzr, #0xfffffffffffffffe +; CHECK: stur x1, [x19] +; CHECK: .set .Lvla_and_realign$frame_escape_0, 32 +; CHECK: stur w0, [x29, #-4] +; CHECK: ldur w8, [x29, #-4] +; CHECK: mov x9, sp +; CHECK: str x9, [x19, #24] +; CHECK: str x8, [x19, #16] +; CHECK: ldr w0, [x19, #32] +; CHECK: bl foo + + %n.addr = alloca i32, align 4 + %o = alloca %struct.S, align 32 + %saved_stack = alloca i8*, align 8 + %__vla_expr0 = alloca i64, align 8 + call void (...) @llvm.localescape(%struct.S* %o) + store i32 %n, i32* %n.addr, align 4 + %0 = load i32, i32* %n.addr, align 4 + %1 = zext i32 %0 to i64 + %2 = call i8* @llvm.stacksave() + store i8* %2, i8** %saved_stack, align 8 + %vla = alloca i32, i64 %1, align 4 + store i64 %1, i64* %__vla_expr0, align 8 + %x = getelementptr inbounds %struct.S, %struct.S* %o, i32 0, i32 0 + %3 = load i32, i32* %x, align 32 + invoke void @foo(i32 %3) #5 + to label %invoke.cont unwind label %ehcleanup + +invoke.cont: ; preds = %entry + %4 = call i8* @llvm.localaddress() + call void @fin_vla_and_realign(i8 0, i8* %4) + %5 = load i8*, i8** %saved_stack, align 8 + call void @llvm.stackrestore(i8* %5) + ret void + +ehcleanup: ; preds = %entry + %6 = cleanuppad within none [] + %7 = call i8* @llvm.localaddress() + call void @fin_vla_and_realign(i8 1, i8* %7) [ "funclet"(token %6) ] + cleanupret from %6 unwind to caller +} + +define void @fin_vla_and_realign(i8 %abnormal_termination, i8* %frame_pointer) { +entry: +; CHECK-LABEL: fin_vla_and_realign +; CHECK: movz x8, #:abs_g1_s:.Lvla_and_realign$frame_escape_0 +; CHECK: movk x8, #:abs_g0_nc:.Lvla_and_realign$frame_escape_0 +; CHECK: strb w0, [sp, #15] +; CHECK: ldr w0, [x1, x8] +; CHECK: bl foo + + %frame_pointer.addr = alloca i8*, align 8 + %abnormal_termination.addr = alloca i8, align 1 + %0 = call i8* @llvm.localrecover(i8* bitcast (void (i32)* @vla_and_realign to i8*), i8* %frame_pointer, i32 0) + %o = bitcast i8* %0 to %struct.S* + store i8* %frame_pointer, i8** %frame_pointer.addr, align 8 + store i8 %abnormal_termination, i8* %abnormal_termination.addr, align 1 + %x = getelementptr inbounds %struct.S, %struct.S* %o, i32 0, i32 0 + %1 = load i32, i32* %x, align 32 + call void @foo(i32 %1) + ret void +} + +declare void @foo(i32) +declare void @llvm.stackrestore(i8*) +declare i8* @llvm.stacksave() +declare i8* @llvm.localrecover(i8*, i8*, i32) +declare i8* @llvm.localaddress() declare void @llvm.localescape(...) +declare i32 @__C_specific_handler(...) + +attributes #0 = { noinline optnone } diff --git a/llvm/test/CodeGen/AArch64/seh-localescape.ll b/llvm/test/CodeGen/AArch64/seh-localescape.ll deleted file mode 100644 index 0a1675014f6..00000000000 --- a/llvm/test/CodeGen/AArch64/seh-localescape.ll +++ /dev/null @@ -1,30 +0,0 @@ -; RUN: llc -mtriple arm64-windows %s -o - | FileCheck %s - -; Function Attrs: noinline nounwind optnone uwtable -define dso_local i32 @foo() { -entry: -; CHECK-LABEL: foo -; CHECK: .set .Lfoo$frame_escape_0, -4 - - %count = alloca i32, align 4 - call void (...) @llvm.localescape(i32* %count) - ret i32 0 -} - -define internal i32 @"?filt$0@0@foo@@"(i8* %exception_pointers, i8* %frame_pointer) { -entry: -; CHECK-LABEL: @"?filt$0@0@foo@@" -; CHECK: movz x8, #:abs_g1_s:.Lfoo$frame_escape_0 -; CHECK: movk x8, #:abs_g0_nc:.Lfoo$frame_escape_0 - - %0 = call i8* @llvm.localrecover(i8* bitcast (i32 ()* @foo to i8*), i8* %frame_pointer, i32 0) - %count = bitcast i8* %0 to i32* - %1 = load i32, i32* %count, align 4 - ret i32 %1 -} - -; Function Attrs: nounwind readnone -declare i8* @llvm.localrecover(i8*, i8*, i32) #2 - -; Function Attrs: nounwind -declare void @llvm.localescape(...) #3 |

