Major Rework all files - Pass pointers to tasks, Remove gloabl variables
- All files: Modify almost all files to adjust functions and classes to work with pointers to objects passed at task creation instead of global variables from config.hpp - Remove/clear config.hpp to get rid of all global variables - main.cpp - Create pointer to all shared (used in multiple tasks) objects in main - remove evaluatedSwitch button object, since joystick library is used to get switch events - changes HTTP-mode - always init http-server (do not enable/disable at mode change) - pass url-handle function to init-htpp function - add lambda function to pass method of instance for thatMajor Rework all files - Remove global variables, pass pointers to tasks - All files: Modify almost all files to adjust functions and classes to work with pointers to objects passed at task creation instead of global variables from config.hpp - Remove/clear config.hpp to get rid of all global variables - main.cpp - Create pointer to all shared (used in multiple tasks) objects in main - remove evaluatedSwitch button object, since joystick library is used to get switch events - changes HTTP-mode - always init http-server (do not enable/disable at mode change) - pass url-handle function to init-htpp function - add lambda function to pass method of instance for that NOTES: - tested on breakoutboard only - known issue that slow encoder events are not recognized (especially in menu) - slowing down motorctl helps
This commit is contained in:
parent
40a0f56208
commit
2fcf17feda
@ -8,9 +8,12 @@ static const char * TAG = "automatedArmchair";
|
|||||||
//=============================
|
//=============================
|
||||||
//======== constructor ========
|
//======== constructor ========
|
||||||
//=============================
|
//=============================
|
||||||
automatedArmchair::automatedArmchair(void) {
|
automatedArmchair_c::automatedArmchair_c(controlledMotor *motorLeft_f, controlledMotor *motorRight_f)
|
||||||
//create command queue
|
{
|
||||||
commandQueue = xQueueCreate( 32, sizeof( commandSimple_t ) ); //TODO add max size to config?
|
motorLeft = motorLeft_f;
|
||||||
|
motorRight = motorRight_f;
|
||||||
|
// create command queue
|
||||||
|
commandQueue = xQueueCreate(32, sizeof(commandSimple_t)); // TODO add max size to config?
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -18,7 +21,7 @@ automatedArmchair::automatedArmchair(void) {
|
|||||||
//==============================
|
//==============================
|
||||||
//====== generateCommands ======
|
//====== generateCommands ======
|
||||||
//==============================
|
//==============================
|
||||||
motorCommands_t automatedArmchair::generateCommands(auto_instruction_t * instruction) {
|
motorCommands_t automatedArmchair_c::generateCommands(auto_instruction_t * instruction) {
|
||||||
//reset instruction
|
//reset instruction
|
||||||
*instruction = auto_instruction_t::NONE;
|
*instruction = auto_instruction_t::NONE;
|
||||||
//check if previous command is finished
|
//check if previous command is finished
|
||||||
@ -29,10 +32,10 @@ motorCommands_t automatedArmchair::generateCommands(auto_instruction_t * instruc
|
|||||||
//copy instruction to be provided to control task
|
//copy instruction to be provided to control task
|
||||||
*instruction = cmdCurrent.instruction;
|
*instruction = cmdCurrent.instruction;
|
||||||
//set acceleration / fading parameters according to command
|
//set acceleration / fading parameters according to command
|
||||||
motorLeft.setFade(fadeType_t::DECEL, cmdCurrent.fadeDecel);
|
motorLeft->setFade(fadeType_t::DECEL, cmdCurrent.fadeDecel);
|
||||||
motorRight.setFade(fadeType_t::DECEL, cmdCurrent.fadeDecel);
|
motorRight->setFade(fadeType_t::DECEL, cmdCurrent.fadeDecel);
|
||||||
motorLeft.setFade(fadeType_t::ACCEL, cmdCurrent.fadeAccel);
|
motorLeft->setFade(fadeType_t::ACCEL, cmdCurrent.fadeAccel);
|
||||||
motorRight.setFade(fadeType_t::ACCEL, cmdCurrent.fadeAccel);
|
motorRight->setFade(fadeType_t::ACCEL, cmdCurrent.fadeAccel);
|
||||||
//calculate timestamp the command is finished
|
//calculate timestamp the command is finished
|
||||||
timestampCmdFinished = esp_log_timestamp() + cmdCurrent.msDuration;
|
timestampCmdFinished = esp_log_timestamp() + cmdCurrent.msDuration;
|
||||||
//copy the new commands
|
//copy the new commands
|
||||||
@ -55,7 +58,7 @@ motorCommands_t automatedArmchair::generateCommands(auto_instruction_t * instruc
|
|||||||
//======== addCommand ========
|
//======== addCommand ========
|
||||||
//============================
|
//============================
|
||||||
//function that adds a basic command to the queue
|
//function that adds a basic command to the queue
|
||||||
void automatedArmchair::addCommand(commandSimple_t command) {
|
void automatedArmchair_c::addCommand(commandSimple_t command) {
|
||||||
//add command to queue
|
//add command to queue
|
||||||
if ( xQueueSend( commandQueue, ( void * )&command, ( TickType_t ) 0 ) ){
|
if ( xQueueSend( commandQueue, ( void * )&command, ( TickType_t ) 0 ) ){
|
||||||
ESP_LOGI(TAG, "Successfully inserted command to queue");
|
ESP_LOGI(TAG, "Successfully inserted command to queue");
|
||||||
@ -64,7 +67,7 @@ void automatedArmchair::addCommand(commandSimple_t command) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void automatedArmchair::addCommands(commandSimple_t commands[], size_t count) {
|
void automatedArmchair_c::addCommands(commandSimple_t commands[], size_t count) {
|
||||||
for (int i = 0; i < count; i++) {
|
for (int i = 0; i < count; i++) {
|
||||||
ESP_LOGI(TAG, "Reading command no. %d from provided array", i);
|
ESP_LOGI(TAG, "Reading command no. %d from provided array", i);
|
||||||
addCommand(commands[i]);
|
addCommand(commands[i]);
|
||||||
@ -77,7 +80,7 @@ void automatedArmchair::addCommands(commandSimple_t commands[], size_t count) {
|
|||||||
//===============================
|
//===============================
|
||||||
//function that deletes all pending/queued commands
|
//function that deletes all pending/queued commands
|
||||||
//e.g. when switching modes
|
//e.g. when switching modes
|
||||||
motorCommands_t automatedArmchair::clearCommands() {
|
motorCommands_t automatedArmchair_c::clearCommands() {
|
||||||
//clear command queue
|
//clear command queue
|
||||||
xQueueReset( commandQueue );
|
xQueueReset( commandQueue );
|
||||||
ESP_LOGW(TAG, "command queue was successfully emptied");
|
ESP_LOGW(TAG, "command queue was successfully emptied");
|
||||||
|
@ -33,13 +33,13 @@ typedef struct commandSimple_t{
|
|||||||
|
|
||||||
|
|
||||||
//------------------------------------
|
//------------------------------------
|
||||||
//----- automatedArmchair class -----
|
//----- automatedArmchair_c class -----
|
||||||
//------------------------------------
|
//------------------------------------
|
||||||
class automatedArmchair {
|
class automatedArmchair_c {
|
||||||
public:
|
public:
|
||||||
//--- methods ---
|
//--- methods ---
|
||||||
//constructor
|
//constructor
|
||||||
automatedArmchair(void);
|
automatedArmchair_c(controlledMotor * motorLeft, controlledMotor * motorRight);
|
||||||
//function to generate motor commands
|
//function to generate motor commands
|
||||||
//can be also seen as handle function
|
//can be also seen as handle function
|
||||||
//TODO: go with other approach: separate task for handling auto mode
|
//TODO: go with other approach: separate task for handling auto mode
|
||||||
@ -62,6 +62,8 @@ class automatedArmchair {
|
|||||||
private:
|
private:
|
||||||
//--- methods ---
|
//--- methods ---
|
||||||
//--- objects ---
|
//--- objects ---
|
||||||
|
controlledMotor * motorLeft;
|
||||||
|
controlledMotor * motorRight;
|
||||||
//TODO: add buzzer here
|
//TODO: add buzzer here
|
||||||
//--- variables ---
|
//--- variables ---
|
||||||
//queue for storing pending commands
|
//queue for storing pending commands
|
||||||
@ -124,7 +126,7 @@ if (trigger){
|
|||||||
.instruction = auto_instruction_t::SWITCH_JOYSTICK_MODE
|
.instruction = auto_instruction_t::SWITCH_JOYSTICK_MODE
|
||||||
};
|
};
|
||||||
|
|
||||||
//send commands to automatedArmchair command queue
|
//send commands to automatedArmchair_c command queue
|
||||||
armchair.addCommands(cmds, 3);
|
armchair.addCommands(cmds, 3);
|
||||||
|
|
||||||
//change mode to AUTO
|
//change mode to AUTO
|
||||||
|
@ -10,29 +10,44 @@ extern "C"
|
|||||||
#include "button.hpp"
|
#include "button.hpp"
|
||||||
#include "encoder.hpp"
|
#include "encoder.hpp"
|
||||||
|
|
||||||
|
// tag for logging
|
||||||
|
static const char *TAG = "button";
|
||||||
|
|
||||||
|
//======================================
|
||||||
//tag for logging
|
//============ button task =============
|
||||||
static const char * TAG = "button";
|
//======================================
|
||||||
|
// task that handles the button interface/commands
|
||||||
|
void task_button(void *task_button_parameters)
|
||||||
|
{
|
||||||
|
task_button_parameters_t *objects = (task_button_parameters_t *)task_button_parameters;
|
||||||
|
ESP_LOGI(TAG, "Initializing command-button and starting handle loop");
|
||||||
|
// create button instance
|
||||||
|
buttonCommands commandButton(objects->control, objects->joystick, objects->encoderQueue, objects->motorLeft, objects->motorRight, objects->buzzer);
|
||||||
|
// start handle loop
|
||||||
|
commandButton.startHandleLoop();
|
||||||
|
}
|
||||||
|
|
||||||
//-----------------------------
|
//-----------------------------
|
||||||
//-------- constructor --------
|
//-------- constructor --------
|
||||||
//-----------------------------
|
//-----------------------------
|
||||||
buttonCommands::buttonCommands(gpio_evaluatedSwitch * button_f, evaluatedJoystick * joystick_f, controlledArmchair * control_f, buzzer_t * buzzer_f, controlledMotor * motorLeft_f, controlledMotor * motorRight_f){
|
buttonCommands::buttonCommands(
|
||||||
//copy object pointers
|
controlledArmchair *control_f,
|
||||||
button = button_f;
|
evaluatedJoystick *joystick_f,
|
||||||
joystick = joystick_f;
|
QueueHandle_t encoderQueue_f,
|
||||||
|
controlledMotor *motorLeft_f,
|
||||||
|
controlledMotor *motorRight_f,
|
||||||
|
buzzer_t *buzzer_f)
|
||||||
|
{
|
||||||
|
// copy object pointers
|
||||||
control = control_f;
|
control = control_f;
|
||||||
buzzer = buzzer_f;
|
joystick = joystick_f;
|
||||||
|
encoderQueue = encoderQueue_f;
|
||||||
motorLeft = motorLeft_f;
|
motorLeft = motorLeft_f;
|
||||||
motorRight = motorRight_f;
|
motorRight = motorRight_f;
|
||||||
//TODO declare / configure evaluatedSwitch here instead of config (unnecessary that button object is globally available - only used here)?
|
buzzer = buzzer_f;
|
||||||
|
// TODO declare / configure evaluatedSwitch here instead of config (unnecessary that button object is globally available - only used here)?
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//----------------------------
|
//----------------------------
|
||||||
//--------- action -----------
|
//--------- action -----------
|
||||||
//----------------------------
|
//----------------------------
|
||||||
@ -40,7 +55,7 @@ buttonCommands::buttonCommands(gpio_evaluatedSwitch * button_f, evaluatedJoystic
|
|||||||
void buttonCommands::action (uint8_t count, bool lastPressLong){
|
void buttonCommands::action (uint8_t count, bool lastPressLong){
|
||||||
//--- variables ---
|
//--- variables ---
|
||||||
bool decelEnabled; //for different beeping when toggling
|
bool decelEnabled; //for different beeping when toggling
|
||||||
commandSimple_t cmds[8]; //array for commands for automatedArmchair
|
commandSimple_t cmds[8]; //array for commands for automatedArmchair_c
|
||||||
|
|
||||||
//--- get joystick position ---
|
//--- get joystick position ---
|
||||||
//in case joystick is used for additional cases:
|
//in case joystick is used for additional cases:
|
||||||
|
@ -17,14 +17,13 @@
|
|||||||
class buttonCommands {
|
class buttonCommands {
|
||||||
public:
|
public:
|
||||||
//--- constructor ---
|
//--- constructor ---
|
||||||
buttonCommands (
|
buttonCommands(
|
||||||
gpio_evaluatedSwitch * button_f,
|
controlledArmchair *control_f,
|
||||||
evaluatedJoystick * joystick_f,
|
evaluatedJoystick *joystick_f,
|
||||||
controlledArmchair * control_f,
|
QueueHandle_t encoderQueue_f,
|
||||||
buzzer_t * buzzer_f,
|
controlledMotor * motorLeft_f,
|
||||||
controlledMotor * motorLeft_f,
|
controlledMotor *motorRight_f,
|
||||||
controlledMotor * motorRight_f
|
buzzer_t *buzzer_f);
|
||||||
);
|
|
||||||
|
|
||||||
//--- functions ---
|
//--- functions ---
|
||||||
//the following function has to be started once in a separate task.
|
//the following function has to be started once in a separate task.
|
||||||
@ -36,12 +35,12 @@ class buttonCommands {
|
|||||||
void action(uint8_t count, bool lastPressLong);
|
void action(uint8_t count, bool lastPressLong);
|
||||||
|
|
||||||
//--- objects ---
|
//--- objects ---
|
||||||
gpio_evaluatedSwitch* button;
|
|
||||||
evaluatedJoystick* joystick;
|
|
||||||
controlledArmchair * control;
|
controlledArmchair * control;
|
||||||
buzzer_t* buzzer;
|
evaluatedJoystick* joystick;
|
||||||
controlledMotor * motorLeft;
|
controlledMotor * motorLeft;
|
||||||
controlledMotor * motorRight;
|
controlledMotor * motorRight;
|
||||||
|
buzzer_t* buzzer;
|
||||||
|
QueueHandle_t encoderQueue;
|
||||||
|
|
||||||
//--- variables ---
|
//--- variables ---
|
||||||
uint8_t count = 0;
|
uint8_t count = 0;
|
||||||
@ -51,3 +50,21 @@ class buttonCommands {
|
|||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//======================================
|
||||||
|
//============ button task =============
|
||||||
|
//======================================
|
||||||
|
// struct with variables passed to task from main
|
||||||
|
typedef struct task_button_parameters_t
|
||||||
|
{
|
||||||
|
controlledArmchair *control;
|
||||||
|
evaluatedJoystick *joystick;
|
||||||
|
QueueHandle_t encoderQueue;
|
||||||
|
controlledMotor *motorLeft;
|
||||||
|
controlledMotor *motorRight;
|
||||||
|
buzzer_t *buzzer;
|
||||||
|
} task_button_parameters_t;
|
||||||
|
|
||||||
|
//task that handles the button interface/commands
|
||||||
|
void task_button( void * task_button_parameters );
|
@ -1,8 +1,54 @@
|
|||||||
#include "config.hpp"
|
// NOTE: this file is included in main.cpp only.
|
||||||
|
// outsourced all configuration related functions and structures to this file:
|
||||||
|
|
||||||
//===================================
|
#include "motordrivers.hpp"
|
||||||
//======= motor configuration =======
|
#include "motorctl.hpp"
|
||||||
//===================================
|
#include "joystick.hpp"
|
||||||
|
#include "http.hpp"
|
||||||
|
#include "speedsensor.hpp"
|
||||||
|
|
||||||
|
#include "buzzer.hpp"
|
||||||
|
#include "control.hpp"
|
||||||
|
#include "fan.hpp"
|
||||||
|
#include "auto.hpp"
|
||||||
|
#include "chairAdjust.hpp"
|
||||||
|
|
||||||
|
//==================================
|
||||||
|
//======== define loglevels ========
|
||||||
|
//==================================
|
||||||
|
void setLoglevels(void)
|
||||||
|
{
|
||||||
|
// 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("buzzer", ESP_LOG_ERROR);
|
||||||
|
// esp_log_level_set("motordriver", ESP_LOG_DEBUG);
|
||||||
|
esp_log_level_set("motor-control", ESP_LOG_WARN);
|
||||||
|
// esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG);
|
||||||
|
// esp_log_level_set("joystickCommands", ESP_LOG_DEBUG);
|
||||||
|
esp_log_level_set("button", ESP_LOG_INFO);
|
||||||
|
esp_log_level_set("control", ESP_LOG_INFO);
|
||||||
|
// esp_log_level_set("fan-control", ESP_LOG_INFO);
|
||||||
|
esp_log_level_set("wifi", ESP_LOG_INFO);
|
||||||
|
esp_log_level_set("http", ESP_LOG_INFO);
|
||||||
|
// esp_log_level_set("automatedArmchair", ESP_LOG_DEBUG);
|
||||||
|
esp_log_level_set("display", ESP_LOG_INFO);
|
||||||
|
// esp_log_level_set("current-sensors", ESP_LOG_INFO);
|
||||||
|
// esp_log_level_set("speedSensor", ESP_LOG_INFO);
|
||||||
|
esp_log_level_set("chair-adjustment", ESP_LOG_INFO);
|
||||||
|
esp_log_level_set("menu", ESP_LOG_INFO);
|
||||||
|
esp_log_level_set("encoder", ESP_LOG_INFO);
|
||||||
|
}
|
||||||
|
|
||||||
|
//==================================
|
||||||
|
//======== configuration ===========
|
||||||
|
//==================================
|
||||||
|
|
||||||
|
//-----------------------------------
|
||||||
|
//------- motor configuration -------
|
||||||
|
//-----------------------------------
|
||||||
/* ==> currently using other driver
|
/* ==> currently using other driver
|
||||||
//--- configure left motor (hardware) ---
|
//--- configure left motor (hardware) ---
|
||||||
single100a_config_t configDriverLeft = {
|
single100a_config_t configDriverLeft = {
|
||||||
@ -11,8 +57,8 @@ single100a_config_t configDriverLeft = {
|
|||||||
.gpio_b = GPIO_NUM_4,
|
.gpio_b = GPIO_NUM_4,
|
||||||
.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)
|
||||||
.bEnabledPinState = false,
|
.bEnabledPinState = false,
|
||||||
.resolution = LEDC_TIMER_11_BIT,
|
.resolution = LEDC_TIMER_11_BIT,
|
||||||
.pwmFreq = 10000
|
.pwmFreq = 10000
|
||||||
};
|
};
|
||||||
@ -24,180 +70,114 @@ single100a_config_t configDriverRight = {
|
|||||||
.gpio_b = GPIO_NUM_14,
|
.gpio_b = GPIO_NUM_14,
|
||||||
.ledc_timer = LEDC_TIMER_1,
|
.ledc_timer = LEDC_TIMER_1,
|
||||||
.ledc_channel = LEDC_CHANNEL_1,
|
.ledc_channel = LEDC_CHANNEL_1,
|
||||||
.aEnabledPinState = false, //-> pin inverted (mosfet)
|
.aEnabledPinState = false, //-> pin inverted (mosfet)
|
||||||
.bEnabledPinState = true, //-> not inverted (direct)
|
.bEnabledPinState = true, //-> not inverted (direct)
|
||||||
.resolution = LEDC_TIMER_11_BIT,
|
.resolution = LEDC_TIMER_11_BIT,
|
||||||
.pwmFreq = 10000
|
.pwmFreq = 10000
|
||||||
};
|
};
|
||||||
*/
|
*/
|
||||||
|
|
||||||
//--- configure sabertooth driver --- (controls both motors in one instance)
|
//--- configure sabertooth driver --- (controls both motors in one instance)
|
||||||
sabertooth2x60_config_t sabertoothConfig = {
|
sabertooth2x60_config_t sabertoothConfig = {
|
||||||
.gpio_TX = GPIO_NUM_25,
|
.gpio_TX = GPIO_NUM_25,
|
||||||
.uart_num = UART_NUM_2
|
.uart_num = UART_NUM_2};
|
||||||
};
|
|
||||||
|
|
||||||
|
// TODO add motor name string -> then use as log tag?
|
||||||
//TODO add motor name string -> then use as log tag?
|
|
||||||
//--- configure left motor (contol) ---
|
//--- configure left motor (contol) ---
|
||||||
motorctl_config_t configMotorControlLeft = {
|
motorctl_config_t configMotorControlLeft = {
|
||||||
.msFadeAccel = 1500, //acceleration of the motor (ms it takes from 0% to 100%)
|
.msFadeAccel = 1500, // 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 = false,
|
.currentLimitEnabled = false,
|
||||||
.currentSensor_adc = ADC1_CHANNEL_4, //GPIO32
|
.currentSensor_adc = ADC1_CHANNEL_4, // GPIO32
|
||||||
.currentSensor_ratedCurrent = 50,
|
.currentSensor_ratedCurrent = 50,
|
||||||
.currentMax = 30,
|
.currentMax = 30,
|
||||||
.deadTimeMs = 0 //minimum time motor is off between direction change
|
.deadTimeMs = 0 // minimum time motor is off between direction change
|
||||||
};
|
};
|
||||||
|
|
||||||
//--- configure right motor (contol) ---
|
//--- configure right motor (contol) ---
|
||||||
motorctl_config_t configMotorControlRight = {
|
motorctl_config_t configMotorControlRight = {
|
||||||
.msFadeAccel = 1500, //acceleration of the motor (ms it takes from 0% to 100%)
|
.msFadeAccel = 1500, // 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 = false,
|
.currentLimitEnabled = false,
|
||||||
.currentSensor_adc = ADC1_CHANNEL_5, //GPIO33
|
.currentSensor_adc = ADC1_CHANNEL_5, // GPIO33
|
||||||
.currentSensor_ratedCurrent = 50,
|
.currentSensor_ratedCurrent = 50,
|
||||||
.currentMax = 30,
|
.currentMax = 30,
|
||||||
.deadTimeMs = 0 //minimum time motor is off between direction change
|
.deadTimeMs = 0 // minimum time motor is off between direction change
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//------------------------------
|
||||||
|
//------- control config -------
|
||||||
//==============================
|
//------------------------------
|
||||||
//======= control config =======
|
|
||||||
//==============================
|
|
||||||
control_config_t configControl = {
|
control_config_t configControl = {
|
||||||
.defaultMode = controlMode_t::JOYSTICK, //default mode after startup and toggling IDLE
|
.defaultMode = controlMode_t::JOYSTICK, // default mode after startup and toggling IDLE
|
||||||
//--- timeout ---
|
//--- timeout ---
|
||||||
.timeoutMs = 3*60*1000, //time of inactivity after which the mode gets switched to IDLE
|
.timeoutMs = 3 * 60 * 1000, // time of inactivity after which the mode gets switched to IDLE
|
||||||
.timeoutTolerancePer = 5, //percentage the duty can vary between timeout checks considered still inactive
|
.timeoutTolerancePer = 5, // percentage the duty can vary between timeout checks considered still inactive
|
||||||
//--- http mode ---
|
//--- http mode ---
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//-------------------------------
|
||||||
|
//----- httpJoystick config -----
|
||||||
//===============================
|
//-------------------------------
|
||||||
//===== httpJoystick config =====
|
|
||||||
//===============================
|
|
||||||
httpJoystick_config_t configHttpJoystickMain{
|
httpJoystick_config_t configHttpJoystickMain{
|
||||||
.toleranceZeroX_Per = 1, //percentage around joystick axis the coordinate snaps to 0
|
.toleranceZeroX_Per = 1, // percentage around joystick axis the coordinate snaps to 0
|
||||||
.toleranceZeroY_Per = 6,
|
.toleranceZeroY_Per = 6,
|
||||||
.toleranceEndPer = 2, //percentage before joystick end the coordinate snaps to 1/-1
|
.toleranceEndPer = 2, // percentage before joystick end the coordinate snaps to 1/-1
|
||||||
.timeoutMs = 2500 //time no new data was received before the motors get turned off
|
.timeoutMs = 2500 // time no new data was received before the motors get turned off
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//--------------------------------------
|
||||||
|
//------- joystick configuration -------
|
||||||
//======================================
|
//--------------------------------------
|
||||||
//======= joystick configuration =======
|
|
||||||
//======================================
|
|
||||||
joystick_config_t configJoystick = {
|
joystick_config_t configJoystick = {
|
||||||
.adc_x = ADC1_CHANNEL_0, //GPIO36
|
.adc_x = ADC1_CHANNEL_0, // GPIO36
|
||||||
.adc_y = ADC1_CHANNEL_3, //GPIO39
|
.adc_y = ADC1_CHANNEL_3, // GPIO39
|
||||||
//percentage of joystick range the coordinate of the axis snaps to 0 (0-100)
|
// percentage of joystick range the coordinate of the axis snaps to 0 (0-100)
|
||||||
.tolerance_zeroX_per = 7, //6
|
.tolerance_zeroX_per = 7, // 6
|
||||||
.tolerance_zeroY_per = 10, //7
|
.tolerance_zeroY_per = 10, // 7
|
||||||
//percentage of joystick range the coordinate snaps to -1 or 1 before configured "_max" or "_min" threshold (mechanical end) is reached (0-100)
|
// percentage of joystick range the coordinate snaps to -1 or 1 before configured "_max" or "_min" threshold (mechanical end) is reached (0-100)
|
||||||
.tolerance_end_per = 4,
|
.tolerance_end_per = 4,
|
||||||
//threshold the radius jumps to 1 before the stick is at max radius (range 0-1)
|
// threshold the radius jumps to 1 before the stick is at max radius (range 0-1)
|
||||||
.tolerance_radius = 0.09,
|
.tolerance_radius = 0.09,
|
||||||
|
|
||||||
//min and max adc values of each axis, !!!AFTER INVERSION!!! is applied:
|
// min and max adc values of each axis, !!!AFTER INVERSION!!! is applied:
|
||||||
.x_min = 1710, //=> x=-1
|
.x_min = 1710, //=> x=-1
|
||||||
.x_max = 2980, //=> x=1
|
.x_max = 2980, //=> x=1
|
||||||
.y_min = 1700, //=> y=-1
|
.y_min = 1700, //=> y=-1
|
||||||
.y_max = 2940, //=> y=1
|
.y_max = 2940, //=> y=1
|
||||||
//invert adc measurement
|
// invert adc measurement
|
||||||
.x_inverted = true,
|
.x_inverted = true,
|
||||||
.y_inverted = true
|
.y_inverted = true};
|
||||||
};
|
|
||||||
|
|
||||||
|
//----------------------------
|
||||||
|
//--- configure fan contol ---
|
||||||
//============================
|
//----------------------------
|
||||||
//=== configure fan contol ===
|
|
||||||
//============================
|
|
||||||
fan_config_t configCooling = {
|
fan_config_t configCooling = {
|
||||||
.gpio_fan = GPIO_NUM_13,
|
.gpio_fan = GPIO_NUM_13,
|
||||||
.dutyThreshold = 40,
|
.dutyThreshold = 40,
|
||||||
.minOnMs = 1500,
|
.minOnMs = 1500,
|
||||||
.minOffMs = 3000,
|
.minOffMs = 3000,
|
||||||
.turnOffDelayMs = 5000,
|
.turnOffDelayMs = 5000,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//--------------------------------------------
|
||||||
|
//-------- speed sensor configuration --------
|
||||||
//============================================
|
//--------------------------------------------
|
||||||
//======== speed sensor configuration ========
|
|
||||||
//============================================
|
|
||||||
speedSensor_config_t speedLeft_config{
|
speedSensor_config_t speedLeft_config{
|
||||||
.gpioPin = GPIO_NUM_5,
|
.gpioPin = GPIO_NUM_5,
|
||||||
.degreePerGroup = 360/5,
|
.degreePerGroup = 360 / 5,
|
||||||
.tireCircumferenceMeter = 210.0*3.141/1000.0,
|
.tireCircumferenceMeter = 210.0 * 3.141 / 1000.0,
|
||||||
.directionInverted = false,
|
.directionInverted = false,
|
||||||
.logName = "speedLeft",
|
.logName = "speedLeft",
|
||||||
};
|
};
|
||||||
|
|
||||||
speedSensor_config_t speedRight_config{
|
speedSensor_config_t speedRight_config{
|
||||||
.gpioPin = GPIO_NUM_14,
|
.gpioPin = GPIO_NUM_14,
|
||||||
.degreePerGroup = 360/12,
|
.degreePerGroup = 360 / 12,
|
||||||
.tireCircumferenceMeter = 210.0*3.141/1000.0,
|
.tireCircumferenceMeter = 210.0 * 3.141 / 1000.0,
|
||||||
.directionInverted = true,
|
.directionInverted = true,
|
||||||
.logName = "speedRight",
|
.logName = "speedRight",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//=================================
|
|
||||||
//===== create global objects =====
|
|
||||||
//=================================
|
|
||||||
//TODO outsource global variables to e.g. global.cpp and only config options here?
|
|
||||||
//create sabertooth motor driver instance
|
|
||||||
sabertooth2x60a sabertoothDriver(sabertoothConfig);
|
|
||||||
|
|
||||||
|
|
||||||
//--- controlledMotor ---
|
|
||||||
//functions for updating the duty via certain/current driver that can then be passed to controlledMotor
|
|
||||||
//-> makes it possible to easily use different motor drivers
|
|
||||||
//note: ignoring warning "capture of variable 'sabertoothDriver' with non-automatic storage duration", since sabertoothDriver object does not get destroyed anywhere - no lifetime issue
|
|
||||||
motorSetCommandFunc_t setLeftFunc = [&sabertoothDriver](motorCommand_t cmd) {
|
|
||||||
sabertoothDriver.setLeft(cmd);
|
|
||||||
};
|
|
||||||
motorSetCommandFunc_t setRightFunc = [&sabertoothDriver](motorCommand_t cmd) {
|
|
||||||
sabertoothDriver.setRight(cmd);
|
|
||||||
};
|
|
||||||
//create controlled motor instances (motorctl.hpp)
|
|
||||||
controlledMotor motorLeft(setLeftFunc, configMotorControlLeft);
|
|
||||||
controlledMotor motorRight(setRightFunc, configMotorControlRight);
|
|
||||||
|
|
||||||
//create speedsensor instances
|
|
||||||
speedSensor speedLeft (speedLeft_config);
|
|
||||||
speedSensor speedRight (speedRight_config);
|
|
||||||
|
|
||||||
//create global joystic instance (joystick.hpp)
|
|
||||||
evaluatedJoystick joystick(configJoystick);
|
|
||||||
|
|
||||||
//create global evaluated switch instance for button next to joystick
|
|
||||||
gpio_evaluatedSwitch buttonJoystick(GPIO_NUM_21, true, false); //pullup true, not inverted (switch to GND use pullup of controller)
|
|
||||||
|
|
||||||
//create buzzer object on pin 12 with gap between queued events of 100ms
|
|
||||||
buzzer_t buzzer(GPIO_NUM_12, 100);
|
|
||||||
|
|
||||||
//create global httpJoystick object (http.hpp)
|
|
||||||
httpJoystick httpJoystickMain(configHttpJoystickMain);
|
|
||||||
|
|
||||||
//create global control object (control.hpp)
|
|
||||||
controlledArmchair control(configControl, &buzzer, &motorLeft, &motorRight, &joystick, &httpJoystickMain);
|
|
||||||
|
|
||||||
//create global automatedArmchair object (for auto-mode) (auto.hpp)
|
|
||||||
automatedArmchair armchair;
|
|
||||||
|
|
||||||
//create global objects for controlling the chair position
|
|
||||||
// gpio_up, gpio_down, name
|
|
||||||
cControlledRest legRest(GPIO_NUM_4, GPIO_NUM_16, "legRest");
|
|
||||||
cControlledRest backRest(GPIO_NUM_2, GPIO_NUM_15, "backRest");
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -13,44 +13,45 @@
|
|||||||
#include "speedsensor.hpp"
|
#include "speedsensor.hpp"
|
||||||
#include "chairAdjust.hpp"
|
#include "chairAdjust.hpp"
|
||||||
|
|
||||||
|
//
|
||||||
//in IDLE mode: set loglevel for evaluatedJoystick to DEBUG
|
////in IDLE mode: set loglevel for evaluatedJoystick to DEBUG
|
||||||
//and repeatedly read joystick e.g. for manually calibrating / testing joystick
|
////and repeatedly read joystick e.g. for manually calibrating / testing joystick
|
||||||
//#define JOYSTICK_LOG_IN_IDLE
|
////#define JOYSTICK_LOG_IN_IDLE
|
||||||
|
//
|
||||||
|
//
|
||||||
//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?
|
||||||
|
//
|
||||||
//create global controlledMotor instances for both motors
|
////create global controlledMotor instances for both motors
|
||||||
extern controlledMotor motorLeft;
|
//extern controlledMotor motorLeft;
|
||||||
extern controlledMotor motorRight;
|
//extern controlledMotor motorRight;
|
||||||
|
//
|
||||||
//create global joystic instance
|
////create global joystic instance
|
||||||
extern evaluatedJoystick joystick;
|
//extern evaluatedJoystick joystick;
|
||||||
|
//
|
||||||
//create global evaluated switch instance for button next to joystick
|
////create global evaluated switch instance for button next to joystick
|
||||||
extern gpio_evaluatedSwitch buttonJoystick;
|
//extern gpio_evaluatedSwitch buttonJoystick;
|
||||||
|
//
|
||||||
//create global buzzer object
|
////create global buzzer object
|
||||||
extern buzzer_t buzzer;
|
//extern buzzer_t buzzer;
|
||||||
|
//
|
||||||
//create global control object
|
////create global control object
|
||||||
extern controlledArmchair control;
|
//extern controlledArmchair control;
|
||||||
|
//
|
||||||
//create global automatedArmchair object (for auto-mode)
|
////create global automatedArmchair object (for auto-mode)
|
||||||
extern automatedArmchair armchair;
|
//extern automatedArmchair_c armchair;
|
||||||
|
//
|
||||||
//create global httpJoystick object
|
////create global httpJoystick object
|
||||||
//extern httpJoystick httpJoystickMain;
|
////extern httpJoystick httpJoystickMain;
|
||||||
|
//
|
||||||
//configuration for fans / cooling
|
////configuration for fans / cooling
|
||||||
extern fan_config_t configCooling;
|
//extern fan_config_t configCooling;
|
||||||
|
//
|
||||||
//create global objects for measuring speed
|
////create global objects for measuring speed
|
||||||
extern speedSensor speedLeft;
|
//extern speedSensor speedLeft;
|
||||||
extern speedSensor speedRight;
|
//extern speedSensor speedRight;
|
||||||
|
//
|
||||||
//create global objects for controlling the chair position
|
////create global objects for controlling the chair position
|
||||||
extern cControlledRest legRest;
|
//extern cControlledRest legRest;
|
||||||
extern cControlledRest backRest;
|
//extern cControlledRest backRest;
|
||||||
|
//
|
||||||
|
//
|
@ -26,14 +26,17 @@ const char* controlModeStr[9] = {"IDLE", "JOYSTICK", "MASSAGE", "HTTP", "MQTT",
|
|||||||
//-----------------------------
|
//-----------------------------
|
||||||
//-------- constructor --------
|
//-------- constructor --------
|
||||||
//-----------------------------
|
//-----------------------------
|
||||||
controlledArmchair::controlledArmchair (
|
controlledArmchair::controlledArmchair(
|
||||||
control_config_t config_f,
|
control_config_t config_f,
|
||||||
buzzer_t * buzzer_f,
|
buzzer_t *buzzer_f,
|
||||||
controlledMotor* motorLeft_f,
|
controlledMotor *motorLeft_f,
|
||||||
controlledMotor* motorRight_f,
|
controlledMotor *motorRight_f,
|
||||||
evaluatedJoystick* joystick_f,
|
evaluatedJoystick *joystick_f,
|
||||||
httpJoystick* httpJoystick_f
|
httpJoystick *httpJoystick_f,
|
||||||
){
|
automatedArmchair_c *automatedArmchair_f,
|
||||||
|
cControlledRest *legRest_f,
|
||||||
|
cControlledRest *backRest_f)
|
||||||
|
{
|
||||||
|
|
||||||
//copy configuration
|
//copy configuration
|
||||||
config = config_f;
|
config = config_f;
|
||||||
@ -43,13 +46,28 @@ controlledArmchair::controlledArmchair (
|
|||||||
motorRight = motorRight_f;
|
motorRight = motorRight_f;
|
||||||
joystick_l = joystick_f,
|
joystick_l = joystick_f,
|
||||||
httpJoystickMain_l = httpJoystick_f;
|
httpJoystickMain_l = httpJoystick_f;
|
||||||
|
automatedArmchair = automatedArmchair_f;
|
||||||
|
legRest = legRest_f;
|
||||||
|
backRest = backRest_f;
|
||||||
//set default mode from config
|
//set default mode from config
|
||||||
modePrevious = config.defaultMode;
|
modePrevious = config.defaultMode;
|
||||||
|
|
||||||
//TODO declare / configure controlled motors here instead of config (unnecessary that button object is globally available - only used here)?
|
//TODO declare / configure controlled motors here instead of config (unnecessary that button object is globally available - only used here)?
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//=======================================
|
||||||
|
//============ control task =============
|
||||||
|
//=======================================
|
||||||
|
//task that controls the armchair modes and initiates commands generation and applies them to driver
|
||||||
|
//parameter: pointer to controlledArmchair object
|
||||||
|
void task_control( void * pvParameters ){
|
||||||
|
//control_task_parameters_t * objects = (control_task_parameters_t *)pvParameters;
|
||||||
|
controlledArmchair * control = (controlledArmchair *)pvParameters;
|
||||||
|
ESP_LOGI(TAG, "Initializing controlledArmchair and starting handle loop");
|
||||||
|
//start handle loop (control object declared in config.hpp)
|
||||||
|
//objects->control->startHandleLoop();
|
||||||
|
control->startHandleLoop();
|
||||||
|
}
|
||||||
|
|
||||||
//----------------------------------
|
//----------------------------------
|
||||||
//---------- Handle loop -----------
|
//---------- Handle loop -----------
|
||||||
@ -70,7 +88,7 @@ void controlledArmchair::startHandleLoop() {
|
|||||||
commands = cmds_bothMotorsIdle;
|
commands = cmds_bothMotorsIdle;
|
||||||
motorRight->setTarget(commands.right.state, commands.right.duty);
|
motorRight->setTarget(commands.right.state, commands.right.duty);
|
||||||
motorLeft->setTarget(commands.left.state, commands.left.duty);
|
motorLeft->setTarget(commands.left.state, commands.left.duty);
|
||||||
vTaskDelay(200 / portTICK_PERIOD_MS);
|
vTaskDelay(300 / portTICK_PERIOD_MS);
|
||||||
#ifdef JOYSTICK_LOG_IN_IDLE
|
#ifdef JOYSTICK_LOG_IN_IDLE
|
||||||
//get joystick data here (without using it)
|
//get joystick data here (without using it)
|
||||||
//since loglevel is DEBUG, calculateion details is output
|
//since loglevel is DEBUG, calculateion details is output
|
||||||
@ -132,7 +150,7 @@ void controlledArmchair::startHandleLoop() {
|
|||||||
case controlMode_t::AUTO:
|
case controlMode_t::AUTO:
|
||||||
vTaskDelay(20 / portTICK_PERIOD_MS);
|
vTaskDelay(20 / portTICK_PERIOD_MS);
|
||||||
//generate commands
|
//generate commands
|
||||||
commands = armchair.generateCommands(&instruction);
|
commands = automatedArmchair->generateCommands(&instruction);
|
||||||
//--- apply commands to motors ---
|
//--- apply commands to motors ---
|
||||||
//TODO make motorctl.setTarget also accept motorcommand struct directly
|
//TODO make motorctl.setTarget also accept motorcommand struct directly
|
||||||
motorRight->setTarget(commands.right.state, commands.right.duty);
|
motorRight->setTarget(commands.right.state, commands.right.duty);
|
||||||
@ -179,7 +197,7 @@ void controlledArmchair::startHandleLoop() {
|
|||||||
motorRight->setTarget(commands.right.state, commands.right.duty);
|
motorRight->setTarget(commands.right.state, commands.right.duty);
|
||||||
motorLeft->setTarget(commands.left.state, commands.left.duty);
|
motorLeft->setTarget(commands.left.state, commands.left.duty);
|
||||||
//--- control armchair position with joystick input ---
|
//--- control armchair position with joystick input ---
|
||||||
controlChairAdjustment(joystick_l->getData(), &legRest, &backRest);
|
controlChairAdjustment(joystick_l->getData(), legRest, backRest);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
|
||||||
@ -359,21 +377,6 @@ void controlledArmchair::changeMode(controlMode_t modeNew) {
|
|||||||
buzzer->beep(1,200,100);
|
buzzer->beep(1,200,100);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case controlMode_t::HTTP:
|
|
||||||
ESP_LOGW(TAG, "switching from http mode -> disabling http and wifi");
|
|
||||||
//stop http server
|
|
||||||
ESP_LOGI(TAG, "disabling http server...");
|
|
||||||
http_stop_server();
|
|
||||||
|
|
||||||
//FIXME: make wifi function work here - currently starting wifi at startup (see notes main.cpp)
|
|
||||||
//stop wifi
|
|
||||||
//TODO: decide whether ap or client is currently used - which has to be disabled?
|
|
||||||
//ESP_LOGI(TAG, "deinit wifi...");
|
|
||||||
//wifi_deinit_client();
|
|
||||||
//wifi_deinit_ap();
|
|
||||||
ESP_LOGI(TAG, "done stopping http mode");
|
|
||||||
break;
|
|
||||||
|
|
||||||
case controlMode_t::MASSAGE:
|
case controlMode_t::MASSAGE:
|
||||||
ESP_LOGW(TAG, "switching from MASSAGE mode -> restoring fading, reset frozen input");
|
ESP_LOGW(TAG, "switching from MASSAGE mode -> restoring fading, reset frozen input");
|
||||||
//TODO: fix issue when downfading was disabled before switching to massage mode - currently it gets enabled again here...
|
//TODO: fix issue when downfading was disabled before switching to massage mode - currently it gets enabled again here...
|
||||||
@ -401,8 +404,8 @@ void controlledArmchair::changeMode(controlMode_t modeNew) {
|
|||||||
case controlMode_t::ADJUST_CHAIR:
|
case controlMode_t::ADJUST_CHAIR:
|
||||||
ESP_LOGW(TAG, "switching from ADJUST_CHAIR mode => turning off adjustment motors...");
|
ESP_LOGW(TAG, "switching from ADJUST_CHAIR mode => turning off adjustment motors...");
|
||||||
//prevent motors from being always on in case of mode switch while joystick is not in center thus motors currently moving
|
//prevent motors from being always on in case of mode switch while joystick is not in center thus motors currently moving
|
||||||
legRest.setState(REST_OFF);
|
legRest->setState(REST_OFF);
|
||||||
backRest.setState(REST_OFF);
|
backRest->setState(REST_OFF);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -427,26 +430,6 @@ void controlledArmchair::changeMode(controlMode_t modeNew) {
|
|||||||
buzzer->beep(4,200,100);
|
buzzer->beep(4,200,100);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case controlMode_t::HTTP:
|
|
||||||
ESP_LOGW(TAG, "switching to http mode -> enabling http and wifi");
|
|
||||||
//start wifi
|
|
||||||
//TODO: decide wether ap or client should be started
|
|
||||||
ESP_LOGI(TAG, "init wifi...");
|
|
||||||
|
|
||||||
//FIXME: make wifi function work here - currently starting wifi at startup (see notes main.cpp)
|
|
||||||
//wifi_init_client();
|
|
||||||
//wifi_init_ap();
|
|
||||||
|
|
||||||
//wait for wifi
|
|
||||||
//ESP_LOGI(TAG, "waiting for wifi...");
|
|
||||||
//vTaskDelay(1000 / portTICK_PERIOD_MS);
|
|
||||||
|
|
||||||
//start http server
|
|
||||||
ESP_LOGI(TAG, "init http server...");
|
|
||||||
http_init_server();
|
|
||||||
ESP_LOGI(TAG, "done initializing http mode");
|
|
||||||
break;
|
|
||||||
|
|
||||||
case controlMode_t::MASSAGE:
|
case controlMode_t::MASSAGE:
|
||||||
ESP_LOGW(TAG, "switching to MASSAGE mode -> reducing fading");
|
ESP_LOGW(TAG, "switching to MASSAGE mode -> reducing fading");
|
||||||
uint32_t shake_msFadeAccel = 500; //TODO: move this to config
|
uint32_t shake_msFadeAccel = 500; //TODO: move this to config
|
||||||
|
@ -5,6 +5,8 @@
|
|||||||
#include "buzzer.hpp"
|
#include "buzzer.hpp"
|
||||||
#include "http.hpp"
|
#include "http.hpp"
|
||||||
#include "auto.hpp"
|
#include "auto.hpp"
|
||||||
|
#include "speedsensor.hpp"
|
||||||
|
#include "chairAdjust.hpp"
|
||||||
|
|
||||||
|
|
||||||
//--------------------------------------------
|
//--------------------------------------------
|
||||||
@ -25,6 +27,13 @@ typedef struct control_config_t {
|
|||||||
} control_config_t;
|
} control_config_t;
|
||||||
|
|
||||||
|
|
||||||
|
//=======================================
|
||||||
|
//============ control task =============
|
||||||
|
//=======================================
|
||||||
|
//task that controls the armchair modes and initiates commands generation and applies them to driver
|
||||||
|
//parameter: pointer to controlledArmchair object
|
||||||
|
void task_control( void * pvParameters );
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//==================================
|
//==================================
|
||||||
@ -41,7 +50,10 @@ class controlledArmchair {
|
|||||||
controlledMotor* motorLeft_f,
|
controlledMotor* motorLeft_f,
|
||||||
controlledMotor* motorRight_f,
|
controlledMotor* motorRight_f,
|
||||||
evaluatedJoystick* joystick_f,
|
evaluatedJoystick* joystick_f,
|
||||||
httpJoystick* httpJoystick_f
|
httpJoystick* httpJoystick_f,
|
||||||
|
automatedArmchair_c* automatedArmchair,
|
||||||
|
cControlledRest * legRest,
|
||||||
|
cControlledRest * backRest
|
||||||
);
|
);
|
||||||
|
|
||||||
//--- functions ---
|
//--- functions ---
|
||||||
@ -85,6 +97,9 @@ class controlledArmchair {
|
|||||||
controlledMotor* motorRight;
|
controlledMotor* motorRight;
|
||||||
httpJoystick* httpJoystickMain_l;
|
httpJoystick* httpJoystickMain_l;
|
||||||
evaluatedJoystick* joystick_l;
|
evaluatedJoystick* joystick_l;
|
||||||
|
automatedArmchair_c *automatedArmchair;
|
||||||
|
cControlledRest * legRest;
|
||||||
|
cControlledRest * backRest;
|
||||||
|
|
||||||
//---variables ---
|
//---variables ---
|
||||||
//struct for motor commands returned by generate functions of each mode
|
//struct for motor commands returned by generate functions of each mode
|
||||||
@ -103,7 +118,7 @@ class controlledArmchair {
|
|||||||
bool freezeInput = false;
|
bool freezeInput = false;
|
||||||
|
|
||||||
//variables for AUTO mode
|
//variables for AUTO mode
|
||||||
auto_instruction_t instruction = auto_instruction_t::NONE; //variable to receive instructions from automatedArmchair
|
auto_instruction_t instruction = auto_instruction_t::NONE; //variable to receive instructions from automatedArmchair_c
|
||||||
|
|
||||||
//variable to store button event
|
//variable to store button event
|
||||||
uint8_t buttonCount = 0;
|
uint8_t buttonCount = 0;
|
||||||
|
@ -204,7 +204,7 @@ float getBatteryPercent(){
|
|||||||
//-----------------------
|
//-----------------------
|
||||||
//shows overview on entire display:
|
//shows overview on entire display:
|
||||||
//percentage, voltage, current, mode, rpm, speed
|
//percentage, voltage, current, mode, rpm, speed
|
||||||
void showScreen1()
|
void showScreen1(display_task_parameters_t * objects)
|
||||||
{
|
{
|
||||||
//-- battery percentage --
|
//-- battery percentage --
|
||||||
// TODO update when no load (currentsensors = ~0A) only
|
// TODO update when no load (currentsensors = ~0A) only
|
||||||
@ -214,24 +214,24 @@ void showScreen1()
|
|||||||
//-- voltage and current --
|
//-- voltage and current --
|
||||||
displayTextLine(&dev, 3, false, false, "%04.1fV %04.1f:%04.1fA",
|
displayTextLine(&dev, 3, false, false, "%04.1fV %04.1f:%04.1fA",
|
||||||
getBatteryVoltage(),
|
getBatteryVoltage(),
|
||||||
fabs(motorLeft.getCurrentA()),
|
fabs(objects->motorLeft->getCurrentA()),
|
||||||
fabs(motorRight.getCurrentA()));
|
fabs(objects->motorRight->getCurrentA()));
|
||||||
|
|
||||||
//-- control state --
|
//-- control state --
|
||||||
//print large line
|
//print large line
|
||||||
displayTextLine(&dev, 4, true, false, "%s ", control.getCurrentModeStr());
|
displayTextLine(&dev, 4, true, false, "%s ", objects->control->getCurrentModeStr());
|
||||||
|
|
||||||
//-- speed and RPM --
|
//-- speed and RPM --
|
||||||
displayTextLine(&dev, 7, false, false, "%3.1fkm/h %03.0f:%03.0fR",
|
displayTextLine(&dev, 7, false, false, "%3.1fkm/h %03.0f:%03.0fR",
|
||||||
fabs((speedLeft.getKmph() + speedRight.getKmph()) / 2),
|
fabs((objects->speedLeft->getKmph() + objects->speedRight->getKmph()) / 2),
|
||||||
speedLeft.getRpm(),
|
objects->speedLeft->getRpm(),
|
||||||
speedRight.getRpm());
|
objects->speedRight->getRpm());
|
||||||
|
|
||||||
// debug speed sensors
|
// debug speed sensors
|
||||||
ESP_LOGD(TAG, "%3.1fkm/h %03.0f:%03.0fR",
|
ESP_LOGD(TAG, "%3.1fkm/h %03.0f:%03.0fR",
|
||||||
fabs((speedLeft.getKmph() + speedRight.getKmph()) / 2),
|
fabs((objects->speedLeft->getKmph() + objects->speedRight->getKmph()) / 2),
|
||||||
speedLeft.getRpm(),
|
objects->speedLeft->getRpm(),
|
||||||
speedRight.getRpm());
|
objects->speedRight->getRpm());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -264,6 +264,9 @@ void showStartupMsg(){
|
|||||||
|
|
||||||
void display_task(void *pvParameters)
|
void display_task(void *pvParameters)
|
||||||
{
|
{
|
||||||
|
//get struct with pointers to all needed global objects from task parameter
|
||||||
|
display_task_parameters_t *objects = (display_task_parameters_t *)pvParameters;
|
||||||
|
|
||||||
// initialize display
|
// initialize display
|
||||||
display_init();
|
display_init();
|
||||||
// TODO check if successfully initialized
|
// TODO check if successfully initialized
|
||||||
@ -276,14 +279,14 @@ void display_task(void *pvParameters)
|
|||||||
// repeatedly update display with content
|
// repeatedly update display with content
|
||||||
while (1)
|
while (1)
|
||||||
{
|
{
|
||||||
if (control.getCurrentMode() == controlMode_t::MENU)
|
if (objects->control->getCurrentMode() == controlMode_t::MENU)
|
||||||
{
|
{
|
||||||
//uses encoder events to control menu and updates display
|
//uses encoder events to control menu and updates display
|
||||||
handleMenu(&dev);
|
handleMenu(objects, &dev);
|
||||||
}
|
}
|
||||||
else //show status screen in any other mode
|
else //show status screen in any other mode
|
||||||
{
|
{
|
||||||
showScreen1();
|
showScreen1(objects);
|
||||||
vTaskDelay(STATUS_SCREEN_UPDATE_INTERVAL / portTICK_PERIOD_MS);
|
vTaskDelay(STATUS_SCREEN_UPDATE_INTERVAL / portTICK_PERIOD_MS);
|
||||||
}
|
}
|
||||||
// TODO add pages and menus
|
// TODO add pages and menus
|
||||||
|
@ -13,7 +13,22 @@ extern "C" {
|
|||||||
#include "font8x8_basic.h"
|
#include "font8x8_basic.h"
|
||||||
}
|
}
|
||||||
|
|
||||||
#include "config.hpp"
|
|
||||||
|
#include "joystick.hpp"
|
||||||
|
#include "control.hpp"
|
||||||
|
#include "speedsensor.hpp"
|
||||||
|
|
||||||
|
// struct with variables passed to task from main()
|
||||||
|
typedef struct display_task_parameters_t {
|
||||||
|
controlledArmchair * control;
|
||||||
|
evaluatedJoystick * joystick;
|
||||||
|
QueueHandle_t encoderQueue;
|
||||||
|
controlledMotor * motorLeft;
|
||||||
|
controlledMotor * motorRight;
|
||||||
|
speedSensor * speedLeft;
|
||||||
|
speedSensor * speedRight;
|
||||||
|
buzzer_t *buzzer;
|
||||||
|
} display_task_parameters_t;
|
||||||
|
|
||||||
|
|
||||||
//task that inititialized the display, displays welcome message
|
//task that inititialized the display, displays welcome message
|
||||||
|
@ -39,11 +39,17 @@ rotary_encoder_t encoderConfig = {
|
|||||||
//==================================
|
//==================================
|
||||||
//========== encoder_init ==========
|
//========== encoder_init ==========
|
||||||
//==================================
|
//==================================
|
||||||
//initialize encoder
|
//initialize encoder //TODO pass config to this function
|
||||||
void encoder_init(){
|
QueueHandle_t encoder_init()
|
||||||
encoderQueue = xQueueCreate(QUEUE_SIZE, sizeof(rotary_encoder_event_t));
|
{
|
||||||
|
QueueHandle_t encoderQueue = xQueueCreate(QUEUE_SIZE, sizeof(rotary_encoder_event_t));
|
||||||
rotary_encoder_init(encoderQueue);
|
rotary_encoder_init(encoderQueue);
|
||||||
rotary_encoder_add(&encoderConfig);
|
rotary_encoder_add(&encoderConfig);
|
||||||
|
if (encoderQueue == NULL)
|
||||||
|
ESP_LOGE(TAG, "Error initializing encoder or queue");
|
||||||
|
else
|
||||||
|
ESP_LOGW(TAG, "Initialized encoder and encoderQueue");
|
||||||
|
return encoderQueue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -52,7 +58,9 @@ void encoder_init(){
|
|||||||
//====== task_encoderExample =======
|
//====== task_encoderExample =======
|
||||||
//==================================
|
//==================================
|
||||||
//receive and handle all available encoder events
|
//receive and handle all available encoder events
|
||||||
void task_encoderExample(void *arg) {
|
void task_encoderExample(void * arg) {
|
||||||
|
//get queue with encoder events from task parameter:
|
||||||
|
QueueHandle_t encoderQueue = (QueueHandle_t)arg;
|
||||||
static rotary_encoder_event_t ev; //store event data
|
static rotary_encoder_event_t ev; //store event data
|
||||||
while (1) {
|
while (1) {
|
||||||
if (xQueueReceive(encoderQueue, &ev, portMAX_DELAY)) {
|
if (xQueueReceive(encoderQueue, &ev, portMAX_DELAY)) {
|
||||||
|
@ -10,11 +10,11 @@ extern "C" {
|
|||||||
#define PIN_B GPIO_NUM_26
|
#define PIN_B GPIO_NUM_26
|
||||||
#define PIN_BUTTON GPIO_NUM_27
|
#define PIN_BUTTON GPIO_NUM_27
|
||||||
|
|
||||||
//global encoder queue
|
|
||||||
extern QueueHandle_t encoderQueue;
|
|
||||||
|
|
||||||
//init encoder with config in encoder.cpp
|
//init encoder with config in encoder.cpp
|
||||||
void encoder_init();
|
QueueHandle_t encoder_init(); //TODO pass config to function
|
||||||
|
|
||||||
|
|
||||||
//task that handles encoder events
|
//task that handles encoder events
|
||||||
void task_encoderExample(void *arg);
|
//note: queue obtained from encoder_init() has to be passed to that task
|
||||||
|
void task_encoderExample(void *encoderQueue);
|
||||||
|
//example: xTaskCreate(&task_encoderExample, "task_buzzer", 2048, encoderQueue, 2, NULL);
|
@ -1,6 +1,3 @@
|
|||||||
#include "hal/uart_types.h"
|
|
||||||
#include "motordrivers.hpp"
|
|
||||||
#include "types.hpp"
|
|
||||||
extern "C"
|
extern "C"
|
||||||
{
|
{
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
@ -14,44 +11,88 @@ extern "C"
|
|||||||
#include "sdkconfig.h"
|
#include "sdkconfig.h"
|
||||||
#include "esp_spiffs.h"
|
#include "esp_spiffs.h"
|
||||||
|
|
||||||
#include "driver/ledc.h"
|
|
||||||
|
|
||||||
//custom C files
|
//custom C files
|
||||||
#include "wifi.h"
|
#include "wifi.h"
|
||||||
}
|
}
|
||||||
|
|
||||||
//custom C++ files
|
//custom C++ files
|
||||||
|
//folder common
|
||||||
|
#include "uart_common.hpp"
|
||||||
|
#include "motordrivers.hpp"
|
||||||
|
#include "http.hpp"
|
||||||
|
#include "types.hpp"
|
||||||
|
#include "speedsensor.hpp"
|
||||||
|
#include "motorctl.hpp"
|
||||||
|
|
||||||
|
//folder single_board
|
||||||
#include "config.hpp"
|
#include "config.hpp"
|
||||||
#include "control.hpp"
|
#include "control.hpp"
|
||||||
#include "button.hpp"
|
#include "button.hpp"
|
||||||
#include "http.hpp"
|
|
||||||
|
|
||||||
#include "uart_common.hpp"
|
|
||||||
|
|
||||||
#include "display.hpp"
|
#include "display.hpp"
|
||||||
#include "encoder.hpp"
|
#include "encoder.hpp"
|
||||||
|
|
||||||
|
|
||||||
|
//only extends this file (no library):
|
||||||
|
//outsourced all configuration related structures
|
||||||
|
#include "config.cpp"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//================================
|
||||||
|
//======== declarations ==========
|
||||||
|
//================================
|
||||||
|
//--- declare all pointers to shared objects ---
|
||||||
|
controlledMotor *motorLeft;
|
||||||
|
controlledMotor *motorRight;
|
||||||
|
|
||||||
|
sabertooth2x60a *sabertoothDriver;
|
||||||
|
|
||||||
|
evaluatedJoystick *joystick;
|
||||||
|
|
||||||
|
buzzer_t *buzzer;
|
||||||
|
|
||||||
|
controlledArmchair *control;
|
||||||
|
|
||||||
|
automatedArmchair_c *automatedArmchair;
|
||||||
|
|
||||||
|
httpJoystick *httpJoystickMain;
|
||||||
|
|
||||||
|
speedSensor *speedLeft;
|
||||||
|
speedSensor *speedRight;
|
||||||
|
|
||||||
|
cControlledRest *legRest;
|
||||||
|
cControlledRest *backRest;
|
||||||
|
|
||||||
|
|
||||||
|
//--- lambda functions motor-driver ---
|
||||||
|
// functions for updating the duty via currently used motor driver (hardware) that can then be passed to controlledMotor
|
||||||
|
//-> makes it possible to easily use different motor drivers
|
||||||
|
motorSetCommandFunc_t setLeftFunc = [&sabertoothDriver](motorCommand_t cmd)
|
||||||
|
{
|
||||||
|
sabertoothDriver->setLeft(cmd);
|
||||||
|
};
|
||||||
|
motorSetCommandFunc_t setRightFunc = [&sabertoothDriver](motorCommand_t cmd)
|
||||||
|
{
|
||||||
|
sabertoothDriver->setRight(cmd);
|
||||||
|
};
|
||||||
|
|
||||||
|
//--- lambda function http-joystick ---
|
||||||
|
// function that initializes the http server requires a function pointer to function that handels each url
|
||||||
|
// the httpd_uri config struct does not accept a pointer to a method of a class instance, directly
|
||||||
|
// thus this lambda function is necessary:
|
||||||
|
// declare pointer to receiveHttpData method of httpJoystick class
|
||||||
|
esp_err_t (httpJoystick::*pointerToReceiveFunc)(httpd_req_t *req) = &httpJoystick::receiveHttpData;
|
||||||
|
esp_err_t on_joystick_url(httpd_req_t *req)
|
||||||
|
{
|
||||||
|
// run pointer to receiveHttpData function of httpJoystickMain instance
|
||||||
|
return (httpJoystickMain->*pointerToReceiveFunc)(req);
|
||||||
|
}
|
||||||
|
|
||||||
//tag for logging
|
//tag for logging
|
||||||
static const char * TAG = "main";
|
static const char * TAG = "main";
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//====================================
|
|
||||||
//========== motorctl task ===========
|
|
||||||
//====================================
|
|
||||||
//task for handling the motors (ramp, current limit, driver)
|
|
||||||
void task_motorctl( void * pvParameters ){
|
|
||||||
ESP_LOGI(TAG, "starting handle loop...");
|
|
||||||
while(1){
|
|
||||||
motorRight.handle();
|
|
||||||
motorLeft.handle();
|
|
||||||
//10khz -> T=100us
|
|
||||||
vTaskDelay(10 / portTICK_PERIOD_MS);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//======================================
|
//======================================
|
||||||
//============ buzzer task =============
|
//============ buzzer task =============
|
||||||
//======================================
|
//======================================
|
||||||
@ -61,33 +102,7 @@ void task_buzzer( void * pvParameters ){
|
|||||||
ESP_LOGI("task_buzzer", "Start of buzzer task...");
|
ESP_LOGI("task_buzzer", "Start of buzzer task...");
|
||||||
//run function that waits for a beep events to arrive in the queue
|
//run function that waits for a beep events to arrive in the queue
|
||||||
//and processes them
|
//and processes them
|
||||||
buzzer.processQueue();
|
buzzer->processQueue();
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//=======================================
|
|
||||||
//============ control task =============
|
|
||||||
//=======================================
|
|
||||||
//task that controls the armchair modes and initiates commands generation and applies them to driver
|
|
||||||
void task_control( void * pvParameters ){
|
|
||||||
ESP_LOGI(TAG, "Initializing controlledArmchair and starting handle loop");
|
|
||||||
//start handle loop (control object declared in config.hpp)
|
|
||||||
control.startHandleLoop();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//======================================
|
|
||||||
//============ button task =============
|
|
||||||
//======================================
|
|
||||||
//task that handles the button interface/commands
|
|
||||||
void task_button( void * pvParameters ){
|
|
||||||
ESP_LOGI(TAG, "Initializing command-button and starting handle loop");
|
|
||||||
//create button instance
|
|
||||||
buttonCommands commandButton(&buttonJoystick, &joystick, &control, &buzzer, &motorLeft, &motorRight);
|
|
||||||
//start handle loop
|
|
||||||
commandButton.startHandleLoop();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -95,11 +110,12 @@ void task_button( void * pvParameters ){
|
|||||||
//=======================================
|
//=======================================
|
||||||
//============== fan task ===============
|
//============== fan task ===============
|
||||||
//=======================================
|
//=======================================
|
||||||
|
//TODO: move this definition to fan.cpp
|
||||||
//task that controlls fans for cooling the drivers
|
//task that controlls fans for cooling the drivers
|
||||||
void task_fans( void * pvParameters ){
|
void task_fans( void * pvParameters ){
|
||||||
ESP_LOGI(TAG, "Initializing fans and starting fan handle loop");
|
ESP_LOGI(TAG, "Initializing fans and starting fan handle loop");
|
||||||
//create fan instances with config defined in config.cpp
|
//create fan instances with config defined in config.cpp
|
||||||
controlledFan fan(configCooling, &motorLeft, &motorRight);
|
controlledFan fan(configCooling, motorLeft, motorRight);
|
||||||
//repeatedly run fan handle function in a slow loop
|
//repeatedly run fan handle function in a slow loop
|
||||||
while(1){
|
while(1){
|
||||||
fan.handle();
|
fan.handle();
|
||||||
@ -132,31 +148,54 @@ void init_spiffs(){
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
//==================================
|
|
||||||
//======== define loglevels ========
|
|
||||||
//==================================
|
|
||||||
void setLoglevels(void){
|
|
||||||
//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("buzzer", ESP_LOG_ERROR);
|
//========= createObjects =========
|
||||||
//esp_log_level_set("motordriver", ESP_LOG_DEBUG);
|
//=================================
|
||||||
//esp_log_level_set("motor-control", ESP_LOG_INFO);
|
//create all shared objects
|
||||||
//esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG);
|
//their references can be passed to the tasks that need access in main
|
||||||
//esp_log_level_set("joystickCommands", ESP_LOG_DEBUG);
|
|
||||||
esp_log_level_set("button", ESP_LOG_INFO);
|
//Note: the configuration structures (e.g. configMotorControlLeft) are outsourced to file 'config.cpp'
|
||||||
esp_log_level_set("control", ESP_LOG_INFO);
|
|
||||||
//esp_log_level_set("fan-control", ESP_LOG_INFO);
|
void createObjects()
|
||||||
esp_log_level_set("wifi", ESP_LOG_INFO);
|
{
|
||||||
esp_log_level_set("http", ESP_LOG_INFO);
|
// create sabertooth motor driver instance
|
||||||
//esp_log_level_set("automatedArmchair", ESP_LOG_DEBUG);
|
// sabertooth2x60a sabertoothDriver(sabertoothConfig);
|
||||||
esp_log_level_set("display", ESP_LOG_INFO);
|
// with configuration above
|
||||||
//esp_log_level_set("current-sensors", ESP_LOG_INFO);
|
sabertoothDriver = new sabertooth2x60a(sabertoothConfig);
|
||||||
//esp_log_level_set("speedSensor", ESP_LOG_INFO);
|
|
||||||
esp_log_level_set("chair-adjustment", ESP_LOG_INFO);
|
// create controlled motor instances (motorctl.hpp)
|
||||||
esp_log_level_set("menu", ESP_LOG_INFO);
|
// with configurations above
|
||||||
|
motorLeft = new controlledMotor(setLeftFunc, configMotorControlLeft);
|
||||||
|
motorRight = new controlledMotor(setRightFunc, configMotorControlRight);
|
||||||
|
|
||||||
|
// create speedsensor instances
|
||||||
|
// with configurations above
|
||||||
|
speedLeft = new speedSensor(speedLeft_config);
|
||||||
|
speedRight = new speedSensor(speedRight_config);
|
||||||
|
|
||||||
|
// create joystic instance (joystick.hpp)
|
||||||
|
joystick = new evaluatedJoystick(configJoystick);
|
||||||
|
|
||||||
|
// create httpJoystick object (http.hpp)
|
||||||
|
httpJoystickMain = new httpJoystick(configHttpJoystickMain);
|
||||||
|
http_init_server(on_joystick_url);
|
||||||
|
|
||||||
|
// create buzzer object on pin 12 with gap between queued events of 100ms
|
||||||
|
buzzer = new buzzer_t(GPIO_NUM_12, 100);
|
||||||
|
|
||||||
|
// create control object (control.hpp)
|
||||||
|
// with configuration above
|
||||||
|
control = new controlledArmchair(configControl, buzzer, motorLeft, motorRight, joystick, httpJoystickMain, automatedArmchair, legRest, backRest);
|
||||||
|
|
||||||
|
// create automatedArmchair_c object (for auto-mode) (auto.hpp)
|
||||||
|
automatedArmchair = new automatedArmchair_c(motorLeft, motorRight);
|
||||||
|
|
||||||
|
// create objects for controlling the chair position
|
||||||
|
// gpio_up, gpio_down, name
|
||||||
|
legRest = new cControlledRest(GPIO_NUM_4, GPIO_NUM_16, "legRest");
|
||||||
|
backRest = new cControlledRest(GPIO_NUM_2, GPIO_NUM_15, "backRest");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -166,25 +205,53 @@ void setLoglevels(void){
|
|||||||
//=========== app_main ============
|
//=========== app_main ============
|
||||||
//=================================
|
//=================================
|
||||||
extern "C" void app_main(void) {
|
extern "C" void app_main(void) {
|
||||||
//enable 5V volate regulator
|
|
||||||
|
ESP_LOGW(TAG, "===== INITIALIZING COMPONENTS =====");
|
||||||
|
//--- define log levels ---
|
||||||
|
setLoglevels();
|
||||||
|
|
||||||
|
//--- enable 5V volate regulator ---
|
||||||
ESP_LOGW(TAG, "enabling 5V regulator...");
|
ESP_LOGW(TAG, "enabling 5V regulator...");
|
||||||
gpio_pad_select_gpio(GPIO_NUM_17);
|
gpio_pad_select_gpio(GPIO_NUM_17);
|
||||||
gpio_set_direction(GPIO_NUM_17, GPIO_MODE_OUTPUT);
|
gpio_set_direction(GPIO_NUM_17, GPIO_MODE_OUTPUT);
|
||||||
gpio_set_level(GPIO_NUM_17, 1);
|
gpio_set_level(GPIO_NUM_17, 1);
|
||||||
|
|
||||||
//---- define log levels ----
|
//--- initialize nvs-flash and netif (needed for wifi) ---
|
||||||
setLoglevels();
|
wifi_initNvs_initNetif();
|
||||||
|
|
||||||
|
//--- initialize spiffs ---
|
||||||
|
init_spiffs();
|
||||||
|
|
||||||
|
//--- initialize and start wifi ---
|
||||||
|
ESP_LOGD(TAG,"starting wifi...");
|
||||||
|
//wifi_init_client(); //connect to existing wifi
|
||||||
|
wifi_init_ap(); //start access point
|
||||||
|
ESP_LOGD(TAG,"done starting wifi");
|
||||||
|
|
||||||
// init encoder
|
|
||||||
//--- initialize encoder ---
|
//--- initialize encoder ---
|
||||||
encoder_init();
|
const QueueHandle_t encoderQueue = encoder_init();
|
||||||
// now global encoderQueue providing all encoder events is available
|
|
||||||
|
|
||||||
|
|
||||||
|
//--- create all objects ---
|
||||||
|
ESP_LOGW(TAG, "===== CREATING SHARED OBJECTS =====");
|
||||||
|
|
||||||
|
//create all class instances used below
|
||||||
|
//see 'createObjects.hpp'
|
||||||
|
createObjects();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef ENCODER_TEST
|
||||||
|
//--- create tasks ---
|
||||||
|
ESP_LOGW(TAG, "===== CREATING TASKS =====");
|
||||||
|
|
||||||
//----------------------------------------------
|
//----------------------------------------------
|
||||||
//--- create task for controlling the motors ---
|
//--- create task for controlling the motors ---
|
||||||
//----------------------------------------------
|
//----------------------------------------------
|
||||||
//task that receives commands, handles ramp and current limit and executes commands using the motordriver function
|
//task that receives commands, handles ramp and current limit and executes commands using the motordriver function
|
||||||
xTaskCreate(&task_motorctl, "task_motor-control", 2*4096, NULL, 6, NULL);
|
task_motorctl_parameters_t motorctl_param = {motorLeft, motorRight};
|
||||||
|
xTaskCreate(&task_motorctl, "task_motor-control", 2*4096, &motorctl_param, 6, NULL);
|
||||||
|
|
||||||
//------------------------------
|
//------------------------------
|
||||||
//--- create task for buzzer ---
|
//--- create task for buzzer ---
|
||||||
@ -195,46 +262,38 @@ extern "C" void app_main(void) {
|
|||||||
//--- create task for control ---
|
//--- create task for control ---
|
||||||
//-------------------------------
|
//-------------------------------
|
||||||
//task that generates motor commands depending on the current mode and sends those to motorctl task
|
//task that generates motor commands depending on the current mode and sends those to motorctl task
|
||||||
xTaskCreate(&task_control, "task_control", 4096, NULL, 5, NULL);
|
//note: pointer to shared object 'control' is passed as task parameter:
|
||||||
|
xTaskCreate(&task_control, "task_control", 4096, control, 5, NULL);
|
||||||
|
|
||||||
//------------------------------
|
//------------------------------
|
||||||
//--- create task for button ---
|
//--- create task for button ---
|
||||||
//------------------------------
|
//------------------------------
|
||||||
//task that evaluates and processes the button input and runs the configured commands
|
//task that handles button/encoder events in any mode except 'MENU' (e.g. switch modes by pressing certain count)
|
||||||
xTaskCreate(&task_button, "task_button", 4096, NULL, 4, NULL);
|
task_button_parameters_t button_param = {control, joystick, encoderQueue, motorLeft, motorRight, buzzer};
|
||||||
|
xTaskCreate(&task_button, "task_button", 4096, &button_param, 4, NULL);
|
||||||
|
|
||||||
//-----------------------------------
|
//-----------------------------------
|
||||||
//--- create task for fan control ---
|
//--- create task for fan control ---
|
||||||
//-----------------------------------
|
//-----------------------------------
|
||||||
//task that evaluates and processes the button input and runs the configured commands
|
//task that controls cooling fans of the motor driver
|
||||||
xTaskCreate(&task_fans, "task_fans", 2048, NULL, 1, NULL);
|
xTaskCreate(&task_fans, "task_fans", 2048, NULL, 1, NULL);
|
||||||
|
|
||||||
|
|
||||||
//-----------------------------------
|
//-----------------------------------
|
||||||
//----- create task for display -----
|
//----- create task for display -----
|
||||||
//-----------------------------------
|
//-----------------------------------
|
||||||
//task that handles the display
|
////task that handles the display (show stats, handle menu in 'MENU' mode)
|
||||||
xTaskCreate(&display_task, "display_task", 3*2048, NULL, 1, NULL);
|
display_task_parameters_t display_param = {control, joystick, encoderQueue, motorLeft, motorRight, speedLeft, speedRight, buzzer};
|
||||||
|
xTaskCreate(&display_task, "display_task", 3*2048, &display_param, 1, NULL);
|
||||||
|
|
||||||
|
|
||||||
//beep at startup
|
#endif
|
||||||
buzzer.beep(3, 70, 50);
|
|
||||||
|
|
||||||
//--- initialize nvs-flash and netif (needed for wifi) ---
|
//--- startup finished ---
|
||||||
wifi_initNvs_initNetif();
|
ESP_LOGW(TAG, "===== STARTUP FINISHED =====");
|
||||||
|
buzzer->beep(3, 70, 50);
|
||||||
//--- initialize spiffs ---
|
|
||||||
init_spiffs();
|
|
||||||
|
|
||||||
//--- initialize and start wifi ---
|
|
||||||
//FIXME: run wifi_init_client or wifi_init_ap as intended from control.cpp when switching state
|
|
||||||
//currently commented out because of error "assert failed: xQueueSemaphoreTake queue.c:1549 (pxQueue->uxItemSize == 0)" when calling control->changeMode from button.cpp
|
|
||||||
//when calling control.changeMode(http) from main.cpp it worked without error for some reason?
|
|
||||||
ESP_LOGI(TAG,"starting wifi...");
|
|
||||||
//wifi_init_client(); //connect to existing wifi
|
|
||||||
wifi_init_ap(); //start access point
|
|
||||||
ESP_LOGI(TAG,"done starting wifi");
|
|
||||||
|
|
||||||
|
//--- testing encoder ---
|
||||||
|
//xTaskCreate(&task_encoderExample, "task_buzzer", 2048, encoderQueue, 2, NULL);
|
||||||
|
|
||||||
//--- testing http server ---
|
//--- testing http server ---
|
||||||
// wifi_init_client(); //connect to existing wifi
|
// wifi_init_client(); //connect to existing wifi
|
||||||
@ -243,8 +302,8 @@ extern "C" void app_main(void) {
|
|||||||
// http_init_server();
|
// http_init_server();
|
||||||
|
|
||||||
|
|
||||||
//--- testing force http mode after startup ---
|
//--- testing force specific mode after startup ---
|
||||||
//control.changeMode(controlMode_t::HTTP);
|
//control->changeMode(controlMode_t::MENU);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -30,12 +30,14 @@ static int value = 0;
|
|||||||
//#########################
|
//#########################
|
||||||
//#### center Joystick ####
|
//#### center Joystick ####
|
||||||
//#########################
|
//#########################
|
||||||
void item_centerJoystick_action(int value, SSD1306_t * display){
|
void item_centerJoystick_action(display_task_parameters_t * objects, SSD1306_t * display, int value){
|
||||||
if (!value) return;
|
if (!value) return;
|
||||||
ESP_LOGW(TAG, "defining joystick center");
|
ESP_LOGW(TAG, "defining joystick center");
|
||||||
joystick.defineCenter();
|
(*objects).joystick->defineCenter();
|
||||||
|
//objects->joystick->defineCenter();
|
||||||
|
//joystick->defineCenter();
|
||||||
}
|
}
|
||||||
int item_centerJoystick_value(){
|
int item_centerJoystick_value(display_task_parameters_t * objects){
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -59,7 +61,7 @@ menuItem_t item_centerJoystick = {
|
|||||||
//#### debug Joystick ####
|
//#### debug Joystick ####
|
||||||
//########################
|
//########################
|
||||||
//continously show/update joystick data on display
|
//continously show/update joystick data on display
|
||||||
void item_debugJoystick_action(int value, SSD1306_t * display)
|
void item_debugJoystick_action(display_task_parameters_t * objects, SSD1306_t * display, int value)
|
||||||
{
|
{
|
||||||
//--- variables ---
|
//--- variables ---
|
||||||
bool running = true;
|
bool running = true;
|
||||||
@ -77,10 +79,10 @@ void item_debugJoystick_action(int value, SSD1306_t * display)
|
|||||||
|
|
||||||
//-- show/update values --
|
//-- show/update values --
|
||||||
// stop when button pressed or control state changes (timeouts to IDLE)
|
// stop when button pressed or control state changes (timeouts to IDLE)
|
||||||
while (running && control.getCurrentMode() == controlMode_t::MENU)
|
while (running && objects->control->getCurrentMode() == controlMode_t::MENU)
|
||||||
{
|
{
|
||||||
// repeatedly print all joystick data
|
// repeatedly print all joystick data
|
||||||
joystickData_t data = joystick.getData();
|
joystickData_t data = objects->joystick->getData();
|
||||||
displayTextLine(display, 1, false, false, "x = %.3f ", data.x);
|
displayTextLine(display, 1, false, false, "x = %.3f ", data.x);
|
||||||
displayTextLine(display, 2, false, false, "y = %.3f ", data.y);
|
displayTextLine(display, 2, false, false, "y = %.3f ", data.y);
|
||||||
displayTextLine(display, 3, false, false, "radius = %.3f", data.radius);
|
displayTextLine(display, 3, false, false, "radius = %.3f", data.radius);
|
||||||
@ -88,7 +90,7 @@ void item_debugJoystick_action(int value, SSD1306_t * display)
|
|||||||
displayTextLine(display, 5, false, false, "pos=%-12s ", joystickPosStr[(int)data.position]);
|
displayTextLine(display, 5, false, false, "pos=%-12s ", joystickPosStr[(int)data.position]);
|
||||||
|
|
||||||
// exit when button pressed
|
// exit when button pressed
|
||||||
if (xQueueReceive(encoderQueue, &event, 20 / portTICK_PERIOD_MS))
|
if (xQueueReceive(objects->encoderQueue, &event, 20 / portTICK_PERIOD_MS))
|
||||||
{
|
{
|
||||||
switch (event.type)
|
switch (event.type)
|
||||||
{
|
{
|
||||||
@ -105,7 +107,7 @@ void item_debugJoystick_action(int value, SSD1306_t * display)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int item_debugJoystick_value(){
|
int item_debugJoystick_value(display_task_parameters_t * objects){
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -128,12 +130,12 @@ menuItem_t item_debugJoystick = {
|
|||||||
//########################
|
//########################
|
||||||
//##### set max duty #####
|
//##### set max duty #####
|
||||||
//########################
|
//########################
|
||||||
void maxDuty_action(int value, SSD1306_t * display)
|
void maxDuty_action(display_task_parameters_t * objects, SSD1306_t * display, int value)
|
||||||
{
|
{
|
||||||
//TODO actually store the value
|
//TODO actually store the value
|
||||||
ESP_LOGW(TAG, "set max duty to %d", value);
|
ESP_LOGW(TAG, "set max duty to %d", value);
|
||||||
}
|
}
|
||||||
int maxDuty_currentValue()
|
int maxDuty_currentValue(display_task_parameters_t * objects)
|
||||||
{
|
{
|
||||||
//TODO get real current value
|
//TODO get real current value
|
||||||
return 84;
|
return 84;
|
||||||
@ -156,14 +158,14 @@ menuItem_t item_maxDuty = {
|
|||||||
//######################
|
//######################
|
||||||
//##### accelLimit #####
|
//##### accelLimit #####
|
||||||
//######################
|
//######################
|
||||||
void item_accelLimit_action(int value, SSD1306_t * display)
|
void item_accelLimit_action(display_task_parameters_t * objects, SSD1306_t * display, int value)
|
||||||
{
|
{
|
||||||
motorLeft.setFade(fadeType_t::ACCEL, (uint32_t)value);
|
objects->motorLeft->setFade(fadeType_t::ACCEL, (uint32_t)value);
|
||||||
motorRight.setFade(fadeType_t::ACCEL, (uint32_t)value);
|
objects->motorRight->setFade(fadeType_t::ACCEL, (uint32_t)value);
|
||||||
}
|
}
|
||||||
int item_accelLimit_value()
|
int item_accelLimit_value(display_task_parameters_t * objects)
|
||||||
{
|
{
|
||||||
return motorLeft.getFade(fadeType_t::ACCEL);
|
return objects->motorLeft->getFade(fadeType_t::ACCEL);
|
||||||
}
|
}
|
||||||
menuItem_t item_accelLimit = {
|
menuItem_t item_accelLimit = {
|
||||||
item_accelLimit_action, // function action
|
item_accelLimit_action, // function action
|
||||||
@ -183,14 +185,14 @@ menuItem_t item_accelLimit = {
|
|||||||
// ######################
|
// ######################
|
||||||
// ##### decelLimit #####
|
// ##### decelLimit #####
|
||||||
// ######################
|
// ######################
|
||||||
void item_decelLimit_action(int value, SSD1306_t * display)
|
void item_decelLimit_action(display_task_parameters_t * objects, SSD1306_t * display, int value)
|
||||||
{
|
{
|
||||||
motorLeft.setFade(fadeType_t::DECEL, (uint32_t)value);
|
objects->motorLeft->setFade(fadeType_t::DECEL, (uint32_t)value);
|
||||||
motorRight.setFade(fadeType_t::DECEL, (uint32_t)value);
|
objects->motorRight->setFade(fadeType_t::DECEL, (uint32_t)value);
|
||||||
}
|
}
|
||||||
int item_decelLimit_value()
|
int item_decelLimit_value(display_task_parameters_t * objects)
|
||||||
{
|
{
|
||||||
return motorLeft.getFade(fadeType_t::DECEL);
|
return objects->motorLeft->getFade(fadeType_t::DECEL);
|
||||||
}
|
}
|
||||||
menuItem_t item_decelLimit = {
|
menuItem_t item_decelLimit = {
|
||||||
item_decelLimit_action, // function action
|
item_decelLimit_action, // function action
|
||||||
@ -210,11 +212,11 @@ menuItem_t item_decelLimit = {
|
|||||||
//#####################
|
//#####################
|
||||||
//###### example ######
|
//###### example ######
|
||||||
//#####################
|
//#####################
|
||||||
void item_example_action(int value, SSD1306_t * display)
|
void item_example_action(display_task_parameters_t * objects, SSD1306_t * display, int value)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
int item_example_value(){
|
int item_example_value(display_task_parameters_t * objects){
|
||||||
return 53;
|
return 53;
|
||||||
}
|
}
|
||||||
menuItem_t item_example = {
|
menuItem_t item_example = {
|
||||||
@ -342,7 +344,7 @@ void showValueSelect(SSD1306_t *display, int selectedItem)
|
|||||||
//function is repeatedly called by display task when in menu state
|
//function is repeatedly called by display task when in menu state
|
||||||
#define QUEUE_TIMEOUT 3000 //timeout no encoder event - to handle timeout and not block the display loop
|
#define QUEUE_TIMEOUT 3000 //timeout no encoder event - to handle timeout and not block the display loop
|
||||||
#define MENU_TIMEOUT 60000 //inactivity timeout (switch to IDLE mode)
|
#define MENU_TIMEOUT 60000 //inactivity timeout (switch to IDLE mode)
|
||||||
void handleMenu(SSD1306_t *display)
|
void handleMenu(display_task_parameters_t * objects, SSD1306_t *display)
|
||||||
{
|
{
|
||||||
static uint32_t lastActivity = 0;
|
static uint32_t lastActivity = 0;
|
||||||
static int selectedItem = 0;
|
static int selectedItem = 0;
|
||||||
@ -358,7 +360,7 @@ void handleMenu(SSD1306_t *display)
|
|||||||
// update display
|
// update display
|
||||||
showItemList(display, selectedItem); // shows list of items with currently selected one on display
|
showItemList(display, selectedItem); // shows list of items with currently selected one on display
|
||||||
// wait for encoder event
|
// wait for encoder event
|
||||||
if (xQueueReceive(encoderQueue, &event, QUEUE_TIMEOUT / portTICK_PERIOD_MS))
|
if (xQueueReceive(objects->encoderQueue, &event, QUEUE_TIMEOUT / portTICK_PERIOD_MS))
|
||||||
{
|
{
|
||||||
lastActivity = esp_log_timestamp();
|
lastActivity = esp_log_timestamp();
|
||||||
switch (event.type)
|
switch (event.type)
|
||||||
@ -387,7 +389,7 @@ void handleMenu(SSD1306_t *display)
|
|||||||
// change state (menu to set value)
|
// change state (menu to set value)
|
||||||
menuState = SET_VALUE;
|
menuState = SET_VALUE;
|
||||||
// get currently configured value
|
// get currently configured value
|
||||||
value = menuItems[selectedItem].currentValue();
|
value = menuItems[selectedItem].currentValue(objects);
|
||||||
// clear display
|
// clear display
|
||||||
ssd1306_clear_screen(display, false);
|
ssd1306_clear_screen(display, false);
|
||||||
break;
|
break;
|
||||||
@ -395,7 +397,7 @@ void handleMenu(SSD1306_t *display)
|
|||||||
//exit menu mode
|
//exit menu mode
|
||||||
case RE_ET_BTN_LONG_PRESSED:
|
case RE_ET_BTN_LONG_PRESSED:
|
||||||
//change to previous mode (e.g. JOYSTICK)
|
//change to previous mode (e.g. JOYSTICK)
|
||||||
control.toggleMode(controlMode_t::MENU); //currently already in MENU -> changes to previous mode
|
objects->control->toggleMode(controlMode_t::MENU); //currently already in MENU -> changes to previous mode
|
||||||
ssd1306_clear_screen(display, false);
|
ssd1306_clear_screen(display, false);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -413,7 +415,7 @@ void handleMenu(SSD1306_t *display)
|
|||||||
// wait for encoder event
|
// wait for encoder event
|
||||||
showValueSelect(display, selectedItem);
|
showValueSelect(display, selectedItem);
|
||||||
|
|
||||||
if (xQueueReceive(encoderQueue, &event, QUEUE_TIMEOUT / portTICK_PERIOD_MS))
|
if (xQueueReceive(objects->encoderQueue, &event, QUEUE_TIMEOUT / portTICK_PERIOD_MS))
|
||||||
{
|
{
|
||||||
lastActivity = esp_log_timestamp();
|
lastActivity = esp_log_timestamp();
|
||||||
switch (event.type)
|
switch (event.type)
|
||||||
@ -434,7 +436,7 @@ void handleMenu(SSD1306_t *display)
|
|||||||
case RE_ET_BTN_CLICKED:
|
case RE_ET_BTN_CLICKED:
|
||||||
//-- apply value --
|
//-- apply value --
|
||||||
ESP_LOGI(TAG, "Button pressed - running action function with value=%d for item '%s'", value, menuItems[selectedItem].title);
|
ESP_LOGI(TAG, "Button pressed - running action function with value=%d for item '%s'", value, menuItems[selectedItem].title);
|
||||||
menuItems[selectedItem].action(value, display);
|
menuItems[selectedItem].action(objects, display, value);
|
||||||
menuState = MAIN_MENU;
|
menuState = MAIN_MENU;
|
||||||
break;
|
break;
|
||||||
case RE_ET_BTN_PRESSED:
|
case RE_ET_BTN_PRESSED:
|
||||||
@ -459,7 +461,7 @@ void handleMenu(SSD1306_t *display)
|
|||||||
menuState = MAIN_MENU;
|
menuState = MAIN_MENU;
|
||||||
ssd1306_clear_screen(display, false);
|
ssd1306_clear_screen(display, false);
|
||||||
// change control mode
|
// change control mode
|
||||||
control.changeMode(controlMode_t::IDLE);
|
objects->control->changeMode(controlMode_t::IDLE);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -13,9 +13,10 @@ typedef enum {
|
|||||||
|
|
||||||
//--- menuItem_t ---
|
//--- menuItem_t ---
|
||||||
// struct describes one menu element (all defined in menu.cpp)
|
// struct describes one menu element (all defined in menu.cpp)
|
||||||
typedef struct {
|
typedef struct
|
||||||
void (*action)(int value, SSD1306_t * display); // pointer to function run when confirmed
|
{
|
||||||
int (*currentValue)(); // pointer to function to get currently configured value
|
void (*action)(display_task_parameters_t * objects, SSD1306_t * display, int value); // pointer to function run when confirmed
|
||||||
|
int (*currentValue)(display_task_parameters_t * objects); // pointer to function to get currently configured value
|
||||||
int valueMin; // min allowed value
|
int valueMin; // min allowed value
|
||||||
int valueMax; // max allowed value
|
int valueMax; // max allowed value
|
||||||
int valueIncrement; // amount changed at one encoder tick (+/-)
|
int valueIncrement; // amount changed at one encoder tick (+/-)
|
||||||
@ -28,5 +29,4 @@ typedef struct {
|
|||||||
const char line7[17]; // below value
|
const char line7[17]; // below value
|
||||||
} menuItem_t;
|
} menuItem_t;
|
||||||
|
|
||||||
|
void handleMenu(display_task_parameters_t * objects, SSD1306_t *display);
|
||||||
void handleMenu(SSD1306_t * display);
|
|
@ -140,10 +140,10 @@ CONFIG_I2C_INTERFACE=y
|
|||||||
# CONFIG_SPI_INTERFACE is not set
|
# CONFIG_SPI_INTERFACE is not set
|
||||||
# CONFIG_SSD1306_128x32 is not set
|
# CONFIG_SSD1306_128x32 is not set
|
||||||
CONFIG_SSD1306_128x64=y
|
CONFIG_SSD1306_128x64=y
|
||||||
CONFIG_OFFSETX=0
|
CONFIG_OFFSETX=2
|
||||||
# CONFIG_FLIP is not set
|
# CONFIG_FLIP is not set
|
||||||
CONFIG_SCL_GPIO=22
|
CONFIG_SCL_GPIO=22
|
||||||
CONFIG_SDA_GPIO=21
|
CONFIG_SDA_GPIO=23
|
||||||
CONFIG_RESET_GPIO=15
|
CONFIG_RESET_GPIO=15
|
||||||
CONFIG_I2C_PORT_0=y
|
CONFIG_I2C_PORT_0=y
|
||||||
# CONFIG_I2C_PORT_1 is not set
|
# CONFIG_I2C_PORT_1 is not set
|
||||||
@ -1246,6 +1246,17 @@ CONFIG_WPA_MBEDTLS_CRYPTO=y
|
|||||||
# CONFIG_WPA_MBO_SUPPORT is not set
|
# CONFIG_WPA_MBO_SUPPORT is not set
|
||||||
# CONFIG_WPA_DPP_SUPPORT is not set
|
# CONFIG_WPA_DPP_SUPPORT is not set
|
||||||
# end of Supplicant
|
# end of Supplicant
|
||||||
|
|
||||||
|
#
|
||||||
|
# Rotary encoders
|
||||||
|
#
|
||||||
|
CONFIG_RE_MAX=1
|
||||||
|
CONFIG_RE_INTERVAL_US=1000
|
||||||
|
CONFIG_RE_BTN_DEAD_TIME_US=10000
|
||||||
|
CONFIG_RE_BTN_PRESSED_LEVEL_0=y
|
||||||
|
# CONFIG_RE_BTN_PRESSED_LEVEL_1 is not set
|
||||||
|
CONFIG_RE_BTN_LONG_PRESS_TIME_US=500000
|
||||||
|
# end of Rotary encoders
|
||||||
# end of Component config
|
# end of Component config
|
||||||
|
|
||||||
#
|
#
|
||||||
|
@ -201,27 +201,17 @@ joystickData_t httpJoystick::getData(){
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
//--------------------------------------------
|
|
||||||
//--- receiveHttpData for httpJoystickMain ---
|
|
||||||
//--------------------------------------------
|
|
||||||
//function that wraps pointer to member function of httpJoystickMain instance in a "normal" function which the webserver can run on joystick URL
|
|
||||||
|
|
||||||
//declare pointer to receiveHttpData method of httpJoystick class
|
|
||||||
esp_err_t (httpJoystick::*pointerToReceiveFunc)(httpd_req_t *req) = &httpJoystick::receiveHttpData;
|
|
||||||
|
|
||||||
esp_err_t on_joystick_url(httpd_req_t *req){
|
|
||||||
//run pointer to receiveHttpData function of httpJoystickMain instance
|
|
||||||
return (httpJoystickMain.*pointerToReceiveFunc)(req);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//============================
|
//============================
|
||||||
//===== init http server =====
|
//===== init http server =====
|
||||||
//============================
|
//============================
|
||||||
//function that initializes http server and configures available urls
|
//function that initializes http server and configures available url's
|
||||||
void http_init_server()
|
|
||||||
|
//parameter: provide pointer to function that handle incomming joystick data (for configuring the url)
|
||||||
|
//TODO add handle functions to future additional endpoints/urls here too
|
||||||
|
void http_init_server(http_handler_t onJoystickUrl)
|
||||||
{
|
{
|
||||||
|
ESP_LOGI(TAG, "initializing HTTP-Server...");
|
||||||
|
|
||||||
//---- configure webserver ----
|
//---- configure webserver ----
|
||||||
httpd_config_t config = HTTPD_DEFAULT_CONFIG();
|
httpd_config_t config = HTTPD_DEFAULT_CONFIG();
|
||||||
@ -236,7 +226,7 @@ void http_init_server()
|
|||||||
httpd_uri_t joystick_url = {
|
httpd_uri_t joystick_url = {
|
||||||
.uri = "/api/joystick",
|
.uri = "/api/joystick",
|
||||||
.method = HTTP_POST,
|
.method = HTTP_POST,
|
||||||
.handler = on_joystick_url,
|
.handler = onJoystickUrl,
|
||||||
};
|
};
|
||||||
httpd_register_uri_handler(server, &joystick_url);
|
httpd_register_uri_handler(server, &joystick_url);
|
||||||
|
|
||||||
@ -265,8 +255,8 @@ void http_init_server()
|
|||||||
//function that destroys the http server
|
//function that destroys the http server
|
||||||
void http_stop_server()
|
void http_stop_server()
|
||||||
{
|
{
|
||||||
printf("stopping http\n");
|
ESP_LOGW(TAG, "stopping HTTP-Server");
|
||||||
httpd_stop(server);
|
httpd_stop(server);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -13,7 +13,18 @@ extern "C"
|
|||||||
//===== init http server =====
|
//===== init http server =====
|
||||||
//============================
|
//============================
|
||||||
//function that initializes http server and configures available urls
|
//function that initializes http server and configures available urls
|
||||||
void http_init_server();
|
//parameter: provide pointer to function that handles incomming joystick data (for configuring the url)
|
||||||
|
//TODO add handle functions to future additional endpoints/urls here too
|
||||||
|
typedef esp_err_t (*http_handler_t)(httpd_req_t *req);
|
||||||
|
void http_init_server(http_handler_t onJoystickUrl);
|
||||||
|
|
||||||
|
//example with lambda function to pass method of a class instance:
|
||||||
|
//esp_err_t (httpJoystick::*pointerToReceiveFunc)(httpd_req_t *req) = &httpJoystick::receiveHttpData;
|
||||||
|
//esp_err_t on_joystick_url(httpd_req_t *req){
|
||||||
|
// //run pointer to receiveHttpData function of httpJoystickMain instance
|
||||||
|
// return (httpJoystickMain->*pointerToReceiveFunc)(req);
|
||||||
|
//}
|
||||||
|
//http_init_server(on_joystick_url);
|
||||||
|
|
||||||
|
|
||||||
//==============================
|
//==============================
|
||||||
@ -27,7 +38,7 @@ void start_mdns_service();
|
|||||||
//===== stop http server =====
|
//===== stop http server =====
|
||||||
//============================
|
//============================
|
||||||
//function that destroys the http server
|
//function that destroys the http server
|
||||||
void http_stop_server();
|
void http_stop_server(httpd_handle_t * httpServer);
|
||||||
|
|
||||||
|
|
||||||
//==============================
|
//==============================
|
||||||
@ -47,7 +58,7 @@ typedef struct httpJoystick_config_t {
|
|||||||
class httpJoystick{
|
class httpJoystick{
|
||||||
public:
|
public:
|
||||||
//--- constructor ---
|
//--- constructor ---
|
||||||
httpJoystick( httpJoystick_config_t config_f );
|
httpJoystick(httpJoystick_config_t config_f);
|
||||||
|
|
||||||
//--- functions ---
|
//--- functions ---
|
||||||
joystickData_t getData(); //wait for and return joystick data from queue, if timeout return CENTER
|
joystickData_t getData(); //wait for and return joystick data from queue, if timeout return CENTER
|
||||||
@ -67,11 +78,4 @@ class httpJoystick{
|
|||||||
.radius = 0,
|
.radius = 0,
|
||||||
.angle = 0
|
.angle = 0
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//===== global object =====
|
|
||||||
//create global instance of httpJoystick
|
|
||||||
//note: is constructed/configured in config.cpp
|
|
||||||
extern httpJoystick httpJoystickMain;
|
|
@ -7,6 +7,24 @@ static const char * TAG = "motor-control";
|
|||||||
|
|
||||||
#define TIMEOUT_IDLE_WHEN_NO_COMMAND 8000
|
#define TIMEOUT_IDLE_WHEN_NO_COMMAND 8000
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//====================================
|
||||||
|
//========== motorctl task ===========
|
||||||
|
//====================================
|
||||||
|
//task for handling the motors (ramp, current limit, driver)
|
||||||
|
void task_motorctl( void * task_motorctl_parameters ){
|
||||||
|
task_motorctl_parameters_t *objects = (task_motorctl_parameters_t *)task_motorctl_parameters;
|
||||||
|
ESP_LOGW(TAG, "Task-motorctl: starting handle loop...");
|
||||||
|
while(1){
|
||||||
|
objects->motorRight->handle();
|
||||||
|
objects->motorLeft->handle();
|
||||||
|
vTaskDelay(20 / portTICK_PERIOD_MS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//=============================
|
//=============================
|
||||||
//======== constructor ========
|
//======== constructor ========
|
||||||
//=============================
|
//=============================
|
||||||
@ -33,6 +51,11 @@ controlledMotor::controlledMotor(motorSetCommandFunc_t setCommandFunc, motorctl
|
|||||||
//============================
|
//============================
|
||||||
void controlledMotor::init(){
|
void controlledMotor::init(){
|
||||||
commandQueue = xQueueCreate( 1, sizeof( struct motorCommand_t ) );
|
commandQueue = xQueueCreate( 1, sizeof( struct motorCommand_t ) );
|
||||||
|
if (commandQueue == NULL)
|
||||||
|
ESP_LOGE(TAG, "Failed to create command-queue");
|
||||||
|
else
|
||||||
|
ESP_LOGW(TAG, "Initialized command-queue");
|
||||||
|
|
||||||
//cSensor.calibrateZeroAmpere(); //currently done in currentsensor constructor TODO do this regularly e.g. in idle?
|
//cSensor.calibrateZeroAmpere(); //currently done in currentsensor constructor TODO do this regularly e.g. in idle?
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -252,11 +275,11 @@ void controlledMotor::setTarget(motorstate_t state_f, float duty_f){
|
|||||||
.state = state_f,
|
.state = state_f,
|
||||||
.duty = duty_f
|
.duty = duty_f
|
||||||
};
|
};
|
||||||
|
ESP_LOGI(TAG, "setTarget: Inserting command to queue: state='%s'(%d), duty=%.2f", motorstateStr[(int)commandSend.state], (int)commandSend.state, commandSend.duty);
|
||||||
ESP_LOGD(TAG, "Inserted command to queue: state=%s, duty=%.2f", motorstateStr[(int)commandSend.state], commandSend.duty);
|
|
||||||
//send command to queue (overwrite if an old command is still in the queue and not processed)
|
//send command to queue (overwrite if an old command is still in the queue and not processed)
|
||||||
xQueueOverwrite( commandQueue, ( void * )&commandSend);
|
xQueueOverwrite( commandQueue, ( void * )&commandSend);
|
||||||
//xQueueSend( commandQueue, ( void * )&commandSend, ( TickType_t ) 0 );
|
//xQueueSend( commandQueue, ( void * )&commandSend, ( TickType_t ) 0 );
|
||||||
|
ESP_LOGD(TAG, "finished inserting new command");
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -86,3 +86,21 @@ class controlledMotor {
|
|||||||
uint32_t timestamp_commandReceived = 0;
|
uint32_t timestamp_commandReceived = 0;
|
||||||
bool receiveTimeout = false;
|
bool receiveTimeout = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// struct with variables passed to task from main
|
||||||
|
typedef struct task_motorctl_parameters_t {
|
||||||
|
controlledMotor * motorLeft;
|
||||||
|
controlledMotor * motorRight;
|
||||||
|
} task_motorctl_parameters_t;
|
||||||
|
|
||||||
|
|
||||||
|
//====================================
|
||||||
|
//========== motorctl task ===========
|
||||||
|
//====================================
|
||||||
|
//task that inititialized the display, displays welcome message
|
||||||
|
//and releatedly updates the display with certain content
|
||||||
|
//note: pointer to required objects have to be provided as task-parameter
|
||||||
|
void task_motorctl( void * task_motorctl_parameters );
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user