diff --git a/main/CMakeLists.txt b/main/CMakeLists.txt index 9eb7ec4..f547649 100644 --- a/main/CMakeLists.txt +++ b/main/CMakeLists.txt @@ -1,2 +1,2 @@ -idf_component_register(SRCS "main.cpp" +idf_component_register(SRCS "main.cpp" "motordrivers.cpp" INCLUDE_DIRS ".") diff --git a/main/main.cpp b/main/main.cpp index d5e92e0..d7112e2 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -9,10 +9,54 @@ extern "C" #include "driver/gpio.h" #include "esp_log.h" #include "sdkconfig.h" -} - -extern "C" void app_main(void) -{ +#include "driver/ledc.h" } + +#include "motordrivers.hpp" + + +extern "C" void app_main(void) { + + //set loglevel for all tags: + esp_log_level_set("*", ESP_LOG_INFO); + //set loglevel for motordriver to DEBUG for testing + esp_log_level_set("motordriver", ESP_LOG_DEBUG); + + //configure motor driver + single100a_config_t configDriverLeft = { + .gpio_pwm = GPIO_NUM_14, + .gpio_a = GPIO_NUM_12, + .gpio_b = GPIO_NUM_13, + .ledc_timer = LEDC_TIMER_0, + .ledc_channel = LEDC_CHANNEL_0, + .abInverted = true, + .resolution = LEDC_TIMER_11_BIT, + .pwmFreq = 10000 + }; + + //create driver instance for motor left + single100a motorLeft(configDriverLeft); + + + while(1){ + + //--- testing the motor driver --- + //fade up duty - forward + for (int duty=0; duty<=100; duty+=5) { + motorLeft.set(motorstate_t::FWD, duty); + vTaskDelay(100 / portTICK_PERIOD_MS); + } + //brake for 1 s + motorLeft.set(motorstate_t::BRAKE); + vTaskDelay(1000 / portTICK_PERIOD_MS); + //stay at 100% - reverse + motorLeft.set(motorstate_t::REV, 150); + vTaskDelay(1000 / portTICK_PERIOD_MS); + //stay at idle (default duty) + motorLeft.set(motorstate_t::IDLE); + vTaskDelay(2000 / portTICK_PERIOD_MS); + + } +} diff --git a/main/motordrivers.cpp b/main/motordrivers.cpp new file mode 100644 index 0000000..fc0f16d --- /dev/null +++ b/main/motordrivers.cpp @@ -0,0 +1,134 @@ +#include "motordrivers.hpp" + +//TODO: move from ledc to mcpwm? +//https://docs.espressif.com/projects/esp-idf/en/v4.3/esp32/api-reference/peripherals/mcpwm.html# +//https://github.com/espressif/esp-idf/tree/v4.3/examples/peripherals/mcpwm/mcpwm_basic_config + +//Note fade functionality provided by LEDC would be very useful but unfortunately is not usable here because: +//"Due to hardware limitations, there is no way to stop a fade before it reaches its target duty." + +//definition of string array to be able to convert state enum to readable string +const char* motorstateStr[4] = {"IDLE", "FWD", "REV", "BRAKE"}; + +//tag for logging +static const char * TAG = "motordriver"; + + + +//==================================== +//===== single100a motor driver ====== +//==================================== + +//----------------------------- +//-------- constructor -------- +//----------------------------- +//copy provided struct with all configuration and run init function +single100a::single100a(single100a_config_t config_f){ + config = config_f; + init(); +} + + + +//---------------------------- +//---------- init ------------ +//---------------------------- +//function to initialize pwm output, gpio pins and calculate maxDuty +void single100a::init(){ + + //--- configure ledc timer --- + ledc_timer_config_t ledc_timer; + ledc_timer.speed_mode = LEDC_HIGH_SPEED_MODE; + ledc_timer.timer_num = config.ledc_timer; + ledc_timer.duty_resolution = config.resolution; //13bit gives max 5khz freq + ledc_timer.freq_hz = config.pwmFreq; + ledc_timer.clk_cfg = LEDC_AUTO_CLK; + //apply configuration + ledc_timer_config(&ledc_timer); + + //--- configure ledc channel --- + ledc_channel_config_t ledc_channel; + ledc_channel.channel = config.ledc_channel; + ledc_channel.duty = 0; + ledc_channel.gpio_num = config.gpio_pwm; + ledc_channel.speed_mode = LEDC_HIGH_SPEED_MODE; + ledc_channel.hpoint = 0; + ledc_channel.timer_sel = config.ledc_timer; + ledc_channel.intr_type = LEDC_INTR_DISABLE; + ledc_channel.flags.output_invert = 0; //TODO: add config option to invert the pwm output? + //apply configuration + ledc_channel_config(&ledc_channel); + + //--- define gpio pins as outputs --- + gpio_pad_select_gpio(config.gpio_a); + gpio_set_direction(config.gpio_a, GPIO_MODE_OUTPUT); + gpio_pad_select_gpio(config.gpio_b); + gpio_set_direction(config.gpio_b, GPIO_MODE_OUTPUT); + + //--- calculate max duty according to selected resolution --- + dutyMax = pow(2, ledc_timer.duty_resolution) -1; + ESP_LOGI(TAG, "initialized single100a driver"); + ESP_LOGI(TAG, "resolution=%dbit, dutyMax value=%d, resolution=%.4f %%", ledc_timer.duty_resolution, dutyMax, 100/(float)dutyMax); +} + + + +//--------------------------- +//----------- set ----------- +//--------------------------- +//function to put the h-bridge module in the desired state and duty cycle +void single100a::set(motorstate_t state_f, float duty_f){ + + //define enabled signal state (gpio high/low) TODO: move this to constructor? + bool enabled; + if (config.abInverted) { + enabled = false; + } else { + enabled = true; + } + + //scale provided target duty in percent to available resolution for ledc + uint32_t dutyScaled; + if (duty_f > 100) { //target duty above 100% + dutyScaled = dutyMax; + } else if (duty_f < 0) { //target duty below 0% + dutyScaled = 0; + } else { //target duty 0-100% + //scale duty to available resolution + dutyScaled = duty_f / 100 * dutyMax; + } + + //put the single100a h-bridge module in the desired state update duty-cycle + switch (state_f){ + case motorstate_t::IDLE: + ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, dutyScaled); + ledc_update_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel); + //TODO: to fix bugged state of h-bridge module when idle and start again, maybe try to leave pwm signal on for some time before updating a/b pins? + //no brake: (freewheel) + gpio_set_level(config.gpio_a, enabled); + gpio_set_level(config.gpio_b, enabled); + break; + case motorstate_t::BRAKE: + ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, 0); + ledc_update_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel); + //brake: + gpio_set_level(config.gpio_a, !enabled); + gpio_set_level(config.gpio_b, !enabled); + break; + case motorstate_t::FWD: + ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, dutyScaled); + ledc_update_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel); + //forward: + gpio_set_level(config.gpio_a, enabled); + gpio_set_level(config.gpio_b, !enabled); + break; + case motorstate_t::REV: + ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, dutyScaled); + ledc_update_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel); + //reverse: + gpio_set_level(config.gpio_a, !enabled); + gpio_set_level(config.gpio_b, enabled); + break; + } + ESP_LOGD(TAG, "set module to state=%s, duty=%d/%d, duty_input=%.3f%%", motorstateStr[(int)state_f], dutyScaled, dutyMax, duty_f); +} diff --git a/main/motordrivers.hpp b/main/motordrivers.hpp new file mode 100644 index 0000000..19da930 --- /dev/null +++ b/main/motordrivers.hpp @@ -0,0 +1,64 @@ +#pragma once + +extern "C" +{ +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "driver/gpio.h" +#include "esp_log.h" + +#include "driver/ledc.h" +#include "esp_err.h" +} + +#include + + +//==================================== +//===== single100a motor driver ====== +//==================================== + +//-------------------------------------------- +//---- struct, enum, variable deklarations --- +//-------------------------------------------- + +//class which controls a motor using a 'single100a' h-bridge module +enum class motorstate_t {IDLE, FWD, REV, BRAKE}; +//definition of string array to be able to convert state enum to readable string (defined in motordrivers.cpp) +extern const char* motorstateStr[4]; + +//struct with all config parameters for single100a motor driver +typedef struct single100a_config_t { + gpio_num_t gpio_pwm; + gpio_num_t gpio_a; + gpio_num_t gpio_b; + ledc_timer_t ledc_timer; + ledc_channel_t ledc_channel; + bool abInverted; + ledc_timer_bit_t resolution; + int pwmFreq; +} single100a_config_t; + + + +//-------------------------------- +//------- single100a class ------- +//-------------------------------- +class single100a { + public: + //--- constructor --- + single100a(single100a_config_t config_f); //provide config struct (see above) + + //--- functions --- + void set(motorstate_t state, float duty_f = 0); //set mode and duty of the motor (see motorstate_t above) + //TODO: add functions to get the current state and duty + + private: + //--- functions --- + void init(); //initialize pwm and gpio outputs, calculate maxDuty + + //--- variables --- + single100a_config_t config; + uint32_t dutyMax; + motorstate_t state = motorstate_t::IDLE; +};