Mirror of BoringSSL (grpc依赖)
https://boringssl.googlesource.com/boringssl
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
181 lines
5.5 KiB
181 lines
5.5 KiB
#!/usr/bin/env perl |
|
# Copyright (c) 2019, Google Inc. |
|
# |
|
# Permission to use, copy, modify, and/or distribute this software for any |
|
# purpose with or without fee is hereby granted, provided that the above |
|
# copyright notice and this permission notice appear in all copies. |
|
# |
|
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES |
|
# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF |
|
# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY |
|
# SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES |
|
# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION |
|
# OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN |
|
# CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
|
|
|
# This file defines helper functions for crypto/test/abi_test.h on 32-bit |
|
# ARM. See that header for details on how to use this. |
|
# |
|
# For convenience, this file is linked into libcrypto, where consuming builds |
|
# already support architecture-specific sources. The static linker should drop |
|
# this code in non-test binaries. This includes a shared library build of |
|
# libcrypto, provided --gc-sections (ELF), -dead_strip (iOS), or equivalent is |
|
# used. |
|
# |
|
# References: |
|
# |
|
# AAPCS: http://infocenter.arm.com/help/topic/com.arm.doc.ihi0042f/IHI0042F_aapcs.pdf |
|
# iOS ARMv6: https://developer.apple.com/library/archive/documentation/Xcode/Conceptual/iPhoneOSABIReference/Articles/ARMv6FunctionCallingConventions.html |
|
# iOS ARMv7: https://developer.apple.com/library/archive/documentation/Xcode/Conceptual/iPhoneOSABIReference/Articles/ARMv7FunctionCallingConventions.html |
|
# Linux: http://sourcery.mentor.com/sgpp/lite/arm/portal/kbattach142/arm_gnu_linux_%20abi.pdf |
|
|
|
use strict; |
|
|
|
my $flavour = shift; |
|
my $output = shift; |
|
if ($flavour =~ /\./) { $output = $flavour; undef $flavour; } |
|
|
|
$0 =~ m/(.*[\/\\])[^\/\\]+$/; |
|
my $dir = $1; |
|
my $xlate; |
|
( $xlate="${dir}arm-xlate.pl" and -f $xlate ) or |
|
( $xlate="${dir}../../perlasm/arm-xlate.pl" and -f $xlate) or |
|
die "can't locate arm-xlate.pl"; |
|
|
|
open OUT, "| \"$^X\" \"$xlate\" $flavour \"$output\""; |
|
*STDOUT = *OUT; |
|
|
|
my ($func, $state, $argv, $argc) = ("r0", "r1", "r2", "r3"); |
|
my $code = <<____; |
|
.syntax unified |
|
|
|
.arch armv7-a |
|
.fpu vfp |
|
|
|
.text |
|
|
|
@ abi_test_trampoline loads callee-saved registers from |state|, calls |func| |
|
@ with |argv|, then saves the callee-saved registers into |state|. It returns |
|
@ the result of |func|. The |unwind| argument is unused. |
|
@ uint32_t abi_test_trampoline(void (*func)(...), CallerState *state, |
|
@ const uint32_t *argv, size_t argc, |
|
@ int unwind); |
|
.type abi_test_trampoline, %function |
|
.globl abi_test_trampoline |
|
.align 4 |
|
abi_test_trampoline: |
|
@ Save parameters and all callee-saved registers. For convenience, we |
|
@ save r9 on iOS even though it's volatile. |
|
vstmdb sp!, {d8-d15} |
|
stmdb sp!, {r0-r11,lr} |
|
|
|
@ Reserve stack space for six (10-4) stack parameters, plus an extra 4 |
|
@ bytes to keep it 8-byte-aligned (see AAPCS, section 5.3). |
|
sub sp, sp, #28 |
|
|
|
@ Every register in AAPCS is either non-volatile or a parameter (except |
|
@ r9 on iOS), so this code, by the actual call, loses all its scratch |
|
@ registers. First fill in stack parameters while there are registers |
|
@ to spare. |
|
cmp $argc, #4 |
|
bls .Lstack_args_done |
|
mov r4, sp @ r4 is the output pointer. |
|
add r5, $argv, $argc, lsl #2 @ Set r5 to the end of argv. |
|
add $argv, $argv, #16 @ Skip four arguments. |
|
.Lstack_args_loop: |
|
ldr r6, [$argv], #4 |
|
cmp $argv, r5 |
|
str r6, [r4], #4 |
|
bne .Lstack_args_loop |
|
|
|
.Lstack_args_done: |
|
@ Load registers from |$state|. |
|
vldmia $state!, {d8-d15} |
|
#if defined(__APPLE__) |
|
@ r9 is not volatile on iOS. |
|
ldmia $state!, {r4-r8,r10-r11} |
|
#else |
|
ldmia $state!, {r4-r11} |
|
#endif |
|
|
|
@ Load register parameters. This uses up our remaining registers, so we |
|
@ repurpose lr as scratch space. |
|
ldr $argc, [sp, #40] @ Reload argc. |
|
ldr lr, [sp, #36] @ Load argv into lr. |
|
cmp $argc, #3 |
|
bhi .Larg_r3 |
|
beq .Larg_r2 |
|
cmp $argc, #1 |
|
bhi .Larg_r1 |
|
beq .Larg_r0 |
|
b .Largs_done |
|
|
|
.Larg_r3: |
|
ldr r3, [lr, #12] @ argv[3] |
|
.Larg_r2: |
|
ldr r2, [lr, #8] @ argv[2] |
|
.Larg_r1: |
|
ldr r1, [lr, #4] @ argv[1] |
|
.Larg_r0: |
|
ldr r0, [lr] @ argv[0] |
|
.Largs_done: |
|
|
|
@ With every other register in use, load the function pointer into lr |
|
@ and call the function. |
|
ldr lr, [sp, #28] |
|
blx lr |
|
|
|
@ r1-r3 are free for use again. The trampoline only supports |
|
@ single-return functions. Pass r4-r11 to the caller. |
|
ldr $state, [sp, #32] |
|
vstmia $state!, {d8-d15} |
|
#if defined(__APPLE__) |
|
@ r9 is not volatile on iOS. |
|
stmia $state!, {r4-r8,r10-r11} |
|
#else |
|
stmia $state!, {r4-r11} |
|
#endif |
|
|
|
@ Unwind the stack and restore registers. |
|
add sp, sp, #44 @ 44 = 28+16 |
|
ldmia sp!, {r4-r11,lr} @ Skip r0-r3 (see +16 above). |
|
vldmia sp!, {d8-d15} |
|
|
|
bx lr |
|
.size abi_test_trampoline,.-abi_test_trampoline |
|
____ |
|
|
|
# abi_test_clobber_* zeros the corresponding register. These are used to test |
|
# the ABI-testing framework. |
|
foreach (0..12) { |
|
# This loop skips r13 (sp), r14 (lr, implicitly clobbered by every call), and |
|
# r15 (pc). |
|
$code .= <<____; |
|
.type abi_test_clobber_r$_, %function |
|
.globl abi_test_clobber_r$_ |
|
.align 4 |
|
abi_test_clobber_r$_: |
|
mov r$_, #0 |
|
bx lr |
|
.size abi_test_clobber_r$_,.-abi_test_clobber_r$_ |
|
____ |
|
} |
|
|
|
foreach (0..15) { |
|
my $lo = "s".(2*$_); |
|
my $hi = "s".(2*$_+1); |
|
$code .= <<____; |
|
.type abi_test_clobber_d$_, %function |
|
.globl abi_test_clobber_d$_ |
|
.align 4 |
|
abi_test_clobber_d$_: |
|
mov r0, #0 |
|
vmov $lo, r0 |
|
vmov $hi, r0 |
|
bx lr |
|
.size abi_test_clobber_d$_,.-abi_test_clobber_d$_ |
|
____ |
|
} |
|
|
|
print $code; |
|
close STDOUT or die "error closing STDOUT: $!";
|
|
|