diff --git a/README.md b/README.md index d91fbe0..62aefc0 100644 --- a/README.md +++ b/README.md @@ -2,9 +2,10 @@ Firmware for a homemade automated electric armchair. More details about this project: https://pfusch.zone/electric-armchair + # Installation ### Install esp-idf -For this project **ESP-IDF v4.4.1** is required (with other versions it might not compile) +For this project **ESP-IDF v4.4.4** is required (with other versions it might not compile) ```bash #download esp-idf yay -S esp-idf #alternatively clone the esp-idf repository from github @@ -93,6 +94,16 @@ A diagram which shows what components are connected to which terminals of the pc - Self driving algorithm - Lights - drinks holder +- improved webinterface + + + +# Todo +**Add switch functions** +- set loglevel +- define max-speed +- calibrate joystick (min, max, center) +- testing mode / dry-run @@ -105,7 +116,7 @@ A diagram which shows what components are connected to which terminals of the pc | 1x | control | [MASSAGE] **freeze** input | when in massage mode: lock or unlock joystick input at current position | | 2x | toggle mode | **IDLE** <=> previous | enable/disable chair armchair e.g. enable after startup or timeout | | 3x | switch mode | **JOYSTICK** | switch to default mode JOYSTICK | -| 4x | toggle mode | **HTTP** <=> JOYSTICK | switch to '**remote control** via web-app' or back to JOYSTICK mode | +| 4x | toggle mode | **HTTP** <=> JOYSTICK | switch to '**remote control** via web-app `http://191.168.4.1`' or back to JOYSTICK mode | | 5x | | | | | 6x | toggle mode | **MASSAGE** <=> JOYSTICK | switch to MASSAGE mode or back to JOYSTICK mode | | 7x | | | | @@ -113,16 +124,20 @@ A diagram which shows what components are connected to which terminals of the pc | | | | | | 12x | toggle option | **alt stick mapping** | toggle between default and alternative stick mapping (reverse swapped) | | >1s | system | **restart** | Restart the controller when pressing the button longer than 1 second | +| 1x short, 1x long | auto command | **eject** foot support | automatically go forward and reverse for certain time with no acceleration limits, so foot support ejects | +## HTTP mode +Control armchair via virtual joystick on a webinterface. -**previous functions - not implemented** -| Count | Action | -| --- | ---| -| 1 | define joystick center | -| 2 | toggle motors | -| 3 | toggle log-level (WARN, DEBUG, INFO) | -| 4 | define max duty | -| 5 | toggle mode MQTT/JOYSTICK | -| 6 | toggle mode SHAKE/JOYSTICK | -| 7 | toggle testing-mode (dry-run) | +**Usage** +- Connect to wifi `armchar`, no password +- Access http://192.168.4.1 (note: **http** NOT https, some browsers automatically add https!) + +**Current Features** +- Control direction and speed with joystick + +**Todo** +- Set parameters +- Control other modes +- Execute preset movement commands diff --git a/components/gpio/gpio_evaluateSwitch.hpp b/components/gpio/gpio_evaluateSwitch.hpp index 23caa14..c4a27e4 100644 --- a/components/gpio/gpio_evaluateSwitch.hpp +++ b/components/gpio/gpio_evaluateSwitch.hpp @@ -22,8 +22,8 @@ extern "C" class gpio_evaluatedSwitch { public: //--- input --- - uint32_t minOnMs = 50; - uint32_t minOffMs = 50; + uint32_t minOnMs = 30; + uint32_t minOffMs = 30; gpio_evaluatedSwitch( //constructor minimal (default parameters pullup=true, inverted=false) gpio_num_t gpio_num_declare ); diff --git a/connection-plan.drawio.pdf b/connection-plan.drawio.pdf index 82ce990..4a430bc 100644 Binary files a/connection-plan.drawio.pdf and b/connection-plan.drawio.pdf differ diff --git a/main/CMakeLists.txt b/main/CMakeLists.txt index 6ea40c3..34de854 100644 --- a/main/CMakeLists.txt +++ b/main/CMakeLists.txt @@ -11,6 +11,7 @@ idf_component_register( "fan.cpp" "wifi.c" "http.cpp" + "auto.cpp" INCLUDE_DIRS "." ) diff --git a/main/auto.cpp b/main/auto.cpp new file mode 100644 index 0000000..4cf4a71 --- /dev/null +++ b/main/auto.cpp @@ -0,0 +1,88 @@ +#include "auto.hpp" +#include "config.hpp" + +//tag for logging +static const char * TAG = "automatedArmchair"; + + +//============================= +//======== constructor ======== +//============================= +automatedArmchair::automatedArmchair(void) { + //create command queue + commandQueue = xQueueCreate( 32, sizeof( commandSimple_t ) ); //TODO add max size to config? +} + + + +//============================== +//====== generateCommands ====== +//============================== +motorCommands_t automatedArmchair::generateCommands(auto_instruction_t * instruction) { + //reset instruction + *instruction = auto_instruction_t::NONE; + //check if previous command is finished + if ( esp_log_timestamp() > timestampCmdFinished ) { + //get next command from queue + if( xQueueReceive( commandQueue, &cmdCurrent, pdMS_TO_TICKS(500) ) ) { + ESP_LOGI(TAG, "running next command from queue..."); + //copy instruction to be provided to control task + *instruction = cmdCurrent.instruction; + //set acceleration / fading parameters according to command + motorLeft.setFade(fadeType_t::DECEL, cmdCurrent.fadeDecel); + motorRight.setFade(fadeType_t::DECEL, cmdCurrent.fadeDecel); + motorLeft.setFade(fadeType_t::ACCEL, cmdCurrent.fadeAccel); + motorRight.setFade(fadeType_t::ACCEL, cmdCurrent.fadeAccel); + //calculate timestamp the command is finished + timestampCmdFinished = esp_log_timestamp() + cmdCurrent.msDuration; + //copy the new commands + motorCommands = cmdCurrent.motorCmds; + } else { //queue empty + ESP_LOGD(TAG, "no new command in queue -> set motors to IDLE"); + motorCommands = motorCmds_bothMotorsIdle; + } + } else { //previous command still running + ESP_LOGD(TAG, "command still running -> no change"); + } + + //TODO also return instructions via call by reference + return motorCommands; +} + + + +//============================ +//======== addCommand ======== +//============================ +//function that adds a basic command to the queue +void automatedArmchair::addCommand(commandSimple_t command) { + //add command to queue + if ( xQueueSend( commandQueue, ( void * )&command, ( TickType_t ) 0 ) ){ + ESP_LOGI(TAG, "Successfully inserted command to queue"); + } else { + ESP_LOGE(TAG, "Failed to insert new command to queue"); + } +} + +void automatedArmchair::addCommands(commandSimple_t commands[], size_t count) { + for (int i = 0; i < count; i++) { + ESP_LOGI(TAG, "Reading command no. %d from provided array", i); + addCommand(commands[i]); + } +} + + +//=============================== +//======== clearCommands ======== +//=============================== +//function that deletes all pending/queued commands +//e.g. when switching modes +motorCommands_t automatedArmchair::clearCommands() { + //clear command queue + xQueueReset( commandQueue ); + ESP_LOGW(TAG, "command queue was successfully emptied"); + //return commands for idling both motors + motorCommands = motorCmds_bothMotorsIdle; + return motorCmds_bothMotorsIdle; +} + diff --git a/main/auto.hpp b/main/auto.hpp new file mode 100644 index 0000000..8b799fb --- /dev/null +++ b/main/auto.hpp @@ -0,0 +1,87 @@ +#pragma once + +extern "C" +{ +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "esp_log.h" +#include "esp_err.h" +} + +#include "freertos/queue.h" +#include +#include "motorctl.hpp" + + + +//-------------------------------------------- +//---- struct, enum, variable declarations --- +//-------------------------------------------- +//enum for special instructions / commands to be run in control task +enum class auto_instruction_t { NONE, SWITCH_PREV_MODE, SWITCH_JOYSTICK_MODE, RESET_ACCEL_DECEL, RESET_ACCEL, RESET_DECEL }; + +//struct for a simple command +//e.g. put motors in a certain state for certain time +typedef struct commandSimple_t{ + motorCommands_t motorCmds; + uint32_t msDuration; + uint32_t fadeDecel; + uint32_t fadeAccel; + auto_instruction_t instruction; +} commandSimple_t; + + + +//------------------------------------ +//----- automatedArmchair class ----- +//------------------------------------ +class automatedArmchair { + public: + //--- methods --- + //constructor + automatedArmchair(void); + //function to generate motor commands + //can be also seen as handle function + //TODO: go with other approach: separate task for handling auto mode + // - receive commands with queue anyways + // - => use delay function + // - have a queue that outputs current motor state/commands -> repeatedly check the queue in control task + //function that handles automatic driving and returns motor commands + //also provides instructions to be executed in control task via pointer + motorCommands_t generateCommands(auto_instruction_t * instruction); + + //function that adds a basic command to the queue + void addCommand(commandSimple_t command); + //function that adds an array of basic commands to queue + void addCommands(commandSimple_t commands[], size_t count); + + //function that deletes all pending/queued commands + motorCommands_t clearCommands(); + + + private: + //--- methods --- + //--- objects --- + //TODO: add buzzer here + //--- variables --- + //queue for storing pending commands + QueueHandle_t commandQueue = NULL; + //current running command + commandSimple_t cmdCurrent; + //timestamp current command is finished + uint32_t timestampCmdFinished = 0; + + motorCommands_t motorCommands; + + //command preset for idling motors + const motorCommand_t motorCmd_motorIdle = { + .state = motorstate_t::IDLE, + .duty = 0 + }; + const motorCommands_t motorCmds_bothMotorsIdle = { + .left = motorCmd_motorIdle, + .right = motorCmd_motorIdle + }; +}; + + diff --git a/main/button.cpp b/main/button.cpp index b717328..1762e6e 100644 --- a/main/button.cpp +++ b/main/button.cpp @@ -19,9 +19,10 @@ static const char * TAG = "button"; //----------------------------- //-------- constructor -------- //----------------------------- -buttonCommands::buttonCommands(gpio_evaluatedSwitch * button_f, controlledArmchair * control_f, buzzer_t * buzzer_f, controlledMotor * motorLeft_f, controlledMotor * motorRight_f){ +buttonCommands::buttonCommands(gpio_evaluatedSwitch * button_f, evaluatedJoystick * joystick_f, controlledArmchair * control_f, buzzer_t * buzzer_f, controlledMotor * motorLeft_f, controlledMotor * motorRight_f){ //copy object pointers button = button_f; + joystick = joystick_f; control = control_f; buzzer = buzzer_f; motorLeft = motorLeft_f; @@ -35,9 +36,13 @@ buttonCommands::buttonCommands(gpio_evaluatedSwitch * button_f, controlledArmcha //--------- action ----------- //---------------------------- //function that runs commands depending on a count value -void buttonCommands::action (uint8_t count){ +void buttonCommands::action (uint8_t count, bool lastPressLong){ //--- variable declarations --- bool decelEnabled; //for different beeping when toggling + commandSimple_t cmds[8]; //array for commands for automatedArmchair + + //--- get joystick position --- + //joystickData_t stickData = joystick->getData(); //--- actions based on count --- switch (count){ @@ -47,28 +52,77 @@ void buttonCommands::action (uint8_t count){ buzzer->beep(3, 400, 100); break; - case 0: //special case when last button press is longer than timeout (>1s) - ESP_LOGW(TAG, "RESTART"); - buzzer->beep(1,1000,1); - vTaskDelay(1000 / portTICK_PERIOD_MS); - esp_restart(); - case 1: + //restart contoller when 1x long pressed + if (lastPressLong){ + ESP_LOGW(TAG, "RESTART"); + buzzer->beep(1,1000,1); + vTaskDelay(1000 / portTICK_PERIOD_MS); + esp_restart(); + return; + } + ESP_LOGW(TAG, "cmd %d: sending button event to control task", count); //-> define joystick center or toggle freeze input (executed in control task) control->sendButtonEvent(count); //TODO: always send button event to control task (not just at count=1) -> control.cpp has to be changed break; + case 2: + //run automatic commands to lift leg support when pressed 1x short 1x long + if (lastPressLong){ + //define commands + cmds[0] = + { + .motorCmds = { + .left = {motorstate_t::REV, 90}, + .right = {motorstate_t::REV, 90} + }, + .msDuration = 1200, + .fadeDecel = 800, + .fadeAccel = 1300, + .instruction = auto_instruction_t::NONE + }; + cmds[1] = + { + .motorCmds = { + .left = {motorstate_t::FWD, 70}, + .right = {motorstate_t::FWD, 70} + }, + .msDuration = 70, + .fadeDecel = 0, + .fadeAccel = 300, + .instruction = auto_instruction_t::NONE + }; + cmds[2] = + { + .motorCmds = { + .left = {motorstate_t::IDLE, 0}, + .right = {motorstate_t::IDLE, 0} + }, + .msDuration = 10, + .fadeDecel = 800, + .fadeAccel = 1300, + .instruction = auto_instruction_t::SWITCH_JOYSTICK_MODE + }; + + //send commands to automatedArmchair command queue + armchair.addCommands(cmds, 3); + + //change mode to AUTO + control->changeMode(controlMode_t::AUTO); + return; + } + + //toggle idle when 2x pressed + ESP_LOGW(TAG, "cmd %d: toggle IDLE", count); + control->toggleIdle(); //toggle between idle and previous/default mode + break; + case 3: ESP_LOGW(TAG, "cmd %d: switch to JOYSTICK", count); control->changeMode(controlMode_t::JOYSTICK); //switch to JOYSTICK mode break; - case 2: - ESP_LOGW(TAG, "cmd %d: toggle IDLE", count); - control->toggleIdle(); //toggle between idle and previous/default mode - break; - case 4: ESP_LOGW(TAG, "cmd %d: toggle between HTTP and JOYSTICK", count); control->toggleModes(controlMode_t::HTTP, controlMode_t::JOYSTICK); //toggle between HTTP and JOYSTICK mode @@ -120,7 +174,7 @@ void buttonCommands::startHandleLoop() { case inputState_t::IDLE: //wait for initial button press if (button->risingEdge) { count = 1; - buzzer->beep(1, 60, 0); + buzzer->beep(1, 65, 0); timestamp_lastAction = esp_log_timestamp(); state = inputState_t::WAIT_FOR_INPUT; ESP_LOGI(TAG, "first button press detected -> waiting for further events"); @@ -131,7 +185,7 @@ void buttonCommands::startHandleLoop() { //button pressed again if (button->risingEdge){ count++; - buzzer->beep(1, 60, 0); + buzzer->beep(1, 65, 0); timestamp_lastAction = esp_log_timestamp(); ESP_LOGI(TAG, "another press detected -> count=%d -> waiting for further events", count); } @@ -143,14 +197,14 @@ void buttonCommands::startHandleLoop() { ESP_LOGI(TAG, "timeout - running action function for count=%d", count); //--- run action function --- //check if still pressed + bool lastPressLong = false; if (button->state == true){ //run special case when last press was longer than timeout - action(0); - } else { - //run action function with current count of button presses - action(count); + lastPressLong = true; + } + //run action function with current count of button presses + action(count, lastPressLong); } - } break; } } diff --git a/main/button.hpp b/main/button.hpp index b5d5900..09de729 100644 --- a/main/button.hpp +++ b/main/button.hpp @@ -4,6 +4,9 @@ #include "buzzer.hpp" #include "control.hpp" #include "motorctl.hpp" +#include "auto.hpp" +#include "config.hpp" +#include "joystick.hpp" @@ -16,6 +19,7 @@ class buttonCommands { //--- constructor --- buttonCommands ( gpio_evaluatedSwitch * button_f, + evaluatedJoystick * joystick_f, controlledArmchair * control_f, buzzer_t * buzzer_f, controlledMotor * motorLeft_f, @@ -29,10 +33,11 @@ class buttonCommands { private: //--- functions --- - void action(uint8_t count); + void action(uint8_t count, bool lastPressLong); //--- objects --- gpio_evaluatedSwitch* button; + evaluatedJoystick* joystick; controlledArmchair * control; buzzer_t* buzzer; controlledMotor * motorLeft; diff --git a/main/config.cpp b/main/config.cpp index 784047e..832f984 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -30,7 +30,7 @@ single100a_config_t configDriverRight = { //--- configure motor contol --- motorctl_config_t configMotorControl = { .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%) + .msFadeDecel = 700, //deceleration of the motor (ms it takes from 100% to 0%) .currentMax = 10 }; @@ -80,11 +80,11 @@ joystick_config_t configJoystick = { //threshold the radius jumps to 1 before the stick is at max radius (range 0-1) .tolerance_radius = 0.05, - //min and max adc values of each axis - .x_min = 975, - .x_max = 2520, - .y_min = 1005, - .y_max = 2550, + //min and max adc values of each axis (after inversion is applied) + .x_min = 1230, //=> x=-1 + .x_max = 2700, //=> x=1 + .y_min = 1260, //=> y=-1 + .y_max = 2700, //=> y=1 //invert adc measurement .x_inverted = true, .y_inverted = true @@ -126,5 +126,7 @@ httpJoystick httpJoystickMain(configHttpJoystickMain); //create global control object controlledArmchair control(configControl, &buzzer, &motorLeft, &motorRight, &joystick, &httpJoystickMain); +//create global automatedArmchair object (for auto-mode) +automatedArmchair armchair; diff --git a/main/config.hpp b/main/config.hpp index 20db9bc..a811190 100644 --- a/main/config.hpp +++ b/main/config.hpp @@ -9,6 +9,12 @@ #include "control.hpp" #include "fan.hpp" #include "http.hpp" +#include "auto.hpp" + + +//in IDLE mode: set loglevel for evaluatedJoystick to DEBUG +//and repeatedly read joystick e.g. for manually calibrating / testing joystick +//#define JOYSTICK_LOG_IN_IDLE //create global controlledMotor instances for both motors @@ -27,6 +33,9 @@ extern buzzer_t buzzer; //create global control object extern controlledArmchair control; +//create global automatedArmchair object (for auto-mode) +extern automatedArmchair armchair; + //create global httpJoystick object extern httpJoystick httpJoystickMain; diff --git a/main/control.cpp b/main/control.cpp index 7bc9faf..ebd4ea8 100644 --- a/main/control.cpp +++ b/main/control.cpp @@ -14,6 +14,10 @@ extern "C" #include "control.hpp" +//used definitions moved from config.hpp: +//#define JOYSTICK_TEST + + //tag for logging static const char * TAG = "control"; @@ -68,6 +72,12 @@ void controlledArmchair::startHandleLoop() { motorRight->setTarget(commands.right.state, commands.right.duty); motorLeft->setTarget(commands.left.state, commands.left.duty); vTaskDelay(200 / portTICK_PERIOD_MS); +#ifdef JOYSTICK_LOG_IN_IDLE + //get joystick data here (without using it) + //since loglevel is DEBUG, calculateion details is output + joystick_l->getData(); //get joystick data here +#endif + break; @@ -121,6 +131,48 @@ void controlledArmchair::startHandleLoop() { motorLeft->setTarget(commands.left.state, commands.left.duty); break; + + case controlMode_t::AUTO: + vTaskDelay(20 / portTICK_PERIOD_MS); + //generate commands + commands = armchair.generateCommands(&instruction); + //--- apply commands to motors --- + //TODO make motorctl.setTarget also accept motorcommand struct directly + motorRight->setTarget(commands.right.state, commands.right.duty); + motorLeft->setTarget(commands.left.state, commands.left.duty); + + //process received instruction + switch (instruction) { + case auto_instruction_t::NONE: + break; + case auto_instruction_t::SWITCH_PREV_MODE: + toggleMode(controlMode_t::AUTO); + break; + case auto_instruction_t::SWITCH_JOYSTICK_MODE: + changeMode(controlMode_t::JOYSTICK); + break; + case auto_instruction_t::RESET_ACCEL_DECEL: + //enable downfading (set to default value) + motorLeft->setFade(fadeType_t::DECEL, true); + motorRight->setFade(fadeType_t::DECEL, true); + //set upfading to default value + motorLeft->setFade(fadeType_t::ACCEL, true); + motorRight->setFade(fadeType_t::ACCEL, true); + break; + case auto_instruction_t::RESET_ACCEL: + //set upfading to default value + motorLeft->setFade(fadeType_t::ACCEL, true); + motorRight->setFade(fadeType_t::ACCEL, true); + break; + case auto_instruction_t::RESET_DECEL: + //enable downfading (set to default value) + motorLeft->setFade(fadeType_t::DECEL, true); + motorRight->setFade(fadeType_t::DECEL, true); + break; + } + break; + + // //TODO: add other modes here } @@ -255,24 +307,38 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { //reset timeout timer resetTimeout(); + //exit if target mode is already active + if (mode == modeNew) { + ESP_LOGE(TAG, "changeMode: Already in target mode '%s' -> nothing to change", controlModeStr[(int)mode]); + return; + } + //copy previous mode - controlMode_t modePrevious = mode; + modePrevious = mode; - ESP_LOGW(TAG, "=== changing mode from %s to %s ===", controlModeStr[(int)mode], controlModeStr[(int)modeNew]); + ESP_LOGW(TAG, "=== changing mode from %s to %s ===", controlModeStr[(int)mode], controlModeStr[(int)modeNew]); - //--- run functions when changing FROM certain mode --- - switch(modePrevious){ - default: - ESP_LOGI(TAG, "noting to execute when changing FROM this mode"); - break; + //========== commands change FROM mode ========== + //run functions when changing FROM certain mode + switch(modePrevious){ + default: + ESP_LOGI(TAG, "noting to execute when changing FROM this mode"); + break; - case controlMode_t::HTTP: - ESP_LOGW(TAG, "switching from http mode -> disabling http and wifi"); - //stop http server - ESP_LOGI(TAG, "disabling http server..."); - http_stop_server(); +#ifdef JOYSTICK_LOG_IN_IDLE + case controlMode_t::IDLE: + ESP_LOGI(TAG, "disabling debug output for 'evaluatedJoystick'"); + esp_log_level_set("evaluatedJoystick", ESP_LOG_WARN); //FIXME: loglevel from config + break; +#endif - //FIXME: make wifi function work here - currently starting wifi at startup (see notes main.cpp) + case controlMode_t::HTTP: + ESP_LOGW(TAG, "switching from http mode -> disabling http and wifi"); + //stop http server + ESP_LOGI(TAG, "disabling http server..."); + http_stop_server(); + + //FIXME: make wifi function work here - currently starting wifi at startup (see notes main.cpp) //stop wifi //TODO: decide whether ap or client is currently used - which has to be disabled? //ESP_LOGI(TAG, "deinit wifi..."); @@ -293,18 +359,34 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { //reset frozen input state freezeInput = false; break; + + case controlMode_t::AUTO: + ESP_LOGW(TAG, "switching from AUTO mode -> restoring fading to default"); + //TODO: fix issue when downfading was disabled before switching to auto mode - currently it gets enabled again here... + //enable downfading (set to default value) + motorLeft->setFade(fadeType_t::DECEL, true); + motorRight->setFade(fadeType_t::DECEL, true); + //set upfading to default value + motorLeft->setFade(fadeType_t::ACCEL, true); + motorRight->setFade(fadeType_t::ACCEL, true); + break; } - //--- run functions when changing TO certain mode --- + //========== commands change TO mode ========== + //run functions when changing TO certain mode switch(modeNew){ default: ESP_LOGI(TAG, "noting to execute when changing TO this mode"); break; - case controlMode_t::IDLE: - buzzer->beep(1, 1500, 0); - break; + case controlMode_t::IDLE: + buzzer->beep(1, 1500, 0); +#ifdef JOYSTICK_LOG_IN_IDLE + esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG); +#endif + + break; case controlMode_t::HTTP: ESP_LOGW(TAG, "switching to http mode -> enabling http and wifi"); @@ -352,15 +434,8 @@ void controlledArmchair::changeMode(controlMode_t modeNew) { //----------------------------------- //function to toggle between IDLE and previous active mode void controlledArmchair::toggleIdle() { - if (mode == controlMode_t::IDLE){ - changeMode(modePrevious); //restore previous mode, or default if not switched yet - buzzer->beep(2, 200, 100); - ESP_LOGW(TAG, "toggle idle: switched mode from IDLE to %s", controlModeStr[(int)mode]); - } else { - modePrevious = mode; //store current mode - changeMode(controlMode_t::IDLE); //set mode to IDLE - ESP_LOGW(TAG, "toggle idle: switched mode from %s to IDLE", controlModeStr[(int)mode]); - } + //toggle between IDLE and previous mode + toggleMode(controlMode_t::IDLE); } @@ -384,3 +459,25 @@ void controlledArmchair::toggleModes(controlMode_t modePrimary, controlMode_t mo changeMode(modePrimary); } } + + + +//----------------------------------- +//----------- toggleMode ------------ +//----------------------------------- +//function that toggles between certain mode and previous mode +void controlledArmchair::toggleMode(controlMode_t modePrimary){ + + //switch to previous mode when primary is already active + if (mode == modePrimary){ + ESP_LOGW(TAG, "toggleMode: switching from primaryMode %s to previousMode %s", controlModeStr[(int)mode], controlModeStr[(int)modePrevious]); + //buzzer->beep(2,200,100); + changeMode(modePrevious); //switch to previous mode + } + //switch to primary mode when any other mode is active + else { + ESP_LOGW(TAG, "toggleModes: switching from %s to primary mode %s", controlModeStr[(int)mode], controlModeStr[(int)modePrimary]); + //buzzer->beep(4,200,100); + changeMode(modePrimary); + } +} diff --git a/main/control.hpp b/main/control.hpp index 69d492a..9715a33 100644 --- a/main/control.hpp +++ b/main/control.hpp @@ -4,6 +4,7 @@ #include "motorctl.hpp" #include "buzzer.hpp" #include "http.hpp" +#include "auto.hpp" //-------------------------------------------- @@ -54,6 +55,8 @@ class controlledArmchair { //function that toggles between two modes, but prefers first argument if entirely different mode is currently active void toggleModes(controlMode_t modePrimary, controlMode_t modeSecondary); + //toggle between certain mode and previous mode + void toggleMode(controlMode_t modePrimary); //function that restarts timer which initiates the automatic timeout (switch to IDLE) after certain time of inactivity void resetTimeout(); @@ -90,6 +93,9 @@ class controlledArmchair { //variables for MASSAGE mode bool freezeInput = false; + + //variables for AUTO mode + auto_instruction_t instruction = auto_instruction_t::NONE; //variable to receive instructions from automatedArmchair //variable to store button event uint8_t buttonCount = 0; diff --git a/main/joystick.cpp b/main/joystick.cpp index 746547e..909fe90 100644 --- a/main/joystick.cpp +++ b/main/joystick.cpp @@ -75,13 +75,21 @@ int evaluatedJoystick::readAdc(adc1_channel_t adc_channel, bool inverted) { joystickData_t evaluatedJoystick::getData() { //get coordinates //TODO individual tolerances for each axis? Otherwise some parameters can be removed - ESP_LOGD(TAG, "getting X coodrinate..."); + ESP_LOGV(TAG, "getting X coodrinate..."); + 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); 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); - ESP_LOGD(TAG, "getting Y coodrinate..."); - float y = scaleCoordinate(readAdc(config.adc_y, config.y_inverted), config.y_min, config.y_max, y_center, config.tolerance_zeroY_per, config.tolerance_end_per); + + 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); 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); //calculate radius data.radius = sqrt(pow(data.x,2) + pow(data.y,2)); diff --git a/main/main.cpp b/main/main.cpp index 49ec363..a9c8e4d 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, &motorLeft, &motorRight); + buttonCommands commandButton(&buttonJoystick, &joystick, &control, &buzzer, &motorLeft, &motorRight); //start handle loop commandButton.startHandleLoop(); } @@ -152,6 +152,7 @@ extern "C" void app_main(void) { esp_log_level_set("fan-control", ESP_LOG_INFO); esp_log_level_set("wifi", ESP_LOG_INFO); esp_log_level_set("http", ESP_LOG_INFO); + esp_log_level_set("automatedArmchair", ESP_LOG_DEBUG); //---------------------------------------------- @@ -175,7 +176,7 @@ extern "C" void app_main(void) { //--- create task for button --- //------------------------------ //task that evaluates and processes the button input and runs the configured commands - xTaskCreate(&task_button, "task_button", 2048, NULL, 4, NULL); + xTaskCreate(&task_button, "task_button", 4096, NULL, 4, NULL); //----------------------------------- //--- create task for fan control --- diff --git a/sdkconfig b/sdkconfig index 43b677a..f7c6b17 100644 --- a/sdkconfig +++ b/sdkconfig @@ -54,6 +54,7 @@ CONFIG_BOOTLOADER_LOG_LEVEL=3 CONFIG_BOOTLOADER_VDDSDIO_BOOST_1_9V=y # CONFIG_BOOTLOADER_FACTORY_RESET is not set # CONFIG_BOOTLOADER_APP_TEST is not set +CONFIG_BOOTLOADER_REGION_PROTECTION_ENABLE=y CONFIG_BOOTLOADER_WDT_ENABLE=y # CONFIG_BOOTLOADER_WDT_DISABLE_IN_USER_CODE is not set CONFIG_BOOTLOADER_WDT_TIME_MS=9000 @@ -95,6 +96,9 @@ CONFIG_ESPTOOLPY_FLASHFREQ="40m" CONFIG_ESPTOOLPY_FLASHSIZE_4MB=y # CONFIG_ESPTOOLPY_FLASHSIZE_8MB is not set # CONFIG_ESPTOOLPY_FLASHSIZE_16MB is not set +# CONFIG_ESPTOOLPY_FLASHSIZE_32MB is not set +# CONFIG_ESPTOOLPY_FLASHSIZE_64MB is not set +# CONFIG_ESPTOOLPY_FLASHSIZE_128MB is not set CONFIG_ESPTOOLPY_FLASHSIZE="4MB" CONFIG_ESPTOOLPY_FLASHSIZE_DETECT=y CONFIG_ESPTOOLPY_BEFORE_RESET=y @@ -263,8 +267,8 @@ CONFIG_EFUSE_MAX_BLK_LEN=192 # CONFIG_ESP_TLS_USING_MBEDTLS=y # CONFIG_ESP_TLS_USE_SECURE_ELEMENT is not set -# CONFIG_ESP_TLS_SERVER is not set # CONFIG_ESP_TLS_CLIENT_SESSION_TICKETS is not set +# CONFIG_ESP_TLS_SERVER is not set # CONFIG_ESP_TLS_PSK_VERIFICATION is not set # CONFIG_ESP_TLS_INSECURE is not set # end of ESP-TLS @@ -472,6 +476,13 @@ CONFIG_ESP_PHY_REDUCE_TX_POWER=y # CONFIG_PM_ENABLE is not set # end of Power Management +# +# ESP Ringbuf +# +# CONFIG_RINGBUF_PLACE_FUNCTIONS_INTO_FLASH is not set +# CONFIG_RINGBUF_PLACE_ISR_FUNCTIONS_INTO_FLASH is not set +# end of ESP Ringbuf + # # ESP System Settings # @@ -555,6 +566,8 @@ CONFIG_ESP32_WIFI_ENABLE_WPA3_SAE=y # CONFIG_ESP_WIFI_STA_DISCONNECTED_PM_ENABLE is not set # CONFIG_ESP_WIFI_GMAC_SUPPORT is not set CONFIG_ESP_WIFI_SOFTAP_SUPPORT=y +# CONFIG_ESP_WIFI_SLP_BEACON_LOST_OPT is not set +CONFIG_ESP_WIFI_ESPNOW_MAX_ENCRYPT_NUM=7 # end of Wi-Fi # @@ -628,11 +641,7 @@ CONFIG_FMB_CONTROLLER_NOTIFY_QUEUE_SIZE=20 CONFIG_FMB_CONTROLLER_STACK_SIZE=4096 CONFIG_FMB_EVENT_QUEUE_TIMEOUT=20 # CONFIG_FMB_TIMER_PORT_ENABLED is not set -CONFIG_FMB_TIMER_GROUP=0 -CONFIG_FMB_TIMER_INDEX=0 -CONFIG_FMB_MASTER_TIMER_GROUP=0 -CONFIG_FMB_MASTER_TIMER_INDEX=0 -# CONFIG_FMB_TIMER_ISR_IN_IRAM is not set +# CONFIG_FMB_TIMER_USE_ISR_DISPATCH_METHOD is not set # end of Modbus configuration # @@ -791,6 +800,7 @@ CONFIG_LWIP_TCP_SYNMAXRTX=12 CONFIG_LWIP_TCP_MSS=1440 CONFIG_LWIP_TCP_TMR_INTERVAL=250 CONFIG_LWIP_TCP_MSL=60000 +CONFIG_LWIP_TCP_FIN_WAIT_TIMEOUT=20000 CONFIG_LWIP_TCP_SND_BUF_DEFAULT=5744 CONFIG_LWIP_TCP_WND_DEFAULT=5744 CONFIG_LWIP_TCP_RECVMBOX_SIZE=6 @@ -901,6 +911,7 @@ CONFIG_MBEDTLS_CERTIFICATE_BUNDLE_DEFAULT_FULL=y # CONFIG_MBEDTLS_CERTIFICATE_BUNDLE_DEFAULT_CMN is not set # CONFIG_MBEDTLS_CERTIFICATE_BUNDLE_DEFAULT_NONE is not set # CONFIG_MBEDTLS_CUSTOM_CERTIFICATE_BUNDLE is not set +CONFIG_MBEDTLS_CERTIFICATE_BUNDLE_MAX_CERTS=200 # end of Certificate Bundle # CONFIG_MBEDTLS_ECP_RESTARTABLE is not set @@ -1048,6 +1059,7 @@ CONFIG_NEWLIB_STDIN_LINE_ENDING_CR=y # # NVS # +# CONFIG_NVS_ASSERT_ERROR_CHECK is not set # end of NVS # @@ -1185,7 +1197,6 @@ CONFIG_VFS_SUPPORT_TERMIOS=y # Host File System I/O (Semihosting) # CONFIG_VFS_SEMIHOSTFS_MAX_MOUNT_POINTS=1 -CONFIG_VFS_SEMIHOSTFS_HOST_PATH_MAX_LEN=128 # end of Host File System I/O (Semihosting) # end of Virtual file system @@ -1202,6 +1213,7 @@ CONFIG_WL_SECTOR_SIZE=4096 # CONFIG_WIFI_PROV_SCAN_MAX_ENTRIES=16 CONFIG_WIFI_PROV_AUTOSTOP_TIMEOUT=30 +# CONFIG_WIFI_PROV_BLE_FORCE_ENCRYPTION is not set # end of Wi-Fi Provisioning Manager # @@ -1214,6 +1226,8 @@ CONFIG_WPA_MBEDTLS_CRYPTO=y # CONFIG_WPA_TESTING_OPTIONS is not set # CONFIG_WPA_WPS_STRICT is not set # CONFIG_WPA_11KV_SUPPORT is not set +# CONFIG_WPA_MBO_SUPPORT is not set +# CONFIG_WPA_DPP_SUPPORT is not set # end of Supplicant # end of Component config @@ -1338,8 +1352,6 @@ CONFIG_MB_CONTROLLER_NOTIFY_QUEUE_SIZE=20 CONFIG_MB_CONTROLLER_STACK_SIZE=4096 CONFIG_MB_EVENT_QUEUE_TIMEOUT=20 # CONFIG_MB_TIMER_PORT_ENABLED is not set -CONFIG_MB_TIMER_GROUP=0 -CONFIG_MB_TIMER_INDEX=0 # CONFIG_ENABLE_STATIC_TASK_CLEAN_UP_HOOK is not set CONFIG_TIMER_TASK_PRIORITY=1 CONFIG_TIMER_TASK_STACK_DEPTH=2048 @@ -1382,5 +1394,4 @@ CONFIG_SPI_FLASH_WRITING_DANGEROUS_REGIONS_ABORTS=y CONFIG_SUPPRESS_SELECT_DEBUG_OUTPUT=y CONFIG_SUPPORT_TERMIOS=y CONFIG_SEMIHOSTFS_MAX_MOUNT_POINTS=1 -CONFIG_SEMIHOSTFS_HOST_PATH_MAX_LEN=128 # End of deprecated options