| /* |
| * Copyright 2017, Data61, CSIRO (ABN 41 687 119 230) |
| * |
| * SPDX-License-Identifier: BSD-2-Clause |
| */ |
| |
| #include <autoconf.h> |
| #include <utils/gen_config.h> |
| #include <utils/stack.h> |
| |
| void * |
| utils_run_on_stack(void *stack_top, void * (*func)(void *arg), void *arg) |
| { |
| #ifdef CONFIG_ARCH_AARCH64 |
| void *ret; |
| asm volatile ( |
| "mov x20, sp\n\t" /* Save sp - x20 is callee saved so we don't need to save it. */ |
| "mov sp, %[new_stack]\n\t" /* Switch to new stack. */ |
| "mov x0, %[arg]\n\t" /* Setup argument to func. */ |
| "blr %[func]\n\t" |
| "mov sp, x20\n\t" |
| "mov %[ret], x0\n\t" |
| : [ret] "=r" (ret) |
| : [new_stack] "r" (stack_top), |
| [func] "r" (func), |
| [arg] "r" (arg) |
| /* x9 - x15 caller saved registers */ |
| /* x20 and LR */ |
| : "x9", "x10", "x11", "x12", "x13", "x14", "x15", "x20", "x30"); |
| #else |
| void *ret; |
| asm volatile ( |
| "mov r4, sp\n\t" /* Save sp - r4 is callee saved so we don't need to save it. */ |
| "mov sp, %[new_stack]\n\t" /* Switch to new stack. */ |
| "mov r0, %[arg]\n\t" /* Setup argument to func. */ |
| "blx %[func]\n\t" |
| "mov sp, r4\n\t" |
| "mov %[ret], r0\n\t" |
| : [ret] "=r" (ret) |
| : [new_stack] "r" (stack_top), |
| [func] "r" (func), |
| [arg] "r" (arg) |
| /* r0 - r3 are caller saved, so they will be clobbered by blx. |
| * blx clobbers lr |
| * we clobber r4 to save the return value in it */ |
| : "r0", "r1", "r2", "r3", "r4", "lr"); |
| #endif /* CONFIG_ARCH_AARCH64 */ |
| return ret; |
| } |