Add timeout feature to control
Add feature that switches to mode IDLE when duty of both motors did not change over certain time. control.cpp/hpp: - add private function that handles timeout - add public function that resets timeout - add slow loop with timeout handle inside control handle loop
This commit is contained in:
parent
20873b4175
commit
d32b0c5671
@ -126,6 +126,78 @@ void controlledArmchair::startHandleLoop() {
|
|||||||
break;
|
break;
|
||||||
//TODO: add other modes here
|
//TODO: add other modes here
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//-----------------------
|
||||||
|
//------ slow loop ------
|
||||||
|
//-----------------------
|
||||||
|
//this section is run about every 5s (+500ms)
|
||||||
|
if (esp_log_timestamp() - timestamp_SlowLoopLastRun > 5000) {
|
||||||
|
ESP_LOGV(TAG, "running slow loop... time since last run: %.1fs", (float)(esp_log_timestamp() - timestamp_SlowLoopLastRun)/1000);
|
||||||
|
timestamp_SlowLoopLastRun = esp_log_timestamp();
|
||||||
|
|
||||||
|
//run function which detects timeout (switch to idle)
|
||||||
|
handleTimeout();
|
||||||
|
}
|
||||||
|
|
||||||
|
}//end while(1)
|
||||||
|
}//end startHandleLoop
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//-----------------------------------
|
||||||
|
//---------- resetTimeout -----------
|
||||||
|
//-----------------------------------
|
||||||
|
void controlledArmchair::resetTimeout(){
|
||||||
|
//TODO mutex
|
||||||
|
timestamp_lastActivity = esp_log_timestamp();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//------------------------------------
|
||||||
|
//---------- handleTimeout -----------
|
||||||
|
//------------------------------------
|
||||||
|
uint32_t msTimeout = 30000; //TODO move this to config #####################
|
||||||
|
float inactivityTolerance = 10; //percentage the duty can vary since last timeout check and still counts as incative
|
||||||
|
|
||||||
|
//local function that checks whether two values differ more than a given tolerance
|
||||||
|
bool validateActivity(float dutyOld, float dutyNow, float tolerance){
|
||||||
|
float dutyDelta = dutyNow - dutyOld;
|
||||||
|
if (fabs(dutyDelta) < tolerance) {
|
||||||
|
return false; //no significant activity detected
|
||||||
|
} else {
|
||||||
|
return true; //there was activity
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//function that evaluates whether there is no activity/change on the motor duty for a certain time. If so, a switch to IDLE is issued. - has to be run repeatedly in a slow interval
|
||||||
|
void controlledArmchair::handleTimeout(){
|
||||||
|
//check for timeout only when not idling already
|
||||||
|
if (mode != controlMode_t::IDLE) {
|
||||||
|
//get current duty from controlled motor objects
|
||||||
|
float dutyLeftNow = motorLeft->getStatus().duty;
|
||||||
|
float dutyRightNow = motorRight->getStatus().duty;
|
||||||
|
|
||||||
|
//activity detected on any of the two motors
|
||||||
|
if (validateActivity(dutyLeft_lastActivity, dutyLeftNow, inactivityTolerance)
|
||||||
|
|| validateActivity(dutyRight_lastActivity, dutyRightNow, inactivityTolerance)
|
||||||
|
){
|
||||||
|
ESP_LOGD(TAG, "timeout check: detected [activity] since last check -> reset");
|
||||||
|
//reset last duty and timestamp
|
||||||
|
timestamp_lastActivity = esp_log_timestamp();
|
||||||
|
dutyLeft_lastActivity = dutyLeftNow;
|
||||||
|
dutyRight_lastActivity = dutyRightNow;
|
||||||
|
}
|
||||||
|
//no activity on any motor and msTimeout exceeded
|
||||||
|
else if (esp_log_timestamp() - timestamp_lastActivity > msTimeout){
|
||||||
|
ESP_LOGI(TAG, "timeout check: [TIMEOUT], no activity for more than %.ds -> switch to idle", msTimeout/1000);
|
||||||
|
//toggle to idle mode
|
||||||
|
toggleIdle();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
ESP_LOGD(TAG, "timeout check: [inactive], last activity %.1f seconds ago", (float)(esp_log_timestamp() - timestamp_lastActivity)/1000);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -136,6 +208,9 @@ void controlledArmchair::startHandleLoop() {
|
|||||||
//-----------------------------------
|
//-----------------------------------
|
||||||
//function to change to a specified control mode
|
//function to change to a specified control mode
|
||||||
void controlledArmchair::changeMode(controlMode_t modeNew) {
|
void controlledArmchair::changeMode(controlMode_t modeNew) {
|
||||||
|
//reset timeout timer
|
||||||
|
resetTimeout();
|
||||||
|
|
||||||
//copy previous mode
|
//copy previous mode
|
||||||
controlMode_t modePrevious = mode;
|
controlMode_t modePrevious = mode;
|
||||||
|
|
||||||
@ -171,6 +246,10 @@ void controlledArmchair::changeMode(controlMode_t modeNew) {
|
|||||||
ESP_LOGI(TAG, "noting to execute when changing TO this mode");
|
ESP_LOGI(TAG, "noting to execute when changing TO this mode");
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case controlMode_t::IDLE:
|
||||||
|
buzzer->beep(1, 1500, 0);
|
||||||
|
break;
|
||||||
|
|
||||||
case controlMode_t::HTTP:
|
case controlMode_t::HTTP:
|
||||||
ESP_LOGW(TAG, "switching to http mode -> enabling http and wifi");
|
ESP_LOGW(TAG, "switching to http mode -> enabling http and wifi");
|
||||||
//start wifi
|
//start wifi
|
||||||
@ -207,12 +286,11 @@ void controlledArmchair::toggleIdle() {
|
|||||||
if (mode == controlMode_t::IDLE){
|
if (mode == controlMode_t::IDLE){
|
||||||
changeMode(modePrevious); //restore previous mode, or default if not switched yet
|
changeMode(modePrevious); //restore previous mode, or default if not switched yet
|
||||||
buzzer->beep(2, 200, 100);
|
buzzer->beep(2, 200, 100);
|
||||||
ESP_LOGW(TAG, "switched mode from IDLE to %s", controlModeStr[(int)mode]);
|
ESP_LOGW(TAG, "toggle idle: switched mode from IDLE to %s", controlModeStr[(int)mode]);
|
||||||
} else {
|
} else {
|
||||||
modePrevious = mode; //store current mode
|
modePrevious = mode; //store current mode
|
||||||
changeMode(controlMode_t::IDLE); //set mode to IDLE
|
changeMode(controlMode_t::IDLE); //set mode to IDLE
|
||||||
buzzer->beep(1, 1000, 0);
|
ESP_LOGW(TAG, "toggle idle: switched mode from %s to IDLE", controlModeStr[(int)mode]);
|
||||||
ESP_LOGW(TAG, "switched mode from IDLE to %s", controlModeStr[(int)mode]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,8 +39,15 @@ class controlledArmchair {
|
|||||||
//function that toggles between two modes, but prefers first argument if entirely different mode is currently active
|
//function that toggles between two modes, but prefers first argument if entirely different mode is currently active
|
||||||
void toggleModes(controlMode_t modePrimary, controlMode_t modeSecondary);
|
void toggleModes(controlMode_t modePrimary, controlMode_t modeSecondary);
|
||||||
|
|
||||||
|
//function that restarts timer which initiates the automatic timeout (switch to IDLE) after certain time of inactivity
|
||||||
|
void resetTimeout();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
|
//--- functions ---
|
||||||
|
//function that evaluates whether there is no activity/change on the motor duty for a certain time, if so a switch to IDLE is issued. - has to be run repeatedly in a slow interval
|
||||||
|
void handleTimeout();
|
||||||
|
|
||||||
//--- objects ---
|
//--- objects ---
|
||||||
buzzer_t* buzzer;
|
buzzer_t* buzzer;
|
||||||
controlledMotor* motorLeft;
|
controlledMotor* motorLeft;
|
||||||
@ -68,6 +75,14 @@ class controlledArmchair {
|
|||||||
.left = cmd_motorIdle,
|
.left = cmd_motorIdle,
|
||||||
.right = cmd_motorIdle
|
.right = cmd_motorIdle
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//variable for slow loop
|
||||||
|
uint32_t timestamp_SlowLoopLastRun = 0;
|
||||||
|
|
||||||
|
//variables for detecting timeout (switch to idle, after inactivity)
|
||||||
|
float dutyLeft_lastActivity = 0;
|
||||||
|
float dutyRight_lastActivity = 0;
|
||||||
|
uint32_t timestamp_lastActivity = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user