/* * Copyright (C) 2018 Gunar Schorcht * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup cpu_esp8266 * @{ * * @file * @brief Implementation of the kernel's architecture dependent thread interface * * @author Gunar Schorcht * * @} */ /* * PLEASE NOTE: Some parts of the code are taken from the FreeRTOS port for * Xtensa processors from Cadence Design Systems. These parts are marked * accordingly. For these parts, the following license is valid: * * Copyright (c) 2003-2015 Cadence Design Systems, Inc. * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be included * in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #define ENABLE_DEBUG 0 #include "debug.h" #include #include #include "board.h" #include "cpu.h" #include "irq.h" #include "log.h" #include "thread.h" #include "sched.h" #include "common.h" #include "esp/common_macros.h" #include "esp/xtensa_ops.h" #include "sdk/ets_task.h" #include "sdk/rom.h" #include "sdk/sdk.h" #include "tools.h" #include "xtensa/xtensa_context.h" /* User exception dispatcher when exiting */ extern void _xt_user_exit(void); /* Switch context to the highest priority ready task without context save */ extern void _frxt_dispatch(void); /* Set an flag indicating that a task switch is required on return from interrupt */ extern void _frxt_setup_switch(void); /* Switch context to the highest priority ready task with context save */ extern void vPortYield(void); extern void vPortYieldFromInt(void); char* thread_stack_init(thread_task_func_t task_func, void *arg, void *stack_start, int stack_size) { /* * Stack layout after task stack initialization * * +------------------------+ * | | TOP * | thread_control_block | * stack_start + stack_size ==> | | * +------------------------+ * top_of_stack ==> | | * | XT_CP_FRAME | * | (optional) | * top_of_stack + 1 - XT_CP_SIZE ==> | | * +------------------------+ * | | * | XT_STK_FRAME | * | | XT_STK_... * | a2 = arg | XT_STK_A2 * | a1 = sp + XT_STK_FRMSZ | XT_STK_A1 * | a0 = sched_task_exit | XT_STK_A0 * | ps = PS_UM | PS_EXCM | XT_STK_PS * | pc = task_func | XT_STK_PC * sp = top_of_stack + 1 - XT_CP_SIZE ==> | exit = _xt_user_exit | XT_STK_EXIT * - XT_STK_FRMSZ +------------------------+ * | | * | remaining stack space | * | available for data | * stack_start (preallocated var) ==> | | BOTTOM * +------------------------+ * * Initialized stack frame represents the registers as set when the * the task function would have been called. * * Registers in a called function * * pc - PC at the beginning in the function * a0 - return address from the function (return address to caller) * a1 - current stack pointer at the beginning in the function * a2 - first argument of the function */ /* stack is [stack_start+0 ... stack_start+stack_size-1] */ uint8_t *top_of_stack; uint8_t *sp; top_of_stack = (uint8_t*)((uint32_t)stack_start + stack_size-1); /* Clear whole stack with a known value to assist debugging */ #if !defined(DEVELHELP) && !defined(SCHED_TEST_STACK) /* Unfortunatly, this affects thread_measure_stack_free function */ memset(stack_start, 0, stack_size); #endif /* BEGIN - code from FreeRTOS port for Xtensa from Cadence */ /* Create interrupt stack frame aligned to 16 byte boundary */ sp = (uint8_t*)(((uint32_t)(top_of_stack+1) - XT_STK_FRMSZ - XT_CP_SIZE) & ~0xf); /* ensure that stack is big enough */ assert (sp > (uint8_t*)stack_start); XtExcFrame* exc_frame = (XtExcFrame*)sp; /* Explicitly initialize certain saved registers */ exc_frame->pc = (uint32_t)task_func; /* task entry point */ exc_frame->a0 = (uint32_t)sched_task_exit; /* task exit point (CHANGED) */ exc_frame->a1 = (uint32_t)sp + XT_STK_FRMSZ; /* physical top of stack frame */ exc_frame->a2 = (uint32_t)arg; /* parameters for task_func */ exc_frame->exit = (uint32_t)_xt_user_exit; /* user exception exit dispatcher */ /* Set initial PS to int level 0, EXCM disabled ('rfe' will enable), user mode. */ /* Also set entry point argument parameter. */ #ifdef __XTENSA_CALL0_ABI__ /* CALL0 ABI */ exc_frame->ps = PS_UM | PS_EXCM; #else /* + for windowed ABI also set WOE and CALLINC (pretend task was 'call4'd). */ exc_frame->ps = PS_UM | PS_EXCM | PS_WOE | PS_CALLINC(1); #endif #ifdef XT_USE_SWPRI /* Set the initial virtual priority mask value to all 1's. */ exc_frame->vpri = 0xFFFFFFFF; #endif #if XCHAL_CP_NUM > 0 /* * Init the coprocessor save area (see xtensa_context.h) * No access to TCB here, so derive indirectly. Stack growth is top to bottom. * p = (uint32_t *) xMPUSettings->coproc_area; */ uint32_t *p; p = (uint32_t *)(((uint32_t) pxTopOfStack - XT_CP_SIZE)); p[0] = 0; p[1] = 0; p[2] = (((uint32_t) p) + 12 + XCHAL_TOTAL_SA_ALIGN - 1) & -XCHAL_TOTAL_SA_ALIGN; #endif /* END - code from FreeRTOS port for Xtensa from Cadence */ DEBUG("%s start=%p size=%d top=%p sp=%p free=%u\n", __func__, stack_start, stack_size, top_of_stack, sp, sp-(uint8_t*)stack_start); return (char*)sp; } unsigned sched_interrupt_nesting = 0; /* Interrupt nesting level */ #ifdef CONTEXT_SWITCH_BY_INT /** * Context switches are realized using software interrupts */ void IRAM thread_yield_isr(void* arg) { /* set the context switch flag (indicates that context has to be switched is switch on exit from interrupt in _frxt_int_exit */ _frxt_setup_switch(); } #endif void thread_yield_higher(void) { /* reset hardware watchdog */ system_soft_wdt_feed(); /* yield next task */ #if defined(ENABLE_DEBUG) && defined(DEVELHELP) if (sched_active_thread) { DEBUG("%u old task %u %s %u\n", phy_get_mactime(), sched_active_thread->pid, sched_active_thread->name, sched_active_thread->sp - sched_active_thread-> stack_start); } #endif if (!irq_is_in()) { #ifdef CONTEXT_SWITCH_BY_INT WSR(BIT(ETS_SOFT_INUM), interrupt); #else vPortYield(); #endif } else { _frxt_setup_switch(); } #if defined(ENABLE_DEBUG) && defined(DEVELHELP) if (sched_active_thread) { DEBUG("%u new task %u %s %u\n", phy_get_mactime(), sched_active_thread->pid, sched_active_thread->name, sched_active_thread->sp - sched_active_thread-> stack_start); } #endif return; } void thread_stack_print(void) { /* Print the current stack to stdout. */ #if defined(DEVELHELP) volatile thread_t* task = thread_get(sched_active_pid); if (task) { char* stack_top = task->stack_start + task->stack_size; int size = stack_top - task->sp; printf("Printing current stack of thread %" PRIkernel_pid "\n", thread_getpid()); esp_hexdump((void*)(task->sp), size >> 2, 'w', 8); } #else NOT_SUPPORTED(); #endif } void thread_print_stack(void) { /* Prints human readable, ps-like thread information for debugging purposes. */ /* because of Xtensa stack structure and call0 ABI, it is not possible to implement */ NOT_YET_IMPLEMENTED(); return; } #if defined(DEVELHELP) extern uint8_t port_IntStack; extern uint8_t port_IntStackTop; void thread_isr_stack_init(void) { /* code from thread.c, please see the copyright notice there */ /* assign each int of the stack the value of it's address */ uintptr_t *stackmax = (uintptr_t *)&port_IntStackTop; uintptr_t *stackp = (uintptr_t *)&port_IntStack; while (stackp < stackmax) { *stackp = (uintptr_t) stackp; stackp++; } } int thread_isr_stack_usage(void) { return &port_IntStackTop - &port_IntStack - thread_measure_stack_free((char*)&port_IntStack); } void *thread_isr_stack_pointer(void) { /* Get the current ISR stack pointer. */ return &port_IntStackTop; } void *thread_isr_stack_start(void) { /* Get the start of the ISR stack. */ return &port_IntStack; } void thread_isr_stack_print(void) { printf("Printing current ISR\n"); esp_hexdump(&port_IntStack, &port_IntStackTop-&port_IntStack, 'w', 8); } #else /* DEVELHELP */ void thread_isr_stack_init(void) {} #endif /* DEVELHELP */ NORETURN void cpu_switch_context_exit(void) { /* Switch context to the highest priority ready task without context save */ _frxt_dispatch(); UNREACHABLE(); }