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("motor-control", ESP_LOG_WARN);
|
||||
// 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("control", ESP_LOG_INFO);
|
||||
// esp_log_level_set("fan-control", ESP_LOG_INFO);
|
||||
@ -108,7 +108,9 @@ motorctl_config_t configMotorControlLeft = {
|
||||
.currentMax = 30,
|
||||
.currentInverted = true,
|
||||
.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) ---
|
||||
@ -124,7 +126,9 @@ motorctl_config_t configMotorControlRight = {
|
||||
.currentMax = 30,
|
||||
.currentInverted = false,
|
||||
.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
|
||||
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
|
||||
handleIteration_mutex = xSemaphoreCreateMutex();
|
||||
|
@ -13,6 +13,8 @@ extern "C"
|
||||
#include "speedsensor.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 ---
|
||||
@ -93,7 +95,11 @@ class controlledArmchair {
|
||||
bool toggleAltStickMapping();
|
||||
|
||||
// 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; };
|
||||
// configure max boost (in joystick or http mode)
|
||||
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
|
||||
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
|
||||
//- traction control -
|
||||
if (tcs_isExceeded) // disable acceleration when slippage is currently detected
|
||||
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)
|
||||
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)
|
||||
dutyIncrementAccel = 100;
|
||||
|
||||
#define DECEL_BOOST_START_THRESHOLD 10 // boost deceleration when stick/target duty is more than that value in opposite direction
|
||||
#define DECEL_BOOST_MIN_DECEL_TIME 350 // milliseconds from 100% to 0%
|
||||
//calculate increment for fading DOWN with passed time since last run and configured fade time
|
||||
// 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)
|
||||
//--- calculate increment (deceleration) ---
|
||||
//- sport mode -
|
||||
if (msFadeDecel == 0){ //no decel limit (immediately reduce to 0)
|
||||
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
|
||||
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)
|
||||
if (dutyDelta > 0) { //difference positive -> increasing duty (-100 -> 100)
|
||||
if (dutyNow < 0) { //reverse, decelerating
|
||||
|
@ -46,6 +46,7 @@ class controlledMotor {
|
||||
void disableTractionControlSystem() {config.tractionControlSystemEnabled = false; tcs_isExceeded = false;};
|
||||
bool getTractionControlSystemStatus() {return config.tractionControlSystemEnabled;};
|
||||
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 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
|
||||
bool tcs_isExceeded = false; //is currently too fast
|
||||
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;
|
||||
float currentSnapToZeroThreshold;
|
||||
uint32_t deadTimeMs; //time motor stays in IDLE before direction change
|
||||
uint32_t brakePauseBeforeResume;
|
||||
uint32_t brakeDecel;
|
||||
} motorctl_config_t;
|
||||
|
||||
//enum fade type (acceleration, deceleration)
|
||||
|
Loading…
x
Reference in New Issue
Block a user