|
|
@@ -15,29 +15,72 @@
|
|
|
#include <esp_expression_with_stack.h>
|
|
|
#include <freertos/xtensa_rtos.h>
|
|
|
#include <freertos/xtensa_context.h>
|
|
|
+#include <setjmp.h>
|
|
|
+#include <string.h>
|
|
|
|
|
|
-StackType_t * esp_switch_stack_setup(StackType_t *stack, size_t stack_size)
|
|
|
+StackType_t *xtensa_shared_stack;
|
|
|
+shared_stack_function xtensa_shared_stack_callback;
|
|
|
+jmp_buf xtensa_shared_stack_env;
|
|
|
+bool xtensa_shared_stack_function_done = false;
|
|
|
+static portMUX_TYPE xtensa_shared_stack_spinlock = portMUX_INITIALIZER_UNLOCKED;
|
|
|
+static void *current_task_stack = NULL;
|
|
|
+
|
|
|
+extern void esp_shared_stack_invoke_function(void);
|
|
|
+
|
|
|
+static void esp_switch_stack_setup(StackType_t *stack, size_t stack_size)
|
|
|
{
|
|
|
#if CONFIG_FREERTOS_WATCHPOINT_END_OF_STACK
|
|
|
esp_clear_watchpoint(1);
|
|
|
- uint32_t watchpoint_place = ((uint32_t)stack + 32) & 0x1f ;
|
|
|
-#endif
|
|
|
- StackType_t *top_of_stack = (StackType_t *)&stack[0] +
|
|
|
- ((stack_size * sizeof(StackType_t)) / sizeof(StackType_t));
|
|
|
+ uint32_t watchpoint_place = ((uint32_t)stack + 32) & ~0x1f ;
|
|
|
+#endif
|
|
|
+ //We need also to tweak current task stackpointer to avoid erroneous
|
|
|
+ //stack overflow indication, so fills the stack with freertos known pattern:
|
|
|
+ memset(stack, 0xa5U, stack_size * sizeof(StackType_t));
|
|
|
|
|
|
- //Align stack to a 16byte boundary, as required by CPU specific:
|
|
|
- top_of_stack = (StackType_t *)(((UBaseType_t)(top_of_stack - 31) -
|
|
|
- ALIGNUP(0x10, sizeof(XtSolFrame) )) &
|
|
|
- ~0xf);
|
|
|
+ StaticTask_t *current = (StaticTask_t *)xTaskGetCurrentTaskHandle();
|
|
|
+ //Then put the fake stack inside of TCB:
|
|
|
+ current_task_stack = current->pxDummy6;
|
|
|
+ current->pxDummy6 = (void *)stack;
|
|
|
|
|
|
- //Fake stack frame to do not break the backtrace
|
|
|
- XtSolFrame *frame = (XtSolFrame *)top_of_stack;
|
|
|
- frame->a0 = 0;
|
|
|
- frame->a1 = (UBaseType_t)top_of_stack;
|
|
|
+ StackType_t *top_of_stack = stack + stack_size;
|
|
|
+
|
|
|
+ //Align stack to a 16byte boundary, as required by CPU specific:
|
|
|
+ top_of_stack = (StackType_t *)(((UBaseType_t)(top_of_stack - 16) & ~0xf));
|
|
|
|
|
|
#if CONFIG_FREERTOS_WATCHPOINT_END_OF_STACK
|
|
|
- esp_set_watchpoint(1, (uint8_t *)watchpoint_place, 32, ESP_WATCHPOINT_STORE);
|
|
|
+ esp_set_watchpoint(1, (uint8_t *)watchpoint_place, 32, ESP_WATCHPOINT_STORE);
|
|
|
#endif
|
|
|
|
|
|
- return top_of_stack;
|
|
|
-}
|
|
|
+ xtensa_shared_stack = top_of_stack;
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+void esp_execute_shared_stack_function(SemaphoreHandle_t lock, void *stack, size_t stack_size, shared_stack_function function)
|
|
|
+{
|
|
|
+ assert(lock);
|
|
|
+ assert(stack);
|
|
|
+ assert(stack_size > 0 && stack_size >= CONFIG_ESP_MINIMAL_SHARED_STACK_SIZE);
|
|
|
+ assert(function);
|
|
|
+
|
|
|
+ xSemaphoreTake(lock, portMAX_DELAY);
|
|
|
+ portENTER_CRITICAL(&xtensa_shared_stack_spinlock);
|
|
|
+ xtensa_shared_stack_function_done = false;
|
|
|
+ esp_switch_stack_setup(stack, stack_size);
|
|
|
+ xtensa_shared_stack_callback = function;
|
|
|
+ portEXIT_CRITICAL(&xtensa_shared_stack_spinlock);
|
|
|
+
|
|
|
+ setjmp(xtensa_shared_stack_env);
|
|
|
+ if(!xtensa_shared_stack_function_done) {
|
|
|
+ esp_shared_stack_invoke_function();
|
|
|
+ }
|
|
|
+
|
|
|
+ portENTER_CRITICAL(&xtensa_shared_stack_spinlock);
|
|
|
+ StaticTask_t *current = (StaticTask_t *)xTaskGetCurrentTaskHandle();
|
|
|
+
|
|
|
+ //Restore current task stack:
|
|
|
+ current->pxDummy6 = (StackType_t *)current_task_stack;
|
|
|
+ vPortSetStackWatchpoint(current->pxDummy6);
|
|
|
+ portEXIT_CRITICAL(&xtensa_shared_stack_spinlock);
|
|
|
+
|
|
|
+ xSemaphoreGive(lock);
|
|
|
+}
|