From 268018832df3c288881e473d02cd6bcd06aba2ae Mon Sep 17 00:00:00 2001 From: jonny_jr9 Date: Tue, 20 Feb 2024 16:34:42 +0100 Subject: [PATCH 1/5] Add nvs in motorctl: Store and Load accel/decel conf Changes in menu for accel/decel time are now persistent after restarts main: initialize nvs and pass pointer to motorctl task motorctl: - add method to get default configured value - add name to config -> adjust logging - add methods to read and write msFadeAccel and msFadeDecel from nvs --- board_single/main/config.cpp | 2 + board_single/main/main.cpp | 26 +++++- common/motorctl.cpp | 158 +++++++++++++++++++++++++++++++---- common/motorctl.hpp | 13 ++- common/types.hpp | 1 + 5 files changed, 178 insertions(+), 22 deletions(-) diff --git a/board_single/main/config.cpp b/board_single/main/config.cpp index d35a2e1..7498e57 100644 --- a/board_single/main/config.cpp +++ b/board_single/main/config.cpp @@ -90,6 +90,7 @@ sabertooth2x60_config_t sabertoothConfig = { // TODO add motor name string -> then use as log tag? //--- configure left motor (contol) --- motorctl_config_t configMotorControlLeft = { + .name = "left", .msFadeAccel = 1500, // acceleration of the motor (ms it takes from 0% to 100%) .msFadeDecel = 1000, // deceleration of the motor (ms it takes from 100% to 0%) .currentLimitEnabled = false, @@ -101,6 +102,7 @@ motorctl_config_t configMotorControlLeft = { //--- configure right motor (contol) --- motorctl_config_t configMotorControlRight = { + .name = "right", .msFadeAccel = 1500, // acceleration of the motor (ms it takes from 0% to 100%) .msFadeDecel = 1000, // deceleration of the motor (ms it takes from 100% to 0%) .currentLimitEnabled = false, diff --git a/board_single/main/main.cpp b/board_single/main/main.cpp index de5fa55..b15bbb8 100644 --- a/board_single/main/main.cpp +++ b/board_single/main/main.cpp @@ -4,6 +4,7 @@ extern "C" #include #include #include +#include "nvs.h" #include "freertos/FreeRTOS.h" #include "freertos/task.h" #include "driver/gpio.h" @@ -93,9 +94,12 @@ esp_err_t on_joystick_url(httpd_req_t *req) return (httpJoystickMain->*pointerToReceiveFunc)(req); } -//-- tag for logging -- +//--- tag for logging --- static const char * TAG = "main"; +//-- handle passed to tasks for accessing nvs -- +nvs_handle_t nvsHandle; + @@ -140,8 +144,8 @@ void createObjects() // create controlled motor instances (motorctl.hpp) // with configurations from config.cpp - motorLeft = new controlledMotor(setLeftFunc, configMotorControlLeft); - motorRight = new controlledMotor(setRightFunc, configMotorControlRight); + motorLeft = new controlledMotor(setLeftFunc, configMotorControlLeft, &nvsHandle); + motorRight = new controlledMotor(setRightFunc, configMotorControlRight, &nvsHandle); // create speedsensor instances // with configurations from config.cpp @@ -207,6 +211,22 @@ extern "C" void app_main(void) { //--- initialize encoder --- const QueueHandle_t encoderQueue = encoder_init(&encoder_config); + //--- initialize nvs-flash --- (for persistant config values) + ESP_LOGW(TAG, "initializing nvs-flash..."); + esp_err_t err = nvs_flash_init(); + if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) + { + ESP_LOGE(TAG, "NVS truncated -> deleting flash"); + // Retry nvs_flash_init + ESP_ERROR_CHECK(nvs_flash_erase()); + err = nvs_flash_init(); + } + ESP_ERROR_CHECK(err); + //--- open nvs-flash --- + err = nvs_open("storage", NVS_READWRITE, &nvsHandle); + if (err != ESP_OK) + ESP_LOGE(TAG, "Error (%s) opening NVS handle!\n", esp_err_to_name(err)); + printf("\n"); diff --git a/common/motorctl.cpp b/common/motorctl.cpp index cb085b4..5083bc8 100644 --- a/common/motorctl.cpp +++ b/common/motorctl.cpp @@ -15,7 +15,7 @@ static const char * TAG = "motor-control"; //task for handling the motors (ramp, current limit, driver) void task_motorctl( void * task_motorctl_parameters ){ task_motorctl_parameters_t *objects = (task_motorctl_parameters_t *)task_motorctl_parameters; - ESP_LOGW(TAG, "Task-motorctl: starting handle loop..."); + ESP_LOGW(TAG, "Task-motorctl: starting handle loops for left and right motor..."); while(1){ objects->motorRight->handle(); objects->motorLeft->handle(); @@ -29,19 +29,17 @@ void task_motorctl( void * task_motorctl_parameters ){ //======== constructor ======== //============================= //constructor, simultaniously initialize instance of motor driver 'motor' and current sensor 'cSensor' with provided config (see below lines after ':') -controlledMotor::controlledMotor(motorSetCommandFunc_t setCommandFunc, motorctl_config_t config_control): +controlledMotor::controlledMotor(motorSetCommandFunc_t setCommandFunc, motorctl_config_t config_control, nvs_handle_t * nvsHandle_f): cSensor(config_control.currentSensor_adc, config_control.currentSensor_ratedCurrent) { //copy parameters for controlling the motor config = config_control; //pointer to update motot dury method motorSetCommand = setCommandFunc; - //copy configured default fading durations to actually used variables - msFadeAccel = config.msFadeAccel; - msFadeDecel = config.msFadeDecel; + //pointer to nvs handle + nvsHandle = nvsHandle_f; + //create queue, initialize config values init(); - //TODO: add currentsensor object here - //currentSensor cSensor(config.currentSensor_adc, config.currentSensor_ratedCurrent); } @@ -54,7 +52,11 @@ void controlledMotor::init(){ if (commandQueue == NULL) ESP_LOGE(TAG, "Failed to create command-queue"); else - ESP_LOGW(TAG, "Initialized command-queue"); + ESP_LOGI(TAG, "[%s] Initialized command-queue", config.name); + + // load config values from nvs, otherwise use default from config object + loadAccelDuration(); + loadDecelDuration(); //cSensor.calibrateZeroAmpere(); //currently done in currentsensor constructor TODO do this regularly e.g. in idle? } @@ -104,7 +106,7 @@ void controlledMotor::handle(){ //--- receive commands from queue --- if( xQueueReceive( commandQueue, &commandReceive, ( TickType_t ) 0 ) ) { - ESP_LOGD(TAG, "Read command from queue: state=%s, duty=%.2f", motorstateStr[(int)commandReceive.state], commandReceive.duty); + ESP_LOGD(TAG, "[%s] Read command from queue: state=%s, duty=%.2f", config.name, motorstateStr[(int)commandReceive.state], commandReceive.duty); state = commandReceive.state; dutyTarget = commandReceive.duty; receiveTimeout = false; @@ -139,7 +141,7 @@ void controlledMotor::handle(){ receiveTimeout = true; state = motorstate_t::IDLE; dutyTarget = 0; - ESP_LOGE(TAG, "TIMEOUT, no target data received for more than %ds -> switch to IDLE", TIMEOUT_IDLE_WHEN_NO_COMMAND/1000); + ESP_LOGE(TAG, "[%s] TIMEOUT, no target data received for more than %ds -> switch to IDLE", config.name, TIMEOUT_IDLE_WHEN_NO_COMMAND/1000); } //--- calculate increment --- @@ -164,7 +166,7 @@ void controlledMotor::handle(){ if (state == motorstate_t::BRAKE){ ESP_LOGD(TAG, "braking - skip fading"); motorSetCommand({motorstate_t::BRAKE, dutyTarget}); - ESP_LOGI(TAG, "Set Motordriver: state=%s, duty=%.2f - Measurements: current=%.2f, speed=N/A", motorstateStr[(int)state], dutyNow, currentNow); + ESP_LOGI(TAG, "[%s] Set Motordriver: state=%s, duty=%.2f - Measurements: current=%.2f, speed=N/A", config.name, motorstateStr[(int)state], dutyNow, currentNow); //dutyNow = 0; return; //no need to run the fade algorithm } @@ -211,7 +213,7 @@ void controlledMotor::handle(){ } else if (dutyNow > currentLimitDecrement) { dutyNow -= currentLimitDecrement; } - ESP_LOGW(TAG, "current limit exceeded! now=%.3fA max=%.1fA => decreased duty from %.3f to %.3f", currentNow, config.currentMax, dutyOld, dutyNow); + ESP_LOGW(TAG, "[%s] current limit exceeded! now=%.3fA max=%.1fA => decreased duty from %.3f to %.3f", config.name, currentNow, config.currentMax, dutyOld, dutyNow); } } @@ -255,7 +257,7 @@ void controlledMotor::handle(){ //--- apply new target to motor --- motorSetCommand({state, (float)fabs(dutyNow)}); - ESP_LOGI(TAG, "Set Motordriver: state=%s, duty=%.2f - Measurements: current=%.2f, speed=N/A", motorstateStr[(int)state], dutyNow, currentNow); + ESP_LOGI(TAG, "[%s] Set Motordriver: state=%s, duty=%.2f - Measurements: current=%.2f, speed=N/A", config.name, motorstateStr[(int)state], dutyNow, currentNow); //note: BRAKE state is handled earlier @@ -275,7 +277,7 @@ void controlledMotor::setTarget(motorstate_t state_f, float duty_f){ .state = state_f, .duty = duty_f }; - ESP_LOGI(TAG, "setTarget: Inserting command to queue: state='%s'(%d), duty=%.2f", motorstateStr[(int)commandSend.state], (int)commandSend.state, commandSend.duty); + ESP_LOGI(TAG, "[%s] setTarget: Inserting command to queue: state='%s'(%d), duty=%.2f", config.name, motorstateStr[(int)commandSend.state], (int)commandSend.state, commandSend.duty); //send command to queue (overwrite if an old command is still in the queue and not processed) xQueueOverwrite( commandQueue, ( void * )&commandSend); //xQueueSend( commandQueue, ( void * )&commandSend, ( TickType_t ) 0 ); @@ -316,6 +318,22 @@ uint32_t controlledMotor::getFade(fadeType_t fadeType){ return 0; } +//============================== +//======= getFadeDefault ======= +//============================== +//return default accel / decel time (from config) +uint32_t controlledMotor::getFadeDefault(fadeType_t fadeType){ + switch(fadeType){ + case fadeType_t::ACCEL: + return config.msFadeAccel; + break; + case fadeType_t::DECEL: + return config.msFadeDecel; + break; + } + return 0; +} + //=============================== @@ -328,12 +346,13 @@ 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: - ESP_LOGW(TAG, "changed fade-up time from %d to %d", msFadeAccel, msFadeNew); - msFadeAccel = msFadeNew; + ESP_LOGW(TAG, "[%s] changed fade-up time from %d to %d", config.name, msFadeAccel, msFadeNew); + writeAccelDuration(msFadeNew); break; case fadeType_t::DECEL: - ESP_LOGW(TAG, "changed fade-down time from %d to %d", msFadeDecel, msFadeNew); + ESP_LOGW(TAG, "[%s] changed fade-down time from %d to %d",config.name, msFadeDecel, msFadeNew); msFadeDecel = msFadeNew; + writeDecelDuration(msFadeNew); break; } } @@ -390,3 +409,108 @@ bool controlledMotor::toggleFade(fadeType_t fadeType){ return enabled; } + + + +//----------------------------- +//----- loadAccelDuration ----- +//----------------------------- +// load stored value from nvs if not successfull uses config default value +void controlledMotor::loadAccelDuration(void) +{ + // load default value + msFadeAccel = config.msFadeAccel; + // read from nvs + uint32_t valueNew; + char key[15]; + snprintf(key, 15, "m-%s-accel", config.name); + esp_err_t err = nvs_get_u32(*nvsHandle, key, &valueNew); + switch (err) + { + case ESP_OK: + ESP_LOGW(TAG, "Successfully read value '%s' from nvs. Overriding default value %d with %d", key, config.msFadeAccel, valueNew); + msFadeAccel = valueNew; + break; + case ESP_ERR_NVS_NOT_FOUND: + ESP_LOGW(TAG, "nvs: the value '%s' is not initialized yet, keeping default value %d", key, msFadeAccel); + break; + default: + ESP_LOGE(TAG, "Error (%s) reading nvs!", esp_err_to_name(err)); + } +} + +//----------------------------- +//----- loadDecelDuration ----- +//----------------------------- +void controlledMotor::loadDecelDuration(void) +{ + // load default value + msFadeDecel = config.msFadeDecel; + // read from nvs + uint32_t valueNew; + char key[15]; + snprintf(key, 15, "m-%s-decel", config.name); + esp_err_t err = nvs_get_u32(*nvsHandle, key, &valueNew); + switch (err) + { + case ESP_OK: + ESP_LOGW(TAG, "Successfully read value '%s' from nvs. Overriding default value %d with %d", key, config.msFadeDecel, valueNew); + msFadeDecel = valueNew; + break; + case ESP_ERR_NVS_NOT_FOUND: + ESP_LOGW(TAG, "nvs: the value '%s' is not initialized yet, keeping default value %d", key, msFadeDecel); + break; + default: + ESP_LOGE(TAG, "Error (%s) reading nvs!", esp_err_to_name(err)); + } +} + + + + +//------------------------------ +//----- writeAccelDuration ----- +//------------------------------ +// write provided value to nvs to be persistent and update the local variable msFadeAccel +void controlledMotor::writeAccelDuration(uint32_t newValue) +{ + // generate nvs storage key + char key[15]; + snprintf(key, 15, "m-%s-accel", config.name); + // update nvs value + ESP_LOGW(TAG, "[%s] updating nvs value '%s' from %d to %d", config.name, key, msFadeAccel, newValue); + esp_err_t err = nvs_set_u32(*nvsHandle, key, newValue); + if (err != ESP_OK) + ESP_LOGE(TAG, "nvs: failed writing"); + err = nvs_commit(*nvsHandle); + if (err != ESP_OK) + ESP_LOGE(TAG, "nvs: failed committing updates"); + else + ESP_LOGI(TAG, "nvs: successfully committed updates"); + // update variable + msFadeAccel = newValue; +} + +//------------------------------ +//----- writeDecelDuration ----- +//------------------------------ +// write provided value to nvs to be persistent and update the local variable msFadeDecel +// TODO: reduce duplicate code +void controlledMotor::writeDecelDuration(uint32_t newValue) +{ + // generate nvs storage key + char key[15]; + snprintf(key, 15, "m-%s-decel", config.name); + // update nvs value + ESP_LOGW(TAG, "[%s] updating nvs value '%s' from %d to %d", config.name, key, msFadeDecel, newValue); + esp_err_t err = nvs_set_u32(*nvsHandle, key, newValue); + if (err != ESP_OK) + ESP_LOGE(TAG, "nvs: failed writing"); + err = nvs_commit(*nvsHandle); + if (err != ESP_OK) + ESP_LOGE(TAG, "nvs: failed committing updates"); + else + ESP_LOGI(TAG, "nvs: successfully committed updates"); + // update variable + msFadeDecel = newValue; +} \ No newline at end of file diff --git a/common/motorctl.hpp b/common/motorctl.hpp index f2ee60c..a8630cb 100644 --- a/common/motorctl.hpp +++ b/common/motorctl.hpp @@ -7,6 +7,8 @@ extern "C" #include "freertos/queue.h" #include "esp_log.h" #include "esp_timer.h" +#include "nvs_flash.h" +#include "nvs.h" } #include "motordrivers.hpp" @@ -28,12 +30,13 @@ typedef void (*motorSetCommandFunc_t)(motorCommand_t cmd); class controlledMotor { public: //--- functions --- - controlledMotor(motorSetCommandFunc_t setCommandFunc, motorctl_config_t config_control); //constructor with structs for configuring motordriver and parameters for control TODO: add configuration for currentsensor + controlledMotor(motorSetCommandFunc_t setCommandFunc, motorctl_config_t config_control, nvs_handle_t * nvsHandle); //constructor with structs for configuring motordriver and parameters for control TODO: add configuration for currentsensor void handle(); //controls motor duty with fade and current limiting feature (has to be run frequently by another task) 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) 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 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 @@ -44,7 +47,11 @@ class controlledMotor { private: //--- functions --- - void init(); //creates currentsensor objects, motordriver objects and queue + void init(); // creates command-queue and initializes config values + void loadAccelDuration(void); // load stored value for msFadeAccel from nvs + void loadDecelDuration(void); + void writeAccelDuration(uint32_t newValue); // write value to nvs and update local variable + void writeDecelDuration(uint32_t newValue); //--- objects --- //queue for sending commands to the separate task running the handle() function very fast @@ -60,6 +67,8 @@ class controlledMotor { //struct for storing control specific parameters motorctl_config_t config; motorstate_t state = motorstate_t::IDLE; + //handle for using the nvs flash (persistent config variables) + nvs_handle_t * nvsHandle; float currentMax; float currentNow; diff --git a/common/types.hpp b/common/types.hpp index b94fc62..450a2e6 100644 --- a/common/types.hpp +++ b/common/types.hpp @@ -41,6 +41,7 @@ typedef struct motorCommands_t { //struct with all config parameters for a motor regarding ramp and current limit typedef struct motorctl_config_t { + char * name; //name for unique nvs storage-key prefix and logging 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%) bool currentLimitEnabled; From c9c371a74244f50b3b11ab9b5f91bfe5a2013277 Mon Sep 17 00:00:00 2001 From: jonny_jr9 Date: Tue, 20 Feb 2024 17:27:40 +0100 Subject: [PATCH 2/5] Add default-value to Menu, Optimize value-screen, Add Beeping menu: - optimize set-value page: send static content only once so only update value -> significant performance boost - formatting - reset control timeout to prevent unintended exit and bugged display - menu item: add option to show a default value (function ptr) - add default value to adjust fading - add beeping motorctl: - dont write to nvs when value is unchanged --- board_single/main/menu.cpp | 260 +++++++++++++++++++++++-------------- board_single/main/menu.hpp | 1 + common/motorctl.cpp | 12 +- 3 files changed, 176 insertions(+), 97 deletions(-) diff --git a/board_single/main/menu.cpp b/board_single/main/menu.cpp index 622b675..ba64bf2 100644 --- a/board_single/main/menu.cpp +++ b/board_single/main/menu.cpp @@ -25,7 +25,9 @@ static int value = 0; //================================ //===== CONFIGURE MENU ITEMS ===== //================================ -// note: when line4 * and line5 * are empty the value is printed large +// Instructions / Behavior: +// - when line4 * and line5 * are empty the value is printed large +// - when 3rd element is not NULL (pointer to defaultValue function) that return value is shown in line 2 //######################### //#### center Joystick #### @@ -33,9 +35,8 @@ static int value = 0; void item_centerJoystick_action(display_task_parameters_t * objects, SSD1306_t * display, int value){ if (!value) return; ESP_LOGW(TAG, "defining joystick center"); - (*objects).joystick->defineCenter(); - //objects->joystick->defineCenter(); - //joystick->defineCenter(); + objects->joystick->defineCenter(); + objects->buzzer->beep(3, 60, 40); } int item_centerJoystick_value(display_task_parameters_t * objects){ return 1; @@ -43,17 +44,18 @@ int item_centerJoystick_value(display_task_parameters_t * objects){ menuItem_t item_centerJoystick = { item_centerJoystick_action, // function action - item_centerJoystick_value, - 0, // valueMin - 1, // valueMAx - 1, // valueIncrement - "Center Joystick", // title - "Center Joystick", // line1 (above value) - "click to confirm", // line2 (above value) - "defines current", // line4 * (below value) - "pos as center", // line5 * - "click to confirm", // line6 - "set 0 to cancel", // line7 + item_centerJoystick_value, // function get initial value + NULL, // function get default value or NULL + 0, // valueMin + 1, // valueMAx + 1, // valueIncrement + "Center Joystick", // title + "Center Joystick", // line1 (above value) + "click to confirm", // line2 (above value) + "defines current", // line4 * (below value) + "pos as center", // line5 * + "click to confirm", // line6 + "set 0 to cancel", // line7 }; @@ -61,6 +63,7 @@ menuItem_t item_centerJoystick = { //#### debug Joystick #### //######################## //continously show/update joystick data on display +#define DEBUG_JOYSTICK_UPDATE_INTERVAL 50 void item_debugJoystick_action(display_task_parameters_t * objects, SSD1306_t * display, int value) { //--- variables --- @@ -90,8 +93,9 @@ void item_debugJoystick_action(display_task_parameters_t * objects, SSD1306_t * displayTextLine(display, 5, false, false, "pos=%-12s ", joystickPosStr[(int)data.position]); // exit when button pressed - if (xQueueReceive(objects->encoderQueue, &event, 20 / portTICK_PERIOD_MS)) + if (xQueueReceive(objects->encoderQueue, &event, DEBUG_JOYSTICK_UPDATE_INTERVAL / portTICK_PERIOD_MS)) { + objects->control->resetTimeout(); switch (event.type) { case RE_ET_BTN_CLICKED: @@ -113,17 +117,18 @@ int item_debugJoystick_value(display_task_parameters_t * objects){ menuItem_t item_debugJoystick = { item_debugJoystick_action, // function action - item_debugJoystick_value, - 0, // valueMin - 1, // valueMAx - 1, // valueIncrement - "Debug joystick", // title - "Debug joystick", // line1 (above value) - "", // line2 (above value) - "click to enter", // line4 * (below value) - "debug screen", // line5 * - "prints values", // line6 - "set 0 to cancel", // line7 + item_debugJoystick_value, // function get initial value + NULL, // function get default value or NULL + 0, // valueMin + 1, // valueMAx + 1, // valueIncrement + "Debug joystick", // title + "Debug joystick", // line1 (above value) + "", // line2 (above value) + "click to enter", // line4 * (below value) + "debug screen", // line5 * + "prints values", // line6 + "set 0 to cancel", // line7 }; @@ -141,20 +146,22 @@ int maxDuty_currentValue(display_task_parameters_t * objects) return 84; } menuItem_t item_maxDuty = { - maxDuty_action, // function action - maxDuty_currentValue, - 1, // valueMin - 99, // valueMAx - 1, // valueIncrement - "max duty", // title - "", // line1 (above value) - " set max-duty: ", // line2 (above value) - "", // line4 * (below value) - "", // line5 * - " 1-99 ", // line6 - " percent ", // line7 + maxDuty_action, // function action + maxDuty_currentValue, // function get initial value + NULL, // function get default value or NULL + 1, // valueMin + 99, // valueMAx + 1, // valueIncrement + "max duty", // title + "", // line1 (above value) + " set max-duty: ", // line2 (above value) + "", // line4 * (below value) + "", // line5 * + " 1-99 ", // line6 + " percent ", // line7 }; + //###################### //##### accelLimit ##### //###################### @@ -167,21 +174,27 @@ int item_accelLimit_value(display_task_parameters_t * objects) { return objects->motorLeft->getFade(fadeType_t::ACCEL); } +int item_accelLimit_default(display_task_parameters_t * objects) +{ + return objects->motorLeft->getFadeDefault(fadeType_t::ACCEL); +} menuItem_t item_accelLimit = { - item_accelLimit_action, // function action - item_accelLimit_value, - 0, // valueMin - 10000, // valueMAx - 100, // valueIncrement - "Accel limit", // title - "Accel limit /", // line1 (above value) - "Fade up time", // line2 (above value) - "", // line4 * (below value) - "", // line5 * - "milliseconds", // line6 - "from 0 to 100%", // line7 + item_accelLimit_action, // function action + item_accelLimit_value, // function get initial value + item_accelLimit_default, // function get default value or NULL + 0, // valueMin + 10000, // valueMAx + 100, // valueIncrement + "Accel limit", // title + " Fade up time", // line1 (above value) + "", // line2 <= showing "default = %d" + "", // line4 * (below value) + "", // line5 * + "milliseconds", // line6 + "from 0 to 100%", // line7 }; + // ###################### // ##### decelLimit ##### // ###################### @@ -194,21 +207,27 @@ int item_decelLimit_value(display_task_parameters_t * objects) { return objects->motorLeft->getFade(fadeType_t::DECEL); } +int item_decelLimit_default(display_task_parameters_t * objects) +{ + return objects->motorLeft->getFadeDefault(fadeType_t::DECEL); +} menuItem_t item_decelLimit = { - item_decelLimit_action, // function action - item_decelLimit_value, - 0, // valueMin - 10000, // valueMAx - 100, // valueIncrement - "Decel limit", // title - "Decel limit /", // line1 (above value) - "Fade down time", // line2 (above value) - "", // line4 * (below value) - "", // line5 * - "milliseconds", // line6 - "from 100 to 0%", // line7 + item_decelLimit_action, // function action + item_decelLimit_value, // function get initial value + item_decelLimit_default, // function get default value or NULL + 0, // valueMin + 10000, // valueMAx + 100, // valueIncrement + "Decel limit", // title + " Fade down time", // line1 (above value) + "", // line2 <= showing "default = %d" + "", // line4 * (below value) + "", // line5 * + "milliseconds", // line6 + "from 100 to 0%", // line7 }; + //##################### //###### example ###### //##################### @@ -217,42 +236,47 @@ void item_example_action(display_task_parameters_t * objects, SSD1306_t * displa return; } int item_example_value(display_task_parameters_t * objects){ - return 53; + return 53; //initial value shown / changed from +} +int item_example_valueDefault(display_task_parameters_t * objects){ + return 931; // optionally shown in line 2 as "default = %d" } menuItem_t item_example = { item_example_action, // function action - item_example_value, - -255, // valueMin - 255, // valueMAx - 2, // valueIncrement - "example-item-max", // title - "line 1 - above", // line1 (above value) - "line 2 - above", // line2 (above value) - "line 4 - below", // line4 * (below value) - "line 5 - below", // line5 * - "line 6 - below", // line6 - "line 7 - last", // line7 + item_example_value, // function get initial value + NULL, // function get default value or NULL + -255, // valueMin + 255, // valueMAx + 2, // valueIncrement + "example-item-max", // title + "line 1 - above", // line1 (above value) + "line 2 - above", // line2 (above value) + "line 4 - below", // line4 * (below value) + "line 5 - below", // line5 * + "line 6 - below", // line6 + "line 7 - last", // line7 }; menuItem_t item_last = { item_example_action, // function action - item_example_value, - -500, // valueMin - 4500, // valueMAx - 50, // valueIncrement - "set large number", // title - "line 1 - above", // line1 (above value) - "line 2 - above", // line2 (above value) - "", // line4 * (below value) - "", // line5 * - "line 6 - below", // line6 - "line 7 - last", // line7 + item_example_value, // function get initial value + item_example_valueDefault, // function get default value or NULL + -500, // valueMin + 4500, // valueMAx + 50, // valueIncrement + "set large number", // title + "line 1 - above", // line1 (above value) + "line 2 - above", // line2 (above value) + "", // line4 * (below value) + "", // line5 * + "line 6 - below", // line6 + "line 7 - last", // line7 }; //store all configured menu items in one array -menuItem_t menuItems[] = {item_centerJoystick, item_debugJoystick, item_accelLimit, item_decelLimit, item_example, item_last}; -int itemCount = 6; +const menuItem_t menuItems[] = {item_centerJoystick, item_debugJoystick, item_accelLimit, item_decelLimit, item_example, item_last}; +const int itemCount = 6; @@ -298,21 +322,28 @@ void showItemList(SSD1306_t *display, int selectedItem) } } -//--------------------------- -//----- showValueSelect ----- -//--------------------------- +//----------------------------- +//--- showValueSelectStatic --- +//----------------------------- // function that renders value-select screen to display (one update) // shows configured text of selected item and currently selected value // TODO show previous value in one line? -// TODO update changed line only (value) -void showValueSelect(SSD1306_t *display, int selectedItem) +void showValueSelectStatic(display_task_parameters_t * objects, SSD1306_t *display, int selectedItem) { //-- show title line -- displayTextLine(display, 0, false, true, " -- set value -- "); // inverted //-- show text above value -- displayTextLine(display, 1, false, false, "%-16s", menuItems[selectedItem].line1); + +//-- show line 2 or default value --- + if (menuItems[selectedItem].defaultValue != NULL){ + displayTextLineCentered(display, 2, false, false, "default = %d", menuItems[selectedItem].defaultValue(objects)); + } + else{ + //displayTextLine(display, 2, false, false, "previous=%d", menuItems[selectedItem].currentValue(objects)); // <= show previous value displayTextLine(display, 2, false, false, "%-16s", menuItems[selectedItem].line2); +} //-- show value and other configured lines -- // print value large, if 2 description lines are empty @@ -334,6 +365,24 @@ void showValueSelect(SSD1306_t *display, int selectedItem) } } +//----------------------------- +//----- updateValueSelect ----- +//----------------------------- +// update line with currently set value only (increses performance significantly) +void updateValueSelect(SSD1306_t *display, int selectedItem) +{ + // print value large, if 2 description lines are empty + if (strlen(menuItems[selectedItem].line4) == 0 && strlen(menuItems[selectedItem].line5) == 0) + { + // print large value + line5 and line6 + displayTextLineCentered(display, 3, true, false, "%d", value); //large centered + } + else + { + displayTextLineCentered(display, 3, false, false, "%d", value); //centered + } +} + @@ -343,11 +392,12 @@ void showValueSelect(SSD1306_t *display, int selectedItem) //controls menu with encoder input and displays the text on oled display //function is repeatedly called by display task when in menu state #define QUEUE_TIMEOUT 3000 //timeout no encoder event - to handle timeout and not block the display loop -#define MENU_TIMEOUT 60000 //inactivity timeout (switch to IDLE mode) +#define MENU_TIMEOUT 60000 //inactivity timeout (switch to IDLE mode) note: should be smaller than IDLE timeout in control task void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) { static uint32_t lastActivity = 0; static int selectedItem = 0; + static bool staticContentUpdated = false; rotary_encoder_event_t event; // store event data //--- handle different menu states --- @@ -363,10 +413,12 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) if (xQueueReceive(objects->encoderQueue, &event, QUEUE_TIMEOUT / portTICK_PERIOD_MS)) { lastActivity = esp_log_timestamp(); + objects->control->resetTimeout(); switch (event.type) { case RE_ET_CHANGED: //--- scroll in list --- + objects->buzzer->beep(1, 10, 5); if (event.diff < 0) { if (selectedItem != itemCount - 1) @@ -385,9 +437,11 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) case RE_ET_BTN_CLICKED: //--- switch to edit value page --- + objects->buzzer->beep(1, 50, 10); ESP_LOGI(TAG, "Button pressed - switching to state SET_VALUE"); // change state (menu to set value) menuState = SET_VALUE; + staticContentUpdated = false; // get currently configured value value = menuItems[selectedItem].currentValue(objects); // clear display @@ -397,6 +451,7 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) //exit menu mode case RE_ET_BTN_LONG_PRESSED: //change to previous mode (e.g. JOYSTICK) + objects->buzzer->beep(4, 15, 5); objects->control->toggleMode(controlMode_t::MENU); //currently already in MENU -> changes to previous mode ssd1306_clear_screen(display, false); break; @@ -413,15 +468,25 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) //------------------------- case SET_VALUE: // wait for encoder event - showValueSelect(display, selectedItem); + if (!staticContentUpdated) + { + showValueSelectStatic(objects, display, selectedItem); + staticContentUpdated = true; + } + else + { + // update line with currently set value only (increses performance significantly) + updateValueSelect(display, selectedItem); + } if (xQueueReceive(objects->encoderQueue, &event, QUEUE_TIMEOUT / portTICK_PERIOD_MS)) { - lastActivity = esp_log_timestamp(); + objects->control->resetTimeout(); switch (event.type) { case RE_ET_CHANGED: //-- change value -- + objects->buzzer->beep(1, 25, 10); // increment value if (event.diff < 0) value += menuItems[selectedItem].valueIncrement; @@ -436,6 +501,7 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) case RE_ET_BTN_CLICKED: //-- apply value -- ESP_LOGI(TAG, "Button pressed - running action function with value=%d for item '%s'", value, menuItems[selectedItem].title); + objects->buzzer->beep(2, 50, 50); menuItems[selectedItem].action(objects, display, value); menuState = MAIN_MENU; break; @@ -444,6 +510,7 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) case RE_ET_BTN_LONG_PRESSED: break; } + lastActivity = esp_log_timestamp(); } break; } @@ -455,6 +522,7 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) //close menu and switch to IDLE mode when no encoder event within MENU_TIMEOUT if (esp_log_timestamp() - lastActivity > MENU_TIMEOUT) { + objects->buzzer->beep(1, 500, 10); ESP_LOGW(TAG, "TIMEOUT - no activity for more than %ds -> closing menu, switching to IDLE", MENU_TIMEOUT/1000); // reset menu selectedItem = 0; diff --git a/board_single/main/menu.hpp b/board_single/main/menu.hpp index 96534ea..b7682ec 100644 --- a/board_single/main/menu.hpp +++ b/board_single/main/menu.hpp @@ -17,6 +17,7 @@ typedef struct { void (*action)(display_task_parameters_t * objects, SSD1306_t * display, int value); // pointer to function run when confirmed int (*currentValue)(display_task_parameters_t * objects); // pointer to function to get currently configured value + int (*defaultValue)(display_task_parameters_t * objects); // pointer to function to get currently configured value int valueMin; // min allowed value int valueMax; // max allowed value int valueIncrement; // amount changed at one encoder tick (+/-) diff --git a/common/motorctl.cpp b/common/motorctl.cpp index 5083bc8..ef2dade 100644 --- a/common/motorctl.cpp +++ b/common/motorctl.cpp @@ -351,7 +351,7 @@ void controlledMotor::setFade(fadeType_t fadeType, uint32_t msFadeNew){ break; case fadeType_t::DECEL: ESP_LOGW(TAG, "[%s] changed fade-down time from %d to %d",config.name, msFadeDecel, msFadeNew); - msFadeDecel = msFadeNew; + // write new value to nvs and update the variable writeDecelDuration(msFadeNew); break; } @@ -474,6 +474,11 @@ void controlledMotor::loadDecelDuration(void) // write provided value to nvs to be persistent and update the local variable msFadeAccel void controlledMotor::writeAccelDuration(uint32_t newValue) { + // check if unchanged + if(msFadeAccel == newValue){ + ESP_LOGW(TAG, "value unchanged at %d, not writing to nvs", newValue); + return; + } // generate nvs storage key char key[15]; snprintf(key, 15, "m-%s-accel", config.name); @@ -498,6 +503,11 @@ void controlledMotor::writeAccelDuration(uint32_t newValue) // TODO: reduce duplicate code void controlledMotor::writeDecelDuration(uint32_t newValue) { + // check if unchanged + if(msFadeDecel == newValue){ + ESP_LOGW(TAG, "value unchanged at %d, not writing to nvs", newValue); + return; + } // generate nvs storage key char key[15]; snprintf(key, 15, "m-%s-decel", config.name); From cdfa64fbc0ce674f330999a8413ca3cb5aed3f66 Mon Sep 17 00:00:00 2001 From: jonny_jr9 Date: Wed, 21 Feb 2024 11:30:36 +0100 Subject: [PATCH 3/5] Add joystick-calibration wizzard (store in nvs), Optimize Menu - menu: - Add item joystick calibration wizzard - Add item RESET (clear nvs and restart) - Add configuration option to not show/change value. Instead only show confirm message apply this to all items only running action - Optimize formatting and comments - main: - pass nvsHandle to display task and joystick class - joystick: - Add methods to write and load calibration values (axis min/max adc value) from nvs - Add method to get raw adc value --- board_single/main/display.hpp | 3 + board_single/main/main.cpp | 8 +- board_single/main/menu.cpp | 416 ++++++++++++++++++++++++---------- common/joystick.cpp | 139 +++++++++++- common/joystick.hpp | 45 ++-- 5 files changed, 467 insertions(+), 144 deletions(-) diff --git a/board_single/main/display.hpp b/board_single/main/display.hpp index fffe95c..9315adf 100644 --- a/board_single/main/display.hpp +++ b/board_single/main/display.hpp @@ -8,6 +8,8 @@ extern "C" { #include "freertos/FreeRTOS.h" #include "freertos/task.h" #include "esp_log.h" +#include "nvs_flash.h" +#include "nvs.h" #include "ssd1306.h" #include "font8x8_basic.h" @@ -42,6 +44,7 @@ typedef struct display_task_parameters_t { speedSensor * speedLeft; speedSensor * speedRight; buzzer_t *buzzer; + nvs_handle_t * nvsHandle; } display_task_parameters_t; diff --git a/board_single/main/main.cpp b/board_single/main/main.cpp index b15bbb8..1640635 100644 --- a/board_single/main/main.cpp +++ b/board_single/main/main.cpp @@ -152,8 +152,8 @@ void createObjects() speedLeft = new speedSensor(speedLeft_config); speedRight = new speedSensor(speedRight_config); - // create joystic instance (joystick.hpp) - joystick = new evaluatedJoystick(configJoystick); + // create joystick instance (joystick.hpp) + joystick = new evaluatedJoystick(configJoystick, &nvsHandle); // create httpJoystick object (http.hpp) httpJoystickMain = new httpJoystick(configHttpJoystickMain); @@ -286,8 +286,8 @@ extern "C" void app_main(void) { //----------------------------------- //----- create task for display ----- //----------------------------------- - ////task that handles the display (show stats, handle menu in 'MENU' mode) - display_task_parameters_t display_param = {display_config, control, joystick, encoderQueue, motorLeft, motorRight, speedLeft, speedRight, buzzer}; + //task that handles the display (show stats, handle menu in 'MENU' mode) + display_task_parameters_t display_param = {display_config, control, joystick, encoderQueue, motorLeft, motorRight, speedLeft, speedRight, buzzer, &nvsHandle}; xTaskCreate(&display_task, "display_task", 3*2048, &display_param, 3, NULL); vTaskDelay(200 / portTICK_PERIOD_MS); //wait for all tasks to finish initializing diff --git a/board_single/main/menu.cpp b/board_single/main/menu.cpp index ba64bf2..6b7fd20 100644 --- a/board_single/main/menu.cpp +++ b/board_single/main/menu.cpp @@ -27,7 +27,8 @@ static int value = 0; //================================ // Instructions / Behavior: // - when line4 * and line5 * are empty the value is printed large -// - when 3rd element is not NULL (pointer to defaultValue function) that return value is shown in line 2 +// - when 3rd element is not NULL (pointer to defaultValue function) return int value of that function is shown in line 2 +// - when 2nd element is NULL (pointer to currentValue function): instead of current value "click to confirm is shown" in line 3 //######################### //#### center Joystick #### @@ -38,24 +39,146 @@ void item_centerJoystick_action(display_task_parameters_t * objects, SSD1306_t * objects->joystick->defineCenter(); objects->buzzer->beep(3, 60, 40); } -int item_centerJoystick_value(display_task_parameters_t * objects){ - return 1; -} - menuItem_t item_centerJoystick = { item_centerJoystick_action, // function action - item_centerJoystick_value, // function get initial value - NULL, // function get default value or NULL + NULL, // function get initial value or NULL(show in line 2) + NULL, // function get default value or NULL(dont set value, show msg) 0, // valueMin - 1, // valueMAx - 1, // valueIncrement - "Center Joystick", // title - "Center Joystick", // line1 (above value) - "click to confirm", // line2 (above value) - "defines current", // line4 * (below value) - "pos as center", // line5 * - "click to confirm", // line6 - "set 0 to cancel", // line7 + 0, // valueMax + 0, // valueIncrement + "Center Joystick ", // title + "Center Joystick ", // line1 (above value) + "", // line2 (above value) + "defines current ", // line4 * (below value) + "pos as center ", // line5 * + "", // line6 + "=>long to cancel", // line7 +}; + +// ############################ +// #### calibrate Joystick #### +// ############################ +// continously show/update joystick data on display +#define CALIBRATE_JOYSTICK_UPDATE_INTERVAL 50 +void item_calibrateJoystick_action(display_task_parameters_t *objects, SSD1306_t *display, int value) +{ + //--- variables --- + bool running = true; + joystickCalibrationMode_t mode = X_MIN; + rotary_encoder_event_t event; + int valueNow = 0; + + //-- pre loop instructions -- + ESP_LOGW(TAG, "starting joystick calibration sequence"); + ssd1306_clear_screen(display, false); + + //-- show static lines -- + // show first line (title) + displayTextLine(display, 0, false, true, "calibrate stick"); + // show last line (info) + displayTextLineCentered(display, 7, false, true, " click: confirm "); + // show initital state + displayTextLineCentered(display, 1, true, false, "%s", "X-min"); + + //-- loop until all positions are defined -- + while (running && objects->control->getCurrentMode() == controlMode_t::MENU) + { + // repeatedly print adc value depending on currently selected axis + switch (mode) + { + case X_MIN: + case X_MAX: + displayTextLineCentered(display, 4, true, false, "%d", valueNow = objects->joystick->getRawX()); // large + break; + case Y_MIN: + case Y_MAX: + displayTextLineCentered(display, 4, true, false, "%d", valueNow = objects->joystick->getRawY()); // large + break; + case X_CENTER: + case Y_CENTER: + displayTextLine(display, 4, false, false, " x = %d", objects->joystick->getRawX()); + displayTextLine(display, 5, false, false, " y = %d", objects->joystick->getRawY()); + displayTextLine(display, 6, false, false, "release & click!"); + break; + } + + // handle encoder event + // save and next when button clicked, exit when long pressed + if (xQueueReceive(objects->encoderQueue, &event, CALIBRATE_JOYSTICK_UPDATE_INTERVAL / portTICK_PERIOD_MS)) + { + objects->control->resetTimeout(); + switch (event.type) + { + case RE_ET_BTN_CLICKED: + objects->buzzer->beep(2, 120, 50); + switch (mode) + { + case X_MIN: + // save x min position + ESP_LOGW(TAG, "calibrate-stick: saving X_MIN"); + objects->joystick->writeCalibration(mode, valueNow); + displayTextLineCentered(display, 1, true, false, "%s", "X-max"); + mode = X_MAX; + break; + case X_MAX: + // save x max position + ESP_LOGW(TAG, "calibrate-stick: saving X_MAX"); + objects->joystick->writeCalibration(mode, valueNow); + displayTextLineCentered(display, 1, true, false, "%s", "Y-min"); + mode = Y_MIN; + break; + case Y_MIN: + // save y min position + ESP_LOGW(TAG, "calibrate-stick: saving Y_MIN"); + objects->joystick->writeCalibration(mode, valueNow); + displayTextLineCentered(display, 1, true, false, "%s", "Y-max"); + mode = Y_MAX; + break; + case Y_MAX: + // save y max position + ESP_LOGW(TAG, "calibrate-stick: saving Y_MAX"); + objects->joystick->writeCalibration(mode, valueNow); + displayTextLineCentered(display, 1, true, false, "%s", "CENTR"); + mode = X_CENTER; + break; + case X_CENTER: + case Y_CENTER: + // save center position + ESP_LOGW(TAG, "calibrate-stick: saving CENTER -> finished"); + objects->joystick->defineCenter(); + // finished + running = false; + break; + } + break; + case RE_ET_BTN_LONG_PRESSED: + //exit to main-menu + objects->buzzer->beep(1, 1000, 10); + ESP_LOGW(TAG, "aborting calibration sqeuence"); + running = false; + case RE_ET_CHANGED: + case RE_ET_BTN_PRESSED: + case RE_ET_BTN_RELEASED: + break; + } + } + } +} + +menuItem_t item_calibrateJoystick = { + item_calibrateJoystick_action, // function action + NULL, // function get initial value or NULL(show in line 2) + NULL, // function get default value or NULL(dont set value, show msg) + 0, // valueMin + 0, // valueMax + 0, // valueIncrement + "Calibrate Stick ", // title + " Calibrate ", // line1 (above value) + " Joystick ", // line2 (above value) + " click to start ", // line4 * (below value) + " sequence ", // line5 * + " ", // line6 + "=>long to cancel", // line7 }; @@ -71,8 +194,6 @@ void item_debugJoystick_action(display_task_parameters_t * objects, SSD1306_t * rotary_encoder_event_t event; //-- pre loop instructions -- - if (!value) // dont open menu when value was set to 0 - return; ESP_LOGW(TAG, "showing joystick debug page"); ssd1306_clear_screen(display, false); // show title @@ -99,36 +220,32 @@ void item_debugJoystick_action(display_task_parameters_t * objects, SSD1306_t * switch (event.type) { case RE_ET_BTN_CLICKED: + case RE_ET_BTN_LONG_PRESSED: running = false; break; case RE_ET_CHANGED: case RE_ET_BTN_PRESSED: case RE_ET_BTN_RELEASED: - case RE_ET_BTN_LONG_PRESSED: break; } } } } -int item_debugJoystick_value(display_task_parameters_t * objects){ - return 1; -} - menuItem_t item_debugJoystick = { item_debugJoystick_action, // function action - item_debugJoystick_value, // function get initial value - NULL, // function get default value or NULL + NULL, // function get initial value or NULL(show in line 2) + NULL, // function get default value or NULL(dont set value, show msg) 0, // valueMin - 1, // valueMAx - 1, // valueIncrement - "Debug joystick", // title - "Debug joystick", // line1 (above value) + 0, // valueMax + 0, // valueIncrement + "Debug joystick ", // title + "Debug joystick ", // line1 (above value) "", // line2 (above value) - "click to enter", // line4 * (below value) - "debug screen", // line5 * - "prints values", // line6 - "set 0 to cancel", // line7 + "", // line4 * (below value) + "debug screen ", // line5 * + "prints values ", // line6 + "=>long to cancel", // line7 }; @@ -147,12 +264,12 @@ int maxDuty_currentValue(display_task_parameters_t * objects) } menuItem_t item_maxDuty = { maxDuty_action, // function action - maxDuty_currentValue, // function get initial value - NULL, // function get default value or NULL + maxDuty_currentValue, // function get initial value or NULL(show in line 2) + NULL, // function get default value or NULL(dont set value, show msg) 1, // valueMin - 99, // valueMAx + 99, // valueMax 1, // valueIncrement - "max duty", // title + "max duty ", // title "", // line1 (above value) " set max-duty: ", // line2 (above value) "", // line4 * (below value) @@ -180,18 +297,18 @@ int item_accelLimit_default(display_task_parameters_t * objects) } menuItem_t item_accelLimit = { item_accelLimit_action, // function action - item_accelLimit_value, // function get initial value - item_accelLimit_default, // function get default value or NULL + item_accelLimit_value, // function get initial value or NULL(show in line 2) + item_accelLimit_default, // function get default value or NULL(dont set value, show msg) 0, // valueMin - 10000, // valueMAx + 10000, // valueMax 100, // valueIncrement - "Accel limit", // title - " Fade up time", // line1 (above value) + "Accel limit ", // title + " Fade up time ", // line1 (above value) "", // line2 <= showing "default = %d" "", // line4 * (below value) "", // line5 * - "milliseconds", // line6 - "from 0 to 100%", // line7 + "milliseconds ", // line6 + "from 0 to 100% ", // line7 }; @@ -213,18 +330,56 @@ int item_decelLimit_default(display_task_parameters_t * objects) } menuItem_t item_decelLimit = { item_decelLimit_action, // function action - item_decelLimit_value, // function get initial value - item_decelLimit_default, // function get default value or NULL + item_decelLimit_value, // function get initial value or NULL(show in line 2) + item_decelLimit_default, // function get default value or NULL(dont set value, show msg) 0, // valueMin - 10000, // valueMAx + 10000, // valueMax 100, // valueIncrement - "Decel limit", // title - " Fade down time", // line1 (above value) + "Decel limit ", // title + " Fade down time ", // line1 (above value) "", // line2 <= showing "default = %d" "", // line4 * (below value) "", // line5 * - "milliseconds", // line6 - "from 100 to 0%", // line7 + "milliseconds ", // line6 + "from 100 to 0% ", // line7 +}; + + +//##################### +//####### RESET ####### +//##################### +void item_reset_action(display_task_parameters_t *objects, SSD1306_t *display, int value) +{ + objects->buzzer->beep(1, 2000, 0); + // close and erase NVS + ESP_LOGW(TAG, "closing and ERASING non-volatile-storage..."); + nvs_close(*(objects->nvsHandle)); + ESP_ERROR_CHECK(nvs_flash_erase()); + // show message restarting + ssd1306_clear_screen(display, false); + displayTextLineCentered(display, 0, false, true, ""); + displayTextLineCentered(display, 1, true, true, "RE-"); + displayTextLineCentered(display, 4, true, true, "START"); + displayTextLineCentered(display, 7, false, true, ""); + vTaskDelay(1000 / portTICK_PERIOD_MS); // wait for buzzer to beep + // restart + ESP_LOGW(TAG, "RESTARTING"); + esp_restart(); +} +menuItem_t item_reset = { + item_reset_action, // function action + NULL, // function get initial value or NULL(show in line 2) + NULL, // function get default value or NULL(dont set value, show msg) + 0, // valueMin + 0, // valueMax + 0, // valueIncrement + "RESET defaults ", // title + " reset nvs ", // line1 (above value) + " and restart ", // line2 <= showing "default = %d" + "reset all stored", // line4 * (below value) + " parameters ", // line5 * + "", // line6 + "=>long to cancel", // line7 }; @@ -243,40 +398,42 @@ int item_example_valueDefault(display_task_parameters_t * objects){ } menuItem_t item_example = { item_example_action, // function action - item_example_value, // function get initial value - NULL, // function get default value or NULL + item_example_value, // function get initial value or NULL(show in line 2) + NULL, // function get default value or NULL(dont set value, show msg) -255, // valueMin - 255, // valueMAx + 255, // valueMax 2, // valueIncrement "example-item-max", // title - "line 1 - above", // line1 (above value) - "line 2 - above", // line2 (above value) - "line 4 - below", // line4 * (below value) - "line 5 - below", // line5 * - "line 6 - below", // line6 - "line 7 - last", // line7 + "line 1 - above ", // line1 (above value) + "line 2 - above ", // line2 (above value) + "line 4 - below ", // line4 * (below value) + "line 5 - below ", // line5 * + "line 6 - below ", // line6 + "line 7 - last ", // line7 }; menuItem_t item_last = { item_example_action, // function action - item_example_value, // function get initial value - item_example_valueDefault, // function get default value or NULL + item_example_value, // function get initial value or NULL(show in line 2) + item_example_valueDefault, // function get default value or NULL(dont set value, show msg) -500, // valueMin - 4500, // valueMAx + 4500, // valueMax 50, // valueIncrement "set large number", // title - "line 1 - above", // line1 (above value) - "line 2 - above", // line2 (above value) + "line 1 - above ", // line1 (above value) + "line 2 - above ", // line2 (above value) "", // line4 * (below value) "", // line5 * - "line 6 - below", // line6 - "line 7 - last", // line7 - + "line 6 - below ", // line6 + "line 7 - last ", // line7 }; -//store all configured menu items in one array -const menuItem_t menuItems[] = {item_centerJoystick, item_debugJoystick, item_accelLimit, item_decelLimit, item_example, item_last}; -const int itemCount = 6; + +//#################################################### +//### store all configured menu items in one array ### +//#################################################### +const menuItem_t menuItems[] = {item_centerJoystick, item_calibrateJoystick, item_debugJoystick, item_accelLimit, item_decelLimit, item_reset, item_example, item_last}; +const int itemCount = 8; @@ -322,12 +479,12 @@ void showItemList(SSD1306_t *display, int selectedItem) } } + //----------------------------- //--- showValueSelectStatic --- //----------------------------- -// function that renders value-select screen to display (one update) -// shows configured text of selected item and currently selected value -// TODO show previous value in one line? +// function that renders lines that do not update of value-select screen to display (initial update) +// shows configured text of currently selected item void showValueSelectStatic(display_task_parameters_t * objects, SSD1306_t *display, int selectedItem) { //-- show title line -- @@ -336,35 +493,46 @@ void showValueSelectStatic(display_task_parameters_t * objects, SSD1306_t *displ //-- show text above value -- displayTextLine(display, 1, false, false, "%-16s", menuItems[selectedItem].line1); -//-- show line 2 or default value --- + //-- show line 2 or default value --- if (menuItems[selectedItem].defaultValue != NULL){ - displayTextLineCentered(display, 2, false, false, "default = %d", menuItems[selectedItem].defaultValue(objects)); - } - else{ - //displayTextLine(display, 2, false, false, "previous=%d", menuItems[selectedItem].currentValue(objects)); // <= show previous value - displayTextLine(display, 2, false, false, "%-16s", menuItems[selectedItem].line2); -} + displayTextLineCentered(display, 2, false, false, "default = %d", menuItems[selectedItem].defaultValue(objects)); + } + else + { + // displayTextLine(display, 2, false, false, "previous=%d", menuItems[selectedItem].currentValue(objects)); // <= show previous value + displayTextLine(display, 2, false, false, "%-16s", menuItems[selectedItem].line2); + } //-- show value and other configured lines -- - // print value large, if 2 description lines are empty + // print value large, if two description lines are empty if (strlen(menuItems[selectedItem].line4) == 0 && strlen(menuItems[selectedItem].line5) == 0) { - // print large value + line5 and line6 - displayTextLineCentered(display, 3, true, false, "%d", value); //large centered + // print less lines: line5 and line6 only (due to large value) + //displayTextLineCentered(display, 3, true, false, "%d", value); //large centered (value shown in separate function) displayTextLine(display, 6, false, false, "%-16s", menuItems[selectedItem].line6); displayTextLine(display, 7, false, false, "%-16s", menuItems[selectedItem].line7); } else { - displayTextLineCentered(display, 3, false, false, "%d", value); //centered + //displayTextLineCentered(display, 3, false, false, "%d", value); //centered (value shown in separate function) // print description lines 4 to 7 displayTextLine(display, 4, false, false, "%-16s", menuItems[selectedItem].line4); displayTextLine(display, 5, false, false, "%-16s", menuItems[selectedItem].line5); displayTextLine(display, 6, false, false, "%-16s", menuItems[selectedItem].line6); displayTextLine(display, 7, false, false, "%-16s", menuItems[selectedItem].line7); } + + //-- show info msg instead of value -- + //when pointer to default value func not defined (set value not used, action only) + if (menuItems[selectedItem].currentValue == NULL) + { + //show static text + displayTextLineCentered(display, 3, false, true, "%s", "click to confirm"); + } + // otherwise value gets updated in next iteration of menu-handle function } + //----------------------------- //----- updateValueSelect ----- //----------------------------- @@ -374,30 +542,29 @@ void updateValueSelect(SSD1306_t *display, int selectedItem) // print value large, if 2 description lines are empty if (strlen(menuItems[selectedItem].line4) == 0 && strlen(menuItems[selectedItem].line5) == 0) { - // print large value + line5 and line6 - displayTextLineCentered(display, 3, true, false, "%d", value); //large centered + // print large and centered value in line 3-5 + displayTextLineCentered(display, 3, true, false, "%d", value); // large centered } else { - displayTextLineCentered(display, 3, false, false, "%d", value); //centered + //print value centered in line 3 + displayTextLineCentered(display, 3, false, false, "%d", value); // centered } } - //======================== //====== handleMenu ====== //======================== //controls menu with encoder input and displays the text on oled display //function is repeatedly called by display task when in menu state -#define QUEUE_TIMEOUT 3000 //timeout no encoder event - to handle timeout and not block the display loop +#define QUEUE_TIMEOUT 3000 //timeout no encoder event - to not block the display loop and actually handle menu-timeout #define MENU_TIMEOUT 60000 //inactivity timeout (switch to IDLE mode) note: should be smaller than IDLE timeout in control task void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) { static uint32_t lastActivity = 0; static int selectedItem = 0; - static bool staticContentUpdated = false; rotary_encoder_event_t event; // store event data //--- handle different menu states --- @@ -412,6 +579,7 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) // wait for encoder event if (xQueueReceive(objects->encoderQueue, &event, QUEUE_TIMEOUT / portTICK_PERIOD_MS)) { + // reset menu- and control-timeout on any encoder event lastActivity = esp_log_timestamp(); objects->control->resetTimeout(); switch (event.type) @@ -441,16 +609,20 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) ESP_LOGI(TAG, "Button pressed - switching to state SET_VALUE"); // change state (menu to set value) menuState = SET_VALUE; - staticContentUpdated = false; - // get currently configured value - value = menuItems[selectedItem].currentValue(objects); // clear display ssd1306_clear_screen(display, false); + //update static content of set-value screen once at change only + showValueSelectStatic(objects, display, selectedItem); + // get currently configured value, when value-select feature is actually used in this item + if (menuItems[selectedItem].currentValue != NULL) + value = menuItems[selectedItem].currentValue(objects); + else + value = 0; break; - //exit menu mode case RE_ET_BTN_LONG_PRESSED: - //change to previous mode (e.g. JOYSTICK) + //--- exit menu mode --- + // change to previous mode (e.g. JOYSTICK) objects->buzzer->beep(4, 15, 5); objects->control->toggleMode(controlMode_t::MENU); //currently already in MENU -> changes to previous mode ssd1306_clear_screen(display, false); @@ -467,18 +639,12 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) //---- State SET VALUE ---- //------------------------- case SET_VALUE: - // wait for encoder event - if (!staticContentUpdated) - { - showValueSelectStatic(objects, display, selectedItem); - staticContentUpdated = true; - } - else - { - // update line with currently set value only (increses performance significantly) + // update currently selected value + // note: static lines are updated at mode change + if (menuItems[selectedItem].currentValue != NULL) // dont update when set-value not used for this item updateValueSelect(display, selectedItem); - } + // wait for encoder event if (xQueueReceive(objects->encoderQueue, &event, QUEUE_TIMEOUT / portTICK_PERIOD_MS)) { objects->control->resetTimeout(); @@ -486,17 +652,21 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) { case RE_ET_CHANGED: //-- change value -- - objects->buzzer->beep(1, 25, 10); - // increment value - if (event.diff < 0) - value += menuItems[selectedItem].valueIncrement; - else - value -= menuItems[selectedItem].valueIncrement; - // limit to min/max range - if (value > menuItems[selectedItem].valueMax) - value = menuItems[selectedItem].valueMax; - if (value < menuItems[selectedItem].valueMin) - value = menuItems[selectedItem].valueMin; + // no need to increment value when item configured to not show value + if (menuItems[selectedItem].currentValue != NULL) + { + objects->buzzer->beep(1, 25, 10); + // increment value + if (event.diff < 0) + value += menuItems[selectedItem].valueIncrement; + else + value -= menuItems[selectedItem].valueIncrement; + // limit to min/max range + if (value > menuItems[selectedItem].valueMax) + value = menuItems[selectedItem].valueMax; + if (value < menuItems[selectedItem].valueMin) + value = menuItems[selectedItem].valueMin; + } break; case RE_ET_BTN_CLICKED: //-- apply value -- @@ -505,12 +675,19 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) menuItems[selectedItem].action(objects, display, value); menuState = MAIN_MENU; break; + case RE_ET_BTN_LONG_PRESSED: + //-- exit value select to main menu -- + objects->buzzer->beep(2, 100, 50); + ssd1306_clear_screen(display, false); + menuState = MAIN_MENU; + break; case RE_ET_BTN_PRESSED: case RE_ET_BTN_RELEASED: - case RE_ET_BTN_LONG_PRESSED: break; } + // reset menu- and control-timeout on any encoder event lastActivity = esp_log_timestamp(); + objects->control->resetTimeout(); } break; } @@ -519,10 +696,9 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) //-------------------- //--- menu timeout --- //-------------------- - //close menu and switch to IDLE mode when no encoder event within MENU_TIMEOUT + //close menu and switch to IDLE mode when no encoder event occured within MENU_TIMEOUT if (esp_log_timestamp() - lastActivity > MENU_TIMEOUT) { - objects->buzzer->beep(1, 500, 10); ESP_LOGW(TAG, "TIMEOUT - no activity for more than %ds -> closing menu, switching to IDLE", MENU_TIMEOUT/1000); // reset menu selectedItem = 0; diff --git a/common/joystick.cpp b/common/joystick.cpp index e9c54b5..eff703f 100644 --- a/common/joystick.cpp +++ b/common/joystick.cpp @@ -19,8 +19,9 @@ static const char * TAG_CMD = "joystickCommands"; //-------- constructor -------- //----------------------------- //copy provided struct with all configuration and run init function -evaluatedJoystick::evaluatedJoystick(joystick_config_t config_f){ +evaluatedJoystick::evaluatedJoystick(joystick_config_t config_f, nvs_handle_t * nvsHandle_f){ config = config_f; + nvsHandle = nvsHandle_f; init(); } @@ -30,7 +31,7 @@ evaluatedJoystick::evaluatedJoystick(joystick_config_t config_f){ //---------- init ------------ //---------------------------- void evaluatedJoystick::init(){ - ESP_LOGI(TAG, "initializing joystick"); + ESP_LOGW(TAG, "initializing ADC's and loading calibration..."); //initialize adc adc1_config_width(ADC_WIDTH_BIT_12); //=> max resolution 4096 @@ -41,6 +42,12 @@ void evaluatedJoystick::init(){ adc1_config_channel_atten(config.adc_x, ADC_ATTEN_DB_11); //max voltage adc1_config_channel_atten(config.adc_y, ADC_ATTEN_DB_11); //max voltage + //load stored calibration values (if not found loads defaults from config) + loadCalibration(X_MIN); + loadCalibration(X_MAX); + loadCalibration(Y_MIN); + loadCalibration(Y_MAX); + //define joystick center from current position defineCenter(); //define joystick center from current position } @@ -81,17 +88,17 @@ joystickData_t evaluatedJoystick::getData() { ESP_LOGV(TAG, "getting X coodrdinate..."); uint32_t adcRead; adcRead = readAdc(config.adc_x, config.x_inverted); - float x = scaleCoordinate(readAdc(config.adc_x, config.x_inverted), config.x_min, config.x_max, x_center, config.tolerance_zeroX_per, config.tolerance_end_per); + float x = scaleCoordinate(readAdc(config.adc_x, config.x_inverted), x_min, x_max, x_center, config.tolerance_zeroX_per, config.tolerance_end_per); data.x = x; ESP_LOGD(TAG, "X: adc-raw=%d \tadc-conv=%d \tmin=%d \t max=%d \tcenter=%d \tinverted=%d => x=%.3f", - adc1_get_raw(config.adc_x), adcRead, config.x_min, config.x_max, x_center, config.x_inverted, x); + adc1_get_raw(config.adc_x), adcRead, x_min, x_max, x_center, config.x_inverted, x); ESP_LOGV(TAG, "getting Y coodrinate..."); adcRead = readAdc(config.adc_y, config.y_inverted); - float y = scaleCoordinate(adcRead, config.y_min, config.y_max, y_center, config.tolerance_zeroY_per, config.tolerance_end_per); + float y = scaleCoordinate(adcRead, y_min, y_max, y_center, config.tolerance_zeroY_per, config.tolerance_end_per); data.y = y; ESP_LOGD(TAG, "Y: adc-raw=%d \tadc-conv=%d \tmin=%d \t max=%d \tcenter=%d \tinverted=%d => y=%.3lf", - adc1_get_raw(config.adc_y), adcRead, config.y_min, config.y_max, y_center, config.y_inverted, y); + adc1_get_raw(config.adc_y), adcRead, y_min, y_max, y_center, config.y_inverted, y); //calculate radius data.radius = sqrt(pow(data.x,2) + pow(data.y,2)); @@ -569,4 +576,124 @@ motorCommands_t joystick_generateCommandsShaking(joystickData_t data){ ESP_LOGI(TAG_CMD, "motor right: state=%s, duty=%.3f", motorstateStr[(int)commands.right.state], commands.right.duty); return commands; +} + + + + +// corresponding storage key strings to each joystickCalibratenMode variable +const char *calibrationStorageKeys[] = {"stick_x-min", "stick_x-max", "stick_y-min", "stick_y-max", "", ""}; + +//------------------------------- +//------- loadCalibration ------- +//------------------------------- +// loads selected calibration value from nvs or default values from config if no data stored +void evaluatedJoystick::loadCalibration(joystickCalibrationMode_t mode) +{ + // determine desired variables + int *configValue, *usedValue; + switch (mode) + { + case X_MIN: + configValue = &(config.x_min); + usedValue = &x_min; + break; + case X_MAX: + configValue = &(config.x_max); + usedValue = &x_max; + break; + case Y_MIN: + configValue = &(config.y_min); + usedValue = &y_min; + break; + case Y_MAX: + configValue = &(config.y_max); + usedValue = &y_max; + break; + case X_CENTER: + case Y_CENTER: + default: + // center position is not stored in nvs, it gets defined at startup or during calibration + ESP_LOGE(TAG, "loadCalibration: 'center_x' and 'center_y' are not stored in nvs -> not assigning anything"); + // defineCenter(); + return; + } + + // read from nvs + int16_t valueRead; + esp_err_t err = nvs_get_i16(*nvsHandle, calibrationStorageKeys[(int)mode], &valueRead); + switch (err) + { + case ESP_OK: + ESP_LOGW(TAG, "Successfully read value '%s' from nvs. Overriding default value %d with %d", calibrationStorageKeys[(int)mode], *configValue, valueRead); + *usedValue = (int)valueRead; + break; + case ESP_ERR_NVS_NOT_FOUND: + ESP_LOGW(TAG, "nvs: the value '%s' is not initialized yet, loading default value %d", calibrationStorageKeys[(int)mode], *configValue); + *usedValue = *configValue; + break; + default: + ESP_LOGE(TAG, "Error (%s) reading nvs!", esp_err_to_name(err)); + *usedValue = *configValue; + } +} + + + +//------------------------------- +//------- loadCalibration ------- +//------------------------------- +// loads selected calibration value from nvs or default values from config if no data stored +void evaluatedJoystick::writeCalibration(joystickCalibrationMode_t mode, int newValue) +{ + // determine desired variables + int *configValue, *usedValue; + switch (mode) + { + case X_MIN: + configValue = &(config.x_min); + usedValue = &x_min; + break; + case X_MAX: + configValue = &(config.x_max); + usedValue = &x_max; + break; + case Y_MIN: + configValue = &(config.y_min); + usedValue = &y_min; + break; + case Y_MAX: + configValue = &(config.y_max); + usedValue = &y_max; + break; + case X_CENTER: + x_center = newValue; + ESP_LOGW(TAG, "writeCalibration: 'center_x' or 'center_y' are not stored in nvs -> loading only"); + return; + case Y_CENTER: + y_center = newValue; + ESP_LOGW(TAG, "writeCalibration: 'center_x' or 'center_y' are not stored in nvs -> loading only"); + default: + return; + } + + // check if unchanged + if (*usedValue == newValue) + { + ESP_LOGW(TAG, "writeCalibration: value '%s' unchanged at %d, not writing to nvs", calibrationStorageKeys[(int)mode], newValue); + return; + } + + // update nvs value + ESP_LOGW(TAG, "writeCalibration: updating nvs value '%s' from %d to %d", calibrationStorageKeys[(int)mode], *usedValue, newValue); + esp_err_t err = nvs_set_i16(*nvsHandle, calibrationStorageKeys[(int)mode], newValue); + if (err != ESP_OK) + ESP_LOGE(TAG, "nvs: failed writing"); + err = nvs_commit(*nvsHandle); + if (err != ESP_OK) + ESP_LOGE(TAG, "nvs: failed committing updates"); + else + ESP_LOGI(TAG, "nvs: successfully committed updates"); + // update variable + *usedValue = newValue; } \ No newline at end of file diff --git a/common/joystick.hpp b/common/joystick.hpp index ccf5bec..9dd86fb 100644 --- a/common/joystick.hpp +++ b/common/joystick.hpp @@ -8,6 +8,8 @@ extern "C" #include "driver/adc.h" #include "esp_log.h" #include "esp_err.h" +#include "nvs_flash.h" +#include "nvs.h" } #include @@ -55,6 +57,7 @@ typedef struct joystick_config_t { enum class joystickPos_t {CENTER, Y_AXIS, X_AXIS, TOP_RIGHT, TOP_LEFT, BOTTOM_LEFT, BOTTOM_RIGHT}; extern const char* joystickPosStr[7]; +typedef enum joystickCalibrationMode_t { X_MIN = 0, X_MAX, Y_MIN, Y_MAX, X_CENTER, Y_CENTER } joystickCalibrationMode_t; //struct with current data of the joystick typedef struct joystickData_t { @@ -70,31 +73,45 @@ typedef struct joystickData_t { //------------------------------------ //----- evaluatedJoystick class ----- //------------------------------------ -class evaluatedJoystick { - public: - //--- constructor --- - evaluatedJoystick(joystick_config_t config_f); +class evaluatedJoystick +{ +public: + //--- constructor --- + evaluatedJoystick(joystick_config_t config_f, nvs_handle_t * nvsHandle); - //--- functions --- - joystickData_t getData(); //read joystick, calculate values and return the data in a struct - void defineCenter(); //define joystick center from current position + //--- functions --- + joystickData_t getData(); // read joystick, calculate values and return the data in a struct + // get raw adc value (inversion applied) + int getRawX() { return readAdc(config.adc_x, config.x_inverted); } + int getRawY() { return readAdc(config.adc_y, config.y_inverted); } + void defineCenter(); // define joystick center from current position + void writeCalibration(joystickCalibrationMode_t mode, int newValue); // load certain new calibration value and store it in nvs - private: - //--- functions --- - //initialize adc inputs, define center - void init(); - //read adc while making multiple samples with option to invert the result - int readAdc(adc1_channel_t adc_channel, bool inverted = false); +private: + //--- functions --- + // initialize adc inputs, define center + void init(); + // loads selected calibration value from nvs or default values from config if no data stored + void loadCalibration(joystickCalibrationMode_t mode); + // read adc while making multiple samples with option to invert the result + int readAdc(adc1_channel_t adc_channel, bool inverted = false); //--- variables --- + // handle for using the nvs flash (persistent config variables) + nvs_handle_t *nvsHandle; joystick_config_t config; + + int x_min; + int x_max; + int y_min; + int y_max; int x_center; int y_center; joystickData_t data; float x; float y; -}; + }; From 0672d08cb85f4d4dbef6c83b6782ca5be062aa0c Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Wed, 21 Feb 2024 14:23:07 +0100 Subject: [PATCH 4/5] Adjust beeping Tested on actual hardware. - adjust count and duration of beeps --- board_single/main/button.cpp | 6 +++--- board_single/main/control.cpp | 2 +- board_single/main/main.cpp | 4 ++-- board_single/main/menu.cpp | 16 +++++++++++----- common/buzzer.cpp | 2 +- 5 files changed, 18 insertions(+), 12 deletions(-) diff --git a/board_single/main/button.cpp b/board_single/main/button.cpp index 40962ac..6b7e618 100644 --- a/board_single/main/button.cpp +++ b/board_single/main/button.cpp @@ -67,7 +67,7 @@ void buttonCommands::action (uint8_t count, bool lastPressLong){ // ## no command ## default: ESP_LOGE(TAG, "no command for count=%d and long=%d defined", count, lastPressLong); - buzzer->beep(3, 400, 100); + buzzer->beep(3, 200, 100); break; case 1: @@ -76,7 +76,7 @@ void buttonCommands::action (uint8_t count, bool lastPressLong){ { control->changeMode(controlMode_t::MENU); ESP_LOGW(TAG, "1x long press -> change to menu mode"); - buzzer->beep(1, 1000, 1); + buzzer->beep(20, 20, 10); vTaskDelay(500 / portTICK_PERIOD_MS); } // ## toggle joystick freeze ## @@ -214,4 +214,4 @@ void buttonCommands::startHandleLoop() } } //end queue } //end while(1) -} //end function \ No newline at end of file +} //end function diff --git a/board_single/main/control.cpp b/board_single/main/control.cpp index e32c448..3d36346 100644 --- a/board_single/main/control.cpp +++ b/board_single/main/control.cpp @@ -419,7 +419,7 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { break; case controlMode_t::IDLE: - buzzer->beep(1, 1500, 0); + buzzer->beep(1, 1000, 0); #ifdef JOYSTICK_LOG_IN_IDLE esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG); #endif diff --git a/board_single/main/main.cpp b/board_single/main/main.cpp index 1640635..afdc31c 100644 --- a/board_single/main/main.cpp +++ b/board_single/main/main.cpp @@ -159,8 +159,8 @@ void createObjects() httpJoystickMain = new httpJoystick(configHttpJoystickMain); http_init_server(on_joystick_url); - // create buzzer object on pin 12 with gap between queued events of 100ms - buzzer = new buzzer_t(GPIO_NUM_12, 100); + // create buzzer object on pin 12 with gap between queued events of 1ms + buzzer = new buzzer_t(GPIO_NUM_12, 1); // create objects for controlling the chair position // gpio_up, gpio_down, name diff --git a/board_single/main/menu.cpp b/board_single/main/menu.cpp index 6b7fd20..2dad775 100644 --- a/board_single/main/menu.cpp +++ b/board_single/main/menu.cpp @@ -222,6 +222,7 @@ void item_debugJoystick_action(display_task_parameters_t * objects, SSD1306_t * case RE_ET_BTN_CLICKED: case RE_ET_BTN_LONG_PRESSED: running = false; + objects->buzzer->beep(1, 100, 10); break; case RE_ET_CHANGED: case RE_ET_BTN_PRESSED: @@ -586,19 +587,24 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) { case RE_ET_CHANGED: //--- scroll in list --- - objects->buzzer->beep(1, 10, 5); if (event.diff < 0) { if (selectedItem != itemCount - 1) + { + objects->buzzer->beep(1, 20, 0); selectedItem++; - ESP_LOGD(TAG, "showing next item: %d '%s'", selectedItem, menuItems[selectedItem].title); + ESP_LOGD(TAG, "showing next item: %d '%s'", selectedItem, menuItems[selectedItem].title); + } //note: display will update at start of next run } else { if (selectedItem != 0) + { + objects->buzzer->beep(1, 20, 0); selectedItem--; - ESP_LOGD(TAG, "showing previous item: %d '%s'", selectedItem, menuItems[selectedItem].title); + ESP_LOGD(TAG, "showing previous item: %d '%s'", selectedItem, menuItems[selectedItem].title); + } //note: display will update at start of next run } break; @@ -623,7 +629,7 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) case RE_ET_BTN_LONG_PRESSED: //--- exit menu mode --- // change to previous mode (e.g. JOYSTICK) - objects->buzzer->beep(4, 15, 5); + objects->buzzer->beep(12, 15, 8); objects->control->toggleMode(controlMode_t::MENU); //currently already in MENU -> changes to previous mode ssd1306_clear_screen(display, false); break; @@ -708,4 +714,4 @@ void handleMenu(display_task_parameters_t * objects, SSD1306_t *display) objects->control->changeMode(controlMode_t::IDLE); return; } -} \ No newline at end of file +} diff --git a/common/buzzer.cpp b/common/buzzer.cpp index e320885..a80ec75 100644 --- a/common/buzzer.cpp +++ b/common/buzzer.cpp @@ -82,7 +82,7 @@ void buzzer_t::processQueue(){ // otherwise waits for at least 7 weeks if( xQueueReceive( beepQueue, &entryRead, portMAX_DELAY ) ) { - ESP_LOGW(TAG_BUZZER, "Read entry from queue: count=%d, msOn=%d, msOff=%d", entryRead.count, entryRead.msOn, entryRead.msOff); + ESP_LOGI(TAG_BUZZER, "Read entry from queue: count=%d, msOn=%d, msOff=%d", entryRead.count, entryRead.msOn, entryRead.msOff); //beep requested count with requested delays for (int i = entryRead.count; i--;){ From cef2a841c87bd5190e44dddba217fb977f5b599d Mon Sep 17 00:00:00 2001 From: jonny_jr9 Date: Thu, 22 Feb 2024 23:59:13 +0100 Subject: [PATCH 5/5] Add menu item Set Max Duty Add nvs to control.cpp --- board_single/main/config.cpp | 11 ++++++ board_single/main/control.cpp | 72 ++++++++++++++++++++++++++++++++--- board_single/main/control.hpp | 22 +++++++++-- board_single/main/main.cpp | 2 +- board_single/main/menu.cpp | 16 ++++---- common/joystick.cpp | 8 ++-- common/joystick.hpp | 10 ++++- 7 files changed, 116 insertions(+), 25 deletions(-) diff --git a/board_single/main/config.cpp b/board_single/main/config.cpp index 7498e57..08ce372 100644 --- a/board_single/main/config.cpp +++ b/board_single/main/config.cpp @@ -220,4 +220,15 @@ rotary_encoder_t encoder_config = { .index = 0, .btn_pressed_time_us = 20000, .btn_state = RE_BTN_RELEASED //default state +}; + + +//----------------------------------- +//--- joystick command generation --- +//----------------------------------- +//configure parameters for motor command generation from joystick data +joystickGenerateCommands_config_t joystickGenerateCommands_config{ + .maxDuty = 100, + .dutyOffset = 5, // duty at which motors start immediately + .altStickMapping = false, }; \ No newline at end of file diff --git a/board_single/main/control.cpp b/board_single/main/control.cpp index 3d36346..f58da0c 100644 --- a/board_single/main/control.cpp +++ b/board_single/main/control.cpp @@ -32,14 +32,17 @@ controlledArmchair::controlledArmchair( controlledMotor *motorLeft_f, controlledMotor *motorRight_f, evaluatedJoystick *joystick_f, + joystickGenerateCommands_config_t *joystickGenerateCommands_config_f, httpJoystick *httpJoystick_f, automatedArmchair_c *automatedArmchair_f, cControlledRest *legRest_f, - cControlledRest *backRest_f) + cControlledRest *backRest_f, + nvs_handle_t * nvsHandle_f) { //copy configuration config = config_f; + joystickGenerateCommands_config = *joystickGenerateCommands_config_f; //copy object pointers buzzer = buzzer_f; motorLeft = motorLeft_f; @@ -49,9 +52,13 @@ controlledArmchair::controlledArmchair( automatedArmchair = automatedArmchair_f; legRest = legRest_f; backRest = backRest_f; + nvsHandle = nvsHandle_f; //set default mode from config modePrevious = config.defaultMode; + // override default config value if maxDuty is found in nvs + loadMaxDuty(); + //TODO declare / configure controlled motors here instead of config (unnecessary that button object is globally available - only used here)? } @@ -104,7 +111,7 @@ void controlledArmchair::startHandleLoop() { //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, altStickMapping); + commands = joystick_generateCommandsDriving(stickData, &joystickGenerateCommands_config); //apply motor commands motorRight->setTarget(commands.right.state, commands.right.duty); motorLeft->setTarget(commands.left.state, commands.left.duty); @@ -138,7 +145,7 @@ void controlledArmchair::startHandleLoop() { 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 - commands = joystick_generateCommandsDriving(stickData, altStickMapping); + commands = joystick_generateCommandsDriving(stickData, &joystickGenerateCommands_config); //--- apply commands to motors --- //TODO make motorctl.setTarget also accept motorcommand struct directly @@ -268,8 +275,8 @@ bool controlledArmchair::toggleFreezeInputMassage() // toggle between normal and alternative stick mapping (joystick reverse position inverted) bool controlledArmchair::toggleAltStickMapping() { - altStickMapping = !altStickMapping; - if (altStickMapping) + joystickGenerateCommands_config.altStickMapping = !joystickGenerateCommands_config.altStickMapping; + if (joystickGenerateCommands_config.altStickMapping) { buzzer->beep(6, 70, 50); ESP_LOGW(TAG, "changed to alternative stick mapping"); @@ -279,7 +286,7 @@ bool controlledArmchair::toggleAltStickMapping() buzzer->beep(1, 500, 100); ESP_LOGW(TAG, "changed to default stick mapping"); } - return altStickMapping; + return joystickGenerateCommands_config.altStickMapping; } @@ -503,3 +510,56 @@ void controlledArmchair::toggleMode(controlMode_t modePrimary){ changeMode(modePrimary); } } + + + + +//------------------------------- +//------ loadDecelDuration ------ +//------------------------------- +// update local config value when maxDuty is stored in nvs +void controlledArmchair::loadMaxDuty(void) +{ + // default value is already loaded (constructor) + // read from nvs + uint16_t valueRead; + esp_err_t err = nvs_get_u16(*nvsHandle, "c-maxDuty", &valueRead); + switch (err) + { + case ESP_OK: + ESP_LOGW(TAG, "Successfully read value '%s' from nvs. Overriding default value %.2f with %.2f", "c-maxDuty", joystickGenerateCommands_config.maxDuty, valueRead/100.0); + joystickGenerateCommands_config.maxDuty = (float)(valueRead/100.0); + break; + case ESP_ERR_NVS_NOT_FOUND: + ESP_LOGW(TAG, "nvs: the value '%s' is not initialized yet, keeping default value %.2f", "c-maxDuty", joystickGenerateCommands_config.maxDuty); + break; + default: + ESP_LOGE(TAG, "Error (%s) reading nvs!", esp_err_to_name(err)); + } +} + + +//----------------------------------- +//---------- writeMaxDuty ----------- +//----------------------------------- +// write provided value to nvs to be persistent and update local variable in joystickGenerateCommmands_config struct +// note: duty percentage gets stored as uint with factor 100 (to get more precision) +void controlledArmchair::writeMaxDuty(float newValue){ + // check if unchanged + if(joystickGenerateCommands_config.maxDuty == newValue){ + ESP_LOGW(TAG, "value unchanged at %.2f, not writing to nvs", newValue); + return; + } + // update nvs value + ESP_LOGW(TAG, "updating nvs value '%s' from %.2f to %.2f", "c-maxDuty", joystickGenerateCommands_config.maxDuty, newValue) ; + esp_err_t err = nvs_set_u16(*nvsHandle, "c-maxDuty", (uint16_t)(newValue*100)); + if (err != ESP_OK) + ESP_LOGE(TAG, "nvs: failed writing"); + err = nvs_commit(*nvsHandle); + if (err != ESP_OK) + ESP_LOGE(TAG, "nvs: failed committing updates"); + else + ESP_LOGI(TAG, "nvs: successfully committed updates"); + // update variable + joystickGenerateCommands_config.maxDuty = newValue; +} \ No newline at end of file diff --git a/board_single/main/control.hpp b/board_single/main/control.hpp index 1ae8e3f..7e7351d 100644 --- a/board_single/main/control.hpp +++ b/board_single/main/control.hpp @@ -1,5 +1,10 @@ #pragma once +extern "C" +{ +#include "nvs_flash.h" +#include "nvs.h" +} #include "motordrivers.hpp" #include "motorctl.hpp" #include "buzzer.hpp" @@ -50,10 +55,12 @@ class controlledArmchair { controlledMotor* motorLeft_f, controlledMotor* motorRight_f, evaluatedJoystick* joystick_f, + joystickGenerateCommands_config_t* joystickGenerateCommands_config_f, httpJoystick* httpJoystick_f, automatedArmchair_c* automatedArmchair, cControlledRest * legRest, - cControlledRest * backRest + cControlledRest * backRest, + nvs_handle_t * nvsHandle_f ); //--- functions --- @@ -79,27 +86,37 @@ class controlledArmchair { controlMode_t getCurrentMode() const {return mode;}; const char *getCurrentModeStr() const { return controlModeStr[(int)mode]; }; + //--- mode specific --- // releases or locks joystick in place when in massage mode, returns true when input is frozen bool toggleFreezeInputMassage(); - // toggle between normal and alternative stick mapping (joystick reverse position inverted), returns true when alt mapping is active bool toggleAltStickMapping(); + // configure max dutycycle (in joystick or http mode) + void setMaxDuty(float maxDutyNew) { writeMaxDuty(maxDutyNew); }; + float getMaxDuty() const {return joystickGenerateCommands_config.maxDuty; }; + private: //--- functions --- //function that evaluates whether there is no activity/change on the motor duty for a certain time, if so a switch to IDLE is issued. - has to be run repeatedly in a slow interval void handleTimeout(); + void loadMaxDuty(); //load stored value for maxDuty from nvs + void writeMaxDuty(float newMaxDuty); //write new value for maxDuty to nvs + //--- objects --- buzzer_t* buzzer; controlledMotor* motorLeft; controlledMotor* motorRight; httpJoystick* httpJoystickMain_l; evaluatedJoystick* joystick_l; + joystickGenerateCommands_config_t joystickGenerateCommands_config; automatedArmchair_c *automatedArmchair; cControlledRest * legRest; cControlledRest * backRest; + //handle for using the nvs flash (persistent config variables) + nvs_handle_t * nvsHandle; //---variables --- //struct for motor commands returned by generate functions of each mode @@ -109,7 +126,6 @@ class controlledArmchair { //store joystick data joystickData_t stickData; - bool altStickMapping; //alternative joystick mapping (reverse mapped differently) //variables for http mode uint32_t http_timestamp_lastData = 0; diff --git a/board_single/main/main.cpp b/board_single/main/main.cpp index afdc31c..50334c8 100644 --- a/board_single/main/main.cpp +++ b/board_single/main/main.cpp @@ -169,7 +169,7 @@ void createObjects() // create control object (control.hpp) // with configuration from config.cpp - control = new controlledArmchair(configControl, buzzer, motorLeft, motorRight, joystick, httpJoystickMain, automatedArmchair, legRest, backRest); + control = new controlledArmchair(configControl, buzzer, motorLeft, motorRight, joystick, &joystickGenerateCommands_config, httpJoystickMain, automatedArmchair, legRest, backRest, &nvsHandle); // create automatedArmchair_c object (for auto-mode) (auto.hpp) automatedArmchair = new automatedArmchair_c(motorLeft, motorRight); diff --git a/board_single/main/menu.cpp b/board_single/main/menu.cpp index 2dad775..937f5b2 100644 --- a/board_single/main/menu.cpp +++ b/board_single/main/menu.cpp @@ -255,27 +255,25 @@ menuItem_t item_debugJoystick = { //######################## void maxDuty_action(display_task_parameters_t * objects, SSD1306_t * display, int value) { - //TODO actually store the value - ESP_LOGW(TAG, "set max duty to %d", value); + objects->control->setMaxDuty(value); } int maxDuty_currentValue(display_task_parameters_t * objects) { - //TODO get real current value - return 84; + return (int)objects->control->getMaxDuty(); } menuItem_t item_maxDuty = { maxDuty_action, // function action maxDuty_currentValue, // function get initial value or NULL(show in line 2) NULL, // function get default value or NULL(dont set value, show msg) 1, // valueMin - 99, // valueMax + 100, // valueMax 1, // valueIncrement - "max duty ", // title + "Set max Duty ", // title "", // line1 (above value) " set max-duty: ", // line2 (above value) "", // line4 * (below value) "", // line5 * - " 1-99 ", // line6 + " 1-100 ", // line6 " percent ", // line7 }; @@ -433,8 +431,8 @@ menuItem_t item_last = { //#################################################### //### store all configured menu items in one array ### //#################################################### -const menuItem_t menuItems[] = {item_centerJoystick, item_calibrateJoystick, item_debugJoystick, item_accelLimit, item_decelLimit, item_reset, item_example, item_last}; -const int itemCount = 8; +const menuItem_t menuItems[] = {item_centerJoystick, item_calibrateJoystick, item_debugJoystick, item_maxDuty, item_accelLimit, item_decelLimit, item_reset, item_example, item_last}; +const int itemCount = 9; diff --git a/common/joystick.cpp b/common/joystick.cpp index eff703f..a22ebbe 100644 --- a/common/joystick.cpp +++ b/common/joystick.cpp @@ -304,7 +304,7 @@ joystickPos_t joystick_evaluatePosition(float x, float y){ //========= joystick_CommandsDriving ========= //============================================ //function that generates commands for both motors from the joystick data -motorCommands_t joystick_generateCommandsDriving(joystickData_t data, bool altStickMapping){ +motorCommands_t joystick_generateCommandsDriving(joystickData_t data, joystickGenerateCommands_config_t * config){ //struct with current data of the joystick //typedef struct joystickData_t { @@ -317,10 +317,8 @@ motorCommands_t joystick_generateCommandsDriving(joystickData_t data, bool altSt //--- variables --- motorCommands_t commands; - float dutyMax = 100; //TODO add this to config, make changeable during runtime - float dutyOffset = 5; //immediately starts with this duty, TODO add this to config - float dutyRange = dutyMax - dutyOffset; + float dutyRange = config->maxDuty - config->dutyOffset; float ratio = fabs(data.angle) / 90; //90degree = x=0 || 0degree = y=0 //--- snap ratio to max at angle threshold --- @@ -334,7 +332,7 @@ motorCommands_t joystick_generateCommandsDriving(joystickData_t data, bool altSt */ //--- experimental alternative control mode --- - if (altStickMapping == true){ + if (config->altStickMapping == true){ //swap BOTTOM_LEFT and BOTTOM_RIGHT if (data.position == joystickPos_t::BOTTOM_LEFT){ data.position = joystickPos_t::BOTTOM_RIGHT; diff --git a/common/joystick.hpp b/common/joystick.hpp index 9dd86fb..af6a06d 100644 --- a/common/joystick.hpp +++ b/common/joystick.hpp @@ -10,6 +10,7 @@ extern "C" #include "esp_err.h" #include "nvs_flash.h" #include "nvs.h" +#include } #include @@ -69,6 +70,13 @@ typedef struct joystickData_t { } joystickData_t; +// struct with parameters provided to joystick_GenerateCommandsDriving() +typedef struct joystickGenerateCommands_config_t { + float maxDuty; + float dutyOffset; + bool altStickMapping; +} joystickGenerateCommands_config_t; + //------------------------------------ //----- evaluatedJoystick class ----- @@ -120,7 +128,7 @@ private: //============================================ //function that generates commands for both motors from the joystick data //motorCommands_t joystick_generateCommandsDriving(evaluatedJoystick joystick); -motorCommands_t joystick_generateCommandsDriving(joystickData_t data, bool altStickMapping = false); +motorCommands_t joystick_generateCommandsDriving(joystickData_t data, joystickGenerateCommands_config_t * config);