2019-05-30 07:57:47 +08:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0-only */
|
2011-06-08 22:29:00 +08:00
|
|
|
/*
|
|
|
|
* arch/arm/lib/call_with_stack.S
|
|
|
|
*
|
|
|
|
* Copyright (C) 2011 ARM Ltd.
|
|
|
|
* Written by Will Deacon <will.deacon@arm.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/linkage.h>
|
|
|
|
#include <asm/assembler.h>
|
2021-10-05 15:15:41 +08:00
|
|
|
#include <asm/unwind.h>
|
2011-06-08 22:29:00 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* void call_with_stack(void (*fn)(void *), void *arg, void *sp)
|
|
|
|
*
|
|
|
|
* Change the stack to that pointed at by sp, then invoke fn(arg) with
|
|
|
|
* the new stack.
|
2021-10-05 15:15:41 +08:00
|
|
|
*
|
|
|
|
* The sequence below follows the APCS frame convention for frame pointer
|
|
|
|
* unwinding, and implements the unwinder annotations needed by the EABI
|
|
|
|
* unwinder.
|
2011-06-08 22:29:00 +08:00
|
|
|
*/
|
|
|
|
|
2021-10-05 15:15:41 +08:00
|
|
|
ENTRY(call_with_stack)
|
|
|
|
#if defined(CONFIG_UNWINDER_FRAME_POINTER) && defined(CONFIG_CC_IS_GCC)
|
|
|
|
mov ip, sp
|
|
|
|
push {fp, ip, lr, pc}
|
|
|
|
sub fp, ip, #4
|
|
|
|
#else
|
|
|
|
UNWIND( .fnstart )
|
|
|
|
UNWIND( .save {fpreg, lr} )
|
|
|
|
push {fpreg, lr}
|
|
|
|
UNWIND( .setfp fpreg, sp )
|
|
|
|
mov fpreg, sp
|
|
|
|
#endif
|
2011-06-08 22:29:00 +08:00
|
|
|
mov sp, r2
|
|
|
|
mov r2, r0
|
|
|
|
mov r0, r1
|
|
|
|
|
2021-10-05 15:15:41 +08:00
|
|
|
bl_r r2
|
2011-06-08 22:29:00 +08:00
|
|
|
|
2021-10-05 15:15:41 +08:00
|
|
|
#if defined(CONFIG_UNWINDER_FRAME_POINTER) && defined(CONFIG_CC_IS_GCC)
|
|
|
|
ldmdb fp, {fp, sp, pc}
|
|
|
|
#else
|
|
|
|
mov sp, fpreg
|
|
|
|
pop {fpreg, pc}
|
|
|
|
UNWIND( .fnend )
|
|
|
|
#endif
|
2011-06-08 22:29:00 +08:00
|
|
|
ENDPROC(call_with_stack)
|