Move mode specific delay outside mutex + logging (control.cpp)

potentially fix rare deadlock assuming handle loop takes mutex too fast
again so changeMode never takes it
This commit is contained in:
jonny 2024-09-08 09:25:32 +02:00
parent 5249ac64e0
commit c13006241e

View File

@ -120,21 +120,55 @@ void controlledArmchair::startHandleLoop()
{
while (1)
{
// === handle current mode ===
// mutex to prevent race condition with actions beeing run at mode change and previous mode still beeing executed
ESP_LOGV(TAG, "handle(): requesting mutex...");
if (xSemaphoreTake(handleIteration_mutex, MUTEX_TIMEOUT / portTICK_PERIOD_MS) == pdTRUE)
{
ESP_LOGV(TAG, "handle(): got mutex!");
//--- handle current mode ---
ESP_LOGV(TAG, "control loop executing... mode='%s'", controlModeStr[(int)mode]);
handle();
ESP_LOGV(TAG, "handle(): releasing mutex");
xSemaphoreGive(handleIteration_mutex);
} // end mutex
else {
else
{
ESP_LOGE(TAG, "mutex timeout - stuck in changeMode? -> RESTART");
esp_restart();
}
//--- slow loop ---
// ==== run mode specific delay ===
// outsourced here to not block the mutex by just waiting
switch (mode)
{
default:
case controlMode_t::IDLE:
vTaskDelay(500 / portTICK_PERIOD_MS);
break;
case controlMode_t::JOYSTICK:
vTaskDelay(50 / portTICK_PERIOD_MS);
break;
case controlMode_t::MASSAGE:
vTaskDelay(10 / portTICK_PERIOD_MS);
break;
case controlMode_t::HTTP:
// has 500ms timeout waiting for new events, thus blocks mutex...
break;
case controlMode_t::AUTO:
vTaskDelay(20 / portTICK_PERIOD_MS);
break;
case controlMode_t::ADJUST_CHAIR:
vTaskDelay(100 / portTICK_PERIOD_MS);
break;
case controlMode_t::MENU_SETTINGS:
case controlMode_t::MENU_MODE_SELECT:
vTaskDelay(500 / portTICK_PERIOD_MS);
break;
}
//=== slow loop, timeout ===
// this section is run approx every 5s (+500ms)
if (esp_log_timestamp() - timestamp_SlowLoopLastRun > 5000)
{
@ -150,12 +184,14 @@ void controlledArmchair::startHandleLoop()
}
//-------------------------------------
//---------- Handle control -----------
//-------------------------------------
// function that repeatedly generates motor commands and runs actions depending on the current mode
void controlledArmchair::handle()
{
//note: mode specific delays are outsourced to startHandleLoop() to not block the mutex
switch (mode)
{
@ -166,7 +202,6 @@ void controlledArmchair::handle()
//------- handle IDLE -------
case controlMode_t::IDLE:
vTaskDelay(500 / portTICK_PERIOD_MS);
// TODO repeatedly set motors to idle, in case driver bugs? Currently 15s motorctl timeout would have to pass
#ifdef JOYSTICK_LOG_IN_IDLE
// get joystick data and log it
@ -180,7 +215,6 @@ void controlledArmchair::handle()
//------- handle JOYSTICK mode -------
case controlMode_t::JOYSTICK:
vTaskDelay(50 / portTICK_PERIOD_MS);
// get current joystick data with getData method of evaluatedJoystick
stickDataLast = stickData;
stickData = joystick_l->getData();
@ -205,7 +239,6 @@ void controlledArmchair::handle()
//------- handle MASSAGE mode -------
case controlMode_t::MASSAGE:
vTaskDelay(10 / portTICK_PERIOD_MS);
//--- read joystick ---
// only update joystick data when input not frozen
stickDataLast = stickData;
@ -250,7 +283,6 @@ void controlledArmchair::handle()
//------- handle AUTO mode -------
case controlMode_t::AUTO:
vTaskDelay(20 / portTICK_PERIOD_MS);
// generate commands
commands = automatedArmchair->generateCommands(&instruction);
//--- apply commands to motors ---
@ -291,7 +323,6 @@ void controlledArmchair::handle()
//------- handle ADJUST_CHAIR mode -------
case controlMode_t::ADJUST_CHAIR:
vTaskDelay(100 / portTICK_PERIOD_MS);
//--- read joystick ---
stickDataLast = stickData;
stickData = joystick_l->getData();
@ -308,7 +339,6 @@ void controlledArmchair::handle()
case controlMode_t::MENU_SETTINGS:
case controlMode_t::MENU_MODE_SELECT:
// nothing to do here, display task handles the menu
vTaskDelay(500 / portTICK_PERIOD_MS);
break;
// TODO: add other modes here
@ -457,8 +487,10 @@ void controlledArmchair::changeMode(controlMode_t modeNew, bool noBeep)
// mutex to wait for current handle iteration (control-task) to finish
// prevents race conditions where operations when changing mode are run but old mode gets handled still
ESP_LOGI(TAG, "changeMode: waiting for current handle() iteration to finish...");
ESP_LOGV(TAG, "changemode(): requesting mutex...");
if (xSemaphoreTake(handleIteration_mutex, MUTEX_TIMEOUT / portTICK_PERIOD_MS) == pdTRUE)
{
ESP_LOGV(TAG, "changemode(): got mutex!");
// copy previous mode
modePrevious = mode;
// store time changed (needed for timeout)
@ -569,6 +601,7 @@ void controlledArmchair::changeMode(controlMode_t modeNew, bool noBeep)
mode = modeNew;
// unlock mutex for control task to continue handling modes
ESP_LOGV(TAG, "changemode(): releasing mutex");
xSemaphoreGive(handleIteration_mutex);
} // end mutex
else