Add logging, mutex + Fixes - works well with MIN-On/Off time now

This commit is contained in:
jonny_l480 2024-09-06 15:27:09 +02:00
parent 89cab00acc
commit c48b162d1f
4 changed files with 86 additions and 40 deletions

View File

@ -228,7 +228,7 @@ void buttonCommands::startHandleLoop()
// ignore first clicks // ignore first clicks
if (rotateCount++ < IGNORE_ROTATE_COUNT) if (rotateCount++ < IGNORE_ROTATE_COUNT)
{ {
buzzer->beep(1, 60, 0); buzzer->beep(1, 20, 0);
break; break;
} }
timestampLastRotate = esp_log_timestamp(); timestampLastRotate = esp_log_timestamp();
@ -253,15 +253,15 @@ void buttonCommands::startHandleLoop()
{ {
// increment target position each click // increment target position each click
if (event.diff > 0) if (event.diff > 0)
legRest->setTargetPercent(legRest->getTargetPercent() - 5); legRest->setTargetPercent(legRest->getTargetPercent() - 10);
else else
legRest->setTargetPercent(legRest->getTargetPercent() + 5); legRest->setTargetPercent(legRest->getTargetPercent() + 10);
// show temporary notification on display // show temporary notification on display
char buf[8]; char buf[8];
snprintf(buf, 8, "%.0f%%", legRest->getTargetPercent()); snprintf(buf, 8, "%.0f%%", legRest->getTargetPercent());
display_showNotification(2500, "moving Rest:", "LEG", buf); display_showNotification(2500, "moving Rest:", "LEG", buf);
} }
buzzer->beep(1, 90, 0); buzzer->beep(1, 40, 0);
break; break;
case RE_ET_BTN_LONG_PRESSED: case RE_ET_BTN_LONG_PRESSED:
case RE_ET_BTN_CLICKED: case RE_ET_BTN_CLICKED:

View File

@ -607,7 +607,7 @@ void display_showNotification(uint32_t showDurationMs, const char *line1, const
displayTextLineCentered(&dev, 1, true, false, "%s", line2Large); displayTextLineCentered(&dev, 1, true, false, "%s", line2Large);
displayTextLineCentered(&dev, 4, true, false, "%s", line3Large); displayTextLineCentered(&dev, 4, true, false, "%s", line3Large);
displayTextLine(&dev, 7, false, false, " "); displayTextLine(&dev, 7, false, false, " ");
ESP_LOGW(TAG, "start showing notification '%s' '%s' for %d ms", line1, line2Large, showDurationMs); ESP_LOGI(TAG, "start showing notification '%s' '%s' for %d ms", line1, line2Large, showDurationMs);
} }

View File

@ -7,10 +7,15 @@ extern "C"
#include "chairAdjust.hpp" #include "chairAdjust.hpp"
#define MUTEX_TIMEOUT (10000 / portTICK_PERIOD_MS)
#define MIN_TIME_ON 1000 //--- config ---
#define MIN_TIME_OFF 2000 #define MUTEX_TIMEOUT (8000 / portTICK_PERIOD_MS)
// thresholds to protect relays from welding stuck
#define MIN_TIME_ON 600 // minimum time in ms motor has to be ON before being able to turn off again
#define MIN_TIME_OFF 800 // minimum time in ms motor has to be OFF before being able to turn on again (other or same direction)
#define TRAVEL_TIME_LIMIT_ADDITION_MS 2000 // traveling longer into limit compensates inaccuracies in time based position tracking
#define CHAIR_ADJUST_HANDLE_TASK_DELAY 100 // interval the stop-condition and state-switching is checked/handled
//--- gloabl variables --- //--- gloabl variables ---
// strings for logging the rest state // strings for logging the rest state
@ -54,14 +59,19 @@ void cControlledRest::init()
state = REST_OFF; state = REST_OFF;
} }
//========================== //==========================
//===== updatePosition ===== //===== updatePosition =====
//========================== //==========================
// calculate and update position in percent based of time running in current direction // calculate and update position in percent based of time running in current direction
void cControlledRest::updatePosition() void cControlledRest::updatePosition()
{ {
// calculate time motor was on
uint32_t now = esp_log_timestamp(); uint32_t now = esp_log_timestamp();
uint32_t timeRan = now - timestamp_lastPosUpdate; uint32_t timeRan = now - timestamp_lastPosUpdate;
// note: timestamp_lastPosUpdate also gets updated when changing to active mode in changemode
timestamp_lastPosUpdate = now; timestamp_lastPosUpdate = now;
float positionOld = positionNow; float positionOld = positionNow;
@ -76,7 +86,7 @@ void cControlledRest::updatePosition()
break; break;
case REST_OFF: case REST_OFF:
// no change // no change
ESP_LOGW(TAG, "updatePosition() unknown direction - cant update position when state is REST_OFF"); ESP_LOGW(TAG, "updatePosition: unknown direction - cant update position when state is REST_OFF");
return; return;
} }
@ -86,9 +96,11 @@ void cControlledRest::updatePosition()
else if (positionNow > 100) else if (positionNow > 100)
positionNow = 100; positionNow = 100;
ESP_LOGD(TAG, "[%s] state='%s' - update pos from %.2f%% to %.2f%% (time ran %dms)", name, restStateStr[state], positionOld, positionNow, timeRan); ESP_LOGD(TAG, "[%s] updatePosition: update pos from %.2f%% to %.2f%% (time ran %dms, prev-state '%s')", name, positionOld, positionNow, timeRan, restStateStr[state] );
} }
//========================== //==========================
//==== setTargetPercent ==== //==== setTargetPercent ====
//========================== //==========================
@ -140,15 +152,22 @@ void cControlledRest::setTargetPercent(float targetPercent)
// queue state change that is executed when valid (respecting min thresholds) // queue state change that is executed when valid (respecting min thresholds)
void cControlledRest::requestStateChange(restState_t targetState) void cControlledRest::requestStateChange(restState_t targetState)
{ {
// check if task is linked
if (taskHandle == NULL)
{
ESP_LOGE(TAG, "[%s] can not activate task! Task is not running", name);
return;
}
// lock the mutex before accessing shared variables // lock the mutex before accessing shared variables
if (xSemaphoreTakeRecursive(mutex, MUTEX_TIMEOUT) == pdTRUE) if (xSemaphoreTakeRecursive(mutex, MUTEX_TIMEOUT) == pdTRUE)
{ {
ESP_LOGV(TAG, "[%s] requesting change to state '%s'", name, restStateStr[targetState]); ESP_LOGD(TAG, "[%s] requesting change to state '%s'", name, restStateStr[targetState]);
nextState = targetState; nextState = targetState;
// activate handle to process the request when on running already // activate task to change, when on running already
if (state == REST_OFF) if (taskIsRunning == false)
xTaskNotifyGive(taskHandle); // activate handle task that stops the rest-motor again xTaskNotifyGive(taskHandle); // activate handle task that handles state change and stops the rest-motor again
// Release the mutex // Release the mutex
xSemaphoreGiveRecursive(mutex); xSemaphoreGiveRecursive(mutex);
} }
@ -177,6 +196,7 @@ void cControlledRest::handleStateChange()
if (state == nextState) if (state == nextState)
{ {
// exit, nothing todo // exit, nothing todo
ESP_LOGV(TAG, "[%s] handleStateChange: already at target state, nothing to do", name);
xSemaphoreGiveRecursive(mutex); xSemaphoreGiveRecursive(mutex);
return; return;
} }
@ -187,8 +207,7 @@ void cControlledRest::handleStateChange()
// exit if not on long enough // exit if not on long enough
if (now - timestamp_lastStateChange < MIN_TIME_ON) if (now - timestamp_lastStateChange < MIN_TIME_ON)
{ {
positionTarget = positionNow; // disable resuming - no unexpected pos when incrementing ESP_LOGD(TAG, "[%s] handleStateChange: not on long enough, not turning off yet", name);
ESP_LOGV(TAG, "SC: not on long enough, not turning off yet");
xSemaphoreGiveRecursive(mutex); xSemaphoreGiveRecursive(mutex);
return; return;
} }
@ -200,7 +219,7 @@ void cControlledRest::handleStateChange()
// exit if not off long enough // exit if not off long enough
if (now - timestamp_lastStateChange < MIN_TIME_OFF) if (now - timestamp_lastStateChange < MIN_TIME_OFF)
{ {
ESP_LOGV(TAG, "SC: not OFF long enough, not turning on yet"); ESP_LOGV(TAG, "[%s] handleStateChange: not OFF long enough, not turning on yet", name);
xSemaphoreGiveRecursive(mutex); xSemaphoreGiveRecursive(mutex);
return; return;
} }
@ -212,14 +231,14 @@ void cControlledRest::handleStateChange()
// exit if not on long enough // exit if not on long enough
if (now - timestamp_lastStateChange < MIN_TIME_ON) if (now - timestamp_lastStateChange < MIN_TIME_ON)
{ {
ESP_LOGV(TAG, "SC: dir change detected: not ON long enough, not turning off yet"); ESP_LOGD(TAG, "[%s] handleStateChange: dir change detected: not ON long enough, not turning off yet", name);
xSemaphoreGiveRecursive(mutex); xSemaphoreGiveRecursive(mutex);
return; return;
} }
// no immediate dir change, turn off first // no immediate dir change, turn off first
else else
{ {
ESP_LOGD(TAG, "SC: dir change detected: turning off first"); ESP_LOGW(TAG, "[%s] handleStateChange: dir change detected: turning off first", name );
changeState(REST_OFF); changeState(REST_OFF);
xSemaphoreGiveRecursive(mutex); xSemaphoreGiveRecursive(mutex);
return; return;
@ -227,7 +246,7 @@ void cControlledRest::handleStateChange()
} }
// not exited by now = no reason to prevent the state change -> update state! // not exited by now = no reason to prevent the state change -> update state!
ESP_LOGV(TAG, "SC: change is allowed now -> applying new state '%s'", restStateStr[nextState]); ESP_LOGV(TAG, "[%s] handleStateChange: change is allowed now -> applying new state '%s'", name , restStateStr[nextState]);
changeState(nextState); changeState(nextState);
// Release the mutex // Release the mutex
@ -251,12 +270,12 @@ void cControlledRest::changeState(restState_t newState)
// check if actually changed // check if actually changed
if (newState == state) if (newState == state)
{ {
ESP_LOGV(TAG, "[%s] state already at '%s', nothing to do", name, restStateStr[state]); ESP_LOGV(TAG, "[%s] changeState: Relay state already at '%s', nothing to do", name, restStateStr[state]);
return; return;
} }
// apply new state to relays // apply new state to relays
ESP_LOGI(TAG, "[%s] switching Relays from state '%s' to '%s'", name, restStateStr[state], restStateStr[newState]); ESP_LOGI(TAG, "[%s] changeState: switching Relays from state '%s' to '%s'", name, restStateStr[state], restStateStr[newState]);
switch (newState) switch (newState)
{ {
case REST_UP: case REST_UP:
@ -274,9 +293,17 @@ void cControlledRest::changeState(restState_t newState)
} }
// apply new state to variables // apply new state to variables
timestamp_lastStateChange = esp_log_timestamp(); uint32_t now = esp_log_timestamp();
updatePosition(); if (state != REST_OFF && newState == REST_OFF) // previously on (turning off now)
{
updatePosition(); // movement finished -> update position
positionTarget = positionNow; // disable resuming - no unexpected pos when incrementing
}
else if (state == REST_OFF && newState != REST_OFF)// previously off (turning on now)
timestamp_lastPosUpdate = now; // pos did not change during off time - reset timestamp
state = newState; state = newState;
timestamp_lastStateChange = now;
} }
@ -285,7 +312,6 @@ void cControlledRest::changeState(restState_t newState)
//=== handle StopAtPosReached === //=== handle StopAtPosReached ===
//=============================== //===============================
// handle automatic stop when target position is reached, should be run repeatedly in a task // handle automatic stop when target position is reached, should be run repeatedly in a task
#define TRAVEL_TIME_LIMIT_ADDITION_MS 2000 // traveling longer into limit compensates inaccuracies in time based position tracking
void cControlledRest::handleStopAtPosReached() void cControlledRest::handleStopAtPosReached()
{ {
// lock the mutex before accessing shared variables // lock the mutex before accessing shared variables
@ -298,8 +324,9 @@ void cControlledRest::handleStopAtPosReached()
return; return;
} }
// calculate time already running and needed time to reach target // calculate time already running
uint32_t timeRan = esp_log_timestamp() - timestamp_lastPosUpdate; uint32_t timeRan = esp_log_timestamp() - timestamp_lastPosUpdate;
// calculate needed time to reach target
uint32_t timeTarget = travelDuration * fabs(positionTarget - positionNow) / 100; uint32_t timeTarget = travelDuration * fabs(positionTarget - positionNow) / 100;
// intentionally travel longer into limit - compensates inaccuracies in time based position tracking // intentionally travel longer into limit - compensates inaccuracies in time based position tracking
@ -309,9 +336,12 @@ void cControlledRest::handleStopAtPosReached()
// target reached // target reached
if (timeRan >= timeTarget) if (timeRan >= timeTarget)
{ {
ESP_LOGW(TAG, "[%s] reached target! run-time (%dms/%dms) for position %.2f%% -> requesting stop", name, timeRan, timeTarget, positionTarget); ESP_LOGI(TAG, "[%s] TARGET REACHED! run-time (%dms/%dms) for target position %.1f%% -> requesting stop", name, timeRan, timeTarget, positionTarget);
requestStateChange(REST_OFF); requestStateChange(REST_OFF);
} }
else
ESP_LOGV(TAG, "[%s] target not reached yet, run-time (%dms/%dms) for target position %.1f%%", name, timeRan, timeTarget, positionTarget);
// Release the mutex // Release the mutex
xSemaphoreGiveRecursive(mutex); xSemaphoreGiveRecursive(mutex);
} }
@ -322,21 +352,23 @@ void cControlledRest::handleStopAtPosReached()
} }
} }
//============================ //============================
//===== chairAdjust_task ===== //===== chairAdjust_task =====
//============================ //============================
#define CHAIR_ADJUST_HANDLE_TASK_DELAY 100
void chairAdjust_task(void *pvParameter) void chairAdjust_task(void *pvParameter)
{ {
cControlledRest *rest = (cControlledRest *)pvParameter; cControlledRest *rest = (cControlledRest *)pvParameter;
ESP_LOGW(TAG, "Starting task for controlling %s...", rest->getName()); ESP_LOGW(TAG, "Starting task for controlling %s...", rest->getName());
// provide taskHandle to rest object for wakeup // provide taskHandle to rest object for wakeup
rest->taskHandle = xTaskGetCurrentTaskHandle(); rest->setTaskHandle(xTaskGetCurrentTaskHandle());
while (1) while (1)
{ {
ulTaskNotifyTake(pdTRUE, portMAX_DELAY); // wait for wakeup by changeState() (rest-motor turned on) ulTaskNotifyTake(pdTRUE, portMAX_DELAY); // wait for wakeup by changeState() (rest-motor turned on)
ESP_LOGW(TAG, "task %s: received notification -> activating task!", rest->getName()); rest->setTaskIsRunning();
ESP_LOGD(TAG, "task %s: received notification -> activating task!", rest->getName());
// running while 1. motor running or 2. not in target state yet // running while 1. motor running or 2. not in target state yet
while ((rest->getState() != REST_OFF) || (rest->getNextState() != rest->getState())) while ((rest->getState() != REST_OFF) || (rest->getNextState() != rest->getState()))
{ {
@ -344,7 +376,8 @@ void chairAdjust_task(void *pvParameter)
rest->handleStopAtPosReached(); rest->handleStopAtPosReached();
vTaskDelay(CHAIR_ADJUST_HANDLE_TASK_DELAY / portTICK_PERIOD_MS); vTaskDelay(CHAIR_ADJUST_HANDLE_TASK_DELAY / portTICK_PERIOD_MS);
} }
ESP_LOGW(TAG, "task %s: motor-off and at target state -> sleeping task", rest->getName()); rest->clearTaskIsRunning();
ESP_LOGD(TAG, "task %s: motor-off and at target state -> sleeping task", rest->getName());
} }
} }
@ -354,8 +387,6 @@ void chairAdjust_task(void *pvParameter)
//====== controlChairAdjustment ====== //====== controlChairAdjustment ======
//==================================== //====================================
//function that controls the two rests according to joystick data (applies threshold, defines direction) //function that controls the two rests according to joystick data (applies threshold, defines direction)
//TODO:
// - control via app
void controlChairAdjustment(joystickData_t data, cControlledRest * legRest, cControlledRest * backRest){ void controlChairAdjustment(joystickData_t data, cControlledRest * legRest, cControlledRest * backRest){
//--- variables --- //--- variables ---
float stickThreshold = 0.3; //min coordinate for motor to start float stickThreshold = 0.3; //min coordinate for motor to start
@ -364,10 +395,12 @@ void controlChairAdjustment(joystickData_t data, cControlledRest * legRest, cCon
//leg rest (x-axis) //leg rest (x-axis)
if (data.x > stickThreshold) legRest->setTargetPercent(100); if (data.x > stickThreshold) legRest->setTargetPercent(100);
else if (data.x < -stickThreshold) legRest->setTargetPercent(0); else if (data.x < -stickThreshold) legRest->setTargetPercent(0);
else legRest->requestStateChange(REST_OFF); else
legRest->requestStateChange(REST_OFF);
//back rest (y-axis) //back rest (y-axis)
if (data.y > stickThreshold) backRest->setTargetPercent(100); if (data.y > stickThreshold) backRest->setTargetPercent(100);
else if (data.y < -stickThreshold) backRest->setTargetPercent(0); else if (data.y < -stickThreshold) backRest->setTargetPercent(0);
else backRest->requestStateChange(REST_OFF); else
backRest->requestStateChange(REST_OFF);
} }

View File

@ -9,6 +9,7 @@ extern "C"
#include "joystick.hpp" #include "joystick.hpp"
typedef enum { typedef enum {
REST_OFF = 0, REST_OFF = 0,
REST_DOWN, REST_DOWN,
@ -18,6 +19,7 @@ typedef enum {
extern const char* restStateStr[]; extern const char* restStateStr[];
//===================================== //=====================================
//======= cControlledRest class ======= //======= cControlledRest class =======
//===================================== //=====================================
@ -26,30 +28,41 @@ extern const char* restStateStr[];
class cControlledRest { class cControlledRest {
public: public:
cControlledRest(gpio_num_t gpio_up, gpio_num_t gpio_down, uint32_t travelDurationMs, const char *name, float defaultPosition = 0); cControlledRest(gpio_num_t gpio_up, gpio_num_t gpio_down, uint32_t travelDurationMs, const char *name, float defaultPosition = 0);
// control the rest:
void requestStateChange(restState_t targetState); //mutex void requestStateChange(restState_t targetState); //mutex
restState_t getState() const {return state;}; restState_t getState() const {return state;};
restState_t getNextState() const {return nextState;}; const char * getName() const {return name;};
float getPercent(); //TODO update position first
void setTargetPercent(float targetPercent); //mutex void setTargetPercent(float targetPercent); //mutex
float getTargetPercent() const {return positionTarget;}; float getTargetPercent() const {return positionTarget;};
float getPercent(); //TODO update position first
// required for task controlling the rest:
void setTaskHandle(TaskHandle_t handle) {taskHandle = handle;};
void setTaskIsRunning() {taskIsRunning = true;};
void clearTaskIsRunning() {taskIsRunning = false;};
void handleStopAtPosReached(); //mutex void handleStopAtPosReached(); //mutex
void handleStateChange(); //mutex void handleStateChange(); //mutex
const char * getName() const {return name;}; restState_t getNextState() const {return nextState;};
TaskHandle_t taskHandle = NULL; //task that repeatedly runs the handle() method, is assigned at task creation
private: private:
void init(); void init();
void updatePosition(); void updatePosition();
void changeState(restState_t newState); void changeState(restState_t newState);
// task related:
TaskHandle_t taskHandle = NULL; //task that repeatedly runs the handle() method, is assigned at task creation
bool taskIsRunning = false;
SemaphoreHandle_t mutex; SemaphoreHandle_t mutex;
// config:
char name[32]; char name[32];
const gpio_num_t gpio_up; const gpio_num_t gpio_up;
const gpio_num_t gpio_down; const gpio_num_t gpio_down;
const uint32_t travelDuration = 12000; const uint32_t travelDuration = 12000;
// variables:
restState_t state = REST_OFF; restState_t state = REST_OFF;
restState_t nextState = REST_OFF; restState_t nextState = REST_OFF;
uint32_t timestamp_lastStateChange = 0; uint32_t timestamp_lastStateChange = 0;
@ -64,7 +77,7 @@ private:
//=========================== //===========================
//==== chairAdjust_task ===== //==== chairAdjust_task =====
//=========================== //===========================
// repeatedly runs handle method of specified ControlledRest object to turn of the rest, when activated by setState() // repeatedly runs handle methods of specified ControlledRest object to turn of the rest, when activated by changeState() method
void chairAdjust_task( void * cControlledRest ); void chairAdjust_task( void * cControlledRest );