2021-08-03 19:05:01 -07:00
|
|
|
#include "pch.h"
|
2020-05-05 05:01:40 -07:00
|
|
|
#include "fuel_math.h"
|
2020-07-25 02:00:24 -07:00
|
|
|
#include "alphan_airmass.h"
|
|
|
|
#include "maf_airmass.h"
|
2020-05-05 05:01:40 -07:00
|
|
|
|
2020-07-25 02:00:24 -07:00
|
|
|
using ::testing::StrictMock;
|
|
|
|
using ::testing::FloatNear;
|
2020-09-07 07:15:42 -07:00
|
|
|
using ::testing::InSequence;
|
|
|
|
using ::testing::_;
|
2020-07-25 02:00:24 -07:00
|
|
|
|
2020-05-05 05:01:40 -07:00
|
|
|
TEST(FuelMath, getStandardAirCharge) {
|
2021-11-16 13:52:11 -08:00
|
|
|
EngineTestHelper eth(TEST_ENGINE);
|
2020-05-05 05:01:40 -07:00
|
|
|
|
|
|
|
// Miata 1839cc 4cyl
|
2021-11-17 00:54:21 -08:00
|
|
|
engineConfiguration->specs.displacement = 1.839f;
|
|
|
|
engineConfiguration->specs.cylindersCount = 4;
|
2020-05-05 05:01:40 -07:00
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
EXPECT_FLOAT_EQ(0.5535934f, getStandardAirCharge());
|
2020-05-05 05:01:40 -07:00
|
|
|
|
|
|
|
// LS 5.3 liter v8
|
2021-11-17 00:54:21 -08:00
|
|
|
engineConfiguration->specs.displacement = 5.327f;
|
|
|
|
engineConfiguration->specs.cylindersCount = 8;
|
2020-05-05 05:01:40 -07:00
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
EXPECT_FLOAT_EQ(0.80179232f, getStandardAirCharge());
|
2020-05-05 05:01:40 -07:00
|
|
|
|
|
|
|
// Chainsaw - single cylinder 32cc
|
2021-11-17 00:54:21 -08:00
|
|
|
engineConfiguration->specs.displacement = 0.032f;
|
|
|
|
engineConfiguration->specs.cylindersCount = 1;
|
2021-11-16 01:15:29 -08:00
|
|
|
EXPECT_FLOAT_EQ(0.038531788f, getStandardAirCharge());
|
2020-05-05 05:01:40 -07:00
|
|
|
|
|
|
|
// Leopard 1 47.666 liter v12
|
2021-11-17 00:54:21 -08:00
|
|
|
engineConfiguration->specs.displacement = 47.666f;
|
|
|
|
engineConfiguration->specs.cylindersCount = 12;
|
2020-05-05 05:01:40 -07:00
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
EXPECT_FLOAT_EQ(4.782959f, getStandardAirCharge());
|
2020-05-05 05:01:40 -07:00
|
|
|
}
|
2020-07-25 02:00:24 -07:00
|
|
|
|
|
|
|
TEST(AirmassModes, AlphaNNormal) {
|
2021-11-16 13:52:11 -08:00
|
|
|
EngineTestHelper eth(TEST_ENGINE);
|
2020-07-25 02:00:24 -07:00
|
|
|
// 4 cylinder 4 liter = easy math
|
|
|
|
engineConfiguration->specs.displacement = 4.0f;
|
|
|
|
engineConfiguration->specs.cylindersCount = 4;
|
|
|
|
|
|
|
|
StrictMock<MockVp3d> veTable;
|
|
|
|
|
|
|
|
EXPECT_CALL(veTable, getValue(1200, FloatNear(0.71f, EPS4D)))
|
|
|
|
.WillOnce(Return(35.0f));
|
|
|
|
|
|
|
|
AlphaNAirmass dut(veTable);
|
|
|
|
|
2021-12-26 10:41:10 -08:00
|
|
|
// that's 0.71% not 71%
|
2020-07-25 02:00:24 -07:00
|
|
|
Sensor::setMockValue(SensorType::Tps1, 0.71f);
|
|
|
|
|
|
|
|
// Mass of 1 liter of air * VE
|
2021-12-26 10:41:10 -08:00
|
|
|
mass_t expectedAirmass = 1.2047f * 0.35f;
|
2020-07-25 02:00:24 -07:00
|
|
|
|
|
|
|
auto result = dut.getAirmass(1200);
|
|
|
|
EXPECT_NEAR(result.CylinderAirmass, expectedAirmass, EPS4D);
|
|
|
|
EXPECT_NEAR(result.EngineLoadPercent, 0.71f, EPS4D);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(AirmassModes, AlphaNFailedTps) {
|
2021-11-16 13:52:11 -08:00
|
|
|
EngineTestHelper eth(TEST_ENGINE);
|
2020-07-25 02:00:24 -07:00
|
|
|
|
|
|
|
// Shouldn't get called
|
|
|
|
StrictMock<MockVp3d> veTable;
|
|
|
|
|
|
|
|
AlphaNAirmass dut(veTable);
|
|
|
|
|
|
|
|
// explicitly reset the sensor
|
|
|
|
Sensor::resetMockValue(SensorType::Tps1);
|
|
|
|
// Ensure that it's actually failed
|
|
|
|
ASSERT_FALSE(Sensor::get(SensorType::Tps1).Valid);
|
|
|
|
|
|
|
|
auto result = dut.getAirmass(1200);
|
|
|
|
EXPECT_EQ(result.CylinderAirmass, 0);
|
|
|
|
}
|
|
|
|
|
2020-08-10 22:11:25 -07:00
|
|
|
TEST(AirmassModes, MafNormal) {
|
2021-11-16 13:52:11 -08:00
|
|
|
EngineTestHelper eth(FORD_ASPIRE_1996);
|
2020-07-25 02:00:24 -07:00
|
|
|
engineConfiguration->fuelAlgorithm = LM_REAL_MAF;
|
|
|
|
engineConfiguration->injector.flow = 200;
|
|
|
|
|
|
|
|
MockVp3d veTable;
|
|
|
|
// Ensure that the correct cell is read from the VE table
|
|
|
|
EXPECT_CALL(veTable, getValue(6000, FloatNear(70.9814f, EPS4D)))
|
|
|
|
.WillOnce(Return(75.0f));
|
|
|
|
|
|
|
|
MafAirmass dut(veTable);
|
|
|
|
|
|
|
|
auto airmass = dut.getAirmassImpl(200, 6000);
|
|
|
|
|
|
|
|
// Check results
|
|
|
|
EXPECT_NEAR(0.277777f * 0.75f, airmass.CylinderAirmass, EPS4D);
|
|
|
|
EXPECT_NEAR(70.9814f, airmass.EngineLoadPercent, EPS4D);
|
|
|
|
}
|
2020-09-07 07:15:42 -07:00
|
|
|
|
|
|
|
TEST(AirmassModes, VeOverride) {
|
|
|
|
StrictMock<MockVp3d> veTable;
|
|
|
|
|
|
|
|
{
|
|
|
|
InSequence is;
|
|
|
|
|
|
|
|
// Default
|
|
|
|
EXPECT_CALL(veTable, getValue(_, 10.0f)).WillOnce(Return(0));
|
|
|
|
// TPS
|
|
|
|
EXPECT_CALL(veTable, getValue(_, 30.0f)).WillOnce(Return(0));
|
|
|
|
}
|
|
|
|
|
2021-05-14 04:17:22 -07:00
|
|
|
struct DummyAirmassModel : public AirmassVeModelBase {
|
|
|
|
DummyAirmassModel(const ValueProvider3D& veTable) : AirmassVeModelBase(veTable) {}
|
2020-09-07 07:15:42 -07:00
|
|
|
|
2021-06-30 21:05:42 -07:00
|
|
|
AirmassResult getAirmass(int rpm) override {
|
2020-09-07 07:15:42 -07:00
|
|
|
// Default load value 10, will be overriden
|
|
|
|
getVe(rpm, 10.0f);
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-11-16 13:52:11 -08:00
|
|
|
EngineTestHelper eth(TEST_ENGINE);
|
2020-09-07 07:15:42 -07:00
|
|
|
DummyAirmassModel dut(veTable);
|
|
|
|
|
|
|
|
// Use default mode - will call with 10
|
|
|
|
dut.getAirmass(0);
|
2021-11-17 00:54:21 -08:00
|
|
|
EXPECT_FLOAT_EQ(engine->engineState.currentVeLoad, 10.0f);
|
2020-09-07 07:15:42 -07:00
|
|
|
|
|
|
|
// Override to TPS
|
2021-11-17 00:54:21 -08:00
|
|
|
engineConfiguration->veOverrideMode = VE_TPS;
|
2020-09-07 07:15:42 -07:00
|
|
|
Sensor::setMockValue(SensorType::Tps1, 30.0f);
|
|
|
|
dut.getAirmass(0);
|
2021-11-17 00:54:21 -08:00
|
|
|
EXPECT_FLOAT_EQ(engine->engineState.currentVeLoad, 30.0f);
|
2020-09-07 07:15:42 -07:00
|
|
|
}
|
2020-10-24 21:12:05 -07:00
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
void setInjectionMode(int value);
|
2020-10-24 21:12:05 -07:00
|
|
|
|
|
|
|
TEST(FuelMath, testDifferentInjectionModes) {
|
2021-11-16 13:52:11 -08:00
|
|
|
EngineTestHelper eth(TEST_ENGINE);
|
2020-10-24 21:12:05 -07:00
|
|
|
setupSimpleTestEngineWithMafAndTT_ONE_trigger(ð);
|
|
|
|
|
|
|
|
EXPECT_CALL(eth.mockAirmass, getAirmass(_))
|
|
|
|
.WillRepeatedly(Return(AirmassResult{1.3440001f, 50.0f}));
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
setInjectionMode((int)IM_BATCH);
|
|
|
|
engine->periodicFastCallback();
|
2020-10-24 21:12:05 -07:00
|
|
|
EXPECT_FLOAT_EQ( 20, engine->injectionDuration) << "injection while batch";
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
setInjectionMode((int)IM_SIMULTANEOUS);
|
|
|
|
engine->periodicFastCallback();
|
2020-10-24 21:12:05 -07:00
|
|
|
EXPECT_FLOAT_EQ( 10, engine->injectionDuration) << "injection while simultaneous";
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
setInjectionMode((int)IM_SEQUENTIAL);
|
|
|
|
engine->periodicFastCallback();
|
2020-10-24 21:12:05 -07:00
|
|
|
EXPECT_FLOAT_EQ( 40, engine->injectionDuration) << "injection while IM_SEQUENTIAL";
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
setInjectionMode((int)IM_SINGLE_POINT);
|
|
|
|
engine->periodicFastCallback();
|
2020-10-24 21:12:05 -07:00
|
|
|
EXPECT_FLOAT_EQ( 40, engine->injectionDuration) << "injection while IM_SINGLE_POINT";
|
2021-06-23 03:37:32 -07:00
|
|
|
EXPECT_EQ( 0, eth.recentWarnings()->getCount()) << "warningCounter#testDifferentInjectionModes";
|
2020-10-24 21:12:05 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(FuelMath, deadtime) {
|
2021-11-16 13:52:11 -08:00
|
|
|
EngineTestHelper eth(TEST_ENGINE);
|
2020-10-24 21:12:05 -07:00
|
|
|
|
|
|
|
setupSimpleTestEngineWithMafAndTT_ONE_trigger(ð);
|
|
|
|
|
|
|
|
EXPECT_CALL(eth.mockAirmass, getAirmass(_))
|
|
|
|
.WillRepeatedly(Return(AirmassResult{1.3440001f, 50.0f}));
|
|
|
|
|
|
|
|
// First test with no deadtime
|
2021-11-16 01:15:29 -08:00
|
|
|
engine->periodicFastCallback();
|
2020-10-24 21:12:05 -07:00
|
|
|
EXPECT_FLOAT_EQ( 20, engine->injectionDuration);
|
|
|
|
|
|
|
|
// Now add some deadtime
|
|
|
|
setArrayValues(engineConfiguration->injector.battLagCorr, 2.0f);
|
|
|
|
|
|
|
|
// Should have deadtime now!
|
2021-11-16 01:15:29 -08:00
|
|
|
engine->periodicFastCallback();
|
2020-10-24 21:12:05 -07:00
|
|
|
EXPECT_FLOAT_EQ( 20 + 2, engine->injectionDuration);
|
|
|
|
}
|
2021-12-31 23:19:59 -08:00
|
|
|
|
|
|
|
TEST(FuelMath, CylinderFuelTrim) {
|
|
|
|
EngineTestHelper eth(TEST_ENGINE);
|
|
|
|
|
|
|
|
EXPECT_CALL(eth.mockAirmass, getAirmass(_))
|
|
|
|
.WillRepeatedly(Return(AirmassResult{1, 50.0f}));
|
|
|
|
|
|
|
|
setTable(config->fuelTrims[0].table, -4);
|
|
|
|
setTable(config->fuelTrims[1].table, -2);
|
|
|
|
setTable(config->fuelTrims[2].table, 2);
|
|
|
|
setTable(config->fuelTrims[3].table, 4);
|
|
|
|
|
|
|
|
// run the fuel math
|
|
|
|
engine->periodicFastCallback();
|
|
|
|
|
|
|
|
// Check that each cylinder gets the expected amount of fuel
|
|
|
|
float unadjusted = 0.072142f;
|
|
|
|
EXPECT_NEAR(engine->injectionMass[0], unadjusted * 0.96, EPS4D);
|
|
|
|
EXPECT_NEAR(engine->injectionMass[1], unadjusted * 0.98, EPS4D);
|
|
|
|
EXPECT_NEAR(engine->injectionMass[2], unadjusted * 1.02, EPS4D);
|
|
|
|
EXPECT_NEAR(engine->injectionMass[3], unadjusted * 1.04, EPS4D);
|
|
|
|
}
|