Rework brake (decel-boost): simplify, config option
Briefly tested this - brake works as intended - Add config options brakeDecel and brakePauseBefureResume - control: update brakeStartThresholdDuty at startup and maxDuty change for each motor - motorctl: drop/simplify decel boost to brake (faster deceleration at certain threshold)
This commit is contained in:
parent
967f2cd8b5
commit
3ebcd6ad3c
@ -32,7 +32,7 @@ void setLoglevels(void)
|
|||||||
// esp_log_level_set("motordriver", ESP_LOG_DEBUG);
|
// esp_log_level_set("motordriver", ESP_LOG_DEBUG);
|
||||||
esp_log_level_set("motor-control", ESP_LOG_WARN);
|
esp_log_level_set("motor-control", ESP_LOG_WARN);
|
||||||
// esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG);
|
// esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG);
|
||||||
esp_log_level_set("joystickCommands", ESP_LOG_DEBUG);
|
esp_log_level_set("joystickCommands", ESP_LOG_WARN);
|
||||||
esp_log_level_set("button", ESP_LOG_INFO);
|
esp_log_level_set("button", ESP_LOG_INFO);
|
||||||
esp_log_level_set("control", ESP_LOG_INFO);
|
esp_log_level_set("control", ESP_LOG_INFO);
|
||||||
// esp_log_level_set("fan-control", ESP_LOG_INFO);
|
// esp_log_level_set("fan-control", ESP_LOG_INFO);
|
||||||
@ -108,7 +108,9 @@ motorctl_config_t configMotorControlLeft = {
|
|||||||
.currentMax = 30,
|
.currentMax = 30,
|
||||||
.currentInverted = true,
|
.currentInverted = true,
|
||||||
.currentSnapToZeroThreshold = 0.15,
|
.currentSnapToZeroThreshold = 0.15,
|
||||||
.deadTimeMs = 0 // minimum time motor is off between direction change
|
.deadTimeMs = 0, // minimum time motor is off between direction change
|
||||||
|
.brakePauseBeforeResume = 1500,
|
||||||
|
.brakeDecel = 400,
|
||||||
};
|
};
|
||||||
|
|
||||||
//--- configure right motor (contol) ---
|
//--- configure right motor (contol) ---
|
||||||
@ -124,7 +126,9 @@ motorctl_config_t configMotorControlRight = {
|
|||||||
.currentMax = 30,
|
.currentMax = 30,
|
||||||
.currentInverted = false,
|
.currentInverted = false,
|
||||||
.currentSnapToZeroThreshold = 0.25,
|
.currentSnapToZeroThreshold = 0.25,
|
||||||
.deadTimeMs = 0 // minimum time motor is off between direction change
|
.deadTimeMs = 0, // minimum time motor is off between direction change
|
||||||
|
.brakePauseBeforeResume = 1500,
|
||||||
|
.brakeDecel = 400,
|
||||||
};
|
};
|
||||||
|
|
||||||
//------------------------------
|
//------------------------------
|
||||||
|
@ -59,6 +59,10 @@ controlledArmchair::controlledArmchair(
|
|||||||
|
|
||||||
// override default config value if maxDuty is found in nvs
|
// override default config value if maxDuty is found in nvs
|
||||||
loadMaxDuty();
|
loadMaxDuty();
|
||||||
|
// update brake start threshold with actual max duty for motorctl
|
||||||
|
ESP_LOGW(TAG, "setting brake start threshold for both motors to %.0f", joystickGenerateCommands_config.maxDutyStraight * BRAKE_START_STICK_PERCENTAGE / 100);
|
||||||
|
motorLeft->setBrakeStartThresholdDuty(joystickGenerateCommands_config.maxDutyStraight * BRAKE_START_STICK_PERCENTAGE / 100);
|
||||||
|
motorRight->setBrakeStartThresholdDuty(joystickGenerateCommands_config.maxDutyStraight * BRAKE_START_STICK_PERCENTAGE / 100);
|
||||||
|
|
||||||
// create semaphore for preventing race condition: mode-change operations while currently still executing certain mode
|
// create semaphore for preventing race condition: mode-change operations while currently still executing certain mode
|
||||||
handleIteration_mutex = xSemaphoreCreateMutex();
|
handleIteration_mutex = xSemaphoreCreateMutex();
|
||||||
|
@ -13,6 +13,8 @@ extern "C"
|
|||||||
#include "speedsensor.hpp"
|
#include "speedsensor.hpp"
|
||||||
#include "chairAdjust.hpp"
|
#include "chairAdjust.hpp"
|
||||||
|
|
||||||
|
//percentage stick has to be moved in the opposite driving direction of current motor direction for braking to start
|
||||||
|
#define BRAKE_START_STICK_PERCENTAGE 95
|
||||||
|
|
||||||
//--------------------------------------------
|
//--------------------------------------------
|
||||||
//---- struct, enum, variable declarations ---
|
//---- struct, enum, variable declarations ---
|
||||||
@ -93,7 +95,11 @@ class controlledArmchair {
|
|||||||
bool toggleAltStickMapping();
|
bool toggleAltStickMapping();
|
||||||
|
|
||||||
// configure max dutycycle (in joystick or http mode)
|
// configure max dutycycle (in joystick or http mode)
|
||||||
void setMaxDuty(float maxDutyNew) { writeMaxDuty(maxDutyNew); };
|
void setMaxDuty(float maxDutyNew) {
|
||||||
|
writeMaxDuty(maxDutyNew);
|
||||||
|
motorLeft->setBrakeStartThresholdDuty(joystickGenerateCommands_config.maxDutyStraight * BRAKE_START_STICK_PERCENTAGE/100);
|
||||||
|
motorRight->setBrakeStartThresholdDuty(joystickGenerateCommands_config.maxDutyStraight * BRAKE_START_STICK_PERCENTAGE/100);
|
||||||
|
};
|
||||||
float getMaxDuty() const {return joystickGenerateCommands_config.maxDutyStraight; };
|
float getMaxDuty() const {return joystickGenerateCommands_config.maxDutyStraight; };
|
||||||
// configure max boost (in joystick or http mode)
|
// configure max boost (in joystick or http mode)
|
||||||
void setMaxRelativeBoostPer(float newValue) { joystickGenerateCommands_config.maxRelativeBoostPercentOfMaxDuty = newValue; };
|
void setMaxRelativeBoostPer(float newValue) { joystickGenerateCommands_config.maxRelativeBoostPercentOfMaxDuty = newValue; };
|
||||||
|
@ -320,40 +320,62 @@ if ( dutyNow != 0 && esp_log_timestamp() - timestamp_commandReceived > TIMEOUT_I
|
|||||||
//calculate passed time since last run
|
//calculate passed time since last run
|
||||||
int64_t usPassed = esp_timer_get_time() - timestampLastRunUs;
|
int64_t usPassed = esp_timer_get_time() - timestampLastRunUs;
|
||||||
|
|
||||||
//--- calculate increment ---
|
//--- calculate increment (acceleration) ---
|
||||||
//calculate increment for fading UP with passed time since last run and configured fade time
|
//calculate increment for fading UP with passed time since last run and configured fade time
|
||||||
|
//- traction control -
|
||||||
if (tcs_isExceeded) // disable acceleration when slippage is currently detected
|
if (tcs_isExceeded) // disable acceleration when slippage is currently detected
|
||||||
dutyIncrementAccel = 0;
|
dutyIncrementAccel = 0;
|
||||||
|
//- recent braking -
|
||||||
|
//FIXME reset timeout when duty less
|
||||||
|
else if (isBraking && (esp_log_timestamp() - timestampBrakeStart) < config.brakePauseBeforeResume) // prevent immediate direction change when currently braking with timeout (eventually currently sliding)
|
||||||
|
{
|
||||||
|
if (log) ESP_LOGI(TAG, "pause after brake... -> accel = 0");
|
||||||
|
dutyIncrementAccel = 0;
|
||||||
|
}
|
||||||
|
//- normal accel -
|
||||||
else if (msFadeAccel > 0)
|
else if (msFadeAccel > 0)
|
||||||
dutyIncrementAccel = (usPassed / ((float)msFadeAccel * 1000)) * 100; // TODO define maximum increment - first run after startup (or long) pause can cause a very large increment
|
dutyIncrementAccel = (usPassed / ((float)msFadeAccel * 1000)) * 100; // TODO define maximum increment - first run after startup (or long) pause can cause a very large increment
|
||||||
|
//- sport mode -
|
||||||
else //no accel limit (immediately set to 100)
|
else //no accel limit (immediately set to 100)
|
||||||
dutyIncrementAccel = 100;
|
dutyIncrementAccel = 100;
|
||||||
|
|
||||||
#define DECEL_BOOST_START_THRESHOLD 10 // boost deceleration when stick/target duty is more than that value in opposite direction
|
//--- calculate increment (deceleration) ---
|
||||||
#define DECEL_BOOST_MIN_DECEL_TIME 350 // milliseconds from 100% to 0%
|
//- sport mode -
|
||||||
//calculate increment for fading DOWN with passed time since last run and configured fade time
|
if (msFadeDecel == 0){ //no decel limit (immediately reduce to 0)
|
||||||
// FIXME: dutyTarget does not represent joystick pos (max maxDuty) -> currently breaks when maxDuty is not set to 100
|
|
||||||
if (msFadeDecel == 0) //no decel limit (immediately reduce to 0)
|
|
||||||
dutyIncrementDecel = 100;
|
dutyIncrementDecel = 100;
|
||||||
//--- dynamic fading ---
|
|
||||||
//detect when quicker brake response is desired (e.g. full speed forward, joystick suddenly is full reverse -> break fast)
|
|
||||||
else if (commandReceive.state != state && fabs(dutyTarget) > DECEL_BOOST_START_THRESHOLD)
|
|
||||||
{
|
|
||||||
float normalIncrementDecel = (usPassed / ((float)msFadeDecel * 1000)) * 100; //TODO limit all increments to 100?
|
|
||||||
//calculate absolute maximum allowed deceleration
|
|
||||||
float maximumIncrementDecel = ((float)usPassed / (DECEL_BOOST_MIN_DECEL_TIME * 1000)) * 100;
|
|
||||||
//calculate how much boost is applied (percent) depending on how much the joystick is in opposite direction
|
|
||||||
float decelBoostFactor = (fabs(dutyTarget) - DECEL_BOOST_START_THRESHOLD) / (100 - DECEL_BOOST_START_THRESHOLD);
|
|
||||||
//calculate total deceleration increment (normal + boost)
|
|
||||||
dutyIncrementDecel = normalIncrementDecel + decelBoostFactor * fabs(maximumIncrementDecel - normalIncrementDecel);
|
|
||||||
if(log) ESP_LOGW(TAG, "boosting deceleration by %.2f%% of remainder to max decel", decelBoostFactor * 100);
|
|
||||||
if(log) ESP_LOGW(TAG, "normalInc=%.5f, maxInc=%.5f, totalInc=%.5f", normalIncrementDecel, maximumIncrementDecel, dutyIncrementDecel);
|
|
||||||
}
|
}
|
||||||
else
|
//- brake -
|
||||||
|
//detect when quicker brake response is desired (e.g. full speed forward, joystick suddenly is full reverse -> break fast)
|
||||||
|
#define NO_BRAKE_THRESHOLD_TOO_SLOW_DUTY 10 //TODO test/adjust this - dont brake when slow already (avoids starting full dead time)
|
||||||
|
else if (commandReceive.state != state && // direction differs
|
||||||
|
fabs(dutyNow) > NO_BRAKE_THRESHOLD_TOO_SLOW_DUTY && // not very slow already
|
||||||
|
fabs(dutyTarget) > brakeStartThreshold) // joystick above threshold
|
||||||
|
{
|
||||||
|
// set braking state and track start time (both for disabling acceleration for some time)
|
||||||
|
if (!isBraking) {
|
||||||
|
if (log) ESP_LOGW(TAG, "started braking...");
|
||||||
|
timestampBrakeStart = esp_log_timestamp();
|
||||||
|
isBraking = true;
|
||||||
|
}
|
||||||
|
// use brake deceleration instead of normal deceleration
|
||||||
|
dutyIncrementDecel = (usPassed / ((float)config.brakeDecel * 1000)) * 100;
|
||||||
|
if(log) ESP_LOGI(TAG, "braking (target duty >%.0f%% in other direction) -> using deceleration %dms", brakeStartThreshold, config.brakeDecel);
|
||||||
|
}
|
||||||
|
//- normal deceleration -
|
||||||
|
else {
|
||||||
// normal deceleration according to configured time
|
// normal deceleration according to configured time
|
||||||
dutyIncrementDecel = (usPassed / ((float)msFadeDecel * 1000)) * 100;
|
dutyIncrementDecel = (usPassed / ((float)msFadeDecel * 1000)) * 100;
|
||||||
|
}
|
||||||
|
|
||||||
//fade duty to target (up and down)
|
// reset braking state when start condition is no longer met (stick below threshold again)
|
||||||
|
if (isBraking &&
|
||||||
|
(fabs(dutyTarget) < brakeStartThreshold || commandReceive.state == state))
|
||||||
|
{
|
||||||
|
ESP_LOGW(TAG, "brake condition no longer met");
|
||||||
|
isBraking = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
//--- fade duty to target (up and down) ---
|
||||||
//TODO: this needs optimization (can be more clear and/or simpler)
|
//TODO: this needs optimization (can be more clear and/or simpler)
|
||||||
if (dutyDelta > 0) { //difference positive -> increasing duty (-100 -> 100)
|
if (dutyDelta > 0) { //difference positive -> increasing duty (-100 -> 100)
|
||||||
if (dutyNow < 0) { //reverse, decelerating
|
if (dutyNow < 0) { //reverse, decelerating
|
||||||
|
@ -46,6 +46,7 @@ class controlledMotor {
|
|||||||
void disableTractionControlSystem() {config.tractionControlSystemEnabled = false; tcs_isExceeded = false;};
|
void disableTractionControlSystem() {config.tractionControlSystemEnabled = false; tcs_isExceeded = false;};
|
||||||
bool getTractionControlSystemStatus() {return config.tractionControlSystemEnabled;};
|
bool getTractionControlSystemStatus() {return config.tractionControlSystemEnabled;};
|
||||||
void setControlMode(motorControlMode_t newMode) {mode = newMode;};
|
void setControlMode(motorControlMode_t newMode) {mode = newMode;};
|
||||||
|
void setBrakeStartThresholdDuty(float duty) {brakeStartThreshold = duty;};
|
||||||
|
|
||||||
uint32_t getFade(fadeType_t fadeType); //get currently set acceleration or deceleration fading time
|
uint32_t getFade(fadeType_t fadeType); //get currently set acceleration or deceleration fading time
|
||||||
uint32_t getFadeDefault(fadeType_t fadeType); //get acceleration or deceleration fading time from config
|
uint32_t getFadeDefault(fadeType_t fadeType); //get acceleration or deceleration fading time from config
|
||||||
@ -129,6 +130,11 @@ class controlledMotor {
|
|||||||
uint32_t tcs_usExceeded = 0; //sum up time
|
uint32_t tcs_usExceeded = 0; //sum up time
|
||||||
bool tcs_isExceeded = false; //is currently too fast
|
bool tcs_isExceeded = false; //is currently too fast
|
||||||
int64_t tcs_timestampLastRun = 0;
|
int64_t tcs_timestampLastRun = 0;
|
||||||
|
|
||||||
|
//brake (decel boost)
|
||||||
|
uint32_t timestampBrakeStart = 0;
|
||||||
|
bool isBraking = false;
|
||||||
|
float brakeStartThreshold = 60;
|
||||||
};
|
};
|
||||||
|
|
||||||
//====================================
|
//====================================
|
||||||
|
@ -53,6 +53,8 @@ typedef struct motorctl_config_t {
|
|||||||
bool currentInverted;
|
bool currentInverted;
|
||||||
float currentSnapToZeroThreshold;
|
float currentSnapToZeroThreshold;
|
||||||
uint32_t deadTimeMs; //time motor stays in IDLE before direction change
|
uint32_t deadTimeMs; //time motor stays in IDLE before direction change
|
||||||
|
uint32_t brakePauseBeforeResume;
|
||||||
|
uint32_t brakeDecel;
|
||||||
} motorctl_config_t;
|
} motorctl_config_t;
|
||||||
|
|
||||||
//enum fade type (acceleration, deceleration)
|
//enum fade type (acceleration, deceleration)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user