- outsource task_fans and task_buzzer from main.cpp to their source files - use task parameters to pass necessary configs and objects - adjust task priorities (display was too low)
105 lines
3.5 KiB
C++
105 lines
3.5 KiB
C++
extern "C"
|
|
{
|
|
#include "freertos/FreeRTOS.h"
|
|
#include "freertos/task.h"
|
|
#include "esp_log.h"
|
|
}
|
|
|
|
#include "fan.hpp"
|
|
|
|
|
|
//tag for logging
|
|
static const char * TAG = "fan-control";
|
|
|
|
|
|
//=======================================
|
|
//============== fan task ===============
|
|
//=======================================
|
|
//task that controlls fans for cooling the drivers
|
|
//turns fan on/off depending on motor duty history
|
|
void task_fans( void * task_fans_parameters ){
|
|
//get configuration struct from task parameter
|
|
task_fans_parameters_t *objects = (task_fans_parameters_t *)task_fans_parameters;
|
|
|
|
//create fan instances with config defined in config.cpp
|
|
ESP_LOGI(TAG, "Initializing fans and starting fan handle loop");
|
|
controlledFan fan(objects->fan_config, objects->motorLeft, objects->motorRight);
|
|
|
|
//repeatedly run fan handle function in a slow loop
|
|
while(1){
|
|
fan.handle();
|
|
vTaskDelay(500 / portTICK_PERIOD_MS);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------
|
|
//-------- constructor --------
|
|
//-----------------------------
|
|
controlledFan::controlledFan (fan_config_t config_f, controlledMotor* motor1_f, controlledMotor* motor2_f ){
|
|
//copy config
|
|
config = config_f;
|
|
//copy pointer to motor objects
|
|
motor1 = motor1_f;
|
|
motor2 = motor2_f;
|
|
//initialize gpio pin
|
|
gpio_pad_select_gpio(config.gpio_fan);
|
|
gpio_set_direction(config.gpio_fan, GPIO_MODE_OUTPUT);
|
|
}
|
|
|
|
|
|
|
|
//--------------------------
|
|
//--------- handle ---------
|
|
//--------------------------
|
|
void controlledFan::handle(){
|
|
//get current state of the motor (motorctl.cpp)
|
|
motor1Status = motor1->getStatus();
|
|
motor2Status = motor2->getStatus();
|
|
|
|
//--- handle duty threshold ---
|
|
//update timestamp if any threshold exceeded
|
|
if (motor1Status.duty > config.dutyThreshold
|
|
|| motor2Status.duty > config.dutyThreshold){ //TODO add temperature threshold
|
|
if (!needsCooling){
|
|
timestamp_needsCoolingSet = esp_log_timestamp();
|
|
needsCooling = true;
|
|
}
|
|
timestamp_lastThreshold = esp_log_timestamp();
|
|
} else {
|
|
needsCooling = false;
|
|
}
|
|
|
|
//--- turn off condition ---
|
|
if (fanRunning
|
|
&& !needsCooling //no more cooling required
|
|
&& (motor1Status.duty == 0) && (motor2Status.duty == 0) //both motors are off
|
|
//-> keeps fans running even when lower than threshold already, however turnOffDelay already started TODO: start turn off delay after motor stop only?
|
|
&& (esp_log_timestamp() - timestamp_lastThreshold) > config.turnOffDelayMs){ //turn off delay passed
|
|
fanRunning = false;
|
|
gpio_set_level(config.gpio_fan, 0);
|
|
timestamp_turnedOff = esp_log_timestamp();
|
|
ESP_LOGI(TAG, "turned fan OFF gpio=%d, minOnMs=%d, WasOnMs=%d", (int)config.gpio_fan, config.minOnMs, esp_log_timestamp()-timestamp_turnedOn);
|
|
}
|
|
|
|
//--- turn on condition ---
|
|
if (!fanRunning
|
|
&& needsCooling
|
|
&& ((esp_log_timestamp() - timestamp_turnedOff) > config.minOffMs) //fans off long enough
|
|
&& ((esp_log_timestamp() - timestamp_needsCoolingSet) > config.minOnMs)){ //motors on long enough
|
|
fanRunning = true;
|
|
gpio_set_level(config.gpio_fan, 1);
|
|
timestamp_turnedOn = esp_log_timestamp();
|
|
ESP_LOGI(TAG, "turned fan ON gpio=%d, minOffMs=%d, WasOffMs=%d", (int)config.gpio_fan, config.minOffMs, esp_log_timestamp()-timestamp_turnedOff);
|
|
}
|
|
|
|
//TODO Add statemachine for more specific control? Exponential average?
|
|
//TODO idea: try other aproach? increment a variable with certain weights e.g. integrate over duty, then turn fans on and decrement the variable again
|
|
|
|
ESP_LOGD(TAG, "fanState=%d, duty1=%f, duty2=%f, needsCooling=%d", fanRunning, motor1Status.duty, motor2Status.duty, needsCooling);
|
|
}
|
|
|
|
|
|
|