From c2cb3b9015797042fe3ad444863e1ea8f469ce5d Mon Sep 17 00:00:00 2001 From: Matthew Kennedy Date: Mon, 13 Nov 2023 14:15:22 -0800 Subject: [PATCH] test heater state machine --- firmware/heater_control.cpp | 6 +-- firmware/heater_control.h | 5 +++ test/tests/test_heater.cpp | 81 ++++++++++++++++++++++++++++++++++++- 3 files changed, 88 insertions(+), 4 deletions(-) diff --git a/firmware/heater_control.cpp b/firmware/heater_control.cpp index 9d12f42..61cadcc 100644 --- a/firmware/heater_control.cpp +++ b/firmware/heater_control.cpp @@ -83,6 +83,8 @@ HeaterState HeaterControllerBase::GetNextState(HeaterState currentState, HeaterA // Stay in preheat - wait for time to elapse break; case HeaterState::WarmupRamp: + timeCounter--; + if (sensorTemp > closedLoopTemp) { return HeaterState::ClosedLoop; @@ -93,8 +95,6 @@ HeaterState HeaterControllerBase::GetNextState(HeaterState currentState, HeaterA return HeaterState::Stopped; } - timeCounter--; - break; case HeaterState::ClosedLoop: // Check that the sensor's ESR is acceptable for normal operation @@ -116,7 +116,7 @@ HeaterState HeaterControllerBase::GetNextState(HeaterState currentState, HeaterA break; } - return heaterState; + return currentState; } float HeaterControllerBase::GetVoltageForState(HeaterState state, float sensorEsr) diff --git a/firmware/heater_control.h b/firmware/heater_control.h index cdbcc0e..9ae3455 100644 --- a/firmware/heater_control.h +++ b/firmware/heater_control.h @@ -42,6 +42,11 @@ public: HeaterState GetNextState(HeaterState currentState, HeaterAllow haeterAllowState, float batteryVoltage, float sensorTemp); float GetVoltageForState(HeaterState state, float sensorEsr); + int GetTimeCounter() const + { + return timeCounter; + } + private: Pid heaterPid = { diff --git a/test/tests/test_heater.cpp b/test/tests/test_heater.cpp index 544f315..c52321f 100644 --- a/test/tests/test_heater.cpp +++ b/test/tests/test_heater.cpp @@ -52,7 +52,86 @@ TEST(HeaterStateOutput, Cases) EXPECT_EQ(0, dut.GetVoltageForState(HeaterState::NoHeaterSupply, 0)); } -TEST(HeaterStateMachine, x) +TEST(HeaterStateMachine, PreheatToWarmupTimeout) { MockHeater dut; + dut.Configure(780, 300); + + for (size_t i = 0; i < HeaterControllerBase::preheatTimeCounter - 1; i++) + { + EXPECT_EQ(HeaterState::Preheat, dut.GetNextState(HeaterState::Preheat, HeaterAllow::Allowed, 12, 500)); + } + + // Timer expired, transition to warmup ramp + EXPECT_EQ(HeaterState::WarmupRamp, dut.GetNextState(HeaterState::Preheat, HeaterAllow::Allowed, 12, 500)); +} + +TEST(HeaterStateMachine, PreheatToWarmupAlreadyWarm) +{ + MockHeater dut; + dut.Configure(780, 300); + + // Preheat for a little while + for (size_t i = 0; i < 10; i++) + { + EXPECT_EQ(HeaterState::Preheat, dut.GetNextState(HeaterState::Preheat, HeaterAllow::Allowed, 12, 500)); + } + + // Sensor magically warms up, skip to warmup ramp! + EXPECT_EQ(HeaterState::WarmupRamp, dut.GetNextState(HeaterState::Preheat, HeaterAllow::Allowed, 12, 780)); +} + +TEST(HeaterStateMachine, WarmupToClosedLoop) +{ + MockHeater dut; + dut.Configure(780, 300); + + // Warm up for a little while + for (size_t i = 0; i < 10; i++) + { + EXPECT_EQ(HeaterState::WarmupRamp, dut.GetNextState(HeaterState::WarmupRamp, HeaterAllow::Allowed, 12, 500)); + } + + // Sensor warms up, time for closed loop! + EXPECT_EQ(HeaterState::ClosedLoop, dut.GetNextState(HeaterState::WarmupRamp, HeaterAllow::Allowed, 12, 780)); +} + +TEST(HeaterStateMachine, WarmupTimeout) +{ + MockHeater dut; + dut.Configure(780, 300); + + size_t timeoutPeriod = dut.GetTimeCounter(); + + // Warm up for a little while + for (size_t i = 0; i < timeoutPeriod - 1; i++) + { + EXPECT_EQ(HeaterState::WarmupRamp, dut.GetNextState(HeaterState::WarmupRamp, HeaterAllow::Allowed, 12, 500)) << "i = " << i; + } + + // Warmup times out, sensor transitions to stopped + EXPECT_EQ(HeaterState::Stopped, dut.GetNextState(HeaterState::WarmupRamp, HeaterAllow::Allowed, 12, 500)); +} + +TEST(HeaterStateMachine, ClosedLoop) +{ + MockHeater dut; + dut.Configure(780, 300); + + // Temperature is reasonable, stay in closed loop + EXPECT_EQ(HeaterState::ClosedLoop, dut.GetNextState(HeaterState::ClosedLoop, HeaterAllow::Allowed, 12, 780)); + + // Temperature is too hot, overheat + EXPECT_EQ(HeaterState::Stopped, dut.GetNextState(HeaterState::ClosedLoop, HeaterAllow::Allowed, 12, 1000)); + + // Temperature is too cold, underheat + EXPECT_EQ(HeaterState::Stopped, dut.GetNextState(HeaterState::ClosedLoop, HeaterAllow::Allowed, 12, 600)); +} + +TEST(HeaterStateMachine, TerminalStates) +{ + MockHeater dut; + dut.Configure(780, 300); + + EXPECT_EQ(HeaterState::Stopped, dut.GetNextState(HeaterState::Stopped, HeaterAllow::Allowed, 12, 780)); }