2018-04-12 00:03:07 +08:00
|
|
|
; RUN: llc -fast-isel-sink-local-values -verify-machineinstrs < %s -mtriple=armv7-apple-ios | FileCheck --check-prefix=CHECK-APPLE --check-prefix=CHECK-ARMV7 %s
|
|
|
|
; RUN: llc -fast-isel-sink-local-values -verify-machineinstrs -O0 < %s -mtriple=armv7-apple-ios | FileCheck --check-prefix=CHECK-O0 %s
|
|
|
|
; RUN: llc -fast-isel-sink-local-values -verify-machineinstrs < %s -mtriple=armv7-linux-androideabi | FileCheck --check-prefix=CHECK-ANDROID %s
|
2016-04-12 05:08:06 +08:00
|
|
|
|
|
|
|
declare i8* @malloc(i64)
|
|
|
|
declare void @free(i8*)
|
|
|
|
%swift_error = type { i64, i8 }
|
|
|
|
%struct.S = type { i32, i32, i32, i32, i32, i32 }
|
|
|
|
|
|
|
|
; This tests the basic usage of a swifterror parameter. "foo" is the function
|
|
|
|
; that takes a swifterror parameter and "caller" is the caller of "foo".
|
|
|
|
define float @foo(%swift_error** swifterror %error_ptr_ref) {
|
|
|
|
; CHECK-APPLE-LABEL: foo:
|
|
|
|
; CHECK-APPLE: mov r0, #16
|
|
|
|
; CHECK-APPLE: malloc
|
|
|
|
; CHECK-APPLE-DAG: mov [[ID:r[0-9]+]], #1
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE-DAG: mov r8, r{{.*}}
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE-DAG: strb [[ID]], [r{{.*}}, #8]
|
|
|
|
|
|
|
|
; CHECK-O0-LABEL: foo:
|
|
|
|
; CHECK-O0: mov r{{.*}}, #16
|
|
|
|
; CHECK-O0: malloc
|
|
|
|
; CHECK-O0: mov [[ID2:r[0-9]+]], r0
|
2020-09-22 17:20:10 +08:00
|
|
|
; CHECK-O0: mov r8, [[ID2]]
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: mov [[ID:r[0-9]+]], #1
|
|
|
|
; CHECK-O0: strb [[ID]], {{\[}}[[ID2]], #8]
|
2016-04-12 05:08:06 +08:00
|
|
|
entry:
|
|
|
|
%call = call i8* @malloc(i64 16)
|
|
|
|
%call.0 = bitcast i8* %call to %swift_error*
|
|
|
|
store %swift_error* %call.0, %swift_error** %error_ptr_ref
|
|
|
|
%tmp = getelementptr inbounds i8, i8* %call, i64 8
|
|
|
|
store i8 1, i8* %tmp
|
|
|
|
ret float 1.0
|
|
|
|
}
|
|
|
|
|
|
|
|
; "caller" calls "foo" that takes a swifterror parameter.
|
|
|
|
define float @caller(i8* %error_ref) {
|
|
|
|
; CHECK-APPLE-LABEL: caller:
|
|
|
|
; CHECK-APPLE-DAG: mov [[ID:r[0-9]+]], r0
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE-DAG: mov r8, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE: bl {{.*}}foo
|
2018-02-16 17:51:01 +08:00
|
|
|
; CHECK-APPLE: mov r0, r8
|
2018-02-28 00:59:10 +08:00
|
|
|
; CHECK-APPLE: cmp r8, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; Access part of the error object and save it to error_ref
|
2018-02-16 17:51:01 +08:00
|
|
|
; CHECK-APPLE: ldrbeq [[CODE:r[0-9]+]], [r0, #8]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE: strbeq [[CODE]], [{{.*}}[[ID]]]
|
2016-10-08 06:06:55 +08:00
|
|
|
; CHECK-APPLE: bl {{.*}}free
|
2016-04-12 05:08:06 +08:00
|
|
|
|
|
|
|
; CHECK-O0-LABEL: caller:
|
|
|
|
; spill r0
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-O0-DAG: mov r8, #0
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0-DAG: str r0, [sp[[SLOT:(, #[0-9]+)?]]]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: bl {{.*}}foo
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-O0: mov [[TMP:r[0-9]+]], r8
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: str [[TMP]], [sp[[SLOT2:(, #[0-9]+)?]]]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: bne
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: ldr [[ID:r[0-9]+]], [sp[[SLOT]]]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: ldrb [[CODE:r[0-9]+]], [r0, #8]
|
|
|
|
; CHECK-O0: strb [[CODE]], [{{.*}}[[ID]]]
|
2016-10-08 06:06:55 +08:00
|
|
|
; reload r0
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: ldr r0, [sp[[SLOT2]]]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: free
|
|
|
|
entry:
|
|
|
|
%error_ptr_ref = alloca swifterror %swift_error*
|
|
|
|
store %swift_error* null, %swift_error** %error_ptr_ref
|
|
|
|
%call = call float @foo(%swift_error** swifterror %error_ptr_ref)
|
|
|
|
%error_from_foo = load %swift_error*, %swift_error** %error_ptr_ref
|
|
|
|
%had_error_from_foo = icmp ne %swift_error* %error_from_foo, null
|
|
|
|
%tmp = bitcast %swift_error* %error_from_foo to i8*
|
|
|
|
br i1 %had_error_from_foo, label %handler, label %cont
|
|
|
|
cont:
|
|
|
|
%v1 = getelementptr inbounds %swift_error, %swift_error* %error_from_foo, i64 0, i32 1
|
|
|
|
%t = load i8, i8* %v1
|
|
|
|
store i8 %t, i8* %error_ref
|
|
|
|
br label %handler
|
|
|
|
handler:
|
|
|
|
call void @free(i8* %tmp)
|
|
|
|
ret float 1.0
|
|
|
|
}
|
|
|
|
|
|
|
|
; "caller2" is the caller of "foo", it calls "foo" inside a loop.
|
|
|
|
define float @caller2(i8* %error_ref) {
|
|
|
|
; CHECK-APPLE-LABEL: caller2:
|
|
|
|
; CHECK-APPLE-DAG: mov [[ID:r[0-9]+]], r0
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE-DAG: mov r8, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE: bl {{.*}}foo
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE: cmp r8, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE: bne
|
|
|
|
; Access part of the error object and save it to error_ref
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE: ldrb [[CODE:r[0-9]+]], [r8, #8]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE: strb [[CODE]], [{{.*}}[[ID]]]
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE: mov r0, r8
|
2016-10-08 06:06:55 +08:00
|
|
|
; CHECK-APPLE: bl {{.*}}free
|
2016-04-12 05:08:06 +08:00
|
|
|
|
|
|
|
; CHECK-O0-LABEL: caller2:
|
|
|
|
; spill r0
|
|
|
|
; CHECK-O0-DAG: str r0,
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-O0-DAG: mov r8, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: bl {{.*}}foo
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-O0: mov r{{.*}}, r8
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: str r0, [sp{{(, #[0-9]+)?}}]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: bne
|
|
|
|
; CHECK-O0: ble
|
|
|
|
; reload r0
|
|
|
|
; CHECK-O0: ldr [[ID:r[0-9]+]],
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: ldrb [[CODE:r[0-9]+]], [r0, #8]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: strb [[CODE]], [{{.*}}[[ID]]]
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: ldr r0, [sp{{(, #[0-9]+)?}}]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: free
|
|
|
|
entry:
|
|
|
|
%error_ptr_ref = alloca swifterror %swift_error*
|
|
|
|
br label %bb_loop
|
|
|
|
bb_loop:
|
|
|
|
store %swift_error* null, %swift_error** %error_ptr_ref
|
|
|
|
%call = call float @foo(%swift_error** swifterror %error_ptr_ref)
|
|
|
|
%error_from_foo = load %swift_error*, %swift_error** %error_ptr_ref
|
|
|
|
%had_error_from_foo = icmp ne %swift_error* %error_from_foo, null
|
|
|
|
%tmp = bitcast %swift_error* %error_from_foo to i8*
|
|
|
|
br i1 %had_error_from_foo, label %handler, label %cont
|
|
|
|
cont:
|
|
|
|
%cmp = fcmp ogt float %call, 1.000000e+00
|
|
|
|
br i1 %cmp, label %bb_end, label %bb_loop
|
|
|
|
bb_end:
|
|
|
|
%v1 = getelementptr inbounds %swift_error, %swift_error* %error_from_foo, i64 0, i32 1
|
|
|
|
%t = load i8, i8* %v1
|
|
|
|
store i8 %t, i8* %error_ref
|
|
|
|
br label %handler
|
|
|
|
handler:
|
|
|
|
call void @free(i8* %tmp)
|
|
|
|
ret float 1.0
|
|
|
|
}
|
|
|
|
|
|
|
|
; "foo_if" is a function that takes a swifterror parameter, it sets swifterror
|
|
|
|
; under a certain condition.
|
|
|
|
define float @foo_if(%swift_error** swifterror %error_ptr_ref, i32 %cc) {
|
|
|
|
; CHECK-APPLE-LABEL: foo_if:
|
|
|
|
; CHECK-APPLE: cmp r0, #0
|
|
|
|
; CHECK-APPLE: eq
|
|
|
|
; CHECK-APPLE: mov r0, #16
|
|
|
|
; CHECK-APPLE: malloc
|
2018-02-16 17:51:01 +08:00
|
|
|
; CHECK-APPLE-DAG: mov [[ID:r[0-9]+]], #1
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE-DAG: mov r8, r{{.*}}
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE-DAG: strb [[ID]], [r{{.*}}, #8]
|
|
|
|
|
|
|
|
; CHECK-O0-LABEL: foo_if:
|
|
|
|
; spill to stack
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-O0: str r8
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: cmp r0, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: beq
|
|
|
|
; CHECK-O0: mov r0, #16
|
|
|
|
; CHECK-O0: malloc
|
|
|
|
; CHECK-O0: mov [[ID:r[0-9]+]], r0
|
|
|
|
; CHECK-O0: mov [[ID2:[a-z0-9]+]], #1
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: strb [[ID2]], {{\[}}[[ID]], #8]
|
2016-04-12 05:08:06 +08:00
|
|
|
; reload from stack
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-O0: ldr r8
|
2016-04-12 05:08:06 +08:00
|
|
|
entry:
|
|
|
|
%cond = icmp ne i32 %cc, 0
|
|
|
|
br i1 %cond, label %gen_error, label %normal
|
|
|
|
|
|
|
|
gen_error:
|
|
|
|
%call = call i8* @malloc(i64 16)
|
|
|
|
%call.0 = bitcast i8* %call to %swift_error*
|
|
|
|
store %swift_error* %call.0, %swift_error** %error_ptr_ref
|
|
|
|
%tmp = getelementptr inbounds i8, i8* %call, i64 8
|
|
|
|
store i8 1, i8* %tmp
|
|
|
|
ret float 1.0
|
|
|
|
|
|
|
|
normal:
|
|
|
|
ret float 0.0
|
|
|
|
}
|
|
|
|
|
|
|
|
; "foo_loop" is a function that takes a swifterror parameter, it sets swifterror
|
|
|
|
; under a certain condition inside a loop.
|
|
|
|
define float @foo_loop(%swift_error** swifterror %error_ptr_ref, i32 %cc, float %cc2) {
|
|
|
|
; CHECK-APPLE-LABEL: foo_loop:
|
|
|
|
; CHECK-APPLE: mov [[CODE:r[0-9]+]], r0
|
|
|
|
; swifterror is kept in a register
|
|
|
|
; CHECK-APPLE: cmp [[CODE]], #0
|
|
|
|
; CHECK-APPLE: beq
|
|
|
|
; CHECK-APPLE: mov r0, #16
|
|
|
|
; CHECK-APPLE: malloc
|
2018-02-28 00:59:10 +08:00
|
|
|
; CHECK-APPLE: strb r{{.*}}, [r0, #8]
|
2019-06-15 07:08:59 +08:00
|
|
|
; CHECK-APPLE: b
|
2016-04-12 05:08:06 +08:00
|
|
|
|
|
|
|
; CHECK-O0-LABEL: foo_loop:
|
|
|
|
; CHECK-O0: cmp r{{.*}}, #0
|
|
|
|
; CHECK-O0: beq
|
[FastISel] Sink local value materializations to first use
Summary:
Local values are constants, global addresses, and stack addresses that
can't be folded into the instruction that uses them. For example, when
storing the address of a global variable into memory, we need to
materialize that address into a register.
FastISel doesn't want to materialize any given local value more than
once, so it generates all local value materialization code at
EmitStartPt, which always dominates the current insertion point. This
allows it to maintain a map of local value registers, and it knows that
the local value area will always dominate the current insertion point.
The downside is that local value instructions are always emitted without
a source location. This is done to prevent jumpy line tables, but it
means that the local value area will be considered part of the previous
statement. Consider this C code:
call1(); // line 1
++global; // line 2
++global; // line 3
call2(&global, &local); // line 4
Today we end up with assembly and line tables like this:
.loc 1 1
callq call1
leaq global(%rip), %rdi
leaq local(%rsp), %rsi
.loc 1 2
addq $1, global(%rip)
.loc 1 3
addq $1, global(%rip)
.loc 1 4
callq call2
The LEA instructions in the local value area have no source location and
are treated as being on line 1. Stepping through the code in a debugger
and correlating it with the assembly won't make much sense, because
these materializations are only required for line 4.
This is actually problematic for the VS debugger "set next statement"
feature, which effectively assumes that there are no registers live
across statement boundaries. By sinking the local value code into the
statement and fixing up the source location, we can make that feature
work. This was filed as https://bugs.llvm.org/show_bug.cgi?id=35975 and
https://crbug.com/793819.
This change is obviously not enough to make this feature work reliably
in all cases, but I felt that it was worth doing anyway because it
usually generates smaller, more comprehensible -O0 code. I measured a
0.12% regression in code generation time with LLC on the sqlite3
amalgamation, so I think this is worth doing.
There are some special cases worth calling out in the commit message:
1. local values materialized for phis
2. local values used by no-op casts
3. dead local value code
Local values can be materialized for phis, and this does not show up as
a vreg use in MachineRegisterInfo. In this case, if there are no other
uses, this patch sinks the value to the first terminator, EH label, or
the end of the BB if nothing else exists.
Local values may also be used by no-op casts, which adds the register to
the RegFixups table. Without reversing the RegFixups map direction, we
don't have enough information to sink these instructions.
Lastly, if the local value register has no other uses, we can delete it.
This comes up when fastisel tries two instruction selection approaches
and the first materializes the value but fails and the second succeeds
without using the local value.
Reviewers: aprantl, dblaikie, qcolombet, MatzeB, vsk, echristo
Subscribers: dotdash, chandlerc, hans, sdardis, amccarth, javed.absar, zturner, llvm-commits, hiraditya
Differential Revision: https://reviews.llvm.org/D43093
llvm-svn: 327581
2018-03-15 05:54:21 +08:00
|
|
|
; CHECK-O0: mov r0, #16
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: malloc
|
|
|
|
; CHECK-O0-DAG: mov [[ID:r[0-9]+]], r0
|
[FastISel] Sink local value materializations to first use
Summary:
Local values are constants, global addresses, and stack addresses that
can't be folded into the instruction that uses them. For example, when
storing the address of a global variable into memory, we need to
materialize that address into a register.
FastISel doesn't want to materialize any given local value more than
once, so it generates all local value materialization code at
EmitStartPt, which always dominates the current insertion point. This
allows it to maintain a map of local value registers, and it knows that
the local value area will always dominate the current insertion point.
The downside is that local value instructions are always emitted without
a source location. This is done to prevent jumpy line tables, but it
means that the local value area will be considered part of the previous
statement. Consider this C code:
call1(); // line 1
++global; // line 2
++global; // line 3
call2(&global, &local); // line 4
Today we end up with assembly and line tables like this:
.loc 1 1
callq call1
leaq global(%rip), %rdi
leaq local(%rsp), %rsi
.loc 1 2
addq $1, global(%rip)
.loc 1 3
addq $1, global(%rip)
.loc 1 4
callq call2
The LEA instructions in the local value area have no source location and
are treated as being on line 1. Stepping through the code in a debugger
and correlating it with the assembly won't make much sense, because
these materializations are only required for line 4.
This is actually problematic for the VS debugger "set next statement"
feature, which effectively assumes that there are no registers live
across statement boundaries. By sinking the local value code into the
statement and fixing up the source location, we can make that feature
work. This was filed as https://bugs.llvm.org/show_bug.cgi?id=35975 and
https://crbug.com/793819.
This change is obviously not enough to make this feature work reliably
in all cases, but I felt that it was worth doing anyway because it
usually generates smaller, more comprehensible -O0 code. I measured a
0.12% regression in code generation time with LLC on the sqlite3
amalgamation, so I think this is worth doing.
There are some special cases worth calling out in the commit message:
1. local values materialized for phis
2. local values used by no-op casts
3. dead local value code
Local values can be materialized for phis, and this does not show up as
a vreg use in MachineRegisterInfo. In this case, if there are no other
uses, this patch sinks the value to the first terminator, EH label, or
the end of the BB if nothing else exists.
Local values may also be used by no-op casts, which adds the register to
the RegFixups table. Without reversing the RegFixups map direction, we
don't have enough information to sink these instructions.
Lastly, if the local value register has no other uses, we can delete it.
This comes up when fastisel tries two instruction selection approaches
and the first materializes the value but fails and the second succeeds
without using the local value.
Reviewers: aprantl, dblaikie, qcolombet, MatzeB, vsk, echristo
Subscribers: dotdash, chandlerc, hans, sdardis, amccarth, javed.absar, zturner, llvm-commits, hiraditya
Differential Revision: https://reviews.llvm.org/D43093
llvm-svn: 327581
2018-03-15 05:54:21 +08:00
|
|
|
; CHECK-O0-DAG: movw [[ID2:.*]], #1
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: strb [[ID2]], [{{.*}}[[ID]], #8]
|
|
|
|
; spill r0
|
|
|
|
; CHECK-O0: str r0, [sp{{.*}}]
|
2019-10-08 16:25:42 +08:00
|
|
|
; CHECK-O0: vcmp
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: ble
|
|
|
|
; reload from stack
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-O0: ldr r8
|
2016-04-12 05:08:06 +08:00
|
|
|
entry:
|
|
|
|
br label %bb_loop
|
|
|
|
|
|
|
|
bb_loop:
|
|
|
|
%cond = icmp ne i32 %cc, 0
|
|
|
|
br i1 %cond, label %gen_error, label %bb_cont
|
|
|
|
|
|
|
|
gen_error:
|
|
|
|
%call = call i8* @malloc(i64 16)
|
|
|
|
%call.0 = bitcast i8* %call to %swift_error*
|
|
|
|
store %swift_error* %call.0, %swift_error** %error_ptr_ref
|
|
|
|
%tmp = getelementptr inbounds i8, i8* %call, i64 8
|
|
|
|
store i8 1, i8* %tmp
|
|
|
|
br label %bb_cont
|
|
|
|
|
|
|
|
bb_cont:
|
|
|
|
%cmp = fcmp ogt float %cc2, 1.000000e+00
|
|
|
|
br i1 %cmp, label %bb_end, label %bb_loop
|
|
|
|
bb_end:
|
|
|
|
ret float 0.0
|
|
|
|
}
|
|
|
|
|
|
|
|
; "foo_sret" is a function that takes a swifterror parameter, it also has a sret
|
|
|
|
; parameter.
|
|
|
|
define void @foo_sret(%struct.S* sret %agg.result, i32 %val1, %swift_error** swifterror %error_ptr_ref) {
|
|
|
|
; CHECK-APPLE-LABEL: foo_sret:
|
|
|
|
; CHECK-APPLE: mov [[SRET:r[0-9]+]], r0
|
|
|
|
; CHECK-APPLE: mov r0, #16
|
|
|
|
; CHECK-APPLE: malloc
|
|
|
|
; CHECK-APPLE: mov [[REG:r[0-9]+]], #1
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE-DAG: mov r8, r0
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE-DAG: strb [[REG]], [r0, #8]
|
|
|
|
; CHECK-APPLE-DAG: str r{{.*}}, [{{.*}}[[SRET]], #4]
|
|
|
|
|
|
|
|
; CHECK-O0-LABEL: foo_sret:
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0-DAG: mov r{{.*}}, #16
|
2016-04-12 05:08:06 +08:00
|
|
|
; spill to stack: sret and val1
|
|
|
|
; CHECK-O0-DAG: str r0
|
|
|
|
; CHECK-O0-DAG: str r1
|
|
|
|
; CHECK-O0: malloc
|
|
|
|
; reload from stack: sret and val1
|
|
|
|
; CHECK-O0: ldr
|
|
|
|
; CHECK-O0: ldr
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0-DAG: mov r8
|
|
|
|
; CHECK-O0-DAG: mov [[ID:r[0-9]+]], #1
|
|
|
|
; CHECK-O0-DAG: strb [[ID]], [{{r[0-9]+}}, #8]
|
|
|
|
; CHECK-O0-DAG: str r{{.*}}, [{{.*}}, #4]
|
2016-04-12 05:08:06 +08:00
|
|
|
entry:
|
|
|
|
%call = call i8* @malloc(i64 16)
|
|
|
|
%call.0 = bitcast i8* %call to %swift_error*
|
|
|
|
store %swift_error* %call.0, %swift_error** %error_ptr_ref
|
|
|
|
%tmp = getelementptr inbounds i8, i8* %call, i64 8
|
|
|
|
store i8 1, i8* %tmp
|
|
|
|
%v2 = getelementptr inbounds %struct.S, %struct.S* %agg.result, i32 0, i32 1
|
|
|
|
store i32 %val1, i32* %v2
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
; "caller3" calls "foo_sret" that takes a swifterror parameter.
|
|
|
|
define float @caller3(i8* %error_ref) {
|
|
|
|
; CHECK-APPLE-LABEL: caller3:
|
|
|
|
; CHECK-APPLE: mov [[ID:r[0-9]+]], r0
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE: mov r8, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE: bl {{.*}}foo_sret
|
2018-02-16 17:51:01 +08:00
|
|
|
; CHECK-APPLE: mov r0, r8
|
2018-02-28 00:59:10 +08:00
|
|
|
; CHECK-APPLE: cmp r8, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; Access part of the error object and save it to error_ref
|
2018-02-16 17:51:01 +08:00
|
|
|
; CHECK-APPLE: ldrbeq [[CODE:r[0-9]+]], [r0, #8]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE: strbeq [[CODE]], [{{.*}}[[ID]]]
|
2016-10-08 06:06:55 +08:00
|
|
|
; CHECK-APPLE: bl {{.*}}free
|
2016-04-12 05:08:06 +08:00
|
|
|
|
|
|
|
; CHECK-O0-LABEL: caller3:
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-O0-DAG: mov r8, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0-DAG: mov r1
|
|
|
|
; CHECK-O0: bl {{.*}}foo_sret
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-O0: mov [[ID2:r[0-9]+]], r8
|
2020-09-22 17:20:10 +08:00
|
|
|
; CHECK-O0: str [[ID2]], [sp[[SLOT:.*]]]
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: cmp r8
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: bne
|
|
|
|
; Access part of the error object and save it to error_ref
|
2020-09-22 17:20:10 +08:00
|
|
|
; CHECK-O0: ldr [[ID:r[0-9]+]]
|
2020-09-22 20:55:54 +08:00
|
|
|
; CHECK-O0: ldrb [[CODE:r[0-9]+]]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-O0: strb [[CODE]], [{{.*}}[[ID]]]
|
2016-10-08 06:06:55 +08:00
|
|
|
; CHECK-O0: ldr r0, [sp[[SLOT]]
|
|
|
|
; CHECK-O0: bl {{.*}}free
|
2016-04-12 05:08:06 +08:00
|
|
|
entry:
|
|
|
|
%s = alloca %struct.S, align 8
|
|
|
|
%error_ptr_ref = alloca swifterror %swift_error*
|
|
|
|
store %swift_error* null, %swift_error** %error_ptr_ref
|
|
|
|
call void @foo_sret(%struct.S* sret %s, i32 1, %swift_error** swifterror %error_ptr_ref)
|
|
|
|
%error_from_foo = load %swift_error*, %swift_error** %error_ptr_ref
|
|
|
|
%had_error_from_foo = icmp ne %swift_error* %error_from_foo, null
|
|
|
|
%tmp = bitcast %swift_error* %error_from_foo to i8*
|
|
|
|
br i1 %had_error_from_foo, label %handler, label %cont
|
|
|
|
cont:
|
|
|
|
%v1 = getelementptr inbounds %swift_error, %swift_error* %error_from_foo, i64 0, i32 1
|
|
|
|
%t = load i8, i8* %v1
|
|
|
|
store i8 %t, i8* %error_ref
|
|
|
|
br label %handler
|
|
|
|
handler:
|
|
|
|
call void @free(i8* %tmp)
|
|
|
|
ret float 1.0
|
|
|
|
}
|
|
|
|
|
|
|
|
; "foo_vararg" is a function that takes a swifterror parameter, it also has
|
|
|
|
; variable number of arguments.
|
|
|
|
declare void @llvm.va_start(i8*) nounwind
|
|
|
|
define float @foo_vararg(%swift_error** swifterror %error_ptr_ref, ...) {
|
|
|
|
; CHECK-APPLE-LABEL: foo_vararg:
|
|
|
|
; CHECK-APPLE: mov r0, #16
|
|
|
|
; CHECK-APPLE: malloc
|
2018-02-16 17:51:01 +08:00
|
|
|
; CHECK-APPLE: mov r8, r0
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE: mov [[ID:r[0-9]+]], #1
|
2018-02-16 17:51:01 +08:00
|
|
|
; CHECK-APPLE-DAG: strb [[ID]], [r8, #8]
|
2016-04-12 05:08:06 +08:00
|
|
|
|
|
|
|
entry:
|
|
|
|
%call = call i8* @malloc(i64 16)
|
|
|
|
%call.0 = bitcast i8* %call to %swift_error*
|
|
|
|
store %swift_error* %call.0, %swift_error** %error_ptr_ref
|
|
|
|
%tmp = getelementptr inbounds i8, i8* %call, i64 8
|
|
|
|
store i8 1, i8* %tmp
|
|
|
|
|
|
|
|
%args = alloca i8*, align 8
|
|
|
|
%a10 = alloca i32, align 4
|
|
|
|
%a11 = alloca i32, align 4
|
|
|
|
%a12 = alloca i32, align 4
|
|
|
|
%v10 = bitcast i8** %args to i8*
|
|
|
|
call void @llvm.va_start(i8* %v10)
|
|
|
|
%v11 = va_arg i8** %args, i32
|
|
|
|
store i32 %v11, i32* %a10, align 4
|
|
|
|
%v12 = va_arg i8** %args, i32
|
|
|
|
store i32 %v12, i32* %a11, align 4
|
|
|
|
%v13 = va_arg i8** %args, i32
|
|
|
|
store i32 %v13, i32* %a12, align 4
|
|
|
|
|
|
|
|
ret float 1.0
|
|
|
|
}
|
|
|
|
|
|
|
|
; "caller4" calls "foo_vararg" that takes a swifterror parameter.
|
|
|
|
define float @caller4(i8* %error_ref) {
|
|
|
|
; CHECK-APPLE-LABEL: caller4:
|
|
|
|
; CHECK-APPLE: mov [[ID:r[0-9]+]], r0
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE: mov r8, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE: bl {{.*}}foo_vararg
|
2018-02-16 17:51:01 +08:00
|
|
|
; CHECK-APPLE: mov r0, r8
|
2018-02-28 00:59:10 +08:00
|
|
|
; CHECK-APPLE: cmp r8, #0
|
2016-04-12 05:08:06 +08:00
|
|
|
; Access part of the error object and save it to error_ref
|
2018-02-16 17:51:01 +08:00
|
|
|
; CHECK-APPLE: ldrbeq [[CODE:r[0-9]+]], [r0, #8]
|
2016-04-12 05:08:06 +08:00
|
|
|
; CHECK-APPLE: strbeq [[CODE]], [{{.*}}[[ID]]]
|
2016-10-08 06:06:55 +08:00
|
|
|
; CHECK-APPLE: bl {{.*}}free
|
2016-04-12 05:08:06 +08:00
|
|
|
entry:
|
|
|
|
%error_ptr_ref = alloca swifterror %swift_error*
|
|
|
|
store %swift_error* null, %swift_error** %error_ptr_ref
|
|
|
|
|
|
|
|
%a10 = alloca i32, align 4
|
|
|
|
%a11 = alloca i32, align 4
|
|
|
|
%a12 = alloca i32, align 4
|
|
|
|
store i32 10, i32* %a10, align 4
|
|
|
|
store i32 11, i32* %a11, align 4
|
|
|
|
store i32 12, i32* %a12, align 4
|
|
|
|
%v10 = load i32, i32* %a10, align 4
|
|
|
|
%v11 = load i32, i32* %a11, align 4
|
|
|
|
%v12 = load i32, i32* %a12, align 4
|
|
|
|
|
|
|
|
%call = call float (%swift_error**, ...) @foo_vararg(%swift_error** swifterror %error_ptr_ref, i32 %v10, i32 %v11, i32 %v12)
|
|
|
|
%error_from_foo = load %swift_error*, %swift_error** %error_ptr_ref
|
|
|
|
%had_error_from_foo = icmp ne %swift_error* %error_from_foo, null
|
|
|
|
%tmp = bitcast %swift_error* %error_from_foo to i8*
|
|
|
|
br i1 %had_error_from_foo, label %handler, label %cont
|
|
|
|
|
|
|
|
cont:
|
|
|
|
%v1 = getelementptr inbounds %swift_error, %swift_error* %error_from_foo, i64 0, i32 1
|
|
|
|
%t = load i8, i8* %v1
|
|
|
|
store i8 %t, i8* %error_ref
|
|
|
|
br label %handler
|
|
|
|
handler:
|
|
|
|
call void @free(i8* %tmp)
|
|
|
|
ret float 1.0
|
|
|
|
}
|
2016-09-22 00:53:36 +08:00
|
|
|
|
|
|
|
; Check that we don't blow up on tail calling swifterror argument functions.
|
|
|
|
define float @tailcallswifterror(%swift_error** swifterror %error_ptr_ref) {
|
|
|
|
entry:
|
|
|
|
%0 = tail call float @tailcallswifterror(%swift_error** swifterror %error_ptr_ref)
|
|
|
|
ret float %0
|
|
|
|
}
|
|
|
|
define swiftcc float @tailcallswifterror_swiftcc(%swift_error** swifterror %error_ptr_ref) {
|
|
|
|
entry:
|
|
|
|
%0 = tail call swiftcc float @tailcallswifterror_swiftcc(%swift_error** swifterror %error_ptr_ref)
|
|
|
|
ret float %0
|
|
|
|
}
|
2016-10-29 01:21:05 +08:00
|
|
|
|
|
|
|
; CHECK-APPLE-LABEL: swifterror_clobber
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE: mov [[REG:r[0-9]+]], r8
|
2016-10-29 01:21:05 +08:00
|
|
|
; CHECK-APPLE: nop
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE: mov r8, [[REG]]
|
2016-10-29 01:21:05 +08:00
|
|
|
define swiftcc void @swifterror_clobber(%swift_error** nocapture swifterror %err) {
|
2017-02-09 09:52:17 +08:00
|
|
|
call void asm sideeffect "nop", "~{r8}"()
|
2016-10-29 01:21:05 +08:00
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-APPLE-LABEL: swifterror_reg_clobber
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE: push {{.*}}r8
|
2016-10-29 01:21:05 +08:00
|
|
|
; CHECK-APPLE: nop
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-APPLE: pop {{.*}}r8
|
2016-10-29 01:21:05 +08:00
|
|
|
define swiftcc void @swifterror_reg_clobber(%swift_error** nocapture %err) {
|
2017-02-09 09:52:17 +08:00
|
|
|
call void asm sideeffect "nop", "~{r8}"()
|
2016-10-29 01:21:05 +08:00
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7-LABEL: _params_in_reg
|
2016-10-29 01:21:05 +08:00
|
|
|
; Store callee saved registers excluding swifterror.
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-ARMV7: push {r4, r5, r6, r7, r10, r11, lr}
|
|
|
|
; Store swiftself (r10) and swifterror (r8).
|
|
|
|
; CHECK-ARMV7-DAG: str r8, [s[[STK1:.*]]]
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7-DAG: str r10, [s[[STK2:.*]]]
|
2016-10-29 01:21:05 +08:00
|
|
|
; Store arguments.
|
2017-06-28 15:07:03 +08:00
|
|
|
; CHECK-ARMV7-DAG: mov r6, r3
|
|
|
|
; CHECK-ARMV7-DAG: mov r4, r2
|
|
|
|
; CHECK-ARMV7-DAG: mov r11, r1
|
|
|
|
; CHECK-ARMV7-DAG: mov r5, r0
|
2016-10-29 01:21:05 +08:00
|
|
|
; Setup call.
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7: mov r0, #1
|
|
|
|
; CHECK-ARMV7: mov r1, #2
|
|
|
|
; CHECK-ARMV7: mov r2, #3
|
|
|
|
; CHECK-ARMV7: mov r3, #4
|
|
|
|
; CHECK-ARMV7: mov r10, #0
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-ARMV7: mov r8, #0
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7: bl _params_in_reg2
|
2016-10-29 01:21:05 +08:00
|
|
|
; Restore original arguments.
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7-DAG: ldr r10, [s[[STK2]]]
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-ARMV7-DAG: ldr r8, [s[[STK1]]]
|
2017-06-28 15:07:03 +08:00
|
|
|
; CHECK-ARMV7-DAG: mov r0, r5
|
|
|
|
; CHECK-ARMV7-DAG: mov r1, r11
|
|
|
|
; CHECK-ARMV7-DAG: mov r2, r4
|
|
|
|
; CHECK-ARMV7-DAG: mov r3, r6
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7: bl _params_in_reg2
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-ARMV7: pop {r4, r5, r6, r7, r10, r11, pc}
|
2016-10-29 01:21:05 +08:00
|
|
|
define swiftcc void @params_in_reg(i32, i32, i32, i32, i8* swiftself, %swift_error** nocapture swifterror %err) {
|
|
|
|
%error_ptr_ref = alloca swifterror %swift_error*, align 8
|
|
|
|
store %swift_error* null, %swift_error** %error_ptr_ref
|
|
|
|
call swiftcc void @params_in_reg2(i32 1, i32 2, i32 3, i32 4, i8* swiftself null, %swift_error** nocapture swifterror %error_ptr_ref)
|
|
|
|
call swiftcc void @params_in_reg2(i32 %0, i32 %1, i32 %2, i32 %3, i8* swiftself %4, %swift_error** nocapture swifterror %err)
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
declare swiftcc void @params_in_reg2(i32, i32, i32, i32, i8* swiftself, %swift_error** nocapture swifterror %err)
|
|
|
|
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7-LABEL: params_and_return_in_reg
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-ARMV7: push {r4, r5, r6, r7, r10, r11, lr}
|
2016-10-29 01:21:05 +08:00
|
|
|
; Store swifterror and swiftself
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-ARMV7: mov r6, r8
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7: str r10, [s[[STK1:.*]]]
|
2016-10-29 01:21:05 +08:00
|
|
|
; Store arguments.
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7: str r3, [s[[STK2:.*]]]
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-ARMV7: mov r4, r2
|
|
|
|
; CHECK-ARMV7: mov r11, r1
|
|
|
|
; CHECK-ARMV7: mov r5, r0
|
2016-10-29 01:21:05 +08:00
|
|
|
; Setup call.
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7: mov r0, #1
|
|
|
|
; CHECK-ARMV7: mov r1, #2
|
|
|
|
; CHECK-ARMV7: mov r2, #3
|
|
|
|
; CHECK-ARMV7: mov r3, #4
|
|
|
|
; CHECK-ARMV7: mov r10, #0
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-ARMV7: mov r8, #0
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7: bl _params_in_reg2
|
2016-10-29 01:21:05 +08:00
|
|
|
; Restore original arguments.
|
2017-06-28 15:07:03 +08:00
|
|
|
; CHECK-ARMV7-DAG: ldr r3, [s[[STK2]]]
|
|
|
|
; CHECK-ARMV7-DAG: ldr r10, [s[[STK1]]]
|
2016-10-29 01:21:05 +08:00
|
|
|
; Store %error_ptr_ref;
|
2017-06-28 15:07:03 +08:00
|
|
|
; CHECK-ARMV7-DAG: str r8, [s[[STK3:.*]]]
|
2016-10-29 01:21:05 +08:00
|
|
|
; Restore original arguments.
|
2017-06-28 15:07:03 +08:00
|
|
|
; CHECK-ARMV7-DAG: mov r0, r5
|
|
|
|
; CHECK-ARMV7-DAG: mov r1, r11
|
|
|
|
; CHECK-ARMV7-DAG: mov r2, r4
|
|
|
|
; CHECK-ARMV7-DAG: mov r8, r6
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7: bl _params_and_return_in_reg2
|
2016-10-29 01:21:05 +08:00
|
|
|
; Store swifterror return %err;
|
2017-06-28 15:07:03 +08:00
|
|
|
; CHECK-ARMV7-DAG: str r8, [s[[STK1]]]
|
2016-10-29 01:21:05 +08:00
|
|
|
; Load swifterror value %error_ptr_ref.
|
2017-06-28 15:07:03 +08:00
|
|
|
; CHECK-ARMV7-DAG: ldr r8, [s[[STK3]]]
|
2016-10-29 01:21:05 +08:00
|
|
|
; Save return values.
|
2017-06-28 15:07:03 +08:00
|
|
|
; CHECK-ARMV7-DAG: mov r4, r0
|
|
|
|
; CHECK-ARMV7-DAG: mov r5, r1
|
|
|
|
; CHECK-ARMV7-DAG: mov r6, r2
|
|
|
|
; CHECK-ARMV7-DAG: mov r11, r3
|
2016-10-29 01:21:05 +08:00
|
|
|
; Setup call.
|
2016-10-29 03:18:09 +08:00
|
|
|
; CHECK-ARMV7: mov r0, #1
|
|
|
|
; CHECK-ARMV7: mov r1, #2
|
|
|
|
; CHECK-ARMV7: mov r2, #3
|
|
|
|
; CHECK-ARMV7: mov r3, #4
|
|
|
|
; CHECK-ARMV7: mov r10, #0
|
|
|
|
; CHECK-ARMV7: bl _params_in_reg2
|
2016-10-29 01:21:05 +08:00
|
|
|
; Load swifterror %err;
|
2017-06-28 15:07:03 +08:00
|
|
|
; CHECK-ARMV7-DAG: ldr r8, [s[[STK1]]]
|
2016-10-29 01:21:05 +08:00
|
|
|
; Restore return values for returning.
|
2017-06-28 15:07:03 +08:00
|
|
|
; CHECK-ARMV7-DAG: mov r0, r4
|
|
|
|
; CHECK-ARMV7-DAG: mov r1, r5
|
|
|
|
; CHECK-ARMV7-DAG: mov r2, r6
|
|
|
|
; CHECK-ARMV7-DAG: mov r3, r11
|
2017-02-09 09:52:17 +08:00
|
|
|
; CHECK-ARMV7: pop {r4, r5, r6, r7, r10, r11, pc}
|
[ARM] Use Swift error registers on non-Darwin targets
Summary:
Remove a check for `ARMSubtarget::isTargetDarwin` when determining
whether to use Swift error registers, so that Swift errors work
properly on non-Darwin ARM32 targets (specifically Android).
Before this patch, generated code would save and restores ARM register r8 at
the entry and returns of a function that throws. As r8 is used as a virtual
return value for the object being thrown, this gets overwritten by the restore,
and calling code is unable to catch the error. In turn this caused Swift code
that used `do`/`try`/`catch` to work improperly on Android ARM32 targets.
Addresses Swift bug report https://bugs.swift.org/browse/SR-5438.
Patch by John Holdsworth.
Reviewers: manmanren, rjmccall, aschwaighofer
Reviewed By: aschwaighofer
Subscribers: srhines, aschwaighofer, aemerson, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D35835
llvm-svn: 312164
2017-08-31 04:03:54 +08:00
|
|
|
|
|
|
|
; CHECK-ANDROID-LABEL: params_and_return_in_reg
|
2017-09-26 01:19:50 +08:00
|
|
|
; CHECK-ANDROID: push {r4, r5, r6, r7, r9, r10, r11, lr}
|
|
|
|
; CHECK-ANDROID: sub sp, sp, #16
|
|
|
|
; CHECK-ANDROID: str r8, [sp, #4] @ 4-byte Spill
|
|
|
|
; CHECK-ANDROID: mov r11, r10
|
|
|
|
; CHECK-ANDROID: mov r6, r3
|
|
|
|
; CHECK-ANDROID: mov r7, r2
|
|
|
|
; CHECK-ANDROID: mov r4, r1
|
|
|
|
; CHECK-ANDROID: mov r5, r0
|
|
|
|
; CHECK-ANDROID: mov r0, #1
|
|
|
|
; CHECK-ANDROID: mov r1, #2
|
|
|
|
; CHECK-ANDROID: mov r2, #3
|
|
|
|
; CHECK-ANDROID: mov r3, #4
|
|
|
|
; CHECK-ANDROID: mov r10, #0
|
|
|
|
; CHECK-ANDROID: mov r8, #0
|
|
|
|
; CHECK-ANDROID: bl params_in_reg2
|
|
|
|
; CHECK-ANDROID: mov r9, r8
|
|
|
|
; CHECK-ANDROID: ldr r8, [sp, #4] @ 4-byte Reload
|
|
|
|
; CHECK-ANDROID: mov r0, r5
|
|
|
|
; CHECK-ANDROID: mov r1, r4
|
|
|
|
; CHECK-ANDROID: mov r2, r7
|
|
|
|
; CHECK-ANDROID: mov r3, r6
|
|
|
|
; CHECK-ANDROID: mov r10, r11
|
|
|
|
; CHECK-ANDROID: bl params_and_return_in_reg2
|
|
|
|
; CHECK-ANDROID: mov r4, r0
|
|
|
|
; CHECK-ANDROID: mov r5, r1
|
|
|
|
; CHECK-ANDROID: mov r6, r2
|
|
|
|
; CHECK-ANDROID: mov r7, r3
|
|
|
|
; CHECK-ANDROID: mov r11, r8
|
|
|
|
; CHECK-ANDROID: mov r0, #1
|
|
|
|
; CHECK-ANDROID: mov r1, #2
|
|
|
|
; CHECK-ANDROID: mov r2, #3
|
|
|
|
; CHECK-ANDROID: mov r3, #4
|
|
|
|
; CHECK-ANDROID: mov r10, #0
|
|
|
|
; CHECK-ANDROID: mov r8, r9
|
|
|
|
; CHECK-ANDROID: bl params_in_reg2
|
|
|
|
; CHECK-ANDROID: mov r0, r4
|
|
|
|
; CHECK-ANDROID: mov r1, r5
|
|
|
|
; CHECK-ANDROID: mov r2, r6
|
|
|
|
; CHECK-ANDROID: mov r3, r7
|
|
|
|
; CHECK-ANDROID: mov r8, r11
|
|
|
|
; CHECK-ANDROID: add sp, sp, #16
|
|
|
|
; CHECK-ANDROID: pop {r4, r5, r6, r7, r9, r10, r11, pc}
|
[ARM] Use Swift error registers on non-Darwin targets
Summary:
Remove a check for `ARMSubtarget::isTargetDarwin` when determining
whether to use Swift error registers, so that Swift errors work
properly on non-Darwin ARM32 targets (specifically Android).
Before this patch, generated code would save and restores ARM register r8 at
the entry and returns of a function that throws. As r8 is used as a virtual
return value for the object being thrown, this gets overwritten by the restore,
and calling code is unable to catch the error. In turn this caused Swift code
that used `do`/`try`/`catch` to work improperly on Android ARM32 targets.
Addresses Swift bug report https://bugs.swift.org/browse/SR-5438.
Patch by John Holdsworth.
Reviewers: manmanren, rjmccall, aschwaighofer
Reviewed By: aschwaighofer
Subscribers: srhines, aschwaighofer, aemerson, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D35835
llvm-svn: 312164
2017-08-31 04:03:54 +08:00
|
|
|
|
2016-10-29 01:21:05 +08:00
|
|
|
define swiftcc { i32, i32, i32, i32} @params_and_return_in_reg(i32, i32, i32, i32, i8* swiftself, %swift_error** nocapture swifterror %err) {
|
|
|
|
%error_ptr_ref = alloca swifterror %swift_error*, align 8
|
|
|
|
store %swift_error* null, %swift_error** %error_ptr_ref
|
|
|
|
call swiftcc void @params_in_reg2(i32 1, i32 2, i32 3, i32 4, i8* swiftself null, %swift_error** nocapture swifterror %error_ptr_ref)
|
|
|
|
%val = call swiftcc { i32, i32, i32, i32 } @params_and_return_in_reg2(i32 %0, i32 %1, i32 %2, i32 %3, i8* swiftself %4, %swift_error** nocapture swifterror %err)
|
|
|
|
call swiftcc void @params_in_reg2(i32 1, i32 2, i32 3, i32 4, i8* swiftself null, %swift_error** nocapture swifterror %error_ptr_ref)
|
|
|
|
ret { i32, i32, i32, i32 }%val
|
|
|
|
}
|
|
|
|
|
|
|
|
declare swiftcc { i32, i32, i32, i32 } @params_and_return_in_reg2(i32, i32, i32, i32, i8* swiftself, %swift_error** nocapture swifterror %err)
|
2017-02-14 03:58:28 +08:00
|
|
|
|
|
|
|
|
|
|
|
declare void @acallee(i8*)
|
|
|
|
|
|
|
|
; Make sure we don't tail call if the caller returns a swifterror value. We
|
|
|
|
; would have to move into the swifterror register before the tail call.
|
|
|
|
; CHECK-APPLE: tailcall_from_swifterror:
|
|
|
|
; CHECK-APPLE-NOT: b _acallee
|
|
|
|
; CHECK-APPLE: bl _acallee
|
[ARM] Use Swift error registers on non-Darwin targets
Summary:
Remove a check for `ARMSubtarget::isTargetDarwin` when determining
whether to use Swift error registers, so that Swift errors work
properly on non-Darwin ARM32 targets (specifically Android).
Before this patch, generated code would save and restores ARM register r8 at
the entry and returns of a function that throws. As r8 is used as a virtual
return value for the object being thrown, this gets overwritten by the restore,
and calling code is unable to catch the error. In turn this caused Swift code
that used `do`/`try`/`catch` to work improperly on Android ARM32 targets.
Addresses Swift bug report https://bugs.swift.org/browse/SR-5438.
Patch by John Holdsworth.
Reviewers: manmanren, rjmccall, aschwaighofer
Reviewed By: aschwaighofer
Subscribers: srhines, aschwaighofer, aemerson, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D35835
llvm-svn: 312164
2017-08-31 04:03:54 +08:00
|
|
|
; CHECK-ANDROID: tailcall_from_swifterror:
|
|
|
|
; CHECK-ANDROID-NOT: b acallee
|
|
|
|
; CHECK-ANDROID: bl acallee
|
2017-02-14 03:58:28 +08:00
|
|
|
|
|
|
|
define swiftcc void @tailcall_from_swifterror(%swift_error** swifterror %error_ptr_ref) {
|
|
|
|
entry:
|
|
|
|
tail call void @acallee(i8* null)
|
|
|
|
ret void
|
|
|
|
}
|
2017-06-16 01:34:42 +08:00
|
|
|
|
|
|
|
|
|
|
|
declare swiftcc void @foo2(%swift_error** swifterror)
|
|
|
|
|
|
|
|
; Make sure we properly assign registers during fast-isel.
|
|
|
|
; CHECK-O0-LABEL: testAssign
|
|
|
|
; CHECK-O0: mov r8, #0
|
|
|
|
; CHECK-O0: bl _foo2
|
|
|
|
; CHECK-O0: str r8, [s[[STK:p.*]]]
|
|
|
|
; CHECK-O0: ldr r0, [s[[STK]]]
|
|
|
|
; CHECK-O0: pop
|
|
|
|
|
|
|
|
; CHECK-APPLE-LABEL: testAssign
|
|
|
|
; CHECK-APPLE: mov r8, #0
|
|
|
|
; CHECK-APPLE: bl _foo2
|
|
|
|
; CHECK-APPLE: mov r0, r8
|
|
|
|
|
[ARM] Use Swift error registers on non-Darwin targets
Summary:
Remove a check for `ARMSubtarget::isTargetDarwin` when determining
whether to use Swift error registers, so that Swift errors work
properly on non-Darwin ARM32 targets (specifically Android).
Before this patch, generated code would save and restores ARM register r8 at
the entry and returns of a function that throws. As r8 is used as a virtual
return value for the object being thrown, this gets overwritten by the restore,
and calling code is unable to catch the error. In turn this caused Swift code
that used `do`/`try`/`catch` to work improperly on Android ARM32 targets.
Addresses Swift bug report https://bugs.swift.org/browse/SR-5438.
Patch by John Holdsworth.
Reviewers: manmanren, rjmccall, aschwaighofer
Reviewed By: aschwaighofer
Subscribers: srhines, aschwaighofer, aemerson, javed.absar, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D35835
llvm-svn: 312164
2017-08-31 04:03:54 +08:00
|
|
|
; CHECK-ANDROID-LABEL: testAssign
|
|
|
|
; CHECK-ANDROID: mov r8, #0
|
|
|
|
; CHECK-ANDROID: bl foo2
|
|
|
|
; CHECK-ANDROID: mov r0, r8
|
|
|
|
|
2017-06-16 01:34:42 +08:00
|
|
|
define swiftcc %swift_error* @testAssign(i8* %error_ref) {
|
|
|
|
entry:
|
|
|
|
%error_ptr = alloca swifterror %swift_error*
|
|
|
|
store %swift_error* null, %swift_error** %error_ptr
|
|
|
|
call swiftcc void @foo2(%swift_error** swifterror %error_ptr)
|
|
|
|
br label %a
|
|
|
|
|
|
|
|
a:
|
|
|
|
%error = load %swift_error*, %swift_error** %error_ptr
|
|
|
|
ret %swift_error* %error
|
|
|
|
}
|