armchair_fw/common/chairAdjust.cpp
jonny 89cab00acc Rework cControlledRest (for MIN_TIME_ON, MIN_TIME_OFF)
- split up in more simple methods
- clearer structure
2024-09-06 11:05:31 +02:00

373 lines
12 KiB
C++

extern "C"
{
#include "esp_log.h"
#include <string.h>
}
#include "chairAdjust.hpp"
#define MUTEX_TIMEOUT (10000 / portTICK_PERIOD_MS)
#define MIN_TIME_ON 1000
#define MIN_TIME_OFF 2000
//--- gloabl variables ---
// strings for logging the rest state
const char* restStateStr[] = {"REST_OFF", "REST_DOWN", "REST_UP"};
//--- local variables ---
//tag for logging
static const char * TAG = "chair-adjustment";
//=============================
//======== constructor ========
//=============================
cControlledRest::cControlledRest(gpio_num_t gpio_up_f, gpio_num_t gpio_down_f, uint32_t travelDurationMs, const char * name_f, float defaultPosition): gpio_up(gpio_up_f), gpio_down(gpio_down_f), travelDuration(travelDurationMs){
strcpy(name, name_f);
positionNow = defaultPosition;
positionTarget = positionNow;
// recursive mutex necessary, because handle() method calls setState() which both have the same mutex
mutex = xSemaphoreCreateRecursiveMutex();
init();
}
//====================
//======= init =======
//====================
// init gpio pins for relays
void cControlledRest::init()
{
ESP_LOGW(TAG, "[%s] initializing gpio pins %d, %d for relays...", name, gpio_up, gpio_down);
// configure 2 gpio pins
gpio_pad_select_gpio(gpio_up);
gpio_set_direction(gpio_up, GPIO_MODE_OUTPUT);
gpio_pad_select_gpio(gpio_down);
gpio_set_direction(gpio_down, GPIO_MODE_OUTPUT);
// both relays off initially
gpio_set_level(gpio_down, 0);
gpio_set_level(gpio_up, 0);
state = REST_OFF;
}
//==========================
//===== updatePosition =====
//==========================
// calculate and update position in percent based of time running in current direction
void cControlledRest::updatePosition()
{
uint32_t now = esp_log_timestamp();
uint32_t timeRan = now - timestamp_lastPosUpdate;
timestamp_lastPosUpdate = now;
float positionOld = positionNow;
// calculate new percentage
switch (state)
{
case REST_UP:
positionNow += (float)timeRan / travelDuration * 100;
break;
case REST_DOWN:
positionNow -= (float)timeRan / travelDuration * 100;
break;
case REST_OFF:
// no change
ESP_LOGW(TAG, "updatePosition() unknown direction - cant update position when state is REST_OFF");
return;
}
// clip to 0-100 (because cant actually happen due to limit switches)
if (positionNow < 0)
positionNow = 0;
else if (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);
}
//==========================
//==== setTargetPercent ====
//==========================
void cControlledRest::setTargetPercent(float targetPercent)
{
// lock the mutex before accessing shared variables
if (xSemaphoreTakeRecursive(mutex, MUTEX_TIMEOUT) == pdTRUE)
{
float positionTargetPrev = positionTarget;
positionTarget = targetPercent;
// limit to 0-100
if (positionTarget > 100)
positionTarget = 100;
else if (positionTarget < 0)
positionTarget = 0;
ESP_LOGI(TAG, "[%s] changed Target position from %.2f%% to %.2f%%", name, positionTargetPrev, positionTarget);
// update actual position positionNow first when already running
if (state != REST_OFF)
updatePosition();
// start rest in required direction
// TODO always run this check in handle()?
// note: when already at 0/100 start anyways (runs for certain threshold in case tracked position out of sync)
if (positionTarget > positionNow || positionTarget >= 100)
requestStateChange(REST_UP);
else if (positionTarget < positionNow || positionTarget <= 0)
requestStateChange(REST_DOWN);
else // already at exact position
requestStateChange(REST_OFF);
// Release the mutex
xSemaphoreGiveRecursive(mutex);
}
else
{
ESP_LOGE(TAG, "mutex timeout while waiting in setTargetPercent -> RESTART");
esp_restart();
}
}
//============================
//==== requestStateChange ====
//============================
// queue state change that is executed when valid (respecting min thresholds)
void cControlledRest::requestStateChange(restState_t targetState)
{
// lock the mutex before accessing shared variables
if (xSemaphoreTakeRecursive(mutex, MUTEX_TIMEOUT) == pdTRUE)
{
ESP_LOGV(TAG, "[%s] requesting change to state '%s'", name, restStateStr[targetState]);
nextState = targetState;
// activate handle to process the request when on running already
if (state == REST_OFF)
xTaskNotifyGive(taskHandle); // activate handle task that stops the rest-motor again
// Release the mutex
xSemaphoreGiveRecursive(mutex);
}
else
{
ESP_LOGE(TAG, "mutex timeout while waiting in requestStateChange -> RESTART");
esp_restart();
}
}
//===========================
//==== handleStateChange ====
//===========================
// ensure MIN_TIME_ON and MIN_TIME_OFF has passed between doing the requested state change
// repeatedly run by task if state is requested
void cControlledRest::handleStateChange()
{
// lock the mutex before accessing shared variables
if (xSemaphoreTakeRecursive(mutex, MUTEX_TIMEOUT) == pdTRUE)
{
uint32_t now = esp_log_timestamp();
// already at target state
if (state == nextState)
{
// exit, nothing todo
xSemaphoreGiveRecursive(mutex);
return;
}
// turn off requested
else if (nextState == REST_OFF)
{
// exit if not on long enough
if (now - timestamp_lastStateChange < MIN_TIME_ON)
{
positionTarget = positionNow; // disable resuming - no unexpected pos when incrementing
ESP_LOGV(TAG, "SC: not on long enough, not turning off yet");
xSemaphoreGiveRecursive(mutex);
return;
}
}
// turn on requested
else if (state == REST_OFF && nextState != REST_OFF)
{
// exit if not off long enough
if (now - timestamp_lastStateChange < MIN_TIME_OFF)
{
ESP_LOGV(TAG, "SC: not OFF long enough, not turning on yet");
xSemaphoreGiveRecursive(mutex);
return;
}
}
// direction change requested
else if (state != REST_OFF && nextState != REST_OFF)
{
// exit if not on long enough
if (now - timestamp_lastStateChange < MIN_TIME_ON)
{
ESP_LOGV(TAG, "SC: dir change detected: not ON long enough, not turning off yet");
xSemaphoreGiveRecursive(mutex);
return;
}
// no immediate dir change, turn off first
else
{
ESP_LOGD(TAG, "SC: dir change detected: turning off first");
changeState(REST_OFF);
xSemaphoreGiveRecursive(mutex);
return;
}
}
// 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]);
changeState(nextState);
// Release the mutex
xSemaphoreGiveRecursive(mutex);
}
else
{
ESP_LOGE(TAG, "mutex timeout while waiting in handleStateChange -> RESTART");
esp_restart();
}
}
//=========================
//====== changeState ======
//=========================
// change state (relays, timestamp) without any validation whether change is allowed
void cControlledRest::changeState(restState_t newState)
{
// check if actually changed
if (newState == state)
{
ESP_LOGV(TAG, "[%s] state already at '%s', nothing to do", name, restStateStr[state]);
return;
}
// apply new state to relays
ESP_LOGI(TAG, "[%s] switching Relays from state '%s' to '%s'", name, restStateStr[state], restStateStr[newState]);
switch (newState)
{
case REST_UP:
gpio_set_level(gpio_down, 0);
gpio_set_level(gpio_up, 1);
break;
case REST_DOWN:
gpio_set_level(gpio_down, 1);
gpio_set_level(gpio_up, 0);
break;
case REST_OFF:
gpio_set_level(gpio_down, 0);
gpio_set_level(gpio_up, 0);
break;
}
// apply new state to variables
timestamp_lastStateChange = esp_log_timestamp();
updatePosition();
state = newState;
}
//===============================
//=== handle StopAtPosReached ===
//===============================
// 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()
{
// lock the mutex before accessing shared variables
if (xSemaphoreTakeRecursive(mutex, MUTEX_TIMEOUT) == pdTRUE)
{
// nothing to do when not running atm
if (state == REST_OFF)
{
xSemaphoreGiveRecursive(mutex);
return;
}
// calculate time already running and needed time to reach target
uint32_t timeRan = esp_log_timestamp() - timestamp_lastPosUpdate;
uint32_t timeTarget = travelDuration * fabs(positionTarget - positionNow) / 100;
// intentionally travel longer into limit - compensates inaccuracies in time based position tracking
if (positionTarget == 0 || positionTarget == 100)
timeTarget += TRAVEL_TIME_LIMIT_ADDITION_MS;
// target reached
if (timeRan >= timeTarget)
{
ESP_LOGW(TAG, "[%s] reached target! run-time (%dms/%dms) for position %.2f%% -> requesting stop", name, timeRan, timeTarget, positionTarget);
requestStateChange(REST_OFF);
}
// Release the mutex
xSemaphoreGiveRecursive(mutex);
}
else
{
ESP_LOGE(TAG, "mutex timeout while waiting in handleStopAtPosReached -> RESTART");
esp_restart();
}
}
//============================
//===== chairAdjust_task =====
//============================
#define CHAIR_ADJUST_HANDLE_TASK_DELAY 100
void chairAdjust_task(void *pvParameter)
{
cControlledRest *rest = (cControlledRest *)pvParameter;
ESP_LOGW(TAG, "Starting task for controlling %s...", rest->getName());
// provide taskHandle to rest object for wakeup
rest->taskHandle = xTaskGetCurrentTaskHandle();
while (1)
{
ulTaskNotifyTake(pdTRUE, portMAX_DELAY); // wait for wakeup by changeState() (rest-motor turned on)
ESP_LOGW(TAG, "task %s: received notification -> activating task!", rest->getName());
// running while 1. motor running or 2. not in target state yet
while ((rest->getState() != REST_OFF) || (rest->getNextState() != rest->getState()))
{
rest->handleStateChange();
rest->handleStopAtPosReached();
vTaskDelay(CHAIR_ADJUST_HANDLE_TASK_DELAY / portTICK_PERIOD_MS);
}
ESP_LOGW(TAG, "task %s: motor-off and at target state -> sleeping task", rest->getName());
}
}
//====================================
//====== controlChairAdjustment ======
//====================================
//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){
//--- variables ---
float stickThreshold = 0.3; //min coordinate for motor to start
//--- control rest motors ---
//leg rest (x-axis)
if (data.x > stickThreshold) legRest->setTargetPercent(100);
else if (data.x < -stickThreshold) legRest->setTargetPercent(0);
else legRest->requestStateChange(REST_OFF);
//back rest (y-axis)
if (data.y > stickThreshold) backRest->setTargetPercent(100);
else if (data.y < -stickThreshold) backRest->setTargetPercent(0);
else backRest->requestStateChange(REST_OFF);
}