Create driver for 'single100a h-bridge'
- create class 'single100a' in motordrivers.hpp and motordrivers.cpp - add code for testing the driver in main.cpp (test with led was successful) - update cmakelists, to also compile the new .cpp file
This commit is contained in:
parent
37e9048daa
commit
b0d0b568a3
@ -1,2 +1,2 @@
|
||||
idf_component_register(SRCS "main.cpp"
|
||||
idf_component_register(SRCS "main.cpp" "motordrivers.cpp"
|
||||
INCLUDE_DIRS ".")
|
||||
|
@ -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);
|
||||
|
||||
}
|
||||
}
|
||||
|
134
main/motordrivers.cpp
Normal file
134
main/motordrivers.cpp
Normal file
@ -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);
|
||||
}
|
64
main/motordrivers.hpp
Normal file
64
main/motordrivers.hpp
Normal file
@ -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 <cmath>
|
||||
|
||||
|
||||
//====================================
|
||||
//===== 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;
|
||||
};
|
Loading…
x
Reference in New Issue
Block a user