From fb90169fa4e40dbe61f2c23afcf929c1ab1c8ab1 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Fri, 10 Jun 2022 08:30:07 +0200 Subject: [PATCH] Add evaluatedSwitch component, Add buttonJoystick Copy gpio component from gate project Add instance of evaluatedswitch for button next to joystick to config Add code for testing the button to main.cpp --- components/gpio/CMakeLists.txt | 4 + components/gpio/gpio_evaluateSwitch.cpp | 171 ++++++++++++++++++++++++ components/gpio/gpio_evaluateSwitch.hpp | 59 ++++++++ main/config.cpp | 3 + main/config.hpp | 6 + main/main.cpp | 15 ++- 6 files changed, 256 insertions(+), 2 deletions(-) create mode 100644 components/gpio/CMakeLists.txt create mode 100644 components/gpio/gpio_evaluateSwitch.cpp create mode 100644 components/gpio/gpio_evaluateSwitch.hpp diff --git a/components/gpio/CMakeLists.txt b/components/gpio/CMakeLists.txt new file mode 100644 index 0000000..386d539 --- /dev/null +++ b/components/gpio/CMakeLists.txt @@ -0,0 +1,4 @@ +idf_component_register( + SRCS "gpio_evaluateSwitch.cpp" + INCLUDE_DIRS "." +) diff --git a/components/gpio/gpio_evaluateSwitch.cpp b/components/gpio/gpio_evaluateSwitch.cpp new file mode 100644 index 0000000..b33e9a2 --- /dev/null +++ b/components/gpio/gpio_evaluateSwitch.cpp @@ -0,0 +1,171 @@ +#include "gpio_evaluateSwitch.hpp" + +static const char *TAG = "evaluateSwitch"; + + +gpio_evaluatedSwitch::gpio_evaluatedSwitch( //minimal (use default values) + gpio_num_t gpio_num_declare + ){ + gpio_num = gpio_num_declare; + pullup = true; + inverted = false; + + init(); +}; + + + +gpio_evaluatedSwitch::gpio_evaluatedSwitch( //optional parameters given + gpio_num_t gpio_num_declare, + bool pullup_declare, + bool inverted_declare + ){ + gpio_num = gpio_num_declare; + pullup = pullup_declare; + inverted = inverted_declare; + + init(); +}; + + + +void gpio_evaluatedSwitch::init(){ + ESP_LOGI(TAG, "initializing gpio pin %d", (int)gpio_num); + + //define gpio pin as input + gpio_pad_select_gpio(gpio_num); + gpio_set_direction(gpio_num, GPIO_MODE_INPUT); + + if (pullup == true){ //enable pullup if desired (default) + gpio_pad_select_gpio(gpio_num); + gpio_set_pull_mode(gpio_num, GPIO_PULLUP_ONLY); + }else{ + gpio_set_pull_mode(gpio_num, GPIO_FLOATING); + gpio_pad_select_gpio(gpio_num); + } + //TODO add pulldown option + //gpio_set_pull_mode(gpio_num, GPIO_PULLDOWN_ONLY); +}; + + +void gpio_evaluatedSwitch::handle(){ //Statemachine for debouncing and edge detection + if (inverted == true){ + //========================================================= + //=========== Statemachine for inverted switch ============ + //=================== (switch to VCC) ===================== + //========================================================= + switch (p_state){ + default: + p_state = switchState::FALSE; + break; + + case switchState::FALSE: //input confirmed high (released) + fallingEdge = false; //reset edge event + if (gpio_get_level(gpio_num) == 1){ //pin high (on) + p_state = switchState::HIGH; + timestampHigh = esp_log_timestamp(); //save timestamp switched from low to high + } else { + msReleased = esp_log_timestamp() - timestampLow; //update duration released + } + break; + + case switchState::HIGH: //input recently switched to high (pressed) + if (gpio_get_level(gpio_num) == 1){ //pin still high (on) + if (esp_log_timestamp() - timestampHigh > minOnMs){ //pin in same state long enough + p_state = switchState::TRUE; + state = true; + risingEdge = true; + msReleased = timestampHigh - timestampLow; //calculate duration the button was released + } + }else{ + p_state = switchState::FALSE; + } + break; + + case switchState::TRUE: //input confirmed high (pressed) + risingEdge = false; //reset edge event + if (gpio_get_level(gpio_num) == 0){ //pin low (off) + timestampLow = esp_log_timestamp(); + p_state = switchState::LOW; + } else { + msPressed = esp_log_timestamp() - timestampHigh; //update duration pressed + } + + break; + + case switchState::LOW: //input recently switched to low (released) + if (gpio_get_level(gpio_num) == 0){ //pin still low (off) + if (esp_log_timestamp() - timestampLow > minOffMs){ //pin in same state long enough + p_state = switchState::FALSE; + msPressed = timestampLow - timestampHigh; //calculate duration the button was pressed + state=false; + fallingEdge=true; + } + }else{ + p_state = switchState::TRUE; + } + break; + } + + }else{ + //========================================================= + //========= Statemachine for not inverted switch ========== + //=================== (switch to GND) ===================== + //========================================================= + switch (p_state){ + default: + p_state = switchState::FALSE; + break; + + case switchState::FALSE: //input confirmed high (released) + fallingEdge = false; //reset edge event + if (gpio_get_level(gpio_num) == 0){ //pin low (on) + p_state = switchState::LOW; + timestampLow = esp_log_timestamp(); //save timestamp switched from high to low + } else { + msReleased = esp_log_timestamp() - timestampHigh; //update duration released + } + break; + + case switchState::LOW: //input recently switched to low (pressed) + if (gpio_get_level(gpio_num) == 0){ //pin still low (on) + if (esp_log_timestamp() - timestampLow > minOnMs){ //pin in same state long enough + p_state = switchState::TRUE; + state = true; + risingEdge = true; + msReleased = timestampLow - timestampHigh; //calculate duration the button was released + } + }else{ + p_state = switchState::FALSE; + } + break; + + case switchState::TRUE: //input confirmed low (pressed) + risingEdge = false; //reset edge event + if (gpio_get_level(gpio_num) == 1){ //pin high (off) + timestampHigh = esp_log_timestamp(); + p_state = switchState::HIGH; + } else { + msPressed = esp_log_timestamp() - timestampLow; //update duration pressed + } + + break; + + case switchState::HIGH: //input recently switched to high (released) + if (gpio_get_level(gpio_num) == 1){ //pin still high (off) + if (esp_log_timestamp() - timestampHigh > minOffMs){ //pin in same state long enough + p_state = switchState::FALSE; + msPressed = timestampHigh - timestampLow; //calculate duration the button was pressed + state=false; + fallingEdge=true; + } + }else{ + p_state = switchState::TRUE; + } + break; + } + } + +} + + diff --git a/components/gpio/gpio_evaluateSwitch.hpp b/components/gpio/gpio_evaluateSwitch.hpp new file mode 100644 index 0000000..23caa14 --- /dev/null +++ b/components/gpio/gpio_evaluateSwitch.hpp @@ -0,0 +1,59 @@ +#pragma once + +#include + +extern "C" +{ +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "driver/gpio.h" +#include "esp_log.h" +} + +//constructor examples: +//switch to gnd and us internal pullup: +//gpio_evaluatedSwitch s3(GPIO_NUM_14); +//switch to gnd dont use internal pullup: +//gpio_evaluatedSwitch s3(GPIO_NUM_14 false); +//switch to VCC (inverted) and dont use internal pullup: +//gpio_evaluatedSwitch s3(GPIO_NUM_14 false, true); + + +class gpio_evaluatedSwitch { + public: + //--- input --- + uint32_t minOnMs = 50; + uint32_t minOffMs = 50; + gpio_evaluatedSwitch( //constructor minimal (default parameters pullup=true, inverted=false) + gpio_num_t gpio_num_declare + ); + gpio_evaluatedSwitch( //constructor with optional parameters + gpio_num_t gpio_num_declare, + bool pullup_declare, + bool inverted_declare=false + ); + + //--- output --- TODO make readonly? (e.g. public section: const int& x = m_x;) + bool state = false; + bool risingEdge = false; + bool fallingEdge = false; + uint32_t msPressed = 0; + uint32_t msReleased = 0; + + //--- functions --- + void handle(); //Statemachine for debouncing and edge detection + + private: + gpio_num_t gpio_num; + bool pullup; + bool inverted; + + enum class switchState {TRUE, FALSE, LOW, HIGH}; + switchState p_state = switchState::FALSE; + uint32_t timestampLow = 0; + uint32_t timestampHigh = 0; + void init(); + +}; + + diff --git a/main/config.cpp b/main/config.cpp index bec50c7..9dfd362 100644 --- a/main/config.cpp +++ b/main/config.cpp @@ -65,3 +65,6 @@ joystick_config_t configJoystick = { //create global joystic instance evaluatedJoystick joystick(configJoystick); + +//create global evaluated switch instance for button next to joystick +gpio_evaluatedSwitch buttonJoystick(GPIO_NUM_33, true, false); //pullup true, not inverted (switch to GND use pullup of controller) diff --git a/main/config.hpp b/main/config.hpp index 2789abc..bf9b5f3 100644 --- a/main/config.hpp +++ b/main/config.hpp @@ -4,9 +4,15 @@ #include "motorctl.hpp" #include "joystick.hpp" +#include "gpio_evaluateSwitch.hpp" + + //create global controlledMotor instances for both motors extern controlledMotor motorLeft; extern controlledMotor motorRight; //create global joystic instance extern evaluatedJoystick joystick; + +//create global evaluated switch instance for button next to joystick +extern gpio_evaluatedSwitch buttonJoystick; diff --git a/main/main.cpp b/main/main.cpp index d14a2c2..510007e 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -59,7 +59,7 @@ extern "C" void app_main(void) { //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("joystickCommands", ESP_LOG_DEBUG); @@ -72,7 +72,18 @@ extern "C" void app_main(void) { while(1){ - vTaskDelay(400 / portTICK_PERIOD_MS); + vTaskDelay(50 / portTICK_PERIOD_MS); + + buttonJoystick.handle(); + + //--- testing button --- + if (buttonJoystick.risingEdge){ + ESP_LOGI(TAG, "button pressed, was released for %d ms", buttonJoystick.msReleased); + }else if (buttonJoystick.fallingEdge){ + ESP_LOGI(TAG, "button released, was pressed for %d ms", buttonJoystick.msPressed); + } + + //--- testing joystick commands --- motorCommands_t commands = joystick_generateCommandsDriving(joystick);