split heater thread and heater logic

This commit is contained in:
Matthew Kennedy 2023-11-13 13:30:23 -08:00
parent ce7d85ccc5
commit ff5b3ac3bb
8 changed files with 173 additions and 144 deletions

View File

@ -153,7 +153,6 @@ CPPSRC = $(ALLCPPSRC) \
pwm.cpp \ pwm.cpp \
dac.cpp \ dac.cpp \
pump_dac.cpp \ pump_dac.cpp \
pid.cpp \
pump_control.cpp \ pump_control.cpp \
max3185x.cpp \ max3185x.cpp \
uart.cpp \ uart.cpp \
@ -165,7 +164,7 @@ CPPSRC = $(ALLCPPSRC) \
console/binary/tunerstudio_io_serial.cpp \ console/binary/tunerstudio_io_serial.cpp \
console/binary/tunerstudio_commands.cpp \ console/binary/tunerstudio_commands.cpp \
sampling_thread.cpp \ sampling_thread.cpp \
heater_control.cpp \ heater_thread.cpp \
main.cpp main.cpp
# List ASM source files here. # List ASM source files here.

View File

@ -1,103 +1,37 @@
#include "heater_control.h" #include "heater_control.h"
#include "ch.h"
#include "hal.h"
#include "port.h"
#include "fault.h" #include "fault.h"
#include "pwm.h"
#include "sampling.h" #include "sampling.h"
struct sensorHeaterParams {
float targetTemp;
float targetESR;
};
static const struct sensorHeaterParams heaterParams49 = {
.targetTemp = 780,
.targetESR = 300,
};
static const struct sensorHeaterParams heaterParams42 = {
.targetTemp = 730,
.targetESR = 80,
};
static const struct sensorHeaterParams heaterParamsAdv = {
.targetTemp = 785,
.targetESR = 300,
};
static const sensorHeaterParams *getHeaterParams(SensorType type) {
switch (type) {
case SensorType::LSU42:
return &heaterParams42;
case SensorType::LSUADV:
return &heaterParamsAdv;
case SensorType::LSU49:
default:
return &heaterParams49;
}
}
using namespace wbo; using namespace wbo;
// 400khz / 1024 = 390hz PWM HeaterControllerBase::HeaterControllerBase(int ch)
static Pwm heaterPwm(HEATER_PWM_DEVICE); : ch(ch)
static const PWMConfig heaterPwmConfig = {
.frequency = 400'000,
.period = 1024,
.callback = nullptr,
.channels = {
{PWM_OUTPUT_ACTIVE_HIGH, nullptr},
{PWM_OUTPUT_ACTIVE_HIGH, nullptr},
{PWM_OUTPUT_ACTIVE_HIGH, nullptr},
{PWM_OUTPUT_ACTIVE_HIGH, nullptr}
},
.cr2 = 0,
#if STM32_PWM_USE_ADVANCED
.bdtr = 0,
#endif
.dier = 0
};
static const struct sensorHeaterParams *heater;
HeaterController::HeaterController(int ch, int pwm_ch)
: ch(ch), pwm_ch(pwm_ch)
{ {
} }
bool HeaterController::IsRunningClosedLoop() const void HeaterControllerBase::Configure(float targetTempC, float targetEsr)
{
m_targetTempC = targetTempC;
m_targetEsr = targetEsr;
}
bool HeaterControllerBase::IsRunningClosedLoop() const
{ {
return heaterState == HeaterState::ClosedLoop; return heaterState == HeaterState::ClosedLoop;
} }
float HeaterController::GetHeaterEffectiveVoltage() const float HeaterControllerBase::GetHeaterEffectiveVoltage() const
{ {
return heaterVoltage; return heaterVoltage;
} }
HeaterState HeaterController::GetHeaterState() const HeaterState HeaterControllerBase::GetHeaterState() const
{ {
return heaterState; return heaterState;
} }
HeaterController heaterControllers[AFR_CHANNELS] = HeaterState HeaterControllerBase::GetNextState(HeaterAllow heaterAllowState, float batteryVoltage, float sensorTemp)
{
{ 0, HEATER_PWM_CHANNEL_0 },
#if AFR_CHANNELS >= 2
{ 1, HEATER_PWM_CHANNEL_1 }
#endif
};
const IHeaterController& GetHeaterController(int ch)
{
return heaterControllers[ch];
}
HeaterState HeaterController::GetNextState(HeaterAllow heaterAllowState, float batteryVoltage, float sensorTemp)
{ {
bool heaterAllowed = heaterAllowState == HeaterAllow::Allowed; bool heaterAllowed = heaterAllowState == HeaterAllow::Allowed;
@ -124,9 +58,9 @@ HeaterState HeaterController::GetNextState(HeaterAllow heaterAllowState, float b
return HeaterState::Preheat; return HeaterState::Preheat;
} }
float overheatTemp = heater->targetTemp + 100; float overheatTemp = m_targetTempC + 100;
float closedLoopTemp = heater->targetTemp - 50; float closedLoopTemp = m_targetTempC - 50;
float underheatTemp = heater->targetTemp - 100; float underheatTemp = m_targetTempC - 100;
switch (heaterState) switch (heaterState)
{ {
@ -185,7 +119,7 @@ HeaterState HeaterController::GetNextState(HeaterAllow heaterAllowState, float b
return heaterState; return heaterState;
} }
float HeaterController::GetVoltageForState(float heaterEsr) float HeaterControllerBase::GetVoltageForState(float heaterEsr)
{ {
switch (heaterState) switch (heaterState)
{ {
@ -207,7 +141,7 @@ float HeaterController::GetVoltageForState(float heaterEsr)
// Negated because lower resistance -> hotter // Negated because lower resistance -> hotter
// TODO: heater PID should operate on temperature, not ESR // TODO: heater PID should operate on temperature, not ESR
return 7.5f - heaterPid.GetOutput(heater->targetESR, heaterEsr); return 7.5f - heaterPid.GetOutput(m_targetEsr, heaterEsr);
case HeaterState::Stopped: case HeaterState::Stopped:
case HeaterState::NoHeaterSupply: case HeaterState::NoHeaterSupply:
// Something has gone wrong, turn off the heater. // Something has gone wrong, turn off the heater.
@ -218,7 +152,7 @@ float HeaterController::GetVoltageForState(float heaterEsr)
return 0; return 0;
} }
void HeaterController::Update(const ISampler& sampler, HeaterAllow heaterAllowState) void HeaterControllerBase::Update(const ISampler& sampler, HeaterAllow heaterAllowState)
{ {
// Read sensor state // Read sensor state
float heaterEsr = sampler.GetSensorInternalResistance(); float heaterEsr = sampler.GetSensorInternalResistance();
@ -258,60 +192,9 @@ void HeaterController::Update(const ISampler& sampler, HeaterAllow heaterAllowSt
duty = 0; duty = 0;
heaterVoltage = 0; heaterVoltage = 0;
} }
// Pipe the output to the heater driver // Pipe the output to the heater driver
heaterPwm.SetDuty(pwm_ch, duty); SetDuty(duty);
heaterVoltage = heaterVoltage;
}
static THD_WORKING_AREA(waHeaterThread, 256);
static void HeaterThread(void*)
{
int i;
chRegSetThreadName("Heater");
// Wait for temperature sensing to stabilize so we don't
// immediately think we overshot the target temperature
chThdSleepMilliseconds(1000);
// Get sensor type and settings
heater = getHeaterParams(GetSensorType());
while (true)
{
auto heaterAllowState = GetHeaterAllowed();
for (i = 0; i < AFR_CHANNELS; i++) {
const auto& sampler = GetSampler(i);
auto& heater = heaterControllers[i];
heater.Update(sampler, heaterAllowState);
}
// Loop at ~20hz
chThdSleepMilliseconds(HEATER_CONTROL_PERIOD);
}
}
void StartHeaterControl()
{
heaterPwm.Start(heaterPwmConfig);
heaterPwm.SetDuty(heaterControllers[0].pwm_ch, 0);
#if (AFR_CHANNELS > 1)
heaterPwm.SetDuty(heaterControllers[1].pwm_ch, 0);
#endif
chThdCreateStatic(waHeaterThread, sizeof(waHeaterThread), NORMALPRIO + 1, HeaterThread, nullptr);
}
float GetHeaterDuty(int ch)
{
return heaterPwm.GetLastDuty(heaterControllers[ch].pwm_ch);
}
HeaterState GetHeaterState(int ch)
{
return heaterControllers[ch].GetHeaterState();
} }
const char* describeHeaterState(HeaterState state) const char* describeHeaterState(HeaterState state)

View File

@ -26,17 +26,19 @@ struct IHeaterController
virtual HeaterState GetHeaterState() const = 0; virtual HeaterState GetHeaterState() const = 0;
}; };
class HeaterController : public IHeaterController class HeaterControllerBase : public IHeaterController
{ {
public: public:
HeaterController(int ch, int pwm_ch); HeaterControllerBase(int ch);
void Configure(float targetTempC, float targetEsr);
void Update(const ISampler& sampler, HeaterAllow heaterAllowState) override; void Update(const ISampler& sampler, HeaterAllow heaterAllowState) override;
bool IsRunningClosedLoop() const override; bool IsRunningClosedLoop() const override;
float GetHeaterEffectiveVoltage() const override; float GetHeaterEffectiveVoltage() const override;
HeaterState GetHeaterState() const override; HeaterState GetHeaterState() const override;
virtual void SetDuty(float duty) const = 0;
protected: protected:
HeaterState GetNextState(HeaterAllow haeterAllowState, float batteryVoltage, float sensorTemp); HeaterState GetNextState(HeaterAllow haeterAllowState, float batteryVoltage, float sensorTemp);
float GetVoltageForState(float heaterEsr); float GetVoltageForState(float heaterEsr);
@ -60,10 +62,12 @@ private:
int cycle; int cycle;
#endif #endif
float m_targetEsr = 0;
float m_targetTempC = 0;
// TODO: private: // TODO: private:
public: public:
const uint8_t ch; const uint8_t ch;
const uint8_t pwm_ch;
static const int preheatTimeCounter = HEATER_PREHEAT_TIME / HEATER_CONTROL_PERIOD; static const int preheatTimeCounter = HEATER_PREHEAT_TIME / HEATER_CONTROL_PERIOD;
static const int batteryStabTimeCounter = HEATER_BATTERY_STAB_TIME / HEATER_CONTROL_PERIOD; static const int batteryStabTimeCounter = HEATER_BATTERY_STAB_TIME / HEATER_CONTROL_PERIOD;

120
firmware/heater_thread.cpp Normal file
View File

@ -0,0 +1,120 @@
#include "ch.h"
#include "hal.h"
#include "pwm.h"
#include "heater_control.h"
#include "port.h"
#include "sampling.h"
// 400khz / 1024 = 390hz PWM
static Pwm heaterPwm(HEATER_PWM_DEVICE);
static const PWMConfig heaterPwmConfig = {
.frequency = 400'000,
.period = 1024,
.callback = nullptr,
.channels = {
{PWM_OUTPUT_ACTIVE_HIGH, nullptr},
{PWM_OUTPUT_ACTIVE_HIGH, nullptr},
{PWM_OUTPUT_ACTIVE_HIGH, nullptr},
{PWM_OUTPUT_ACTIVE_HIGH, nullptr}
},
.cr2 = 0,
#if STM32_PWM_USE_ADVANCED
.bdtr = 0,
#endif
.dier = 0
};
class HeaterController : public HeaterControllerBase {
public:
HeaterController(int ch, int pwm_ch)
: HeaterControllerBase(ch)
, pwm_ch(pwm_ch)
{
}
void SetDuty(float duty) const override
{
heaterPwm.SetDuty(pwm_ch, duty);
}
// TODO: private:
public:
const uint8_t pwm_ch;
};
HeaterController heaterControllers[AFR_CHANNELS] =
{
{ 0, HEATER_PWM_CHANNEL_0 },
#if AFR_CHANNELS >= 2
{ 1, HEATER_PWM_CHANNEL_1 }
#endif
};
const IHeaterController& GetHeaterController(int ch)
{
return heaterControllers[ch];
}
static THD_WORKING_AREA(waHeaterThread, 256);
static void HeaterThread(void*)
{
int i;
chRegSetThreadName("Heater");
// Wait for temperature sensing to stabilize so we don't
// immediately think we overshot the target temperature
chThdSleepMilliseconds(1000);
// Configure heater controllers for sensor type
for (i = 0; i < AFR_CHANNELS; i++) {
auto& h = heaterControllers[i];
switch (GetSensorType()) {
case SensorType::LSU42:
return h.Configure(730, 80);
case SensorType::LSUADV:
return h.Configure(785, 300);
case SensorType::LSU49:
default:
return h.Configure(780, 300);
}
}
while (true)
{
auto heaterAllowState = GetHeaterAllowed();
for (i = 0; i < AFR_CHANNELS; i++) {
const auto& sampler = GetSampler(i);
auto& heater = heaterControllers[i];
heater.Update(sampler, heaterAllowState);
}
// Loop at ~20hz
chThdSleepMilliseconds(HEATER_CONTROL_PERIOD);
}
}
void StartHeaterControl()
{
heaterPwm.Start(heaterPwmConfig);
heaterPwm.SetDuty(heaterControllers[0].pwm_ch, 0);
#if (AFR_CHANNELS > 1)
heaterPwm.SetDuty(heaterControllers[1].pwm_ch, 0);
#endif
chThdCreateStatic(waHeaterThread, sizeof(waHeaterThread), NORMALPRIO + 1, HeaterThread, nullptr);
}
float GetHeaterDuty(int ch)
{
return heaterPwm.GetLastDuty(heaterControllers[ch].pwm_ch);
}
HeaterState GetHeaterState(int ch)
{
return heaterControllers[ch].GetHeaterState();
}

View File

@ -1,2 +1,4 @@
WIDEBANDSRC = \ WIDEBANDSRC = \
$(FIRMWARE_DIR)/pid.cpp \
$(FIRMWARE_DIR)/sampling.cpp \ $(FIRMWARE_DIR)/sampling.cpp \
$(FIRMWARE_DIR)/heater_control.cpp \

View File

@ -27,6 +27,7 @@ CPPSRC += \
gtest-all.cpp \ gtest-all.cpp \
gmock-all.cpp \ gmock-all.cpp \
gtest_main.cpp \ gtest_main.cpp \
test_stubs.cpp \
tests/test_sampler.cpp \ tests/test_sampler.cpp \
tests/test_heater.cpp \ tests/test_heater.cpp \

11
test/test_stubs.cpp Normal file
View File

@ -0,0 +1,11 @@
#include "fault.h"
#include "can.h"
void SetFault(int, wbo::Fault)
{
}
float GetRemoteBatteryVoltage()
{
return 0;
}

View File

@ -1,6 +1,15 @@
#include <gtest/gtest.h> #include <gtest/gtest.h>
#include <gmock/gmock.h>
#include "heater_control.h"
struct MockHeater : public HeaterControllerBase {
MockHeater() : HeaterControllerBase(0) { }
MOCK_METHOD(void, SetDuty, (float), (const, override));
};
TEST(Heater, Basic) TEST(Heater, Basic)
{ {
MockHeater dut;
} }