blob: 1d8de7b7b7db052497f51528606453296a4a1f22 [file] [log] [blame]
/*
* Copyright 2017, Data61
* Commonwealth Scientific and Industrial Research Organisation (CSIRO)
* ABN 41 687 119 230.
*
* This software may be distributed and modified according to the terms of
* the BSD 2-Clause license. Note that NO WARRANTY is provided.
* See "LICENSE_BSD2.txt" for details.
*
* @TAG(DATA61_BSD)
*/
#include <autoconf.h>
#include <utils/gen_config.h>
#include <utils/stack.h>
#include <utils/arith.h>
#include <utils/util.h>
void *
utils_run_on_stack(void *stack_top, void * (*func)(void *arg), void *arg)
{
if (stack_top == NULL) {
ZF_LOGE("Invalid stack address \n");
return NULL;
} else if (!IS_ALIGNED((uintptr_t) stack_top, 4)) {
/* Stack has to be aligned to 16-bytes */
ZF_LOGE("Invalid stack alignment. Stack has to be 16-bytes aligned \n");
return NULL;
}
/*
* Note that x86-64 ABI requires that at least SSE and
* SSE2 are supported, and thus the API uses xmm registers
* for passing floating point parameters. The GCC compiler
* may use the movaps instruction to store the context
* xmm regsiters on stack, and these store operations
* require that the stack addresses are 16-byte
* aligned, so we push oen more dummy number to the stack
* for alignment purpose.
*/
#ifdef CONFIG_ARCH_X86_64
void *ret;
asm volatile (
"movq %%rsp, %%rcx\n\t"
"movq %[new_stack], %%rsp\n\t"
"push %%rcx\n\t"
"pushq $0\n\t" /* dummy number for stack alignment */
"movq %[arg], %%rdi\n\t"
"call *%[func]\n\t"
"add $0x8, %%rsp\n\t"
"pop %%rsp\n\t"
: [ret] "=a" (ret)
: [new_stack] "r" (stack_top),
[func] "r" (func),
[arg] "r" (arg)
: "rcx", "rbx", "rdi"
);
#else
void *ret;
asm volatile (
"mov %%esp, %%ecx\n\t" /* Save sp. */
"mov %[new_stack], %%esp\n\t" /* Switch to new stack. */
"pushl %%ecx\n\t" /* Push old sp onto new stack. */
"subl $8, %%esp\n\t" /* dummy padding for 16-byte stack alignment */
"pushl %[arg]\n\t" /* Setup argument to func. */
"call *%[func]\n\t"
"add $12, %%esp\n\t"
"popl %%esp\n\t" /* Switch back! */
: [ret] "=a" (ret)
: [new_stack] "r" (stack_top),
[func] "r" (func),
[arg] "r" (arg)
: "ecx", "ebx");
#endif /* CONFIG_ARCH_X86_64 */
return ret;
}