#ifndef BELL_TASK_H #define BELL_TASK_H #ifdef ESP_PLATFORM #include #include #include #include #include #elif _WIN32 #include #else #include #endif #include namespace bell { class Task { public: std::string taskName; int stackSize, core; bool runOnPSRAM; Task(std::string taskName, int stackSize, int priority, int core, bool runOnPSRAM = true) { this->taskName = taskName; this->stackSize = stackSize; this->core = core; this->runOnPSRAM = runOnPSRAM; #ifdef ESP_PLATFORM this->xStack = NULL; this->priority = CONFIG_ESP32_PTHREAD_TASK_PRIO_DEFAULT + priority; if (this->priority <= ESP_TASK_PRIO_MIN) this->priority = ESP_TASK_PRIO_MIN + 1; if (runOnPSRAM) { this->xStack = (StackType_t*) heap_caps_malloc(this->stackSize, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT); } #endif } virtual ~Task() { #ifdef ESP_PLATFORM if (xStack) heap_caps_free(xStack); #endif } bool startTask() { #ifdef ESP_PLATFORM if (runOnPSRAM) { xTaskBuffer = (StaticTask_t*) heap_caps_malloc(sizeof(StaticTask_t), MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); return (xTaskCreateStaticPinnedToCore(taskEntryFuncPSRAM, this->taskName.c_str(), this->stackSize, this, this->priority, xStack, xTaskBuffer, this->core) != NULL); } else { printf("task on internal %s", this->taskName.c_str()); esp_pthread_cfg_t cfg = esp_pthread_get_default_config(); cfg.stack_size = stackSize; cfg.inherit_cfg = true; cfg.thread_name = this->taskName.c_str(); cfg.pin_to_core = core; cfg.prio = this->priority; esp_pthread_set_cfg(&cfg); } #endif #if _WIN32 thread = CreateThread(NULL, stackSize, (LPTHREAD_START_ROUTINE) taskEntryFunc, this, 0, NULL); return thread != NULL; #else return (pthread_create(&thread, NULL, taskEntryFunc, this) == 0); #endif } protected: virtual void runTask() = 0; private: #if _WIN32 HANDLE thread; #else pthread_t thread; #endif #ifdef ESP_PLATFORM int priority; StaticTask_t *xTaskBuffer; StackType_t *xStack; static void taskEntryFuncPSRAM(void *This) { Task* self = (Task*) This; self->runTask(); // TCB are cleanup in IDLE task, so give it some time TimerHandle_t timer = xTimerCreate( "cleanup", pdMS_TO_TICKS(5000), pdFALSE, self->xTaskBuffer, [](TimerHandle_t xTimer) { heap_caps_free(pvTimerGetTimerID(xTimer)); xTimerDelete(xTimer, portMAX_DELAY); } ); xTimerStart(timer, portMAX_DELAY); vTaskDelete(NULL); } #endif static void *taskEntryFunc(void *This) { Task* self = (Task*) This; self->runTask(); #if _WIN32 WaitForSingleObject(self->thread, INFINITE); #else pthread_join(self->thread, NULL); #endif return NULL; } }; } #endif