From adb517c7edff1b0a1103d4e87c3e5d87a0a79222 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Fri, 10 Jun 2022 22:57:27 +0200 Subject: [PATCH] Create 'control.hpp/cpp', Create 'button.hpp/cpp' Add control.hpp and control.cpp - task that repeatedly generates motor commands depending on the current mode - function to change to a specified control mode Add button.hpp and button.cpp - class which runs commands depending on the count a button was pressed Update main.cpp - create button task - create control task - comment out previous testing code - remove unnecessary includes (already included in config.hpp) Add control.cpp and button.cpp to CMakeLists Notes: Tested this state on the armchair: All currently implemented features work. You can switch between IDLE and JOYSTICK by pressing the button 2 or 3 times. Also driving works well (limited to 60% duty, with no fans yet). --- main/CMakeLists.txt | 2 +- main/button.cpp | 114 ++++++++++++++++++++++++++++++++++++++++++++ main/button.hpp | 40 ++++++++++++++++ main/control.cpp | 72 ++++++++++++++++++++++++++++ main/control.hpp | 15 ++++++ main/main.cpp | 64 ++++++++++++++++++------- 6 files changed, 290 insertions(+), 17 deletions(-) create mode 100644 main/button.cpp create mode 100644 main/button.hpp create mode 100644 main/control.cpp create mode 100644 main/control.hpp diff --git a/main/CMakeLists.txt b/main/CMakeLists.txt index 2caa7ed..6909b71 100644 --- a/main/CMakeLists.txt +++ b/main/CMakeLists.txt @@ -1,2 +1,2 @@ -idf_component_register(SRCS "main.cpp" "motordrivers.cpp" "motorctl.cpp" "config.cpp" "joystick.cpp" "buzzer.cpp" +idf_component_register(SRCS "main.cpp" "motordrivers.cpp" "motorctl.cpp" "config.cpp" "joystick.cpp" "buzzer.cpp" "control.cpp" "button.cpp" INCLUDE_DIRS ".") diff --git a/main/button.cpp b/main/button.cpp new file mode 100644 index 0000000..b4a5042 --- /dev/null +++ b/main/button.cpp @@ -0,0 +1,114 @@ +extern "C" +{ +#include +#include +#include +#include "freertos/FreeRTOS.h" +#include "esp_log.h" +} + +#include "control.hpp" +#include "button.hpp" + + + +//tag for logging +static const char * TAG = "button"; + + + +//----------------------------- +//-------- constructor -------- +//----------------------------- +buttonCommands::buttonCommands(gpio_evaluatedSwitch * button_f, buzzer_t * buzzer_f ){ + //copy object pointers + button = button_f; + buzzer = buzzer_f; + //TODO declare / configure evaluatedSwitch here instead of config (unnecessary that button object is globally available - only used here)? +} + + + +//---------------------------- +//--------- action ----------- +//---------------------------- +//function that runs commands depending on a count value +void buttonCommands::action (uint8_t count){ + switch (count){ + //no such command + default: + ESP_LOGE(TAG, "no command for count=%d defined", count); + buzzer->beep(3, 400, 100); + break; + + case 1: + ESP_LOGW(TAG, "running command for count 1"); + buzzer->beep(1,500,1); + break; + + case 2: + ESP_LOGW(TAG, "cmd %d: switching to IDLE", count); + control_changeMode(controlMode_t::IDLE); + buzzer->beep(1,1000,1); + break; + + case 3: + ESP_LOGW(TAG, "cmd %d: switching to JOYSTICK", count); + control_changeMode(controlMode_t::JOYSTICK); + buzzer->beep(2,400,100); + break; + } +} + + + + +//----------------------------- +//------ startHandleLoop ------ +//----------------------------- +//this function has to be started once in a separate task +//repeatedly evaluates and processes button events then takes the corresponding action +void buttonCommands::startHandleLoop() { + + while(1) { + vTaskDelay(20 / portTICK_PERIOD_MS); + //run handle function of evaluatedSwitch object + button->handle(); + + //--- count button presses and run action --- + switch(state) { + case inputState_t::IDLE: //wait for initial button press + if (button->risingEdge) { + count = 1; + buzzer->beep(1, 60, 0); + timestamp_lastAction = esp_log_timestamp(); + state = inputState_t::WAIT_FOR_INPUT; + ESP_LOGI(TAG, "first button press detected -> waiting for further events"); + } + break; + + case inputState_t::WAIT_FOR_INPUT: //wait for further presses + //button pressed again + if (button->risingEdge){ + count++; + buzzer->beep(1, 60, 0); + timestamp_lastAction = esp_log_timestamp(); + ESP_LOGI(TAG, "another press detected -> count=%d -> waiting for further events", count); + } + //timeout + else if (esp_log_timestamp() - timestamp_lastAction > 1000) { + state = inputState_t::IDLE; + buzzer->beep(count, 50, 50); + //TODO: add optional "bool wait" parameter to beep function to delay until finished beeping + //run action function with current count of button presses + ESP_LOGI(TAG, "timeout - running action function for count=%d", count); + action(count); + } + break; + } + } +} + + + + diff --git a/main/button.hpp b/main/button.hpp new file mode 100644 index 0000000..a4f3d65 --- /dev/null +++ b/main/button.hpp @@ -0,0 +1,40 @@ +#pragma once + +#include "gpio_evaluateSwitch.hpp" +#include "buzzer.hpp" + + + +//=================================== +//====== buttonCommands class ======= +//=================================== +//class which runs commands depending on the count a button was pressed +class buttonCommands { + public: + //--- constructor --- + buttonCommands ( + gpio_evaluatedSwitch * button_f, + buzzer_t * buzzer_f + ); + + //--- functions --- + //the following function has to be started once in a separate task. + //repeatedly evaluates and processes button events then takes the corresponding action + void startHandleLoop(); + + private: + //--- functions --- + void action(uint8_t count); + + //--- objects --- + gpio_evaluatedSwitch* button; + buzzer_t* buzzer; + + //--- variables --- + uint8_t count = 0; + uint32_t timestamp_lastAction = 0; + enum class inputState_t {IDLE, WAIT_FOR_INPUT}; + inputState_t state = inputState_t::IDLE; + +}; + diff --git a/main/control.cpp b/main/control.cpp new file mode 100644 index 0000000..a72aa7e --- /dev/null +++ b/main/control.cpp @@ -0,0 +1,72 @@ +extern "C" +{ +#include +#include "freertos/FreeRTOS.h" +#include "esp_log.h" +} + +#include "motordrivers.hpp" +#include "motorctl.hpp" +#include "joystick.hpp" +#include "config.hpp" +#include "control.hpp" + + +//================================= +//===== variable declaration ====== +//================================= +//tag for logging +static const char * TAG = "control"; + +//definition of mode enum +controlMode_t mode = controlMode_t::IDLE; +const char* controlModeStr[6] = {"IDLE", "JOYSTICK", "MASSAGE", "MQTT", "BLUETOOTH", "AUTO"}; + + + +//================================== +//========== handle task =========== +//================================== +//task that repeatedly generates motor commands depending on the current mode +void task_control ( void * pvParameters ) { + while (1){ + vTaskDelay(20 / portTICK_PERIOD_MS); + + ESP_LOGV(TAG, "control task executing... mode=%s", controlModeStr[(int)mode]); + + switch(mode) { + default: + mode = controlMode_t::IDLE; + break; + + case controlMode_t::IDLE: + motorRight.setTarget(motorstate_t::IDLE, 0); + motorLeft.setTarget(motorstate_t::IDLE, 0); + break; + + case controlMode_t::JOYSTICK: + motorCommands_t commands = joystick_generateCommandsDriving(joystick); + motorRight.setTarget(commands.right.state, commands.right.duty); + motorLeft.setTarget(commands.left.state, commands.left.duty); + //TODO make motorctl.setTarget also accept motorcommand struct directly + break; + + //TODO: add other modes here + } + } +} + + + +//=================================== +//=========== changeMode ============ +//=================================== +//function to change to a specified control mode +void control_changeMode(controlMode_t modeNew) { + ESP_LOGW(TAG, "changing mode from %s to %s", controlModeStr[(int)mode], controlModeStr[(int)modeNew]); + mode = modeNew; + //TODO: add mutex +} + + + diff --git a/main/control.hpp b/main/control.hpp new file mode 100644 index 0000000..9c7478f --- /dev/null +++ b/main/control.hpp @@ -0,0 +1,15 @@ +#pragma once + + +//enum that decides how the motors get controlled +enum class controlMode_t {IDLE, JOYSTICK, MASSAGE, MQTT, BLUETOOTH, AUTO}; +//extern controlMode_t mode; + +//task that repeatedly generates motor commands depending on the current mode +void task_control(void * pvParameters); + +//function that changes to a specified control mode +void control_changeMode(controlMode_t modeNew); + + + diff --git a/main/main.cpp b/main/main.cpp index 97e95ed..ee66d09 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -15,6 +15,8 @@ extern "C" } #include "config.hpp" +#include "control.hpp" +#include "button.hpp" //tag for logging static const char * TAG = "main"; @@ -31,7 +33,7 @@ void task_motorctl( void * pvParameters ){ motorRight.handle(); motorLeft.handle(); //10khz -> T=100us - vTaskDelay(50 / portTICK_PERIOD_MS); + vTaskDelay(20 / portTICK_PERIOD_MS); } } @@ -51,6 +53,19 @@ void task_buzzer( void * pvParameters ){ +//====================================== +//============ button task ============= +//====================================== +void task_button( void * pvParameters ){ + ESP_LOGI(TAG, "Initializing command-button and starting handle loop"); + //create button instance + buttonCommands commandButton(&buttonJoystick, &buzzer); + //start handle loop + commandButton.startHandleLoop(); +} + + + //================================= //=========== app_main ============ //================================= @@ -61,22 +76,26 @@ extern "C" void app_main(void) { gpio_set_level(GPIO_NUM_17, 1); + //------------------------------- //---------- log level ---------- //------------------------------- //set loglevel for all tags: esp_log_level_set("*", ESP_LOG_WARN); - //set loglevel for individual tags: + esp_log_level_set("main", ESP_LOG_INFO); //esp_log_level_set("motordriver", ESP_LOG_DEBUG); //esp_log_level_set("motor-control", ESP_LOG_DEBUG); //esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG); //esp_log_level_set("joystickCommands", ESP_LOG_DEBUG); + esp_log_level_set("button", ESP_LOG_INFO); + //---------------------------------------------- //--- create task for controlling the motors --- //---------------------------------------------- + //task that receives commands, handles ramp and current limit and executes commands using the motordriver function xTaskCreate(&task_motorctl, "task_motor-control", 2048, NULL, 5, NULL); //------------------------------ @@ -84,33 +103,46 @@ extern "C" void app_main(void) { //------------------------------ xTaskCreate(&task_buzzer, "task_buzzer", 2048, NULL, 5, NULL); + //------------------------------- + //--- create task for control --- + //------------------------------- + //task that generates motor commands depending on the current mode and sends those to motorctl task (task_control is defined in control.cpp) + xTaskCreate(&task_control, "task_control", 2048, NULL, 5, NULL); + + //------------------------------ + //--- create task for button --- + //------------------------------ + //task that evaluates and processes the button input and runs the configured commands + xTaskCreate(&task_button, "task_buzzer", 2048, NULL, 5, NULL); + + //beep at startup buzzer.beep(3, 70, 50); while(1){ - vTaskDelay(50 / portTICK_PERIOD_MS); + vTaskDelay(500 / portTICK_PERIOD_MS); - buttonJoystick.handle(); - //--- testing button --- - if (buttonJoystick.risingEdge){ - ESP_LOGI(TAG, "button pressed, was released for %d ms", buttonJoystick.msReleased); - buzzer.beep(2, 100, 50); + //--- testing button --- + //buttonJoystick.handle(); + // if (buttonJoystick.risingEdge){ + // ESP_LOGI(TAG, "button pressed, was released for %d ms", buttonJoystick.msReleased); + // buzzer.beep(2, 100, 50); - }else if (buttonJoystick.fallingEdge){ - ESP_LOGI(TAG, "button released, was pressed for %d ms", buttonJoystick.msPressed); - buzzer.beep(1, 200, 0); - } + // }else if (buttonJoystick.fallingEdge){ + // ESP_LOGI(TAG, "button released, was pressed for %d ms", buttonJoystick.msPressed); + // buzzer.beep(1, 200, 0); + // } //--- testing joystick commands --- - motorCommands_t commands = joystick_generateCommandsDriving(joystick); - motorRight.setTarget(commands.right.state, commands.right.duty); //TODO make motorctl.setTarget also accept motorcommand struct directly - motorLeft.setTarget(commands.left.state, commands.left.duty); //TODO make motorctl.setTarget also accept motorcommand struct directly - //motorRight.setTarget(commands.right.state, commands.right.duty); + // motorCommands_t commands = joystick_generateCommandsDriving(joystick); + // motorRight.setTarget(commands.right.state, commands.right.duty); //TODO make motorctl.setTarget also accept motorcommand struct directly + // motorLeft.setTarget(commands.left.state, commands.left.duty); //TODO make motorctl.setTarget also accept motorcommand struct directly + // //motorRight.setTarget(commands.right.state, commands.right.duty);