2022-09-01 10:55:03 -07:00
|
|
|
|
2022-09-01 17:40:38 -07:00
|
|
|
// here am flirting with not using pch.h and not including at least Engine
|
|
|
|
#include <rusefi/interpolation.h>
|
|
|
|
#include <rusefi/arrays.h>
|
|
|
|
#include "engine_configuration.h"
|
|
|
|
#include "sensor.h"
|
|
|
|
#include "error_handling.h"
|
2021-07-25 22:05:17 -07:00
|
|
|
|
2020-08-17 02:22:25 -07:00
|
|
|
#include "injector_model.h"
|
2021-11-05 19:29:56 -07:00
|
|
|
#include "fuel_computer.h"
|
2020-08-17 02:22:25 -07:00
|
|
|
|
|
|
|
void InjectorModelBase::prepare() {
|
2022-10-20 19:25:39 -07:00
|
|
|
float flowRatio = getInjectorFlowRatio();
|
|
|
|
|
|
|
|
// "large pulse" flow rate
|
|
|
|
m_massFlowRate = flowRatio * getBaseFlowRate();
|
2022-09-01 07:58:16 -07:00
|
|
|
m_deadtime = getDeadtime();
|
2022-10-20 19:25:39 -07:00
|
|
|
|
|
|
|
if (getNonlinearMode() == INJ_FordModel) {
|
|
|
|
m_smallPulseFlowRate = flowRatio * getSmallPulseFlowRate();
|
|
|
|
m_smallPulseBreakPoint = getSmallPulseBreakPoint();
|
|
|
|
|
|
|
|
// amount added to small pulses to correct for the "kink" from low flow region
|
|
|
|
m_smallPulseOffset = 1000 * ((m_smallPulseBreakPoint / m_massFlowRate) - (m_smallPulseBreakPoint / m_smallPulseFlowRate));
|
|
|
|
}
|
2020-08-17 02:22:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
constexpr float convertToGramsPerSecond(float ccPerMinute) {
|
2021-11-05 19:29:56 -07:00
|
|
|
return ccPerMinute * (fuelDensity / 60.f);
|
2020-08-17 02:22:25 -07:00
|
|
|
}
|
|
|
|
|
2024-01-16 23:11:01 -08:00
|
|
|
float InjectorModelWithConfig::getBaseFlowRate() const {
|
2022-10-20 19:25:39 -07:00
|
|
|
if (engineConfiguration->injectorFlowAsMassFlow) {
|
2024-01-16 23:11:01 -08:00
|
|
|
return m_cfg->flow;
|
2022-10-20 19:25:39 -07:00
|
|
|
} else {
|
2024-01-16 23:11:01 -08:00
|
|
|
return convertToGramsPerSecond(m_cfg->flow);
|
2022-10-20 19:25:39 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-16 23:11:01 -08:00
|
|
|
float InjectorModelPrimary::getSmallPulseFlowRate() const {
|
2022-10-20 19:25:39 -07:00
|
|
|
return engineConfiguration->fordInjectorSmallPulseSlope;
|
|
|
|
}
|
|
|
|
|
2024-01-16 23:11:01 -08:00
|
|
|
float InjectorModelPrimary::getSmallPulseBreakPoint() const {
|
2022-10-20 19:25:39 -07:00
|
|
|
// convert milligrams -> grams
|
|
|
|
return 0.001f * engineConfiguration->fordInjectorSmallPulseBreakPoint;
|
|
|
|
}
|
|
|
|
|
2024-01-16 23:11:01 -08:00
|
|
|
InjectorNonlinearMode InjectorModelPrimary::getNonlinearMode() const {
|
2022-10-20 19:25:39 -07:00
|
|
|
return engineConfiguration->injectorNonlinearMode;
|
|
|
|
}
|
|
|
|
|
2024-01-16 23:11:01 -08:00
|
|
|
float InjectorModelSecondary::getSmallPulseFlowRate() const {
|
|
|
|
// not supported on second bank
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
float InjectorModelSecondary::getSmallPulseBreakPoint() const {
|
|
|
|
// not supported on second bank
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
InjectorNonlinearMode InjectorModelSecondary::getNonlinearMode() const {
|
|
|
|
// nonlinear not supported on second bank
|
|
|
|
return InjectorNonlinearMode::INJ_None;
|
|
|
|
}
|
|
|
|
|
|
|
|
expected<float> InjectorModelWithConfig::getFuelDifferentialPressure() const {
|
2023-03-25 13:53:16 -07:00
|
|
|
auto map = Sensor::get(SensorType::Map);
|
2023-12-22 21:41:31 -08:00
|
|
|
auto baro = Sensor::get(SensorType::BarometricPressure);
|
|
|
|
|
|
|
|
float baroKpa = baro.Value;
|
|
|
|
if (!baro || baro.Value > 120 || baro.Value < 50) {
|
|
|
|
baroKpa = 101.325f;
|
|
|
|
}
|
2023-03-25 13:53:16 -07:00
|
|
|
|
2021-11-17 00:54:21 -08:00
|
|
|
switch (engineConfiguration->injectorCompensationMode) {
|
2020-11-10 20:11:22 -08:00
|
|
|
case ICM_FixedRailPressure:
|
2021-05-30 16:23:29 -07:00
|
|
|
// Add barometric pressure, as "fixed" really means "fixed pressure above atmosphere"
|
2023-03-25 13:53:16 -07:00
|
|
|
return
|
|
|
|
engineConfiguration->fuelReferencePressure
|
2023-12-22 21:41:31 -08:00
|
|
|
+ baroKpa
|
2023-03-25 13:53:16 -07:00
|
|
|
- map.value_or(101.325);
|
|
|
|
case ICM_SensedRailPressure: {
|
2021-03-12 20:32:41 -08:00
|
|
|
if (!Sensor::hasSensor(SensorType::FuelPressureInjector)) {
|
2023-06-23 01:27:24 -07:00
|
|
|
warning(ObdCode::OBD_Fuel_Pressure_Sensor_Missing, "Fuel pressure compensation is set to use a pressure sensor, but none is configured.");
|
2021-03-12 20:32:41 -08:00
|
|
|
return unexpected;
|
|
|
|
}
|
|
|
|
|
2023-03-25 13:53:16 -07:00
|
|
|
auto fps = Sensor::get(SensorType::FuelPressureInjector);
|
|
|
|
|
2020-11-10 20:11:22 -08:00
|
|
|
// TODO: what happens when the sensor fails?
|
2023-03-25 13:53:16 -07:00
|
|
|
if (!fps) {
|
|
|
|
return unexpected;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (engineConfiguration->fuelPressureSensorMode) {
|
|
|
|
case FPM_Differential:
|
|
|
|
// This sensor directly measures delta-P, no math needed!
|
|
|
|
return fps.Value;
|
|
|
|
case FPM_Gauge:
|
|
|
|
if (!map) {
|
|
|
|
return unexpected;
|
|
|
|
}
|
|
|
|
|
2023-12-22 21:41:31 -08:00
|
|
|
return fps.Value + baroKpa - map.Value;
|
2023-03-25 13:53:16 -07:00
|
|
|
case FPM_Absolute:
|
|
|
|
default:
|
|
|
|
if (!map) {
|
|
|
|
return unexpected;
|
|
|
|
}
|
|
|
|
|
|
|
|
return fps.Value - map.Value;
|
|
|
|
}
|
|
|
|
} default: return unexpected;
|
2020-11-10 20:11:22 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-16 23:11:01 -08:00
|
|
|
float InjectorModelWithConfig::getInjectorFlowRatio() {
|
2020-11-10 20:11:22 -08:00
|
|
|
// Compensation disabled, use reference flow.
|
2021-11-17 00:54:21 -08:00
|
|
|
if (engineConfiguration->injectorCompensationMode == ICM_None) {
|
2020-11-10 20:11:22 -08:00
|
|
|
return 1.0f;
|
|
|
|
}
|
|
|
|
|
2021-11-17 00:54:21 -08:00
|
|
|
float referencePressure = engineConfiguration->fuelReferencePressure;
|
2022-08-31 17:32:37 -07:00
|
|
|
|
|
|
|
if (referencePressure < 50) {
|
|
|
|
// impossibly low fuel ref pressure
|
2023-08-20 19:23:44 -07:00
|
|
|
criticalError("Impossible fuel reference pressure: %f", referencePressure);
|
2022-08-31 17:32:37 -07:00
|
|
|
|
|
|
|
return 1.0f;
|
|
|
|
}
|
|
|
|
|
2023-03-25 13:53:16 -07:00
|
|
|
expected<float> diffPressure = getFuelDifferentialPressure();
|
2020-11-10 20:11:22 -08:00
|
|
|
|
|
|
|
// If sensor failed, best we can do is disable correction
|
2023-03-25 13:53:16 -07:00
|
|
|
if (!diffPressure) {
|
2021-01-02 16:13:10 -08:00
|
|
|
return 1.0f;
|
|
|
|
}
|
2020-11-10 20:11:22 -08:00
|
|
|
|
2023-03-25 13:53:16 -07:00
|
|
|
pressureDelta = diffPressure.Value;
|
2021-03-12 20:32:41 -08:00
|
|
|
|
|
|
|
// Somehow pressure delta is less than 0, assume failed sensor and return default flow
|
|
|
|
if (pressureDelta <= 0) {
|
|
|
|
return 1.0f;
|
|
|
|
}
|
|
|
|
|
2021-12-26 09:59:53 -08:00
|
|
|
pressureRatio = pressureDelta / referencePressure;
|
|
|
|
// todo: live data model?
|
2020-11-10 20:11:22 -08:00
|
|
|
float flowRatio = sqrtf(pressureRatio);
|
|
|
|
|
|
|
|
// TODO: should the flow ratio be clamped?
|
|
|
|
return flowRatio;
|
|
|
|
}
|
|
|
|
|
2024-01-16 23:11:01 -08:00
|
|
|
float InjectorModelWithConfig::getDeadtime() const {
|
2020-08-17 02:22:25 -07:00
|
|
|
return interpolate2d(
|
2021-03-11 20:07:18 -08:00
|
|
|
Sensor::get(SensorType::BatteryVoltage).value_or(VBAT_FALLBACK_VALUE),
|
2024-01-16 23:11:01 -08:00
|
|
|
m_cfg->battLagCorrBins,
|
|
|
|
m_cfg->battLagCorr
|
2020-08-17 02:22:25 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
float InjectorModelBase::getInjectionDuration(float fuelMassGram) const {
|
2022-10-20 19:25:39 -07:00
|
|
|
if (fuelMassGram <= 0) {
|
|
|
|
// If 0 mass, don't do any math, just skip the injection.
|
2020-10-26 04:23:13 -07:00
|
|
|
return 0.0f;
|
|
|
|
}
|
2021-07-07 20:46:44 -07:00
|
|
|
|
2022-10-20 19:25:39 -07:00
|
|
|
// Get the no-offset duration
|
|
|
|
float baseDuration = getBaseDurationImpl(fuelMassGram);
|
2021-07-07 20:46:44 -07:00
|
|
|
|
2022-10-20 19:25:39 -07:00
|
|
|
// Add deadtime offset
|
2021-07-07 20:46:44 -07:00
|
|
|
return baseDuration + m_deadtime;
|
2020-08-17 02:22:25 -07:00
|
|
|
}
|
2021-03-03 04:30:56 -08:00
|
|
|
|
|
|
|
float InjectorModelBase::getFuelMassForDuration(floatms_t duration) const {
|
|
|
|
// Convert from ms -> grams
|
|
|
|
return duration * m_massFlowRate * 0.001f;
|
|
|
|
}
|
2021-07-07 20:46:44 -07:00
|
|
|
|
2022-10-20 19:25:39 -07:00
|
|
|
float InjectorModelBase::getBaseDurationImpl(float fuelMassGram) const {
|
|
|
|
floatms_t baseDuration = fuelMassGram / m_massFlowRate * 1000;
|
|
|
|
|
|
|
|
switch (getNonlinearMode()) {
|
|
|
|
case INJ_FordModel:
|
|
|
|
if (fuelMassGram < m_smallPulseBreakPoint) {
|
|
|
|
// Small pulse uses a different slope, and adds the "zero fuel pulse" offset
|
|
|
|
return (fuelMassGram / m_smallPulseFlowRate * 1000) + m_smallPulseOffset;
|
|
|
|
} else {
|
|
|
|
// Large pulse
|
|
|
|
return baseDuration;
|
|
|
|
}
|
2021-07-07 20:46:44 -07:00
|
|
|
case INJ_PolynomialAdder:
|
|
|
|
return correctInjectionPolynomial(baseDuration);
|
|
|
|
case INJ_None:
|
|
|
|
default:
|
|
|
|
return baseDuration;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-20 19:25:39 -07:00
|
|
|
float InjectorModelBase::correctInjectionPolynomial(float baseDuration) const {
|
2022-04-28 05:16:02 -07:00
|
|
|
if (baseDuration > engineConfiguration->applyNonlinearBelowPulse) {
|
2021-07-07 20:46:44 -07:00
|
|
|
// Large pulse, skip correction.
|
|
|
|
return baseDuration;
|
|
|
|
}
|
|
|
|
|
2021-11-17 00:54:21 -08:00
|
|
|
auto& is = engineConfiguration->injectorCorrectionPolynomial;
|
2021-07-07 20:46:44 -07:00
|
|
|
float xi = 1;
|
|
|
|
|
|
|
|
float adder = 0;
|
|
|
|
|
|
|
|
// Add polynomial terms, starting with x^0
|
|
|
|
for (size_t i = 0; i < efi::size(is); i++) {
|
|
|
|
adder += is[i] * xi;
|
|
|
|
xi *= baseDuration;
|
|
|
|
}
|
|
|
|
|
|
|
|
return baseDuration + adder;
|
|
|
|
}
|
2024-01-16 23:11:01 -08:00
|
|
|
|
|
|
|
InjectorModelWithConfig::InjectorModelWithConfig(const injector_s* const cfg)
|
|
|
|
: m_cfg(cfg)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
InjectorModelPrimary::InjectorModelPrimary()
|
|
|
|
: InjectorModelWithConfig(&engineConfiguration->injector)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: actual separate config for second bank!
|
|
|
|
InjectorModelSecondary::InjectorModelSecondary()
|
2024-01-17 02:20:16 -08:00
|
|
|
: InjectorModelWithConfig(&engineConfiguration->injectorSecondary)
|
2024-01-16 23:11:01 -08:00
|
|
|
{
|
|
|
|
}
|