From 9db5736f77ddeb919c41b10c0b80747fa18a686c Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Sat, 25 Jun 2022 14:00:57 +0200 Subject: [PATCH 01/12] Create shake mode (basic functionality) - add function joystick_generateCommandsShaking that generates motor commands from joystick data - pulses motors: - intervals depend on joystick radius - direction depends on joystick position (currently only on-x-axis and on-y-axis supported) --- main/control.cpp | 9 ++-- main/joystick.cpp | 118 ++++++++++++++++++++++++++++++++++++++++++++++ main/joystick.hpp | 10 ++++ 3 files changed, 134 insertions(+), 3 deletions(-) diff --git a/main/control.cpp b/main/control.cpp index b672977..eb83bbb 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -79,9 +79,12 @@ void controlledArmchair::startHandleLoop() { break; case controlMode_t::MASSAGE: - motorRight->setTarget(motorstate_t::IDLE, 0); - motorLeft->setTarget(motorstate_t::IDLE, 0); - //TODO add actual command generation here + //generate motor commands + //pass joystick data from getData method of evaluatedJoystick to generateCommandsShaking function + commands = joystick_generateCommandsShaking(joystick.getData()); + //apply motor commands + motorRight->setTarget(commands.right.state, commands.right.duty); + motorLeft->setTarget(commands.left.state, commands.left.duty); vTaskDelay(20 / portTICK_PERIOD_MS); break; diff --git a/main/joystick.cpp b/main/joystick.cpp index 9796a55..129b83e 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -299,3 +299,121 @@ motorCommands_t joystick_generateCommandsDriving(joystickData_t data){ ESP_LOGI(TAG_CMD, "motor right: state=%s, duty=%.3f", motorstateStr[(int)commands.right.state], commands.right.duty); return commands; } + + + +//============================================ +//========= joystick_CommandsShaking ========= +//============================================ +//--- variable declarations --- +uint32_t shake_timestamp_turnedOn = 0; +uint32_t shake_timestamp_turnedOff = 0; +bool shake_state = false; + +//--- configure shake mode --- TODO: move this to config +uint32_t shake_msOffMax = 90; +uint32_t shake_msOnMax = 180; +float dutyShake = 60; + +//function that generates commands for both motors from the joystick data +motorCommands_t joystick_generateCommandsShaking(joystickData_t data){ + + + //struct with current data of the joystick + //typedef struct joystickData_t { + // joystickPos_t position; + // float x; + // float y; + // float radius; + // float angle; + //} joystickData_t; + + + motorCommands_t commands; + float ratio = fabs(data.angle) / 90; //90degree = x=0 || 0degree = y=0 + + //--- calculate on/off duration --- + uint32_t msOn = shake_msOnMax * data.radius; + uint32_t msOff = shake_msOffMax * data.radius; + + //--- evaluate state (on/off) --- + if (data.radius > 0 ){ + //currently off + if (shake_state == false){ + //off long enough + if (esp_log_timestamp() - shake_timestamp_turnedOff > msOff) { + //turn on + shake_state = true; + shake_timestamp_turnedOn = esp_log_timestamp(); + } + } + //currently on + else { + //on long enough + if (esp_log_timestamp() - shake_timestamp_turnedOn > msOn) { + //turn off + shake_state = false; + shake_timestamp_turnedOff = esp_log_timestamp(); + } + } + } + //joystick is at center + else { + shake_state = false; + shake_timestamp_turnedOff = esp_log_timestamp(); + } + + + + + + if (shake_state){ + switch (data.position){ + + default: + commands.left.state = motorstate_t::IDLE; + commands.right.state = motorstate_t::IDLE; + commands.left.duty = 0; + commands.right.duty = 0; + break; + + case joystickPos_t::Y_AXIS: + if (data.y > 0){ + commands.left.state = motorstate_t::FWD; + commands.right.state = motorstate_t::FWD; + } else { + commands.left.state = motorstate_t::REV; + commands.right.state = motorstate_t::REV; + } + //set duty to shake + commands.left.duty = dutyShake; + commands.right.duty = dutyShake; + break; + + case joystickPos_t::X_AXIS: + if (data.x > 0) { + commands.left.state = motorstate_t::FWD; + commands.right.state = motorstate_t::REV; + } else { + commands.left.state = motorstate_t::REV; + commands.right.state = motorstate_t::FWD; + } + //set duty to shake + commands.left.duty = dutyShake; + commands.right.duty = dutyShake; + break; + } + } else { //shake state off + commands.left.state = motorstate_t::IDLE; + commands.right.state = motorstate_t::IDLE; + commands.left.duty = 0; + commands.right.duty = 0; + } + + + ESP_LOGI(TAG_CMD, "generated commands from data: state=%s, angle=%.3f, ratio=%.3f/%.3f, radius=%.2f, x=%.2f, y=%.2f", + joystickPosStr[(int)data.position], data.angle, ratio, (1-ratio), data.radius, data.x, data.y); + ESP_LOGI(TAG_CMD, "motor left: state=%s, duty=%.3f", motorstateStr[(int)commands.left.state], commands.left.duty); + ESP_LOGI(TAG_CMD, "motor right: state=%s, duty=%.3f", motorstateStr[(int)commands.right.state], commands.right.duty); + return commands; +} diff --git a/main/joystick.hpp b/main/joystick.hpp index 663a6da..b7335e4 100644 --- a/main/joystick.hpp +++ b/main/joystick.hpp @@ -110,6 +110,16 @@ class evaluatedJoystick { motorCommands_t joystick_generateCommandsDriving(joystickData_t data ); + +//============================================ +//========= joystick_CommandsShaking ========= +//============================================ +//function that generates commands for both motors from the joystick data +//motorCommands_t joystick_generateCommandsDriving(evaluatedJoystick joystick); +motorCommands_t joystick_generateCommandsShaking(joystickData_t data ); + + + //============================== //====== scaleCoordinate ======= //============================== From 1c6586c29eb75b8b27c34f87be18559fb08b6623 Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Sat, 2 Jul 2022 20:04:14 +0200 Subject: [PATCH 02/12] Add exponential scaling of stick coordinates - joystick.hpp/cpp - add function joystick_scaleCoordinatesExp(joystickData_t * data, float exponent) which updates joystick data with its scaled coordinates and re-calculated radius - control.cpp - scale coordinates exponential (pow 2) in JOYSTICK mode - scale coordinates exponential (pow 2) in HTTP mode - config.cpp - fixed swapped x/y zero tolerances for hardware joystick Note: tested armchair with scaling exponents 1.5; 2; 3 and noticed that 2 works best --- main/config.cpp | 4 ++-- main/control.cpp | 18 ++++++++++-------- main/control.hpp | 3 +++ main/joystick.cpp | 31 ++++++++++++++++++++++++++++++- main/joystick.hpp | 9 +++++++++ 5 files changed, 54 insertions(+), 11 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index c6ef330..69f4dfa 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -72,8 +72,8 @@ joystick_config_t configJoystick = { .adc_x = ADC1_CHANNEL_3, //GPIO39 .adc_y = ADC1_CHANNEL_0, //GPIO36 //percentage of joystick range the coordinate of the axis snaps to 0 (0-100) - .tolerance_zeroX_per = 7, - .tolerance_zeroY_per = 3, + .tolerance_zeroX_per = 3, + .tolerance_zeroY_per = 7, //percentage of joystick range the coordinate snaps to -1 or 1 before configured "_max" or "_min" threshold (mechanical end) is reached (0-100) .tolerance_end_per = 5, //threshold the radius jumps to 1 before the stick is at max radius (range 0-1) diff --git a/main/control.cpp b/main/control.cpp index eb83bbb..9fa5903 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -68,9 +68,12 @@ void controlledArmchair::startHandleLoop() { break; case controlMode_t::JOYSTICK: + //get current joystick data with getData method of evaluatedJoystick + stickData = joystick.getData(); + //additionaly scale coordinates exponentionally (more detail in slower area) + joystick_scaleCoordinatesExp(&stickData, 2); //TODO: add scaling exponent to config //generate motor commands - //pass joystick data from getData method of evaluatedJoystick to generateCommandsDriving function - commands = joystick_generateCommandsDriving(joystick.getData()); + commands = joystick_generateCommandsDriving(stickData); //TODO: pass pointer to joystick object to control class instead of accessing it directly globally motorRight->setTarget(commands.right.state, commands.right.duty); motorLeft->setTarget(commands.left.state, commands.left.duty); @@ -89,17 +92,16 @@ void controlledArmchair::startHandleLoop() { break; case controlMode_t::HTTP: - //create emptry struct for receiving data from http function - joystickData_t dataRead = { }; - //--- get joystick data from queue --- //Note this function waits several seconds (httpconfig.timeoutMs) for data to arrive, otherwise Center data or NULL is returned //TODO: as described above, when changing modes it might delay a few seconds for the change to apply - dataRead = httpJoystickMain_l->getData(); + stickData = httpJoystickMain_l->getData(); + //scale coordinates additionally (more detail in slower area) + joystick_scaleCoordinatesExp(&stickData, 2); //TODO: add scaling exponent to config + ESP_LOGD(TAG, "generating commands from x=%.3f y=%.3f radius=%.3f angle=%.3f", stickData.x, stickData.y, stickData.radius, stickData.angle); //--- generate motor commands --- - ESP_LOGD(TAG, "generating commands from x=%.3f y=%.3f radius=%.3f angle=%.3f", dataRead.x, dataRead.y, dataRead.radius, dataRead.angle); //Note: timeout (no data received) is handled in getData method - commands = joystick_generateCommandsDriving(dataRead); + commands = joystick_generateCommandsDriving(stickData); //--- apply commands to motors --- //TODO make motorctl.setTarget also accept motorcommand struct directly diff --git a/main/control.hpp b/main/control.hpp index cc83964..0acafad 100644 --- a/main/control.hpp +++ b/main/control.hpp @@ -75,6 +75,9 @@ class controlledArmchair { //struct with config parameters control_config_t config; + //store joystick data + joystickData_t stickData; + //variables for http mode uint32_t http_timestamp_lastData = 0; diff --git a/main/joystick.cpp b/main/joystick.cpp index 129b83e..93239c1 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -163,6 +163,35 @@ float scaleCoordinate(float input, float min, float max, float center, float tol +//=========================================== +//====== joystick_scaleCoordinatesExp ======= +//=========================================== +//local function that scales the absolute value of a variable exponentionally +float scaleExp(float value, float exponent){ + float result = powf(fabs(value), exponent); + if (value >= 0) { + return result; + } else { + return -result; + } +} +//function that updates a joystickData object with exponentionally scaling applied to coordinates +void joystick_scaleCoordinatesExp(joystickData_t * data, float exponent){ + //scale x and y coordinate + data->x = scaleExp(data->x, exponent); + data->y = scaleExp(data->y, exponent); + //re-calculate radius + data->radius = sqrt(pow(data->x,2) + pow(data->y,2)); + if (data->radius > 1-0.07) {//FIXME hardcoded radius tolerance + data->radius = 1; + } +} + + + + + + //============================================= //========= joystick_evaluatePosition ========= //============================================= @@ -227,7 +256,7 @@ motorCommands_t joystick_generateCommandsDriving(joystickData_t data){ motorCommands_t commands; float dutyMax = 94; //TODO add this to config, make changeable during runtime - float dutyOffset = 5; //immedeately starts with this duty, TODO add this to config + float dutyOffset = 10; //immedeately starts with this duty, TODO add this to config float dutyRange = dutyMax - dutyOffset; float ratio = fabs(data.angle) / 90; //90degree = x=0 || 0degree = y=0 diff --git a/main/joystick.hpp b/main/joystick.hpp index b7335e4..42a62c3 100644 --- a/main/joystick.hpp +++ b/main/joystick.hpp @@ -127,6 +127,15 @@ motorCommands_t joystick_generateCommandsShaking(joystickData_t data ); float scaleCoordinate(float input, float min, float max, float center, float tolerance_zero_per, float tolerance_end_per); + +//=========================================== +//====== joystick_scaleCoordinatesExp ======= +//=========================================== +//function that updates a joystickData object with exponentionally scaling applied to coordinates +void joystick_scaleCoordinatesExp(joystickData_t * data, float exponent); + + + //============================================= //========= joystick_evaluatePosition ========= //============================================= From b460a521811bf028e27fd2b4cdb33b403e7e2fe2 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Wed, 6 Jul 2022 14:25:31 +0200 Subject: [PATCH 03/12] Add linear scaling of stick coordinates - joystick.hpp/cpp - add function joystick_scaleCoordinatesLinear(joystickData_t * data, float pointX, float pointY) that scales the coordinates with 2 different slopes before and after certain point - control.cpp - scale coordinates linear with new function instead of exponential in JOYSTICK mode - scale coordinates linear with new function instead of exponential in HTTP mode - config.cpp - adjust / decrease http joystick tolerances Note: tested the armchair in http and joystick mode briefly and optimized the scaling point slightly --- main/config.cpp | 6 +++--- main/control.cpp | 6 +++--- main/joystick.cpp | 38 ++++++++++++++++++++++++++++++++++++++ main/joystick.hpp | 14 ++++++++++++++ 4 files changed, 58 insertions(+), 6 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index 69f4dfa..b41d0d4 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -57,10 +57,10 @@ control_config_t configControl = { //----- httpJoystick config ----- //------------------------------- httpJoystick_config_t configHttpJoystickMain{ - .toleranceZeroX_Per = 3, //percentage around joystick axis the coordinate snaps to 0 - .toleranceZeroY_Per = 10, + .toleranceZeroX_Per = 1, //percentage around joystick axis the coordinate snaps to 0 + .toleranceZeroY_Per = 6, .toleranceEndPer = 2, //percentage before joystick end the coordinate snaps to 1/-1 - .timeoutMs = 3000 //time no new data was received before the motors get turned off + .timeoutMs = 2500 //time no new data was received before the motors get turned off }; diff --git a/main/control.cpp b/main/control.cpp index 9fa5903..539013f 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -70,8 +70,8 @@ void controlledArmchair::startHandleLoop() { case controlMode_t::JOYSTICK: //get current joystick data with getData method of evaluatedJoystick stickData = joystick.getData(); - //additionaly scale coordinates exponentionally (more detail in slower area) - joystick_scaleCoordinatesExp(&stickData, 2); //TODO: add scaling exponent to config + //additionaly scale coordinates (more detail in slower area) + joystick_scaleCoordinatesLinear(&stickData, 0.6, 0.35); //TODO: add scaling parameters to config //generate motor commands commands = joystick_generateCommandsDriving(stickData); //TODO: pass pointer to joystick object to control class instead of accessing it directly globally @@ -97,7 +97,7 @@ void controlledArmchair::startHandleLoop() { //TODO: as described above, when changing modes it might delay a few seconds for the change to apply stickData = httpJoystickMain_l->getData(); //scale coordinates additionally (more detail in slower area) - joystick_scaleCoordinatesExp(&stickData, 2); //TODO: add scaling exponent to config + joystick_scaleCoordinatesLinear(&stickData, 0.6, 0.4); //TODO: add scaling parameters to config ESP_LOGD(TAG, "generating commands from x=%.3f y=%.3f radius=%.3f angle=%.3f", stickData.x, stickData.y, stickData.radius, stickData.angle); //--- generate motor commands --- //Note: timeout (no data received) is handled in getData method diff --git a/main/joystick.cpp b/main/joystick.cpp index 93239c1..fa4bd12 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -190,6 +190,44 @@ void joystick_scaleCoordinatesExp(joystickData_t * data, float exponent){ +//============================================== +//====== joystick_scaleCoordinatesLinear ======= +//============================================== +//local function that scales value from -1-1 to -1-1 with two different slopes before and after a specified point +//slope1: for value from 0 to pointX -> scale linear from 0 to pointY +//slope2: for value from pointX to 1 -> scale linear from pointY to 1 +float scaleLinPoint(float value, float pointX, float pointY){ + float result; + if (fabs(value) <= pointX) { + //--- scale on line from 0 to point --- + result = fabs(value) * (pointY/pointX); + } else { + //--- scale on line from point to 1 --- + float m = (1-pointY) / (1-pointX); + result = fabs(value) * m + (1 - m); + } + + //--- return result with same sign as input --- + if (value >= 0) { + return result; + } else { + return -result; + } +} +//function that updates a joystickData object with linear scaling applied to coordinates +//e.g. use to use more joystick resolution for lower speeds +void joystick_scaleCoordinatesLinear(joystickData_t * data, float pointX, float pointY){ + //scale x and y coordinate + data->x = scaleLinPoint(data->x, pointX, pointY); + data->y = scaleLinPoint(data->y, pointX, pointY); + //re-calculate radius + data->radius = sqrt(pow(data->x,2) + pow(data->y,2)); + if (data->radius > 1-0.07) {//FIXME hardcoded radius tolerance + data->radius = 1; + } +} + + //============================================= diff --git a/main/joystick.hpp b/main/joystick.hpp index 42a62c3..c631003 100644 --- a/main/joystick.hpp +++ b/main/joystick.hpp @@ -132,10 +132,24 @@ float scaleCoordinate(float input, float min, float max, float center, float tol //====== joystick_scaleCoordinatesExp ======= //=========================================== //function that updates a joystickData object with exponentionally scaling applied to coordinates +//e.g. use to use more joystick resolution for lower speeds void joystick_scaleCoordinatesExp(joystickData_t * data, float exponent); +//============================================== +//====== joystick_scaleCoordinatesLinear ======= +//============================================== +//function that updates a joystickData object with linear scaling applied to coordinates +//scales coordinates with two different slopes before and after a specified point +//slope1: for value from 0 to pointX -> scale linear from 0 to pointY +//slope2: for value from pointX to 1 -> scale linear from pointY to 1 +//=> best to draw the lines and point in a graph +//e.g. use to use more joystick resolution for lower speeds +void joystick_scaleCoordinatesLinear(joystickData_t * data, float pointX, float pointY); + + + //============================================= //========= joystick_evaluatePosition ========= //============================================= From 77a32c15f73a3918cbb38ba8e4799c156121eb86 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Fri, 8 Jul 2022 18:49:17 +0200 Subject: [PATCH 04/12] Optimize control: use reference to joystick object Use reference to joystick object in control class instead of accessing the global evaluatedJoystick object control.hpp: - add joystick reference to constructor - add local joystick pointer variable control.cpp: - constructor: copy pointer to joystick object - use methods of joystick reference instead of global object - update log output of timeout check config.cpp: - add joystick object to control object construction --- main/config.cpp | 2 +- main/control.cpp | 14 ++++++++------ main/control.hpp | 2 ++ 3 files changed, 11 insertions(+), 7 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index b41d0d4..99ccc45 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -123,7 +123,7 @@ buzzer_t buzzer(GPIO_NUM_12, 100); httpJoystick httpJoystickMain(configHttpJoystickMain); //create global control object -controlledArmchair control(configControl, &buzzer, &motorLeft, &motorRight, &httpJoystickMain); +controlledArmchair control(configControl, &buzzer, &motorLeft, &motorRight, &joystick, &httpJoystickMain); diff --git a/main/control.cpp b/main/control.cpp index 539013f..ff77931 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -27,6 +27,7 @@ controlledArmchair::controlledArmchair ( buzzer_t * buzzer_f, controlledMotor* motorLeft_f, controlledMotor* motorRight_f, + evaluatedJoystick* joystick_f, httpJoystick* httpJoystick_f ){ @@ -36,6 +37,7 @@ controlledArmchair::controlledArmchair ( buzzer = buzzer_f; motorLeft = motorLeft_f; motorRight = motorRight_f; + joystick_l = joystick_f, httpJoystickMain_l = httpJoystick_f; //set default mode from config modePrevious = config.defaultMode; @@ -69,12 +71,12 @@ void controlledArmchair::startHandleLoop() { case controlMode_t::JOYSTICK: //get current joystick data with getData method of evaluatedJoystick - stickData = joystick.getData(); + stickData = joystick_l->getData(); //additionaly scale coordinates (more detail in slower area) joystick_scaleCoordinatesLinear(&stickData, 0.6, 0.35); //TODO: add scaling parameters to config //generate motor commands commands = joystick_generateCommandsDriving(stickData); - //TODO: pass pointer to joystick object to control class instead of accessing it directly globally + //apply motor commands motorRight->setTarget(commands.right.state, commands.right.duty); motorLeft->setTarget(commands.left.state, commands.left.duty); //TODO make motorctl.setTarget also accept motorcommand struct directly @@ -84,7 +86,7 @@ void controlledArmchair::startHandleLoop() { case controlMode_t::MASSAGE: //generate motor commands //pass joystick data from getData method of evaluatedJoystick to generateCommandsShaking function - commands = joystick_generateCommandsShaking(joystick.getData()); + commands = joystick_generateCommandsShaking(joystick_l->getData()); //apply motor commands motorRight->setTarget(commands.right.state, commands.right.duty); motorLeft->setTarget(commands.left.state, commands.left.duty); @@ -167,11 +169,11 @@ void controlledArmchair::handleTimeout(){ if (validateActivity(dutyLeft_lastActivity, dutyLeftNow, inactivityTolerance) || validateActivity(dutyRight_lastActivity, dutyRightNow, inactivityTolerance) ){ - ESP_LOGD(TAG, "timeout check: detected [activity] since last check -> reset"); + ESP_LOGD(TAG, "timeout check: [activity] detected since last check -> reset"); //reset last duty and timestamp - timestamp_lastActivity = esp_log_timestamp(); dutyLeft_lastActivity = dutyLeftNow; dutyRight_lastActivity = dutyRightNow; + resetTimeout(); } //no activity on any motor and msTimeout exceeded else if (esp_log_timestamp() - timestamp_lastActivity > config.timeoutMs){ @@ -180,7 +182,7 @@ void controlledArmchair::handleTimeout(){ toggleIdle(); } else { - ESP_LOGD(TAG, "timeout check: [inactive], last activity %.1f seconds ago", (float)(esp_log_timestamp() - timestamp_lastActivity)/1000); + ESP_LOGD(TAG, "timeout check: [inactive], last activity %.1f s ago, timeout after %d s", (float)(esp_log_timestamp() - timestamp_lastActivity)/1000, config.timeoutMs/1000); } } } diff --git a/main/control.hpp b/main/control.hpp index 0acafad..52dc4b1 100644 --- a/main/control.hpp +++ b/main/control.hpp @@ -38,6 +38,7 @@ class controlledArmchair { buzzer_t* buzzer_f, controlledMotor* motorLeft_f, controlledMotor* motorRight_f, + evaluatedJoystick* joystick_f, httpJoystick* httpJoystick_f ); @@ -68,6 +69,7 @@ class controlledArmchair { controlledMotor* motorLeft; controlledMotor* motorRight; httpJoystick* httpJoystickMain_l; + evaluatedJoystick* joystick_l; //---variables --- //struct for motor commands returned by generate functions of each mode From b2b12fe5de1e16ea2f25dfa4b0c3ac6845024c71 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Fri, 8 Jul 2022 19:38:29 +0200 Subject: [PATCH 05/12] Add hysteresis to joystick position X-Axis [WIP] modified function joystickPos_t joystick_evaluatePosition(float x, float y, joystickPos_t* prevPos) because prevPos has to be stored http.cpp/hpp and joystick.cpp/hpp had to be updated as well TODO: add function without hysteresis again? WIP this has to be tested --- main/http.cpp | 2 +- main/http.hpp | 2 ++ main/joystick.cpp | 46 ++++++++++++++++++++++++++++++++++++---------- main/joystick.hpp | 5 ++++- 4 files changed, 43 insertions(+), 12 deletions(-) diff --git a/main/http.cpp b/main/http.cpp index 0c76184..0231c56 100644 --- a/main/http.cpp +++ b/main/http.cpp @@ -159,7 +159,7 @@ esp_err_t httpJoystick::receiveHttpData(httpd_req_t *req){ //--- calculate angle --- data.angle = (atan(data.y/data.x) * 180) / 3.141; //--- evaluate position --- - data.position = joystick_evaluatePosition(data.x, data.y); + data.position = joystick_evaluatePosition(data.x, data.y, &stickPosPrevious); //log processed values ESP_LOGI(TAG, "processed values: x=%.3f y=%.3f radius=%.3f angle=%.3f pos=%s", diff --git a/main/http.hpp b/main/http.hpp index eab53ef..55e8a85 100644 --- a/main/http.hpp +++ b/main/http.hpp @@ -68,4 +68,6 @@ class httpJoystick{ .radius = 0, .angle = 0 }; + //store last joystick position for position hysteresis + joystickPos_t stickPosPrevious = joystickPos_t::CENTER; }; diff --git a/main/joystick.cpp b/main/joystick.cpp index fa4bd12..c4e15cc 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -93,7 +93,7 @@ joystickData_t evaluatedJoystick::getData() { data.angle = (atan(data.y/data.x) * 180) / 3.141; //define position - data.position = joystick_evaluatePosition(x, y); + data.position = joystick_evaluatePosition(x, y, &stickPosPrevious); return data; } @@ -234,41 +234,67 @@ void joystick_scaleCoordinatesLinear(joystickData_t * data, float pointX, float //========= joystick_evaluatePosition ========= //============================================= //function that defines and returns enum joystickPos from x and y coordinates -joystickPos_t joystick_evaluatePosition(float x, float y){ +joystickPos_t joystick_evaluatePosition(float x, float y, joystickPos_t* prevPos){ + joystickPos_t newPos; //define position //--- center --- if((fabs(x) == 0) && (fabs(y) == 0)){ - return joystickPos_t::CENTER; + newPos = joystickPos_t::CENTER; } //--- x axis --- else if(fabs(y) == 0){ - return joystickPos_t::X_AXIS; + newPos = joystickPos_t::X_AXIS; } //--- y axis --- else if(fabs(x) == 0){ - return joystickPos_t::Y_AXIS; + newPos = joystickPos_t::Y_AXIS; } //--- top right --- else if(x > 0 && y > 0){ - return joystickPos_t::TOP_RIGHT; + newPos = joystickPos_t::TOP_RIGHT; } //--- top left --- else if(x < 0 && y > 0){ - return joystickPos_t::TOP_LEFT; + newPos = joystickPos_t::TOP_LEFT; } //--- bottom left --- else if(x < 0 && y < 0){ - return joystickPos_t::BOTTOM_LEFT; + newPos = joystickPos_t::BOTTOM_LEFT; } //--- bottom right --- else if(x > 0 && y < 0){ - return joystickPos_t::BOTTOM_RIGHT; + newPos = joystickPos_t::BOTTOM_RIGHT; } //--- other --- else { - return joystickPos_t::CENTER; + newPos = joystickPos_t::CENTER; } + + + //--- apply hysteresis on change from X-AXIS --- + //=> prevent frequent switching between X-AXIS and other positions + //otherwise this results in a very jerky motor action when in driving mode + float xAxisHyst = 0.2; + + //change in position + if (newPos != *prevPos) { + + //switched FROM x-axis to other position + if(*prevPos == joystickPos_t::X_AXIS) { //switch FROM X_AXIS + + //verify coordinate changed more than hysteresis + if (fabs(y) < xAxisHyst) { //less offset than hysteresis + newPos = joystickPos_t::X_AXIS; //stay at X_AXIS position + } else { //switch is valid (enough change) + *prevPos = newPos; + } + + } else { //switched to any other position + *prevPos = newPos; + } + } + return newPos; } diff --git a/main/joystick.hpp b/main/joystick.hpp index c631003..4dd4f8a 100644 --- a/main/joystick.hpp +++ b/main/joystick.hpp @@ -96,6 +96,8 @@ class evaluatedJoystick { joystickData_t data; float x; float y; + //store last joystick position for position hysteresis + joystickPos_t stickPosPrevious = joystickPos_t::CENTER; }; @@ -154,4 +156,5 @@ void joystick_scaleCoordinatesLinear(joystickData_t * data, float pointX, float //========= joystick_evaluatePosition ========= //============================================= //function that defines and returns enum joystickPos from x and y coordinates -joystickPos_t joystick_evaluatePosition(float x, float y); +//joystickPos_t joystick_evaluatePosition(float x, float y); +joystickPos_t joystick_evaluatePosition(float x, float y, joystickPos_t* prevPos); From d573cc5b2419188e9cdbde772c449e1482b12a61 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Thu, 14 Jul 2022 18:50:36 +0200 Subject: [PATCH 06/12] Revert "Add hysteresis to joystick position X-Axis" While testing this feature briefly that approach did not appear to be a good idea. - noticed that the driving with joystick was a little unintuitive/unexpected (maybe just not used to it?) - BUG that the state does not reset to center when previous state was X-AXIS? Resulting in the armchair to slightly shake randomly at joystick center -> this has to be fixed if this approach is tested again. This reverts commit b2b12fe5de1e16ea2f25dfa4b0c3ac6845024c71. --- main/http.cpp | 2 +- main/http.hpp | 2 -- main/joystick.cpp | 46 ++++++++++------------------------------------ main/joystick.hpp | 5 +---- 4 files changed, 12 insertions(+), 43 deletions(-) diff --git a/main/http.cpp b/main/http.cpp index 0231c56..0c76184 100644 --- a/main/http.cpp +++ b/main/http.cpp @@ -159,7 +159,7 @@ esp_err_t httpJoystick::receiveHttpData(httpd_req_t *req){ //--- calculate angle --- data.angle = (atan(data.y/data.x) * 180) / 3.141; //--- evaluate position --- - data.position = joystick_evaluatePosition(data.x, data.y, &stickPosPrevious); + data.position = joystick_evaluatePosition(data.x, data.y); //log processed values ESP_LOGI(TAG, "processed values: x=%.3f y=%.3f radius=%.3f angle=%.3f pos=%s", diff --git a/main/http.hpp b/main/http.hpp index 55e8a85..eab53ef 100644 --- a/main/http.hpp +++ b/main/http.hpp @@ -68,6 +68,4 @@ class httpJoystick{ .radius = 0, .angle = 0 }; - //store last joystick position for position hysteresis - joystickPos_t stickPosPrevious = joystickPos_t::CENTER; }; diff --git a/main/joystick.cpp b/main/joystick.cpp index c4e15cc..fa4bd12 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -93,7 +93,7 @@ joystickData_t evaluatedJoystick::getData() { data.angle = (atan(data.y/data.x) * 180) / 3.141; //define position - data.position = joystick_evaluatePosition(x, y, &stickPosPrevious); + data.position = joystick_evaluatePosition(x, y); return data; } @@ -234,67 +234,41 @@ void joystick_scaleCoordinatesLinear(joystickData_t * data, float pointX, float //========= joystick_evaluatePosition ========= //============================================= //function that defines and returns enum joystickPos from x and y coordinates -joystickPos_t joystick_evaluatePosition(float x, float y, joystickPos_t* prevPos){ - joystickPos_t newPos; +joystickPos_t joystick_evaluatePosition(float x, float y){ //define position //--- center --- if((fabs(x) == 0) && (fabs(y) == 0)){ - newPos = joystickPos_t::CENTER; + return joystickPos_t::CENTER; } //--- x axis --- else if(fabs(y) == 0){ - newPos = joystickPos_t::X_AXIS; + return joystickPos_t::X_AXIS; } //--- y axis --- else if(fabs(x) == 0){ - newPos = joystickPos_t::Y_AXIS; + return joystickPos_t::Y_AXIS; } //--- top right --- else if(x > 0 && y > 0){ - newPos = joystickPos_t::TOP_RIGHT; + return joystickPos_t::TOP_RIGHT; } //--- top left --- else if(x < 0 && y > 0){ - newPos = joystickPos_t::TOP_LEFT; + return joystickPos_t::TOP_LEFT; } //--- bottom left --- else if(x < 0 && y < 0){ - newPos = joystickPos_t::BOTTOM_LEFT; + return joystickPos_t::BOTTOM_LEFT; } //--- bottom right --- else if(x > 0 && y < 0){ - newPos = joystickPos_t::BOTTOM_RIGHT; + return joystickPos_t::BOTTOM_RIGHT; } //--- other --- else { - newPos = joystickPos_t::CENTER; + return joystickPos_t::CENTER; } - - - //--- apply hysteresis on change from X-AXIS --- - //=> prevent frequent switching between X-AXIS and other positions - //otherwise this results in a very jerky motor action when in driving mode - float xAxisHyst = 0.2; - - //change in position - if (newPos != *prevPos) { - - //switched FROM x-axis to other position - if(*prevPos == joystickPos_t::X_AXIS) { //switch FROM X_AXIS - - //verify coordinate changed more than hysteresis - if (fabs(y) < xAxisHyst) { //less offset than hysteresis - newPos = joystickPos_t::X_AXIS; //stay at X_AXIS position - } else { //switch is valid (enough change) - *prevPos = newPos; - } - - } else { //switched to any other position - *prevPos = newPos; - } - } - return newPos; } diff --git a/main/joystick.hpp b/main/joystick.hpp index 4dd4f8a..c631003 100644 --- a/main/joystick.hpp +++ b/main/joystick.hpp @@ -96,8 +96,6 @@ class evaluatedJoystick { joystickData_t data; float x; float y; - //store last joystick position for position hysteresis - joystickPos_t stickPosPrevious = joystickPos_t::CENTER; }; @@ -156,5 +154,4 @@ void joystick_scaleCoordinatesLinear(joystickData_t * data, float pointX, float //========= joystick_evaluatePosition ========= //============================================= //function that defines and returns enum joystickPos from x and y coordinates -//joystickPos_t joystick_evaluatePosition(float x, float y); -joystickPos_t joystick_evaluatePosition(float x, float y, joystickPos_t* prevPos); +joystickPos_t joystick_evaluatePosition(float x, float y); From 436c528eaf7b467d4b97011d5749458682f621bf Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Sat, 16 Jul 2022 12:46:56 +0200 Subject: [PATCH 07/12] Add fade-down functionality to motorctl (bug) With this changes the motors are faded down (same as already existing fading up) when the target duty suddenly decreases. It can be configured or disabled with setting msFadeDown to 0 in config struct of motorctl object. - Add new config parameter msFadeDown - Change msFade to msFadeDown - Add fade down functionality to motorctl instead of just setting the duty to lower value !!!TODO: fix major BUG when motor direction changes suddenly (before fade got to 0) the motor immediately starts in other direction with previous duty (no/less fading up happening) --- main/config.cpp | 3 ++- main/motorctl.cpp | 33 +++++++++++++++++++++++++-------- main/motorctl.hpp | 6 ++++-- 3 files changed, 31 insertions(+), 11 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index 99ccc45..2a83bd1 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -29,7 +29,8 @@ single100a_config_t configDriverRight = { //--- configure motor contol --- motorctl_config_t configMotorControl = { - .msFade = 900, + .msFadeUp = 1500, + .msFadeDown = 3000, .currentMax = 10 }; diff --git a/main/motorctl.cpp b/main/motorctl.cpp index fd8cfc8..9746a9e 100644 --- a/main/motorctl.cpp +++ b/main/motorctl.cpp @@ -46,9 +46,16 @@ void controlledMotor::handle(){ } //--- calculate increment --- - //calculate increment 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 int64_t usPassed = esp_timer_get_time() - timestampLastRunUs; - dutyIncrement = ( usPassed / ((float)config.msFade * 1000) ) * 100; //TODO define maximum increment - first run after startup (or long) pause can cause a very large increment + dutyIncrementUp = ( usPassed / ((float)config.msFadeUp * 1000) ) * 100; //TODO define maximum increment - first run after startup (or long) pause can cause a very large increment + //calculate increment for fading DOWN with passed time since last run and configured fade time + if (config.msFadeDown > 0){ + dutyIncrementDown = ( usPassed / ((float)config.msFadeDown * 1000) ) * 100; + } else { + dutyIncrementDown = 100; + } + //--- calculate difference --- dutyDelta = dutyTarget - dutyNow; @@ -56,14 +63,24 @@ void controlledMotor::handle(){ //negative: need to decrease //--- fade up --- - if(dutyDelta > dutyIncrement){ //target duty his higher than current duty -> fade up - ESP_LOGV(TAG, "*fading up*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrement, (int)usPassed); - dutyNow += dutyIncrement; //increase duty by increment + if(dutyDelta > dutyIncrementUp){ //target duty his higher than current duty -> fade up + ESP_LOGV(TAG, "*fading up*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrementUp, (int)usPassed); + dutyNow += dutyIncrementUp; //increase duty by increment //--- set lower --- - } else { //target duty is lower than current duty -> immediately set to target - ESP_LOGV(TAG, "*setting to target*: target=%.2f%% - previous=%.2f%% ", dutyTarget, dutyNow); - dutyNow = dutyTarget; //set target duty + //} else { //target duty is lower than current duty -> immediately set to target + // ESP_LOGV(TAG, "*setting to target*: target=%.2f%% - previous=%.2f%% ", dutyTarget, dutyNow); + // dutyNow = dutyTarget; //set target duty + //} + + //--- fade down --- + } else { //target duty is lower than current duty -> fade down + ESP_LOGV(TAG, "*fading up*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrementDown, (int)usPassed); + if (dutyTarget < dutyIncrementDown){ //immediately set to target when closer than increment (avoid negative duty) + dutyNow = dutyTarget; + } else { + dutyNow -= dutyIncrementDown; //decrease duty by increment + } } //--- apply to motor --- diff --git a/main/motorctl.hpp b/main/motorctl.hpp index 86708fe..593ff11 100644 --- a/main/motorctl.hpp +++ b/main/motorctl.hpp @@ -30,7 +30,8 @@ typedef struct motorCommands_t { //struct with all config parameters for a motor regarding ramp and current limit typedef struct motorctl_config_t { - uint32_t msFade; //acceleration of the motor (ms it should take from 0 to 100%) + uint32_t msFadeUp; //acceleration of the motor (ms it should take from 0% to 100%) + uint32_t msFadeDown; //acceleration of the motor (ms it should take from 100% to 0%) float currentMax; } motorctl_config_t; @@ -67,7 +68,8 @@ class controlledMotor { float dutyTarget; float dutyNow; - float dutyIncrement; + float dutyIncrementUp; + float dutyIncrementDown; float dutyDelta; uint32_t ramp; From cd804452e42ba872ca8a7a6a772f4b105c0de706 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Sun, 17 Jul 2022 15:23:42 +0200 Subject: [PATCH 08/12] Change motorctl.handle from STATE to duty -100-100 For fading-down and delay-between-state-change to work correctly when switching between motorstates (e.g. FWD -> REV) the handle function was changed. using a duty value -100 for reverse 100% and +100 for forward 100% allows are more mathematical approach changes to motorctl.cpp: - convert state and duty to duty -100 to 100 when receiving target command at start - convert duty -100-100 back to state and duty at the end for applying motor commands --- main/motorctl.cpp | 38 ++++++++++++++++++++++++++++++++++---- 1 file changed, 34 insertions(+), 4 deletions(-) diff --git a/main/motorctl.cpp b/main/motorctl.cpp index 9746a9e..b0f8a2a 100644 --- a/main/motorctl.cpp +++ b/main/motorctl.cpp @@ -43,12 +43,28 @@ void controlledMotor::handle(){ ESP_LOGD(TAG, "Read command from queue: state=%s, duty=%.2f", motorstateStr[(int)commandReceive.state], commandReceive.duty); state = commandReceive.state; dutyTarget = commandReceive.duty; + + //--- convert duty --- + //define target duty (-100 to 100) from provided duty and motorstate + switch(commandReceive.state){ + case motorstate_t::BRAKE: + case motorstate_t::IDLE: + dutyTarget = 0; + break; + case motorstate_t::FWD: + dutyTarget = fabs(commandReceive.duty); + break; + case motorstate_t::REV: + dutyTarget = - fabs(commandReceive.duty); + break; + } } //--- calculate increment --- //calculate increment for fading UP with passed time since last run and configured fade time int64_t usPassed = esp_timer_get_time() - timestampLastRunUs; dutyIncrementUp = ( usPassed / ((float)config.msFadeUp * 1000) ) * 100; //TODO define maximum increment - first run after startup (or long) pause can cause a very large increment + //calculate increment for fading DOWN with passed time since last run and configured fade time if (config.msFadeDown > 0){ dutyIncrementDown = ( usPassed / ((float)config.msFadeDown * 1000) ) * 100; @@ -72,7 +88,7 @@ void controlledMotor::handle(){ // ESP_LOGV(TAG, "*setting to target*: target=%.2f%% - previous=%.2f%% ", dutyTarget, dutyNow); // dutyNow = dutyTarget; //set target duty //} - + //--- fade down --- } else { //target duty is lower than current duty -> fade down ESP_LOGV(TAG, "*fading up*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrementDown, (int)usPassed); @@ -82,11 +98,25 @@ void controlledMotor::handle(){ dutyNow -= dutyIncrementDown; //decrease duty by increment } } - + //--- apply to motor --- + //convert duty -100 to 100 back to motorstate and duty 0-100 //apply target duty and state to motor driver - motor.set(state, dutyNow); - + //forward + if(dutyNow > 0){ + motor.set(motorstate_t::FWD, dutyNow); + } + //reverse + else if (dutyNow < 0){ + motor.set(motorstate_t::REV, fabs(dutyNow)); + } + //idle + //TODO: add BRAKE case!!! + else { + motor.set(motorstate_t::IDLE, 0); + } + + //--- update timestamp --- timestampLastRunUs = esp_timer_get_time(); //update timestamp last run with current timestamp in microseconds From 1360832f5ec6240d6429d4a591ee51834f86bbcb Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Sun, 17 Jul 2022 17:55:57 +0200 Subject: [PATCH 09/12] Fix bug no fwd downfade; Add BRAKE; cleanup; [bug] - Fix bug where downfade did not work when driving forward - Add BRAKE command functionality to motorctl (untested, since not used anywhere atm) - Cleanup: Optimize code, add comments !!!TODO: there is a fault in the current concept/logic: fading up/down generally works, however when accelerating REVERSE fade-down is utilized instead of fade-up. => Fix that bug, fade down should only be used when decelerating. --- main/config.cpp | 2 +- main/motorctl.cpp | 58 +++++++++++++++++++++++++++++++---------------- 2 files changed, 39 insertions(+), 21 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index 2a83bd1..89d40bf 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -30,7 +30,7 @@ single100a_config_t configDriverRight = { //--- configure motor contol --- motorctl_config_t configMotorControl = { .msFadeUp = 1500, - .msFadeDown = 3000, + .msFadeDown = 1000, .currentMax = 10 }; diff --git a/main/motorctl.cpp b/main/motorctl.cpp index b0f8a2a..55b60f8 100644 --- a/main/motorctl.cpp +++ b/main/motorctl.cpp @@ -46,8 +46,13 @@ void controlledMotor::handle(){ //--- convert duty --- //define target duty (-100 to 100) from provided duty and motorstate + //this value is more suitable for the fading algorithm switch(commandReceive.state){ case motorstate_t::BRAKE: + //update state + state = motorstate_t::BRAKE; + dutyTarget = 0; + break; case motorstate_t::IDLE: dutyTarget = 0; break; @@ -72,6 +77,16 @@ void controlledMotor::handle(){ dutyIncrementDown = 100; } + + //--- BRAKE --- + //brake immediately, update state, duty and exit this cycle of handle function + if (state == motorstate_t::BRAKE){ + motor.set(motorstate_t::BRAKE, 0); + dutyNow = 0; + return; //no need to run the fade algorithm + } + + //--- calculate difference --- dutyDelta = dutyTarget - dutyNow; @@ -79,42 +94,45 @@ void controlledMotor::handle(){ //negative: need to decrease //--- fade up --- - if(dutyDelta > dutyIncrementUp){ //target duty his higher than current duty -> fade up + //dutyDelta is higher than IncrementUp -> fade up + if(dutyDelta > dutyIncrementUp){ ESP_LOGV(TAG, "*fading up*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrementUp, (int)usPassed); dutyNow += dutyIncrementUp; //increase duty by increment - - //--- set lower --- - //} else { //target duty is lower than current duty -> immediately set to target - // ESP_LOGV(TAG, "*setting to target*: target=%.2f%% - previous=%.2f%% ", dutyTarget, dutyNow); - // dutyNow = dutyTarget; //set target duty - //} + } //--- fade down --- - } else { //target duty is lower than current duty -> fade down - ESP_LOGV(TAG, "*fading up*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrementDown, (int)usPassed); - if (dutyTarget < dutyIncrementDown){ //immediately set to target when closer than increment (avoid negative duty) - dutyNow = dutyTarget; - } else { - dutyNow -= dutyIncrementDown; //decrease duty by increment - } + //dutyDelta is more negative than -IncrementDown -> fade down + else if (dutyDelta < -dutyIncrementDown){ + ESP_LOGV(TAG, "*fading down*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrementDown, (int)usPassed); + dutyNow -= dutyIncrementDown; //decrease duty by increment + } + + //--- set to target --- + //duty is already very close to target (closer than IncrementUp or IncrementDown) + else{ + //immediately set to target duty + dutyNow = dutyTarget; } - //--- apply to motor --- - //convert duty -100 to 100 back to motorstate and duty 0-100 + + //define motorstate from converted duty -100 to 100 //apply target duty and state to motor driver //forward if(dutyNow > 0){ - motor.set(motorstate_t::FWD, dutyNow); + state = motorstate_t::FWD; } //reverse else if (dutyNow < 0){ - motor.set(motorstate_t::REV, fabs(dutyNow)); + state = motorstate_t::REV; } //idle - //TODO: add BRAKE case!!! else { - motor.set(motorstate_t::IDLE, 0); + state = motorstate_t::IDLE; } + + //--- apply to motor --- + motor.set(state, fabs(dutyNow)); + //note: BRAKE state is handled earlier //--- update timestamp --- From 0a2695f45f6b6af8b315e5ec6680ce65c1a179a7 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Mon, 18 Jul 2022 16:58:40 +0200 Subject: [PATCH 10/12] Fix fading: correctly use different accel, decel ramps - Change fading algorithm to handle Acceleration and Deceleration for forward and reverse separately (4 different cases) - rename variables to make fading direction more obvious e.g. msFadeUp -> msFadeAccel or dutyIncrementDon -> dutyIncrementDecel TODO: fading needs optimization --- main/config.cpp | 4 +-- main/motorctl.cpp | 90 ++++++++++++++++++++++++++++++++++++----------- main/motorctl.hpp | 8 ++--- 3 files changed, 75 insertions(+), 27 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index 89d40bf..40d9703 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -29,8 +29,8 @@ single100a_config_t configDriverRight = { //--- configure motor contol --- motorctl_config_t configMotorControl = { - .msFadeUp = 1500, - .msFadeDown = 1000, + .msFadeAccel = 5000, //acceleration of the motor (ms it takes from 0% to 100%) + .msFadeDecel = 3000, //deceleration of the motor (ms it takes from 100% to 0%) .currentMax = 10 }; diff --git a/main/motorctl.cpp b/main/motorctl.cpp index 55b60f8..55d7435 100644 --- a/main/motorctl.cpp +++ b/main/motorctl.cpp @@ -27,6 +27,26 @@ void controlledMotor::init(){ +//---------------- +//----- fade ----- +//---------------- +//local function that fades a variable +//- increments a variable (pointer) by given value +//- sets to target if already closer than increment +//TODO this needs testing +void fade(float * dutyNow, float dutyTarget, float dutyIncrement){ + float dutyDelta = dutyTarget - *dutyNow; + if ( fabs(dutyDelta) > fabs(dutyIncrement) ) { //check if already close to target + *dutyNow = *dutyNow + dutyIncrement; + } + //already closer to target than increment + else { + *dutyNow = dutyTarget; + } +} + + + //============================== //=========== handle =========== //============================== @@ -65,16 +85,17 @@ void controlledMotor::handle(){ } } + //--- calculate increment --- //calculate increment for fading UP with passed time since last run and configured fade time int64_t usPassed = esp_timer_get_time() - timestampLastRunUs; - dutyIncrementUp = ( usPassed / ((float)config.msFadeUp * 1000) ) * 100; //TODO define maximum increment - first run after startup (or long) pause can cause a very large increment + dutyIncrementAccel = ( usPassed / ((float)config.msFadeAccel * 1000) ) * 100; //TODO define maximum increment - first run after startup (or long) pause can cause a very large increment //calculate increment for fading DOWN with passed time since last run and configured fade time - if (config.msFadeDown > 0){ - dutyIncrementDown = ( usPassed / ((float)config.msFadeDown * 1000) ) * 100; + if (config.msFadeDecel > 0){ + dutyIncrementDecel = ( usPassed / ((float)config.msFadeDecel * 1000) ) * 100; } else { - dutyIncrementDown = 100; + dutyIncrementDecel = 100; } @@ -93,28 +114,55 @@ void controlledMotor::handle(){ //positive: need to increase by that value //negative: need to decrease - //--- fade up --- - //dutyDelta is higher than IncrementUp -> fade up - if(dutyDelta > dutyIncrementUp){ - ESP_LOGV(TAG, "*fading up*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrementUp, (int)usPassed); - dutyNow += dutyIncrementUp; //increase duty by increment + + + //--- 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 + fade(&dutyNow, dutyTarget, dutyIncrementDecel); + } + else if (dutyNow > 0) { //forward, accelerating + fade(&dutyNow, dutyTarget, dutyIncrementAccel); + } + } + else if (dutyDelta < 0) { //difference negative -> decreasing duty (100 -> -100) + if (dutyNow < 0) { //reverse, accelerating + fade(&dutyNow, dutyTarget, - dutyIncrementAccel); + + } + else if (&dutyNow > 0) { //forward, decelerating + fade(&dutyNow, dutyTarget, - dutyIncrementDecel); + } } - //--- fade down --- - //dutyDelta is more negative than -IncrementDown -> fade down - else if (dutyDelta < -dutyIncrementDown){ - ESP_LOGV(TAG, "*fading down*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrementDown, (int)usPassed); - dutyNow -= dutyIncrementDown; //decrease duty by increment - } - //--- set to target --- - //duty is already very close to target (closer than IncrementUp or IncrementDown) - else{ - //immediately set to target duty - dutyNow = dutyTarget; - } + + //previous approach: (resulted in bug where accel/decel fade is swaped in reverse) +// //--- fade up --- +// //dutyDelta is higher than IncrementUp -> fade up +// if(dutyDelta > dutyIncrementUp){ +// ESP_LOGV(TAG, "*fading up*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrementUp, (int)usPassed); +// dutyNow += dutyIncrementUp; //increase duty by increment +// } +// +// //--- fade down --- +// //dutyDelta is more negative than -IncrementDown -> fade down +// else if (dutyDelta < -dutyIncrementDown){ +// ESP_LOGV(TAG, "*fading down*: target=%.2f%% - previous=%.2f%% - increment=%.6f%% - usSinceLastRun=%d", dutyTarget, dutyNow, dutyIncrementDown, (int)usPassed); +// +// dutyNow -= dutyIncrementDown; //decrease duty by increment +// } +// +// //--- set to target --- +// //duty is already very close to target (closer than IncrementUp or IncrementDown) +// else{ +// //immediately set to target duty +// dutyNow = dutyTarget; +// } + //define motorstate from converted duty -100 to 100 //apply target duty and state to motor driver //forward diff --git a/main/motorctl.hpp b/main/motorctl.hpp index 593ff11..4528147 100644 --- a/main/motorctl.hpp +++ b/main/motorctl.hpp @@ -30,8 +30,8 @@ typedef struct motorCommands_t { //struct with all config parameters for a motor regarding ramp and current limit typedef struct motorctl_config_t { - uint32_t msFadeUp; //acceleration of the motor (ms it should take from 0% to 100%) - uint32_t msFadeDown; //acceleration of the motor (ms it should take from 100% to 0%) + uint32_t msFadeAccel; //acceleration of the motor (ms it takes from 0% to 100%) + uint32_t msFadeDecel; //deceleration of the motor (ms it takes from 100% to 0%) float currentMax; } motorctl_config_t; @@ -68,8 +68,8 @@ class controlledMotor { float dutyTarget; float dutyNow; - float dutyIncrementUp; - float dutyIncrementDown; + float dutyIncrementAccel; + float dutyIncrementDecel; float dutyDelta; uint32_t ramp; From bcbd2578f692a030dc3442e656ac828adec1a621 Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Fri, 22 Jul 2022 20:11:12 +0200 Subject: [PATCH 11/12] Fix bug: fade stuck at zero; Adjust fade durations - Fix bug where case duty=0 was not handled in fading if structure thus no incrementing action happened anymore (stuck at duty = 0) - Set fade duration (acceleration, deceleration) to realistic value in config --- main/config.cpp | 4 ++-- main/motorctl.cpp | 5 ++--- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/main/config.cpp b/main/config.cpp index 40d9703..1b515e1 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -29,8 +29,8 @@ single100a_config_t configDriverRight = { //--- configure motor contol --- motorctl_config_t configMotorControl = { - .msFadeAccel = 5000, //acceleration of the motor (ms it takes from 0% to 100%) - .msFadeDecel = 3000, //deceleration of the motor (ms it takes from 100% to 0%) + .msFadeAccel = 2000, //acceleration of the motor (ms it takes from 0% to 100%) + .msFadeDecel = 1000, //deceleration of the motor (ms it takes from 100% to 0%) .currentMax = 10 }; diff --git a/main/motorctl.cpp b/main/motorctl.cpp index 55d7435..c887c97 100644 --- a/main/motorctl.cpp +++ b/main/motorctl.cpp @@ -122,14 +122,13 @@ void controlledMotor::handle(){ if (dutyNow < 0) { //reverse, decelerating fade(&dutyNow, dutyTarget, dutyIncrementDecel); } - else if (dutyNow > 0) { //forward, accelerating + else if (dutyNow >= 0) { //forward, accelerating fade(&dutyNow, dutyTarget, dutyIncrementAccel); } } else if (dutyDelta < 0) { //difference negative -> decreasing duty (100 -> -100) - if (dutyNow < 0) { //reverse, accelerating + if (dutyNow <= 0) { //reverse, accelerating fade(&dutyNow, dutyTarget, - dutyIncrementAccel); - } else if (&dutyNow > 0) { //forward, decelerating fade(&dutyNow, dutyTarget, - dutyIncrementDecel); From 0b113eafb5dd84a10bd5056f319678d2d5bf1381 Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Sat, 23 Jul 2022 14:48:38 +0200 Subject: [PATCH 12/12] Add functions to modify/toggle fading (button x8) - add 3 functions to controlledMotor class in motorctl.cpp/hpp - setFade: set specific msFade duration for acceleration or deceleration - setFade: disable or set to default ramp for acceleration or deceleration - toggleFade: toggle between fading enabled or disables for acceleration or deceleration - button.cpp - add controlledMotor objects to constructor - add new command 8x press which now toggles fade-deceleration of both motors - config.cpp - slightly decrease fading durations --- main/button.cpp | 17 ++++++++- main/button.hpp | 7 +++- main/config.cpp | 4 +-- main/main.cpp | 2 +- main/motorctl.cpp | 88 ++++++++++++++++++++++++++++++++++++++++++++--- main/motorctl.hpp | 17 ++++++--- 6 files changed, 122 insertions(+), 13 deletions(-) diff --git a/main/button.cpp b/main/button.cpp index 0781953..927dac3 100644 --- a/main/button.cpp +++ b/main/button.cpp @@ -19,11 +19,13 @@ static const char * TAG = "button"; //----------------------------- //-------- constructor -------- //----------------------------- -buttonCommands::buttonCommands(gpio_evaluatedSwitch * button_f, controlledArmchair * control_f, buzzer_t * buzzer_f ){ +buttonCommands::buttonCommands(gpio_evaluatedSwitch * button_f, controlledArmchair * control_f, buzzer_t * buzzer_f, controlledMotor * motorLeft_f, controlledMotor * motorRight_f){ //copy object pointers button = button_f; control = control_f; buzzer = buzzer_f; + motorLeft = motorLeft_f; + motorRight = motorRight_f; //TODO declare / configure evaluatedSwitch here instead of config (unnecessary that button object is globally available - only used here)? } @@ -63,6 +65,19 @@ void buttonCommands::action (uint8_t count){ ESP_LOGW(TAG, "cmd %d: toggle between MASSAGE and JOYSTICK", count); control->toggleModes(controlMode_t::MASSAGE, controlMode_t::JOYSTICK); //toggle between MASSAGE and JOYSTICK mode break; + + case 8: + //toggle deceleration fading between on and off + bool decelEnabled = motorLeft->toggleFade(fadeType_t::DECEL); + motorRight->toggleFade(fadeType_t::DECEL); + ESP_LOGW(TAG, "cmd %d: toggle deceleration fading to: %d", count, (int)decelEnabled); + if (decelEnabled){ + buzzer->beep(3, 60, 50); + } else { + buzzer->beep(1, 1000, 1); + } + + } } diff --git a/main/button.hpp b/main/button.hpp index acaf520..b5d5900 100644 --- a/main/button.hpp +++ b/main/button.hpp @@ -3,6 +3,7 @@ #include "gpio_evaluateSwitch.hpp" #include "buzzer.hpp" #include "control.hpp" +#include "motorctl.hpp" @@ -16,7 +17,9 @@ class buttonCommands { buttonCommands ( gpio_evaluatedSwitch * button_f, controlledArmchair * control_f, - buzzer_t * buzzer_f + buzzer_t * buzzer_f, + controlledMotor * motorLeft_f, + controlledMotor * motorRight_f ); //--- functions --- @@ -32,6 +35,8 @@ class buttonCommands { gpio_evaluatedSwitch* button; controlledArmchair * control; buzzer_t* buzzer; + controlledMotor * motorLeft; + controlledMotor * motorRight; //--- variables --- uint8_t count = 0; diff --git a/main/config.cpp b/main/config.cpp index 1b515e1..784047e 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -29,8 +29,8 @@ single100a_config_t configDriverRight = { //--- configure motor contol --- motorctl_config_t configMotorControl = { - .msFadeAccel = 2000, //acceleration of the motor (ms it takes from 0% to 100%) - .msFadeDecel = 1000, //deceleration of the motor (ms it takes from 100% to 0%) + .msFadeAccel = 1300, //acceleration of the motor (ms it takes from 0% to 100%) + .msFadeDecel = 800, //deceleration of the motor (ms it takes from 100% to 0%) .currentMax = 10 }; diff --git a/main/main.cpp b/main/main.cpp index 8b701a6..f5bff17 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -75,7 +75,7 @@ void task_control( void * pvParameters ){ void task_button( void * pvParameters ){ ESP_LOGI(TAG, "Initializing command-button and starting handle loop"); //create button instance - buttonCommands commandButton(&buttonJoystick, &control, &buzzer); + buttonCommands commandButton(&buttonJoystick, &control, &buzzer, &motorLeft, &motorRight); //start handle loop commandButton.startHandleLoop(); } diff --git a/main/motorctl.cpp b/main/motorctl.cpp index c887c97..62ff8c6 100644 --- a/main/motorctl.cpp +++ b/main/motorctl.cpp @@ -11,6 +11,9 @@ static const char * TAG = "motor-control"; controlledMotor::controlledMotor(single100a_config_t config_driver, motorctl_config_t config_control): motor(config_driver) { //copy parameters for controlling the motor config = config_control; + //copy configured default fading durations to actually used variables + msFadeAccel = config.msFadeAccel; + msFadeDecel = config.msFadeDecel; init(); //TODO: add currentsensor object here @@ -89,11 +92,15 @@ void controlledMotor::handle(){ //--- calculate increment --- //calculate increment for fading UP with passed time since last run and configured fade time int64_t usPassed = esp_timer_get_time() - timestampLastRunUs; - dutyIncrementAccel = ( usPassed / ((float)config.msFadeAccel * 1000) ) * 100; //TODO define maximum increment - first run after startup (or long) pause can cause a very large increment + 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 + } else { + dutyIncrementAccel = 100; + } //calculate increment for fading DOWN with passed time since last run and configured fade time - if (config.msFadeDecel > 0){ - dutyIncrementDecel = ( usPassed / ((float)config.msFadeDecel * 1000) ) * 100; + if (msFadeDecel > 0){ + dutyIncrementDecel = ( usPassed / ((float)msFadeDecel * 1000) ) * 100; } else { dutyIncrementDecel = 100; } @@ -130,7 +137,7 @@ void controlledMotor::handle(){ if (dutyNow <= 0) { //reverse, accelerating fade(&dutyNow, dutyTarget, - dutyIncrementAccel); } - else if (&dutyNow > 0) { //forward, decelerating + else if (dutyNow > 0) { //forward, decelerating fade(&dutyNow, dutyTarget, - dutyIncrementDecel); } } @@ -222,3 +229,76 @@ motorCommand_t controlledMotor::getStatus(){ return status; }; + + +//=============================== +//=========== setFade =========== +//=============================== +//function for editing or enabling the fading/ramp of the motor control + +//set/update fading duration/amount +void controlledMotor::setFade(fadeType_t fadeType, uint32_t msFadeNew){ + //TODO: mutex for msFade variable also used in handle function + switch(fadeType){ + case fadeType_t::ACCEL: + msFadeAccel = msFadeNew; + break; + case fadeType_t::DECEL: + msFadeDecel = msFadeNew; + break; + } +} + + +//enable (set to default value) or disable fading +void controlledMotor::setFade(fadeType_t fadeType, bool enabled){ + uint32_t msFadeNew = 0; //define new fade time as default disabled + if(enabled){ //enable + //set to default/configured value + switch(fadeType){ + case fadeType_t::ACCEL: + msFadeNew = config.msFadeAccel; + break; + case fadeType_t::DECEL: + msFadeNew = config.msFadeDecel; + break; + } + } + //apply new Fade value + setFade(fadeType, msFadeNew); +} + + + +//================================== +//=========== toggleFade =========== +//================================== +//toggle fading between OFF and default value +bool controlledMotor::toggleFade(fadeType_t fadeType){ + uint32_t msFadeNew = 0; + bool enabled = false; + switch(fadeType){ + case fadeType_t::ACCEL: + if (msFadeAccel == 0){ + msFadeNew = config.msFadeAccel; + enabled = true; + } else { + msFadeNew = 0; + } + break; + case fadeType_t::DECEL: + if (msFadeDecel == 0){ + msFadeNew = config.msFadeAccel; + enabled = true; + } else { + msFadeNew = 0; + } + break; + } + //apply new Fade value + setFade(fadeType, msFadeNew); + + //return new state (fading enabled/disabled) + return enabled; +} + diff --git a/main/motorctl.hpp b/main/motorctl.hpp index 4528147..322b0f5 100644 --- a/main/motorctl.hpp +++ b/main/motorctl.hpp @@ -13,7 +13,7 @@ extern "C" //------------------------------------- -//-------- struct declarations ------- +//-------- struct/type declarations ------- //------------------------------------- //struct for sending command for one motor in the queue @@ -35,6 +35,10 @@ typedef struct motorctl_config_t { float currentMax; } motorctl_config_t; +//enum fade type (acceleration, deceleration) +//e.g. used for specifying which fading should be modified with setFade, togleFade functions +enum class fadeType_t {ACCEL, DECEL}; + class controlledMotor { @@ -45,6 +49,10 @@ class controlledMotor { void setTarget(motorstate_t state_f, float duty_f = 0); //adds target command to queue for handle function motorCommand_t getStatus(); //get current status of the motor (returns struct with state and duty) + void setFade(fadeType_t fadeType, bool enabled); //enable/disable acceleration or deceleration fading + void setFade(fadeType_t fadeType, uint32_t msFadeNew); //set acceleration or deceleration fade time + bool toggleFade(fadeType_t fadeType); //toggle acceleration or deceleration on/off + private: //--- functions --- @@ -60,7 +68,7 @@ class controlledMotor { //--- variables --- //struct for storing control specific parameters motorctl_config_t config; - + motorstate_t state = motorstate_t::IDLE; float currentMax; @@ -72,12 +80,13 @@ class controlledMotor { float dutyIncrementDecel; float dutyDelta; + uint32_t msFadeAccel; + uint32_t msFadeDecel; + uint32_t ramp; int64_t timestampLastRunUs; struct motorCommand_t commandReceive = {}; struct motorCommand_t commandSend = {}; - - };