Merge branch 'brake' into dev - brake relay support

brake-mode support
in BRAKE state relays are powered that connects a brake resistor
(not tested with actual motors)
Also brake mode is currently not used at all
This commit is contained in:
jonny_jr9 2023-09-05 21:55:46 +02:00
commit 66fc0cf4e9
5 changed files with 123 additions and 30 deletions

View File

@ -1,3 +1,4 @@
#include "config.hpp" #include "config.hpp"
//=================================== //===================================
@ -6,8 +7,9 @@
//--- configure left motor (hardware) --- //--- configure left motor (hardware) ---
single100a_config_t configDriverLeft = { single100a_config_t configDriverLeft = {
.gpio_pwm = GPIO_NUM_26, .gpio_pwm = GPIO_NUM_26,
.gpio_a = GPIO_NUM_6, .gpio_a = GPIO_NUM_4,
.gpio_b = GPIO_NUM_16, .gpio_b = GPIO_NUM_16,
.gpio_brakeRelay = GPIO_NUM_5, //power mosfet 2
.ledc_timer = LEDC_TIMER_0, .ledc_timer = LEDC_TIMER_0,
.ledc_channel = LEDC_CHANNEL_0, .ledc_channel = LEDC_CHANNEL_0,
.aEnabledPinState = false, //-> pins inverted (mosfets) .aEnabledPinState = false, //-> pins inverted (mosfets)
@ -21,6 +23,7 @@ single100a_config_t configDriverRight = {
.gpio_pwm = GPIO_NUM_27, .gpio_pwm = GPIO_NUM_27,
.gpio_a = GPIO_NUM_2, .gpio_a = GPIO_NUM_2,
.gpio_b = GPIO_NUM_15, .gpio_b = GPIO_NUM_15,
.gpio_brakeRelay = GPIO_NUM_18, //power mosfet 1
.ledc_timer = LEDC_TIMER_1, .ledc_timer = LEDC_TIMER_1,
.ledc_channel = LEDC_CHANNEL_1, .ledc_channel = LEDC_CHANNEL_1,
.aEnabledPinState = false, //-> pins inverted (mosfets) .aEnabledPinState = false, //-> pins inverted (mosfets)
@ -35,7 +38,7 @@ single100a_config_t configDriverRight = {
motorctl_config_t configMotorControlLeft = { motorctl_config_t configMotorControlLeft = {
.msFadeAccel = 1900, //acceleration of the motor (ms it takes from 0% to 100%) .msFadeAccel = 1900, //acceleration of the motor (ms it takes from 0% to 100%)
.msFadeDecel = 1000, //deceleration of the motor (ms it takes from 100% to 0%) .msFadeDecel = 1000, //deceleration of the motor (ms it takes from 100% to 0%)
.currentLimitEnabled = true, .currentLimitEnabled = false,
.currentSensor_adc = ADC1_CHANNEL_0, //GPIO36 .currentSensor_adc = ADC1_CHANNEL_0, //GPIO36
.currentSensor_ratedCurrent = 50, .currentSensor_ratedCurrent = 50,
.currentMax = 30, .currentMax = 30,
@ -46,7 +49,7 @@ motorctl_config_t configMotorControlLeft = {
motorctl_config_t configMotorControlRight = { motorctl_config_t configMotorControlRight = {
.msFadeAccel = 1900, //acceleration of the motor (ms it takes from 0% to 100%) .msFadeAccel = 1900, //acceleration of the motor (ms it takes from 0% to 100%)
.msFadeDecel = 1000, //deceleration of the motor (ms it takes from 100% to 0%) .msFadeDecel = 1000, //deceleration of the motor (ms it takes from 100% to 0%)
.currentLimitEnabled = true, .currentLimitEnabled = false,
.currentSensor_adc = ADC1_CHANNEL_3, //GPIO39 .currentSensor_adc = ADC1_CHANNEL_3, //GPIO39
.currentSensor_ratedCurrent = 50, .currentSensor_ratedCurrent = 50,
.currentMax = 30, .currentMax = 30,
@ -70,7 +73,7 @@ fan_config_t configCooling = {
//================================= //=================================
//===== create global object s ===== //===== create global objects =====
//================================= //=================================
//TODO outsource global variables to e.g. global.cpp and only config options here? //TODO outsource global variables to e.g. global.cpp and only config options here?

View File

@ -28,12 +28,17 @@ extern "C"
//disables other functionality //disables other functionality
//#define UART_TEST_ONLY //#define UART_TEST_ONLY
//==========================
//======= BRAKE TEST =======
//==========================
//only run brake-test (ignore uart input)
#define BRAKE_TEST_ONLY
//tag for logging //tag for logging
static const char * TAG = "main"; static const char * TAG = "main";
#ifndef UART_TEST_ONLY #ifndef UART_TEST_ONLY
//==================================== //====================================
//========== motorctl task =========== //========== motorctl task ===========
@ -92,8 +97,8 @@ void setLoglevels(void){
//--- set loglevel for individual tags --- //--- set loglevel for individual tags ---
esp_log_level_set("main", ESP_LOG_INFO); esp_log_level_set("main", ESP_LOG_INFO);
esp_log_level_set("buzzer", ESP_LOG_ERROR); esp_log_level_set("buzzer", ESP_LOG_ERROR);
//esp_log_level_set("motordriver", ESP_LOG_INFO); esp_log_level_set("motordriver", ESP_LOG_VERBOSE);
//esp_log_level_set("motor-control", ESP_LOG_DEBUG); esp_log_level_set("motor-control", ESP_LOG_INFO);
//esp_log_level_set("evaluatedJoystick", 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);
esp_log_level_set("button", ESP_LOG_INFO); esp_log_level_set("button", ESP_LOG_INFO);
@ -150,15 +155,52 @@ extern "C" void app_main(void) {
//------------------------------------------- //-------------------------------------------
//--- create tasks for uart communication --- //--- create tasks for uart communication ---
//------------------------------------------- //-------------------------------------------
#ifndef BRAKE_TEST_ONLY
uart_init(); uart_init();
xTaskCreate(task_uartReceive, "task_uartReceive", 4096, NULL, 10, NULL); xTaskCreate(task_uartReceive, "task_uartReceive", 4096, NULL, 10, NULL);
xTaskCreate(task_uartSend, "task_uartSend", 4096, NULL, 10, NULL); xTaskCreate(task_uartSend, "task_uartSend", 4096, NULL, 10, NULL);
#endif
//--- main loop --- //--- main loop ---
//does nothing except for testing things //does nothing except for testing things
while(1){ while(1){
vTaskDelay(500 / portTICK_PERIOD_MS);
#ifdef BRAKE_TEST_ONLY
//test relays at standstill
ESP_LOGW("brake-test", "test relays via motorctl");
//turn on
motorRight.setTarget(motorstate_t::BRAKE, 0);
vTaskDelay(500 / portTICK_PERIOD_MS);
motorRight.setTarget(motorstate_t::BRAKE, 0);
vTaskDelay(1000 / portTICK_PERIOD_MS);
//turn off
motorRight.setTarget(motorstate_t::IDLE, 0);
vTaskDelay(500 / portTICK_PERIOD_MS);
motorRight.setTarget(motorstate_t::IDLE, 0);
vTaskDelay(1000 / portTICK_PERIOD_MS);
//go forward and brake
ESP_LOGW("brake-test", "go forward 30%% then brake");
motorRight.setTarget(motorstate_t::FWD, 30);
vTaskDelay(1000 / portTICK_PERIOD_MS);
motorRight.setTarget(motorstate_t::BRAKE, 0);
vTaskDelay(3000 / portTICK_PERIOD_MS);
//brake partial
ESP_LOGW("brake-test", "go forward 30%% then brake partial 10%%, hold for 5sec");
motorRight.setTarget(motorstate_t::FWD, 30);
vTaskDelay(1000 / portTICK_PERIOD_MS);
motorRight.setTarget(motorstate_t::BRAKE, 10);
vTaskDelay(5000 / portTICK_PERIOD_MS); vTaskDelay(5000 / portTICK_PERIOD_MS);
//reset to idle
motorRight.setTarget(motorstate_t::IDLE, 0);
#endif
//--- test controlledMotor --- (ramp) //--- test controlledMotor --- (ramp)
// //brake for 1 s // //brake for 1 s

View File

@ -5,6 +5,7 @@
//tag for logging //tag for logging
static const char * TAG = "motor-control"; static const char * TAG = "motor-control";
#define TIMEOUT_IDLE_WHEN_NO_COMMAND 8000
//============================= //=============================
//======== constructor ======== //======== constructor ========
@ -79,7 +80,7 @@ void controlledMotor::handle(){
//--- receive commands from queue --- //--- receive commands from queue ---
if( xQueueReceive( commandQueue, &commandReceive, ( TickType_t ) 0 ) ) if( xQueueReceive( commandQueue, &commandReceive, ( TickType_t ) 0 ) )
{ {
ESP_LOGD(TAG, "Read command from queue: state=%s, duty=%.2f", motorstateStr[(int)commandReceive.state], commandReceive.duty); ESP_LOGI(TAG, "Read command from queue: state=%s, duty=%.2f", motorstateStr[(int)commandReceive.state], commandReceive.duty);
state = commandReceive.state; state = commandReceive.state;
dutyTarget = commandReceive.duty; dutyTarget = commandReceive.duty;
receiveTimeout = false; receiveTimeout = false;
@ -92,7 +93,8 @@ void controlledMotor::handle(){
case motorstate_t::BRAKE: case motorstate_t::BRAKE:
//update state //update state
state = motorstate_t::BRAKE; state = motorstate_t::BRAKE;
dutyTarget = 0; //dutyTarget = 0;
dutyTarget = fabs(commandReceive.duty);
break; break;
case motorstate_t::IDLE: case motorstate_t::IDLE:
dutyTarget = 0; dutyTarget = 0;
@ -108,11 +110,12 @@ void controlledMotor::handle(){
//--- timeout, no data --- //--- timeout, no data ---
//turn motors off if no data received for long time (e.g. no uart data / control offline) //turn motors off if no data received for long time (e.g. no uart data / control offline)
if ((esp_log_timestamp() - timestamp_commandReceived) > 3000 && !receiveTimeout){ //TODO no timeout when braking?
if ((esp_log_timestamp() - timestamp_commandReceived) > TIMEOUT_IDLE_WHEN_NO_COMMAND && !receiveTimeout){
receiveTimeout = true; receiveTimeout = true;
state = motorstate_t::IDLE; state = motorstate_t::IDLE;
dutyTarget = 0; dutyTarget = 0;
ESP_LOGE(TAG, "TIMEOUT, no target data received for more than 3s -> switch to IDLE"); ESP_LOGE(TAG, "TIMEOUT, no target data received for more than %ds -> switch to IDLE", TIMEOUT_IDLE_WHEN_NO_COMMAND/1000);
} }
//--- calculate increment --- //--- calculate increment ---
@ -131,14 +134,15 @@ void controlledMotor::handle(){
dutyIncrementDecel = 100; dutyIncrementDecel = 100;
} }
//--- BRAKE --- //--- BRAKE ---
//brake immediately, update state, duty and exit this cycle of handle function //brake immediately, update state, duty and exit this cycle of handle function
if (state == motorstate_t::BRAKE){ if (state == motorstate_t::BRAKE){
motor.set(motorstate_t::BRAKE, 0); ESP_LOGD(TAG, "braking - skip fading");
dutyNow = 0; motor.set(motorstate_t::BRAKE, dutyTarget);
return; //no need to run the fade algorithm //dutyNow = 0;
} return; //no need to run the fade algorithm
}
//--- calculate difference --- //--- calculate difference ---

View File

@ -1,4 +1,6 @@
#include "motordrivers.hpp" #include "motordrivers.hpp"
#include "esp_log.h"
#include "types.hpp"
//TODO: move from ledc to mcpwm? //TODO: move from ledc to mcpwm?
//https://docs.espressif.com/projects/esp-idf/en/v4.3/esp32/api-reference/peripherals/mcpwm.html# //https://docs.espressif.com/projects/esp-idf/en/v4.3/esp32/api-reference/peripherals/mcpwm.html#
@ -10,6 +12,8 @@
//tag for logging //tag for logging
static const char * TAG = "motordriver"; static const char * TAG = "motordriver";
//ms to wait in IDLE before BRAKE until relay actually switched
#define BRAKE_RELAY_DELAY_MS 300
//==================================== //====================================
@ -61,11 +65,13 @@ void single100a::init(){
gpio_set_direction(config.gpio_a, GPIO_MODE_OUTPUT); gpio_set_direction(config.gpio_a, GPIO_MODE_OUTPUT);
gpio_pad_select_gpio(config.gpio_b); gpio_pad_select_gpio(config.gpio_b);
gpio_set_direction(config.gpio_b, GPIO_MODE_OUTPUT); gpio_set_direction(config.gpio_b, GPIO_MODE_OUTPUT);
gpio_pad_select_gpio(config.gpio_brakeRelay);
gpio_set_direction(config.gpio_brakeRelay, GPIO_MODE_OUTPUT);
//--- calculate max duty according to selected resolution --- //--- calculate max duty according to selected resolution ---
dutyMax = pow(2, ledc_timer.duty_resolution) -1; dutyMax = pow(2, ledc_timer.duty_resolution) -1;
ESP_LOGI(TAG, "initialized single100a driver"); ESP_LOGW(TAG, "initialized single100a driver");
ESP_LOGI(TAG, "resolution=%dbit, dutyMax value=%d, resolution=%.4f %%", ledc_timer.duty_resolution, dutyMax, 100/(float)dutyMax); ESP_LOGW(TAG, "resolution=%dbit, dutyMax value=%d, resolution=%.4f %%", ledc_timer.duty_resolution, dutyMax, 100/(float)dutyMax);
} }
@ -80,7 +86,7 @@ void single100a::set(motorstate_t state_f, float duty_f){
uint32_t dutyScaled; uint32_t dutyScaled;
if (duty_f > 100) { //target duty above 100% if (duty_f > 100) { //target duty above 100%
dutyScaled = dutyMax; dutyScaled = dutyMax;
} else if (duty_f <= 0) { //target at or below 0% } else if (duty_f < 0) { //target at or below 0%
state_f = motorstate_t::IDLE; state_f = motorstate_t::IDLE;
dutyScaled = 0; dutyScaled = 0;
} else { //target duty 0-100% } else { //target duty 0-100%
@ -88,7 +94,18 @@ void single100a::set(motorstate_t state_f, float duty_f){
dutyScaled = duty_f / 100 * dutyMax; dutyScaled = duty_f / 100 * dutyMax;
} }
ESP_LOGV(TAG, "target-state=%s, duty=%d/%d, duty_input=%.3f%%", motorstateStr[(int)state_f], dutyScaled, dutyMax, duty_f);
//TODO: only when previous mode was BRAKE?
if (state_f != motorstate_t::BRAKE){
//reset brake wait state
brakeWaitingForRelay = false;
//turn of brake relay
gpio_set_level(config.gpio_brakeRelay, 0);
}
//put the single100a h-bridge module in the desired state update duty-cycle //put the single100a h-bridge module in the desired state update duty-cycle
//TODO maybe outsource mode code from once switch case? e.g. idle() brake()...
switch (state_f){ switch (state_f){
case motorstate_t::IDLE: case motorstate_t::IDLE:
ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, dutyScaled); ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, dutyScaled);
@ -100,13 +117,37 @@ void single100a::set(motorstate_t state_f, float duty_f){
gpio_set_level(config.gpio_a, config.aEnabledPinState); gpio_set_level(config.gpio_a, config.aEnabledPinState);
gpio_set_level(config.gpio_b, config.bEnabledPinState); gpio_set_level(config.gpio_b, config.bEnabledPinState);
break; break;
case motorstate_t::BRAKE:
ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, 0); case motorstate_t::BRAKE:
ledc_update_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel); //prevent full short (no brake resistors) due to slow relay, also reduces switching load
//brake: if (!brakeWaitingForRelay){
gpio_set_level(config.gpio_a, !config.aEnabledPinState); ESP_LOGW(TAG, "BRAKE: turned on relay, waiting in IDLE for %d ms, then apply brake", BRAKE_RELAY_DELAY_MS);
gpio_set_level(config.gpio_b, !config.bEnabledPinState); //switch driver to IDLE for now
break; gpio_set_level(config.gpio_a, config.aEnabledPinState);
gpio_set_level(config.gpio_b, config.bEnabledPinState);
//start brake relay
gpio_set_level(config.gpio_brakeRelay, 1);
timestamp_brakeRelayPowered = esp_log_timestamp();
brakeWaitingForRelay = true;
}
//check if delay for relay to switch has passed
else if ((esp_log_timestamp() - timestamp_brakeRelayPowered) > BRAKE_RELAY_DELAY_MS) {
ESP_LOGD(TAG, "applying brake with brake-resistors at duty=%.2f%%", duty_f);
//switch driver to BRAKE
gpio_set_level(config.gpio_a, !config.aEnabledPinState);
gpio_set_level(config.gpio_b, !config.bEnabledPinState);
//apply duty
//FIXME switch between BREAK and IDLE with PWM and ignore pwm-pin? (needs test)
ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, dutyScaled);
ledc_update_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel);
} else {
//waiting... stay in IDLE
ESP_LOGD(TAG, "waiting for brake relay to close (IDLE)...");
gpio_set_level(config.gpio_a, config.aEnabledPinState);
gpio_set_level(config.gpio_b, config.bEnabledPinState);
}
break;
case motorstate_t::FWD: case motorstate_t::FWD:
ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, dutyScaled); ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, dutyScaled);
ledc_update_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel); ledc_update_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel);
@ -114,6 +155,7 @@ void single100a::set(motorstate_t state_f, float duty_f){
gpio_set_level(config.gpio_a, config.aEnabledPinState); gpio_set_level(config.gpio_a, config.aEnabledPinState);
gpio_set_level(config.gpio_b, !config.bEnabledPinState); gpio_set_level(config.gpio_b, !config.bEnabledPinState);
break; break;
case motorstate_t::REV: case motorstate_t::REV:
ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, dutyScaled); ledc_set_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel, dutyScaled);
ledc_update_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel); ledc_update_duty(LEDC_HIGH_SPEED_MODE, config.ledc_channel);
@ -122,5 +164,4 @@ void single100a::set(motorstate_t state_f, float duty_f){
gpio_set_level(config.gpio_b, config.bEnabledPinState); gpio_set_level(config.gpio_b, config.bEnabledPinState);
break; break;
} }
ESP_LOGD(TAG, "set module to state=%s, duty=%d/%d, duty_input=%.3f%%", motorstateStr[(int)state_f], dutyScaled, dutyMax, duty_f);
} }

View File

@ -29,6 +29,7 @@ typedef struct single100a_config_t {
gpio_num_t gpio_pwm; gpio_num_t gpio_pwm;
gpio_num_t gpio_a; gpio_num_t gpio_a;
gpio_num_t gpio_b; gpio_num_t gpio_b;
gpio_num_t gpio_brakeRelay;
ledc_timer_t ledc_timer; ledc_timer_t ledc_timer;
ledc_channel_t ledc_channel; ledc_channel_t ledc_channel;
bool aEnabledPinState; bool aEnabledPinState;
@ -59,4 +60,6 @@ class single100a {
single100a_config_t config; single100a_config_t config;
uint32_t dutyMax; uint32_t dutyMax;
motorstate_t state = motorstate_t::IDLE; motorstate_t state = motorstate_t::IDLE;
bool brakeWaitingForRelay = false;
uint32_t timestamp_brakeRelayPowered;
}; };