rusefi/firmware/controllers/engine_cycle/prime_injection.cpp

118 lines
3.4 KiB
C++
Raw Normal View History

2022-09-07 13:25:50 -07:00
/*
* @file prime_injection.cpp
*/
2022-09-07 13:27:56 -07:00
#include "pch.h"
2022-09-07 13:25:50 -07:00
#include "prime_injection.h"
2022-09-07 16:35:52 -07:00
#include "injection_gpio.h"
2022-09-07 13:27:56 -07:00
#include "sensor.h"
2022-09-07 17:31:04 -07:00
#include "backup_ram.h"
2022-09-07 13:25:50 -07:00
2022-09-07 13:27:56 -07:00
floatms_t PrimeController::getPrimeDuration() const {
auto clt = Sensor::get(SensorType::Clt);
// If the coolant sensor is dead, skip the prime. The engine will still start fine, but may take a little longer.
if (!clt) {
return 0;
}
auto primeMass =
0.001f * // convert milligram to gram
interpolate2d(clt.Value, engineConfiguration->primeBins, engineConfiguration->primeValues);
return engine->module<InjectorModelPrimary>()->getInjectionDuration(primeMass);
2022-09-07 13:27:56 -07:00
}
2022-09-28 04:53:12 -07:00
2022-09-07 18:29:11 -07:00
// Check if the engine is not stopped or cylinder cleanup is activated
static bool isPrimeInjectionPulseSkipped() {
2022-09-28 04:53:12 -07:00
// Skip if the engine is already spinning
if (!getEngineRotationState()->isStopped()) {
2022-09-07 18:29:11 -07:00
return true;
2022-09-28 04:53:12 -07:00
}
// Skip if cylinder cleanup is active
2022-09-07 18:29:11 -07:00
return engineConfiguration->isCylinderCleanupEnabled && (Sensor::getOrZero(SensorType::Tps1) > CLEANUP_MODE_TPS);
}
void PrimeController::onIgnitionStateChanged(bool ignitionOn) {
if (!ignitionOn) {
// don't prime on ignition-off
return;
}
// First, we need a protection against 'fake' ignition switch on and off (i.e. no engine started), to avoid repeated prime pulses.
// So we check and update the ignition switch counter in non-volatile backup-RAM
2022-09-28 04:53:12 -07:00
uint32_t ignSwitchCounter = getKeyCycleCounter();
2022-09-07 18:29:11 -07:00
// if we're just toying with the ignition switch, give it another chance eventually...
2022-09-28 04:53:12 -07:00
if (ignSwitchCounter > 10) {
2022-09-07 18:29:11 -07:00
ignSwitchCounter = 0;
2022-09-28 04:53:12 -07:00
}
2022-09-07 18:29:11 -07:00
// If we're going to skip this pulse, then save the counter as 0.
// That's because we'll definitely need the prime pulse next time (either due to the cylinder cleanup or the engine spinning)
2022-09-28 04:53:12 -07:00
if (isPrimeInjectionPulseSkipped()) {
2022-09-07 18:29:11 -07:00
ignSwitchCounter = -1;
2022-09-28 04:53:12 -07:00
}
2022-09-07 18:29:11 -07:00
// start prime injection if this is a 'fresh start'
if (ignSwitchCounter == 0) {
auto primeDelayMs = engineConfiguration->primingDelay * 1000;
auto startTime = getTimeNowNt() + MS2NT(primeDelayMs);
getExecutorInterface()->scheduleByTimestampNt("prime", nullptr, startTime, { PrimeController::onPrimeStartAdapter, this });
2022-09-07 18:29:11 -07:00
} else {
efiPrintf("Skipped priming pulse since ignSwitchCounter = %d", ignSwitchCounter);
}
2022-09-28 04:53:12 -07:00
2022-09-07 18:29:11 -07:00
// we'll reset it later when the engine starts
2022-09-28 04:53:12 -07:00
setKeyCycleCounter(ignSwitchCounter + 1);
}
void PrimeController::setKeyCycleCounter(uint32_t count) {
2023-05-24 08:12:00 -07:00
#if EFI_BACKUP_SRAM
2024-02-06 11:04:34 -08:00
backupRamSave(backup_ram_e::IgnCounter, count);
2023-05-24 08:12:00 -07:00
#endif // EFI_BACKUP_SRAM
2022-09-28 04:53:12 -07:00
}
2023-05-24 08:12:00 -07:00
2022-09-28 04:53:12 -07:00
uint32_t PrimeController::getKeyCycleCounter() const {
2023-05-24 08:12:00 -07:00
#if EFI_BACKUP_SRAM
2024-02-06 11:04:34 -08:00
return backupRamLoad(backup_ram_e::IgnCounter);
2023-05-24 08:12:00 -07:00
#else // not EFI_BACKUP_SRAM
2022-09-28 04:53:12 -07:00
return 0;
2023-05-24 08:12:00 -07:00
#endif // EFI_BACKUP_SRAM
2022-09-07 18:29:11 -07:00
}
void PrimeController::onPrimeStart() {
auto durationMs = getPrimeDuration();
// Don't prime a zero-duration pulse
if (durationMs <= 0) {
efiPrintf("Skipped zero-duration priming pulse.");
return;
}
efiPrintf("Firing priming pulse of %.2f ms", durationMs);
auto endTime = getTimeNowNt() + MS2NT(durationMs);
// Open all injectors, schedule closing later
m_isPriming = true;
startSimultaneousInjection();
getExecutorInterface()->scheduleByTimestampNt("prime", nullptr, endTime, { onPrimeEndAdapter, this });
2022-09-07 18:29:11 -07:00
}
2022-09-07 17:31:04 -07:00
2022-09-28 04:53:12 -07:00
void PrimeController::onPrimeEnd() {
endSimultaneousInjectionOnlyTogglePins();
2022-09-07 17:31:04 -07:00
2022-09-28 04:53:12 -07:00
m_isPriming = false;
}
void PrimeController::onSlowCallback() {
2022-09-07 17:31:04 -07:00
if (!getEngineRotationState()->isStopped()) {
2023-05-24 08:15:02 -07:00
#if EFI_BACKUP_SRAM
2024-02-06 11:04:34 -08:00
backupRamSave(backup_ram_e::IgnCounter, 0);
2023-05-24 08:15:02 -07:00
#endif /* EFI_BACKUP_SRAM */
2022-09-07 17:31:04 -07:00
}
}