| /* |
| * 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. |
| * |
| * @LICENSE(DATA61_BSD) |
| */ |
| |
| #include <sel4debug/stack.h> |
| #include <utils/stack.h> |
| |
| /* A simple spin lock that we will use to protect usage of the emergency stack. |
| * Note that this is not advisable in the presence of threads of different |
| * priorities, but we assume callers are only invoking this as a last ditch |
| * debugging effort anyway. If you use this functionality with threads of |
| * different priorities you will likely livelock your system. |
| */ |
| static int stack_lock; |
| |
| /* An emergency stack that we will switch to in order to run the caller's |
| * function. 8K is more or less empirically chosen as "enough to run a trivial |
| * printf." |
| */ |
| static unsigned char emergency_stack[8 * 1024] |
| __attribute__((aligned(sizeof(long long)))); |
| |
| void *debug_run_on_emergency_stack(void *(*f)(void *), void *arg) { |
| /* Take lock */ |
| while (!__sync_bool_compare_and_swap(&stack_lock, 0, 1)); |
| |
| void *ret = (void*)utils_run_on_stack( |
| (void*)emergency_stack + sizeof(emergency_stack), f, arg); |
| |
| /* Release lock */ |
| while (!__sync_bool_compare_and_swap(&stack_lock, 1, 0)); |
| |
| return ret; |
| } |