2022-09-03 08:05:18 -07:00
|
|
|
|
|
|
|
#include "engine_configuration.h"
|
|
|
|
#include "sensor.h"
|
|
|
|
#include "error_handling.h"
|
|
|
|
|
2020-07-23 01:23:57 -07:00
|
|
|
#include "maf_airmass.h"
|
|
|
|
#include "maf.h"
|
2022-09-03 08:05:18 -07:00
|
|
|
#include "fuel_math.h"
|
2020-07-23 01:23:57 -07:00
|
|
|
|
2022-10-24 09:41:05 -07:00
|
|
|
float MafAirmass::getMaf() const {
|
|
|
|
auto maf = Sensor::get(SensorType::Maf);
|
|
|
|
|
|
|
|
if (Sensor::hasSensor(SensorType::Maf2)) {
|
|
|
|
auto maf2 = Sensor::get(SensorType::Maf2);
|
|
|
|
|
|
|
|
if (maf && maf2) {
|
|
|
|
// Both MAFs work, return the sum
|
|
|
|
return maf.Value + maf2.Value;
|
|
|
|
} else if (maf) {
|
|
|
|
// MAF 1 works, but not MAF 2, so double the value from #1
|
|
|
|
return 2 * maf.Value;
|
|
|
|
} else if (maf2) {
|
|
|
|
// MAF 2 works, but not MAF 1, so double the value from #2
|
|
|
|
return 2 * maf2.Value;
|
|
|
|
} else {
|
|
|
|
// Both MAFs are broken, give up.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return maf.value_or(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-30 21:05:42 -07:00
|
|
|
AirmassResult MafAirmass::getAirmass(int rpm) {
|
2022-10-24 09:41:05 -07:00
|
|
|
float maf = getMaf();
|
|
|
|
|
2020-07-24 11:44:54 -07:00
|
|
|
return getAirmassImpl(maf, rpm);
|
2020-07-23 01:23:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function block now works to create a standardised load from the cylinder filling as well as tune fuel via VE table.
|
|
|
|
* @return total duration of fuel injection per engine cycle, in milliseconds
|
|
|
|
*/
|
|
|
|
AirmassResult MafAirmass::getAirmassImpl(float massAirFlow, int rpm) const {
|
|
|
|
// If the engine is stopped, MAF is meaningless
|
|
|
|
if (rpm == 0) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
// kg/hr -> g/s
|
|
|
|
float gramPerSecond = massAirFlow * 1000 / 3600;
|
|
|
|
|
|
|
|
// 1/min -> 1/s
|
|
|
|
float revsPerSecond = rpm / 60.0f;
|
2021-12-26 10:41:10 -08:00
|
|
|
mass_t airPerRevolution = gramPerSecond / revsPerSecond;
|
2020-07-23 01:23:57 -07:00
|
|
|
|
2021-12-26 09:33:32 -08:00
|
|
|
// Now we have to divide among cylinders - on a 4 stroke, half of the cylinders happen every revolution
|
|
|
|
// This math is floating point to work properly on engines with odd cylinder count
|
2023-03-27 00:58:18 -07:00
|
|
|
float halfCylCount = engineConfiguration->cylindersCount / 2.0f;
|
2020-07-23 01:23:57 -07:00
|
|
|
|
2021-12-26 10:41:10 -08:00
|
|
|
mass_t cylinderAirmass = airPerRevolution / halfCylCount;
|
2020-07-23 01:23:57 -07:00
|
|
|
|
2021-12-26 09:33:32 -08:00
|
|
|
//Create % load for fuel table using relative naturally aspirated cylinder filling
|
2022-09-03 08:05:18 -07:00
|
|
|
float airChargeLoad = 100 * cylinderAirmass / getStandardAirCharge();
|
2020-07-23 01:23:57 -07:00
|
|
|
|
|
|
|
//Correct air mass by VE table
|
2021-12-26 10:41:10 -08:00
|
|
|
mass_t correctedAirmass = cylinderAirmass * getVe(rpm, airChargeLoad);
|
2020-07-23 01:23:57 -07:00
|
|
|
|
|
|
|
return {
|
|
|
|
correctedAirmass,
|
|
|
|
airChargeLoad, // AFR/VE/ignition table Y axis
|
|
|
|
};
|
|
|
|
}
|