diff --git a/board_single/main/config.cpp b/board_single/main/config.cpp index a5efdf7..e665ce9 100644 --- a/board_single/main/config.cpp +++ b/board_single/main/config.cpp @@ -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, }; //------------------------------ diff --git a/board_single/main/control.cpp b/board_single/main/control.cpp index baa6384..684f562 100644 --- a/board_single/main/control.cpp +++ b/board_single/main/control.cpp @@ -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(); diff --git a/board_single/main/control.hpp b/board_single/main/control.hpp index ebca9b3..419e977 100644 --- a/board_single/main/control.hpp +++ b/board_single/main/control.hpp @@ -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; }; diff --git a/common/motorctl.cpp b/common/motorctl.cpp index b0c06f9..de39ef9 100644 --- a/common/motorctl.cpp +++ b/common/motorctl.cpp @@ -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 diff --git a/common/motorctl.hpp b/common/motorctl.hpp index 1ec9a41..7c0d50f 100644 --- a/common/motorctl.hpp +++ b/common/motorctl.hpp @@ -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; }; //==================================== diff --git a/common/types.hpp b/common/types.hpp index 5421a3c..23f5d04 100644 --- a/common/types.hpp +++ b/common/types.hpp @@ -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)