ignition controller detects rising edge on voltage (#3636)
* ignition controller detects rising edge on voltage * update test * comment * ignore negative transients * tweak * test
This commit is contained in:
parent
aae32fd3ca
commit
91ec135cf8
|
@ -4,10 +4,10 @@
|
||||||
#include "fuel_pump.h"
|
#include "fuel_pump.h"
|
||||||
|
|
||||||
void FuelPumpController::onSlowCallback() {
|
void FuelPumpController::onSlowCallback() {
|
||||||
auto uptime = getTimeNowSeconds();
|
auto timeSinceIgn = m_ignOnTimer.getElapsedSeconds();
|
||||||
|
|
||||||
// If the ECU just started, turn on the fuel pump to prime
|
// If the ignition just turned on, turn on the fuel pump to prime
|
||||||
isPrime = uptime >= 0 && uptime < engineConfiguration->startUpFuelPumpDuration;
|
isPrime = timeSinceIgn >= 0 && timeSinceIgn < engineConfiguration->startUpFuelPumpDuration;
|
||||||
|
|
||||||
// If there was a trigger event recently, turn on the pump, the engine is running!
|
// If there was a trigger event recently, turn on the pump, the engine is running!
|
||||||
auto timeSinceTrigger = engine->triggerCentral.getTimeSinceTriggerEvent(getTimeNowNt());
|
auto timeSinceTrigger = engine->triggerCentral.getTimeSinceTriggerEvent(getTimeNowNt());
|
||||||
|
@ -21,3 +21,9 @@ void FuelPumpController::onSlowCallback() {
|
||||||
engine->outputChannels.isFuelPumpOn = isPumpOn;
|
engine->outputChannels.isFuelPumpOn = isPumpOn;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void FuelPumpController::onIgnitionStateChanged(bool ignitionOn) {
|
||||||
|
if (ignitionOn) {
|
||||||
|
m_ignOnTimer.reset();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -3,6 +3,11 @@
|
||||||
#include "engine_module.h"
|
#include "engine_module.h"
|
||||||
#include "fuel_pump_generated.h"
|
#include "fuel_pump_generated.h"
|
||||||
|
|
||||||
struct FuelPumpController : public EngineModule, public fuel_pump_control_s {
|
class FuelPumpController : public EngineModule, public fuel_pump_control_s {
|
||||||
|
public:
|
||||||
void onSlowCallback() override;
|
void onSlowCallback() override;
|
||||||
|
void onIgnitionStateChanged(bool ignitionOn) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
Timer m_ignOnTimer;
|
||||||
};
|
};
|
||||||
|
|
|
@ -6,8 +6,6 @@ void MainRelayController::onSlowCallback() {
|
||||||
isBenchTest = engine->isInMainRelayBench();
|
isBenchTest = engine->isInMainRelayBench();
|
||||||
|
|
||||||
#if EFI_MAIN_RELAY_CONTROL
|
#if EFI_MAIN_RELAY_CONTROL
|
||||||
hasIgnitionVoltage = Sensor::getOrZero(SensorType::BatteryVoltage) > 5;
|
|
||||||
|
|
||||||
mainRelayState = isBenchTest | hasIgnitionVoltage;
|
mainRelayState = isBenchTest | hasIgnitionVoltage;
|
||||||
#else // not EFI_MAIN_RELAY_CONTROL
|
#else // not EFI_MAIN_RELAY_CONTROL
|
||||||
mainRelayState = !isBenchTest;
|
mainRelayState = !isBenchTest;
|
||||||
|
@ -15,3 +13,7 @@ void MainRelayController::onSlowCallback() {
|
||||||
|
|
||||||
enginePins.mainRelay.setValue(mainRelayState);
|
enginePins.mainRelay.setValue(mainRelayState);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MainRelayController::onIgnitionStateChanged(bool ignitionOn) {
|
||||||
|
hasIgnitionVoltage = ignitionOn;
|
||||||
|
}
|
||||||
|
|
|
@ -5,4 +5,5 @@
|
||||||
|
|
||||||
struct MainRelayController : public EngineModule, public main_relay_s {
|
struct MainRelayController : public EngineModule, public main_relay_s {
|
||||||
void onSlowCallback() override;
|
void onSlowCallback() override;
|
||||||
|
void onIgnitionStateChanged(bool ignitionOn) override;
|
||||||
};
|
};
|
||||||
|
|
|
@ -34,6 +34,7 @@
|
||||||
#include "ac_control.h"
|
#include "ac_control.h"
|
||||||
#include "type_list.h"
|
#include "type_list.h"
|
||||||
#include "boost_control.h"
|
#include "boost_control.h"
|
||||||
|
#include "ignition_controller.h"
|
||||||
#include "alternator_controller.h"
|
#include "alternator_controller.h"
|
||||||
|
|
||||||
#ifndef EFI_UNIT_TEST
|
#ifndef EFI_UNIT_TEST
|
||||||
|
@ -154,6 +155,7 @@ public:
|
||||||
#endif /* EFI_ALTERNATOR_CONTROL */
|
#endif /* EFI_ALTERNATOR_CONTROL */
|
||||||
FuelPumpController,
|
FuelPumpController,
|
||||||
MainRelayController,
|
MainRelayController,
|
||||||
|
IgnitionController,
|
||||||
AcController,
|
AcController,
|
||||||
EngineModule // dummy placeholder so the previous entries can all have commas
|
EngineModule // dummy placeholder so the previous entries can all have commas
|
||||||
> engineModules;
|
> engineModules;
|
||||||
|
|
|
@ -21,6 +21,7 @@ CONTROLLERS_SRC_CPP = \
|
||||||
$(CONTROLLERS_DIR)/actuators/idle_thread_io.cpp \
|
$(CONTROLLERS_DIR)/actuators/idle_thread_io.cpp \
|
||||||
$(CONTROLLERS_DIR)/actuators/idle_hardware.cpp \
|
$(CONTROLLERS_DIR)/actuators/idle_hardware.cpp \
|
||||||
$(CONTROLLERS_DIR)/actuators/idle_thread.cpp \
|
$(CONTROLLERS_DIR)/actuators/idle_thread.cpp \
|
||||||
|
$(CONTROLLERS_DIR)/actuators/ignition_controller.cpp \
|
||||||
$(CONTROLLERS_DIR)/actuators/main_relay.cpp \
|
$(CONTROLLERS_DIR)/actuators/main_relay.cpp \
|
||||||
$(CONTROLLERS_DIR)/actuators/pwm_tester.cpp \
|
$(CONTROLLERS_DIR)/actuators/pwm_tester.cpp \
|
||||||
$(CONTROLLERS_DIR)/actuators/vvt.cpp \
|
$(CONTROLLERS_DIR)/actuators/vvt.cpp \
|
||||||
|
|
|
@ -10,4 +10,7 @@ public:
|
||||||
|
|
||||||
// Called approx 200Hz
|
// Called approx 200Hz
|
||||||
virtual void onFastCallback() { }
|
virtual void onFastCallback() { }
|
||||||
|
|
||||||
|
// Called whenever the ignition switch state changes
|
||||||
|
virtual void onIgnitionStateChanged(bool ignitionOn) { }
|
||||||
};
|
};
|
||||||
|
|
|
@ -0,0 +1,26 @@
|
||||||
|
#include "pch.h"
|
||||||
|
|
||||||
|
void IgnitionController::onSlowCallback() {
|
||||||
|
// default to 0 if failed sensor to prevent accidental ign-on if battery
|
||||||
|
// input misconfigured (or the ADC hasn't started yet)
|
||||||
|
auto hasIgnVoltage = Sensor::get(SensorType::BatteryVoltage).value_or(0) > 5;
|
||||||
|
|
||||||
|
if (hasIgnVoltage) {
|
||||||
|
m_timeSinceIgnVoltage.reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hasIgnVoltage == m_lastState) {
|
||||||
|
// nothing to do, states match
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ignore low voltage transients - we may see this at the start of cranking
|
||||||
|
// and we don't want to
|
||||||
|
if (!hasIgnVoltage && !m_timeSinceIgnVoltage.hasElapsedSec(0.2f)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Store state and notify other modules of the change
|
||||||
|
m_lastState = hasIgnVoltage;
|
||||||
|
engine->engineModules.apply_all([&](auto& m) { m.onIgnitionStateChanged(hasIgnVoltage); });
|
||||||
|
}
|
|
@ -0,0 +1,14 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "engine_module.h"
|
||||||
|
|
||||||
|
#include "timer.h"
|
||||||
|
|
||||||
|
class IgnitionController : public EngineModule {
|
||||||
|
public:
|
||||||
|
void onSlowCallback() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
Timer m_timeSinceIgnVoltage;
|
||||||
|
bool m_lastState = false;
|
||||||
|
};
|
|
@ -254,9 +254,10 @@ TEST(fsio, testLogicExpressions) {
|
||||||
extern int timeNowUs;
|
extern int timeNowUs;
|
||||||
|
|
||||||
TEST(fsio, fuelPump) {
|
TEST(fsio, fuelPump) {
|
||||||
// this will init fuel pump fsio logic
|
|
||||||
EngineTestHelper eth(TEST_ENGINE);
|
EngineTestHelper eth(TEST_ENGINE);
|
||||||
|
|
||||||
|
FuelPumpController dut;
|
||||||
|
|
||||||
// Mock a fuel pump pin
|
// Mock a fuel pump pin
|
||||||
engineConfiguration->fuelPumpPin = GPIOA_0;
|
engineConfiguration->fuelPumpPin = GPIOA_0;
|
||||||
// Re-init so it picks up the new config
|
// Re-init so it picks up the new config
|
||||||
|
@ -264,27 +265,30 @@ TEST(fsio, fuelPump) {
|
||||||
|
|
||||||
// ECU just started, haven't seen trigger yet
|
// ECU just started, haven't seen trigger yet
|
||||||
timeNowUs = 0.5e6;
|
timeNowUs = 0.5e6;
|
||||||
engine->module<FuelPumpController>().unmock().onSlowCallback();
|
dut.onIgnitionStateChanged(true);
|
||||||
|
dut.onSlowCallback();
|
||||||
// Pump should be on!
|
// Pump should be on!
|
||||||
EXPECT_TRUE(efiReadPin(GPIOA_0));
|
EXPECT_TRUE(efiReadPin(GPIOA_0));
|
||||||
|
|
||||||
// Long time since ecu start, haven't seen trigger yet
|
// Long time since ecu start, haven't seen trigger yet
|
||||||
timeNowUs = 60e6;
|
dut.onIgnitionStateChanged(true);
|
||||||
engine->module<FuelPumpController>().unmock().onSlowCallback();
|
timeNowUs += 10e6;
|
||||||
|
dut.onSlowCallback();
|
||||||
// Pump should be off!
|
// Pump should be off!
|
||||||
EXPECT_FALSE(efiReadPin(GPIOA_0));
|
EXPECT_FALSE(efiReadPin(GPIOA_0));
|
||||||
|
|
||||||
// Long time since ecu start, just saw a trigger!
|
// Long time since ecu start, just saw a trigger!
|
||||||
timeNowUs = 10e6;
|
dut.onIgnitionStateChanged(true);
|
||||||
|
timeNowUs += 10e6;
|
||||||
engine->triggerCentral.handleShaftSignal(SHAFT_PRIMARY_FALLING, timeNowUs * US_TO_NT_MULTIPLIER);
|
engine->triggerCentral.handleShaftSignal(SHAFT_PRIMARY_FALLING, timeNowUs * US_TO_NT_MULTIPLIER);
|
||||||
engine->module<FuelPumpController>().unmock().onSlowCallback();
|
dut.onSlowCallback();
|
||||||
// Pump should be on!
|
// Pump should be on!
|
||||||
EXPECT_TRUE(efiReadPin(GPIOA_0));
|
EXPECT_TRUE(efiReadPin(GPIOA_0));
|
||||||
|
|
||||||
// ECU just started, and we just saw a trigger!
|
// ECU just started, and we just saw a trigger!
|
||||||
timeNowUs = 10e6;
|
dut.onIgnitionStateChanged(true);
|
||||||
engine->triggerCentral.handleShaftSignal(SHAFT_PRIMARY_FALLING, timeNowUs * US_TO_NT_MULTIPLIER);
|
engine->triggerCentral.handleShaftSignal(SHAFT_PRIMARY_FALLING, timeNowUs * US_TO_NT_MULTIPLIER);
|
||||||
engine->module<FuelPumpController>().unmock().onSlowCallback();
|
dut.onSlowCallback();
|
||||||
// Pump should be on!
|
// Pump should be on!
|
||||||
EXPECT_TRUE(efiReadPin(GPIOA_0));
|
EXPECT_TRUE(efiReadPin(GPIOA_0));
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,13 +7,13 @@ TEST(MainRelay, mr) {
|
||||||
|
|
||||||
MainRelayController dut;
|
MainRelayController dut;
|
||||||
|
|
||||||
// Low battery voltage, MR is off
|
// Ignition is off, MR is off
|
||||||
Sensor::setMockValue(SensorType::BatteryVoltage, 1);
|
dut.onIgnitionStateChanged(false);
|
||||||
dut.onSlowCallback();
|
dut.onSlowCallback();
|
||||||
EXPECT_EQ(enginePins.mainRelay.getLogicValue(), false);
|
EXPECT_EQ(enginePins.mainRelay.getLogicValue(), false);
|
||||||
|
|
||||||
// Ignition is now on, MR is on
|
// Ignition is now on, MR is on
|
||||||
Sensor::setMockValue(SensorType::BatteryVoltage, 12);
|
dut.onIgnitionStateChanged(true);
|
||||||
dut.onSlowCallback();
|
dut.onSlowCallback();
|
||||||
EXPECT_EQ(enginePins.mainRelay.getLogicValue(), true);
|
EXPECT_EQ(enginePins.mainRelay.getLogicValue(), true);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue