/** * @file BinarySemaphore.cpp * * @date 25.02.2020 */ #include #include #include #include "portmacro.h" #include "task.h" BinarySemaphore::BinarySemaphore() { xSemaphoreCreateBinary(handle); if(handle == nullptr) { error << "Binary semaphore creation failure" << std::endl; } } BinarySemaphore::~BinarySemaphore() { vSemaphoreDelete(handle); } ReturnValue_t BinarySemaphore::takeBinarySemaphore(uint32_t timeoutMs) { if(handle == nullptr) { return HasReturnvaluesIF::RETURN_FAILED; } TickType_t timeout = portMAX_DELAY; if(timeoutMs != 0) { timeout = pdMS_TO_TICKS(timeoutMs); } BaseType_t returncode = xSemaphoreTake(handle, timeout); if (returncode == pdPASS) { return HasReturnvaluesIF::RETURN_OK; } else { return SEMAPHORE_NOT_FOUND; } } ReturnValue_t BinarySemaphore::takeBinarySemaphoreTickTimeout(TickType_t timeoutTicks) { if(handle == nullptr) { return SEMAPHORE_NOT_FOUND; } BaseType_t returncode = xSemaphoreTake(handle, timeoutTicks); if (returncode == pdPASS) { return HasReturnvaluesIF::RETURN_OK; } else { return SEMAPHORE_TIMEOUT; } } ReturnValue_t BinarySemaphore::giveBinarySemaphore() { if (handle == nullptr) { return HasReturnvaluesIF::RETURN_FAILED; } BaseType_t returncode = xSemaphoreGive(handle); if (returncode == pdPASS) { return HasReturnvaluesIF::RETURN_OK; } else { return SEMAPHORE_NOT_OWNED; } } SemaphoreHandle_t BinarySemaphore::getSemaphore() { return handle; } ReturnValue_t BinarySemaphore::giveBinarySemaphore(SemaphoreHandle_t semaphore) { if (semaphore == nullptr) { return HasReturnvaluesIF::RETURN_FAILED; } BaseType_t returncode = xSemaphoreGive(semaphore); if (returncode == pdPASS) { return HasReturnvaluesIF::RETURN_OK; } else { return HasReturnvaluesIF::RETURN_FAILED; } } void BinarySemaphore::resetSemaphore() { if(handle != nullptr) { vSemaphoreDelete(handle); xSemaphoreCreateBinary(handle); } } // Be careful with the stack size here. This is called from an ISR! ReturnValue_t BinarySemaphore::giveBinarySemaphoreFromISR(SemaphoreHandle_t semaphore, BaseType_t * higherPriorityTaskWoken) { if (semaphore == nullptr) { return HasReturnvaluesIF::RETURN_FAILED; } BaseType_t returncode = xSemaphoreGiveFromISR(semaphore, higherPriorityTaskWoken); if (returncode == pdPASS) { if(*higherPriorityTaskWoken == pdPASS) { // Request context switch // TODO: I don't know if this will ever happen but if it does, // I want to to know in case this causes issues. If it doesn't // we should remove this. TRACE_INFO("Binary Semaphore: Higher priority task unblocked!"); TaskManagement::requestContextSwitch(CallContext::isr); } return HasReturnvaluesIF::RETURN_OK; } else { return HasReturnvaluesIF::RETURN_FAILED; } }