// Copyright 2015-2019 Espressif Systems (Shanghai) PTE LTD // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #ifndef __ESP_EXPRESSION_WITH_STACK_H #define __ESP_EXPRESSION_WITH_STACK_H #include "freertos/FreeRTOS.h" #include "freertos/semphr.h" #include "esp_debug_helpers.h" /** * @brief Executes a 1-line expression with a application alocated stack * @param lock Mutex object to protect in case of shared stack * @param stack Pointer to user alocated stack * @param stack_size Size of current stack in bytes * @param expression Expression or function to be executed using the stack */ #define ESP_EXECUTE_EXPRESSION_WITH_STACK(lock, stack, stack_size, expression) \ ({ \ if(lock && stack && stack_size) { \ uint32_t backup; \ int watchpoint_place=(int)stack; \ portSTACK_TYPE *top_of_stack = &stack[0] + \ (sizeof(stack_size * sizeof(portSTACK_TYPE)) / \ sizeof(portSTACK_TYPE)); \ watchpoint_place=(watchpoint_place+31)&(~31); \ xSemaphoreTake(lock, portMAX_DELAY); \ esp_set_watchpoint(2, (char*)watchpoint_place, 32, ESP_WATCHPOINT_STORE);\ esp_switch_stack_enter(top_of_stack, &backup); \ { \ expression; \ } \ esp_clear_watchpoint(2); \ esp_switch_stack_exit(&backup); \ xSemaphoreGive(lock); \ } \ }) /** * @brief Changes CPU sp-register to use another stack space and save the previous one * @param stack Caller allocated stack pointer * @param backup_stack Pointer to a place to save the current stack * @note Application must not call this function directly */ extern void esp_switch_stack_enter(portSTACK_TYPE *stack, uint32_t *backup_stack); /** * @brief Restores the previous CPU sp-register * @param backup_stack Pointer to the place where stack was saved * @note Application must not call this function directly */ extern void esp_switch_stack_exit(uint32_t *backup_stack); #endif