/* * @file test_etb.cpp * * @date Dec 13, 2019 * @author Andrey Belomutskiy, (c) 2012-2020 */ #include "pch.h" #include "electronic_throttle_impl.h" #include "dc_motor.h" #include "idle_thread.h" #include "mocks.h" using ::testing::_; using ::testing::Ne; using ::testing::Return; using ::testing::StrictMock; TEST(etb, initializationNoPedal) { StrictMock mocks[ETB_COUNT]; EngineTestHelper eth(engine_type_e::TEST_ENGINE); for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &mocks[i]; } EXPECT_CALL(mocks[0], init(DC_Throttle1, _, _, _, false)).WillOnce(Return(false)); EXPECT_CALL(mocks[1], init(DC_Throttle2, _, _, _, false)).WillOnce(Return(false)); // This shouldn't throw, since no throttles are configured, but no pedal is configured either EXPECT_NO_FATAL_ERROR(doInitElectronicThrottle()); } TEST(etb, initializationMissingThrottle) { StrictMock mocks[ETB_COUNT]; EngineTestHelper eth(engine_type_e::TEST_ENGINE, [](engine_configuration_s* engineConfiguration) { engineConfiguration->etbFunctions[0] = DC_None; engineConfiguration->etbFunctions[1] = DC_None; }); for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &mocks[i]; } EXPECT_CALL(mocks[0], init(DC_None, _, _, _, true)).Times(0); EXPECT_CALL(mocks[1], init(DC_None, _, _, _, true)).Times(0); // Must have a sensor configured before init Sensor::setMockValue(SensorType::AcceleratorPedal, 0, true); Sensor::setMockValue(SensorType::AcceleratorPedalPrimary, 0); // This should throw: a pedal is configured but no throttles EXPECT_FATAL_ERROR(doInitElectronicThrottle()); } TEST(etb, initializationSingleThrottle) { StrictMock mocks[ETB_COUNT]; EXPECT_CALL(mocks[0], isEtbMode()) .WillOnce(Return(TRUE)); EngineTestHelper eth(engine_type_e::TEST_ENGINE, [](engine_configuration_s* engineConfiguration) { engineConfiguration->etbFunctions[0] = DC_Throttle1; engineConfiguration->etbFunctions[1] = DC_None; }); for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &mocks[i]; } // Must have a sensor configured before init Sensor::setMockValue(SensorType::AcceleratorPedal, 0, true); Sensor::setMockValue(SensorType::AcceleratorPedalPrimary, 0); // Expect mock0 to be init as throttle 1, and PID params EXPECT_CALL(mocks[0], init(DC_Throttle1, _, &engineConfiguration->etb, Ne(nullptr), true)).WillOnce(Return(true)); // Expect mock1 to be init as none EXPECT_CALL(mocks[1], init(DC_None, _, _, _, true)).Times(0); doInitElectronicThrottle(); } TEST(etb, initializationSingleThrottleInSecondSlot) { StrictMock mocks[ETB_COUNT]; EXPECT_CALL(mocks[1], isEtbMode()) .WillOnce(Return(TRUE)); EngineTestHelper eth(engine_type_e::TEST_ENGINE, [](engine_configuration_s* engineConfiguration) { engineConfiguration->etbFunctions[0] = DC_None; engineConfiguration->etbFunctions[1] = DC_Throttle1; }); for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &mocks[i]; } // Must have a sensor configured before init Sensor::setMockValue(SensorType::AcceleratorPedal, 0, true); Sensor::setMockValue(SensorType::AcceleratorPedalPrimary, 0, false); // Expect mock0 to be init as none EXPECT_CALL(mocks[0], init(DC_None, _, _, _, true)).Times(0); // Expect mock1 to be init as throttle 1, and PID params EXPECT_CALL(mocks[1], init(DC_Throttle1, _, &engineConfiguration->etb, Ne(nullptr), true)).WillOnce(Return(true)); doInitElectronicThrottle(); } TEST(etb, initializationDualThrottle) { StrictMock mocks[ETB_COUNT]; EXPECT_CALL(mocks[0], isEtbMode()) .WillOnce(Return(TRUE)); EXPECT_CALL(mocks[1], isEtbMode()) .WillOnce(Return(TRUE)); EngineTestHelper eth(engine_type_e::TEST_ENGINE); for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &mocks[i]; } // Must have a sensor configured before init Sensor::setMockValue(SensorType::AcceleratorPedal, 0, true); Sensor::setMockValue(SensorType::AcceleratorPedalPrimary, 0, false); Sensor::setMockValue(SensorType::Tps1, 25.0f, true); Sensor::setMockValue(SensorType::Tps2, 25.0f, true); engineConfiguration->etbFunctions[0] = DC_Throttle1; engineConfiguration->etbFunctions[1] = DC_Throttle2; // Expect mock0 to be init as throttle 1, and PID params EXPECT_CALL(mocks[0], init(DC_Throttle1, _, &engineConfiguration->etb, Ne(nullptr), true)).WillOnce(Return(true)); // Expect mock1 to be init as throttle 2, and PID params EXPECT_CALL(mocks[1], init(DC_Throttle2, _, &engineConfiguration->etb, Ne(nullptr), true)).WillOnce(Return(true)); doInitElectronicThrottle(); } TEST(etb, initializationWastegate) { StrictMock mocks[ETB_COUNT]; EXPECT_CALL(mocks[0], isEtbMode()) .WillOnce(Return(false)); EngineTestHelper eth(engine_type_e::TEST_ENGINE, [](engine_configuration_s* engineConfiguration) { engineConfiguration->etbFunctions[0] = DC_Wastegate; engineConfiguration->etbFunctions[1] = DC_None; }); for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &mocks[i]; } // Expect mock0 to be init as throttle 1, and PID wastegate params EXPECT_CALL(mocks[0], init(DC_Wastegate, _, &engineConfiguration->etbWastegatePid, Ne(nullptr), false)).WillOnce(Return(true)); // Expect mock1 to be init as none EXPECT_CALL(mocks[1], init(DC_None, _, _, _, false)).Times(0); doInitElectronicThrottle(); ASSERT_FALSE(engineConfiguration->etb1configured); } TEST(etb, initializationNoFunction) { StrictMock motor; EtbController dut; // When init called with DC_None, should ignore the provided params and return false EXPECT_FALSE(dut.init(DC_None, &motor, nullptr, nullptr, false)); // This should no-op, it shouldn't call motor.set(float duty) dut.setOutput(0.5f); } TEST(etb, initializationNotRedundantTps) { EtbController dut; // Needs pedal for init Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); // Not redundant, should fail upon init Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, false); EXPECT_FATAL_ERROR(dut.init(DC_Throttle1, nullptr, nullptr, nullptr, true)); } TEST(etb, initializationNotRedundantPedal) { EtbController dut; // Init pedal without redundancy Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, false); // Normal redundant TPS Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); EXPECT_FATAL_ERROR(dut.init(DC_Throttle1, nullptr, nullptr, nullptr, true)); } TEST(etb, initializationNoSensor) { Sensor::resetAllMocks(); EtbController dut; // Needs pedal for init Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EXPECT_FALSE(dut.init(DC_Throttle1, nullptr, nullptr, nullptr, true)); // Redundant Sensor::setMockValue(SensorType::Tps1, /*value*/0, /*mockRedundant*/true); // With primary TPS, should return true (ie, throttle was configured) EXPECT_TRUE(dut.init(DC_Throttle1, nullptr, nullptr, nullptr, true)); } TEST(etb, initializationNoThrottles) { // This tests the case where you don't want an ETB, and expect everything to go fine EtbController duts[2]; EngineTestHelper eth(engine_type_e::TEST_ENGINE); for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &duts[i]; } // Configure ETB functions, but no pedal engineConfiguration->etbFunctions[0] = DC_Throttle1; engineConfiguration->etbFunctions[1] = DC_Throttle2; // No pedal - don't init ETBs Sensor::resetMockValue(SensorType::AcceleratorPedal); // Not redundant TPS (aka cable throttle) Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, false); EXPECT_NO_FATAL_ERROR(initElectronicThrottle()); } TEST(etb, idlePlumbing) { StrictMock mocks[ETB_COUNT]; EngineTestHelper eth(engine_type_e::TEST_ENGINE); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f, true); for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &mocks[i]; EXPECT_CALL(mocks[i], setIdlePosition(33.0f)); } applyIACposition(33.0f); } TEST(etb, testSetpointOnlyPedal) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); EtbController etb; // Mock pedal map that's just passthru pedal -> target StrictMock pedalMap; EXPECT_CALL(pedalMap, getValue(_, _)) .WillRepeatedly([](float xRpm, float y) { return y; }); // Uninitialized ETB must return unexpected (and not deference a null pointer) EXPECT_EQ(etb.getSetpoint(), unexpected); // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); etb.init(DC_Throttle1, nullptr, nullptr, &pedalMap, true); // Check endpoints and midpoint Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EXPECT_EQ(1, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f, true); EXPECT_EQ(50, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 100.0f, true); EXPECT_EQ(100, etb.getSetpoint().value_or(-1)); // Test some floating point pedal/output values Sensor::setMockValue(SensorType::AcceleratorPedal, 50.8302f, true); EXPECT_NEAR(50.8302, etb.getSetpoint().value_or(-1), EPS4D); Sensor::setMockValue(SensorType::AcceleratorPedal, 51.6605f, true); EXPECT_NEAR(51.6605, etb.getSetpoint().value_or(-1), EPS4D); // Valid but out of range - should clamp to [0, 100] Sensor::setMockValue(SensorType::AcceleratorPedal, -5, true); EXPECT_EQ(1, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 105, true); EXPECT_EQ(100, etb.getSetpoint().value_or(-1)); // Test invalid pedal position - should give 0 position Sensor::resetMockValue(SensorType::AcceleratorPedal); EXPECT_EQ(1, etb.getSetpoint().value_or(-1)); // Check that adjustable clamping works correctly Sensor::setMockValue(SensorType::AcceleratorPedal, 0, true); EXPECT_EQ(1, etb.getSetpoint().value_or(-1)); engineConfiguration->etbMaximumPosition = 90; Sensor::setMockValue(SensorType::AcceleratorPedal, 85, true); EXPECT_EQ(85, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 90, true); EXPECT_EQ(90, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 95, true); EXPECT_EQ(90, etb.getSetpoint().value_or(-1)); } TEST(etb, setpointSecondThrottleTrim) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); // Mock pedal map that's just passthru pedal -> target StrictMock pedalMap; EXPECT_CALL(pedalMap, getValue(_, _)) .WillRepeatedly([](float xRpm, float y) { return y; }); // Should get called with the un-adjusted setpoint StrictMock throttleTrimTable; EXPECT_CALL(throttleTrimTable, getValue(0, 47)) .WillOnce(Return(4)); // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController2 etb(throttleTrimTable); etb.init(DC_Throttle1, nullptr, nullptr, &pedalMap, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 47, true); EXPECT_EQ(51, etb.getSetpoint().value_or(-1)); } TEST(etb, setpointIdle) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); // Use ETB for idle, but don't give it any range (yet) engineConfiguration->etbIdleThrottleRange = 0; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController etb; // Mock pedal map that's just passthru pedal -> target StrictMock pedalMap; EXPECT_CALL(pedalMap, getValue(_, _)) .WillRepeatedly([](float xRpm, float y) { return y; }); etb.init(DC_Throttle1, nullptr, nullptr, &pedalMap, true); // No idle range, should just pass pedal Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EXPECT_EQ(1, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f, true); EXPECT_EQ(50, etb.getSetpoint().value_or(-1)); // Idle should now have 10% range engineConfiguration->etbIdleThrottleRange = 10; // 50% idle position should increase setpoint by 5% when closed, and 0% when open etb.setIdlePosition(50); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EXPECT_FLOAT_EQ(5, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f, true); EXPECT_FLOAT_EQ(52.5, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 100.0f, true); EXPECT_FLOAT_EQ(100, etb.getSetpoint().value_or(-1)); // 100% setpoint should increase by 10% closed, scaled 0% at wot etb.setIdlePosition(100); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EXPECT_FLOAT_EQ(10, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f, true); EXPECT_FLOAT_EQ(55, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 100.0f, true); EXPECT_FLOAT_EQ(100, etb.getSetpoint().value_or(-1)); // 125% setpoint should clamp to 10% increase etb.setIdlePosition(125); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EXPECT_FLOAT_EQ(10, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f, true); EXPECT_FLOAT_EQ(55, etb.getSetpoint().value_or(-1)); } TEST(etb, setpointRevLimit) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); // Configure 5000 limit start, with 750 rpm taper engineConfiguration->etbRevLimitStart = 5000; engineConfiguration->etbRevLimitRange = 750; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController etb; // Mock pedal map to just return 80% StrictMock pedalMap; EXPECT_CALL(pedalMap, getValue(_, _)) .WillRepeatedly([](float, float) { return 80; }); etb.init(DC_Throttle1, nullptr, nullptr, &pedalMap, true); // Below threshold, should return unadjusted throttle Sensor::setMockValue(SensorType::Rpm, 1000); EXPECT_EQ(80, etb.getSetpoint().value_or(-1)); // At threshold, should return unadjusted throttle Sensor::setMockValue(SensorType::Rpm, 5000); EXPECT_EQ(80, etb.getSetpoint().value_or(-1)); // Middle of range, should return half of unadjusted Sensor::setMockValue(SensorType::Rpm, 5375); EXPECT_EQ(40, etb.getSetpoint().value_or(-1)); // At limit+range, should return 0 Sensor::setMockValue(SensorType::Rpm, 5750); EXPECT_EQ(1, etb.getSetpoint().value_or(-1)); // Above limit+range, should return 0 Sensor::setMockValue(SensorType::Rpm, 6000); EXPECT_EQ(1, etb.getSetpoint().value_or(-1)); } TEST(etb, setpointNoPedalMap) { EtbController etb; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); // Don't pass a pedal map etb.init(DC_Throttle1, nullptr, nullptr, nullptr, true); EXPECT_EQ(etb.getSetpoint(), unexpected); } TEST(etb, setpointIdleValveController) { EtbController etb; etb.init(DC_IdleValve, nullptr, nullptr, nullptr, false); etb.setIdlePosition(0); EXPECT_FLOAT_EQ(0, etb.getSetpoint().value_or(-1)); etb.setIdlePosition(50); EXPECT_FLOAT_EQ(50, etb.getSetpoint().value_or(-1)); etb.setIdlePosition(100); EXPECT_FLOAT_EQ(100, etb.getSetpoint().value_or(-1)); // Out of range should be clamped etb.setIdlePosition(-10); EXPECT_FLOAT_EQ(0, etb.getSetpoint().value_or(-1)); etb.setIdlePosition(110); EXPECT_FLOAT_EQ(100, etb.getSetpoint().value_or(-1)); } TEST(etb, setpointWastegateController) { EtbController etb; etb.init(DC_Wastegate, nullptr, nullptr, nullptr, false); etb.setWastegatePosition(0); EXPECT_FLOAT_EQ(0, etb.getSetpoint().value_or(-1)); etb.setWastegatePosition(50); EXPECT_FLOAT_EQ(50, etb.getSetpoint().value_or(-1)); etb.setWastegatePosition(100); EXPECT_FLOAT_EQ(100, etb.getSetpoint().value_or(-1)); // Out of range should be clamped etb.setWastegatePosition(-10); EXPECT_FLOAT_EQ(0, etb.getSetpoint().value_or(-1)); etb.setWastegatePosition(110); EXPECT_FLOAT_EQ(100, etb.getSetpoint().value_or(-1)); } TEST(etb, setpointLuaAdder) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController etb; // Mock pedal map to just return 50% StrictMock pedalMap; EXPECT_CALL(pedalMap, getValue(_, _)) .WillRepeatedly([](float, float) { return 50; }); etb.init(DC_Throttle1, nullptr, nullptr, &pedalMap, true); // No adjustment, should be unadjusted etb.setLuaAdjustment(0); EXPECT_EQ(50, etb.getSetpoint().value_or(-1)); // Normal adjustments should do as expected etb.setLuaAdjustment(10); EXPECT_EQ(60, etb.getSetpoint().value_or(-1)); etb.setLuaAdjustment(-10); EXPECT_EQ(40, etb.getSetpoint().value_or(-1)); // Crazy adjustments don't cause unreasonable target etb.setLuaAdjustment(1000); EXPECT_EQ(100, etb.getSetpoint().value_or(-1)); etb.setLuaAdjustment(-1000); EXPECT_EQ(1, etb.getSetpoint().value_or(-1)); extern int timeNowUs; int startTime = 1e6; timeNowUs = startTime; // Adjustment works immediately after setting etb.setLuaAdjustment(10); EXPECT_EQ(60, etb.getSetpoint().value_or(-1)); // Adjustment works 0.19 second after setting timeNowUs = startTime + 0.19 * 1e6; EXPECT_EQ(60, etb.getSetpoint().value_or(-1)); // Adjustment resets to 0 after 0.21 second timeNowUs = startTime + 0.21 * 1e6; EXPECT_EQ(50, etb.getSetpoint().value_or(-1)); } TEST(etb, etbTpsSensor) { // Throw some distinct values on the TPS sensors so we can identify that we're getting the correct one Sensor::setMockValue(SensorType::Tps1, 25.0f, true); Sensor::setMockValue(SensorType::Tps2, 75.0f, true); Sensor::setMockValue(SensorType::WastegatePosition, 33.0f); Sensor::setMockValue(SensorType::IdlePosition, 66.0f); // Redundant accelerator pedal required for init Sensor::setMockValue(SensorType::AcceleratorPedal, 0, true); // Test first throttle { EtbController etb; etb.init(DC_Throttle1, nullptr, nullptr, nullptr, true); EXPECT_EQ(etb.observePlant().Value, 25.0f); } // Test second throttle { EtbController etb; etb.init(DC_Throttle2, nullptr, nullptr, nullptr, true); EXPECT_EQ(etb.observePlant().Value, 75.0f); } // Test wastegate control { EtbController etb; etb.init(DC_Wastegate, nullptr, nullptr, nullptr, true); EXPECT_EQ(etb.observePlant().Value, 33.0f); } // Test idle valve control { EtbController etb; etb.init(DC_IdleValve, nullptr, nullptr, nullptr, true); EXPECT_EQ(etb.observePlant().Value, 66.0f); } } TEST(etb, setOutputInvalid) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); // Redundant TPS & accelerator pedal required for init Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0, true); StrictMock motor; EtbController etb; etb.init(DC_Throttle1, &motor, nullptr, nullptr, true); // Should be disabled in case of unexpected EXPECT_CALL(motor, disable(_)); etb.setOutput(unexpected); } TEST(etb, setOutputValid) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); StrictMock motor; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController etb; etb.init(DC_Throttle1, &motor, nullptr, nullptr, true); // Should be enabled and value set EXPECT_CALL(motor, enable()); EXPECT_CALL(motor, set(0.25f)) .WillOnce(Return(false)); etb.setOutput(25.0f); } TEST(etb, setOutputValid2) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); StrictMock motor; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController etb; etb.init(DC_Throttle1, &motor, nullptr, nullptr, true); // Should be enabled and value set EXPECT_CALL(motor, enable()); EXPECT_CALL(motor, set(-0.25f)) .WillOnce(Return(false)); etb.setOutput(-25.0f); } TEST(etb, setOutputOutOfRangeHigh) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); StrictMock motor; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController etb; etb.init(DC_Throttle1, &motor, nullptr, nullptr, true); // Should be enabled and value set EXPECT_CALL(motor, enable()); EXPECT_CALL(motor, set(0.90f)); // Off scale - should get clamped to 90% etb.setOutput(110); } TEST(etb, setOutputOutOfRangeLow) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); StrictMock motor; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController etb; etb.init(DC_Throttle1, &motor, nullptr, nullptr, true); // Should be enabled and value set EXPECT_CALL(motor, enable()); EXPECT_CALL(motor, set(-0.90f)); // Off scale - should get clamped to -90% etb.setOutput(-110); } TEST(etb, setOutputPauseControl) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); StrictMock motor; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController etb; etb.init(DC_Throttle1, &motor, nullptr, nullptr, true); // Pause control - should get no output engineConfiguration->pauseEtbControl = true; // Disable should be called, and set shouldn't be called EXPECT_CALL(motor, disable(_)); etb.setOutput(25.0f); } TEST(etb, setOutputLimpHome) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); StrictMock motor; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController etb; etb.init(DC_Throttle1, &motor, nullptr, nullptr, true); // Should be disabled when in ETB limp mode EXPECT_CALL(motor, disable(_)); // Trip a fatal error getLimpManager()->fatalError(); etb.setOutput(25.0f); } TEST(etb, closedLoopPid) { pid_s pid = {}; pid.pFactor = 5; pid.maxValue = 75; pid.minValue = -60; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); EtbController etb; etb.init(DC_Throttle1, nullptr, &pid, nullptr, true); // Disable autotune for now Engine e; EngineTestHelperBase base(&e, nullptr, nullptr); e.etbAutoTune = false; // Setpoint greater than actual, should be positive output EXPECT_FLOAT_EQ(etb.getClosedLoop(50, 40).value_or(-1), 50); // Likewise but negative EXPECT_FLOAT_EQ(etb.getClosedLoop(50, 52).value_or(-1), -10); // Test PID limiting EXPECT_FLOAT_EQ(etb.getClosedLoop(50, 70).value_or(-1), -60); EXPECT_FLOAT_EQ(etb.getClosedLoop(50, 30).value_or(-1), 75); } extern int timeNowUs; TEST(etb, jamDetection) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); pid_s pid = {}; // I-only since we're testing out the integrator pid.pFactor = 0; pid.iFactor = 1; pid.dFactor = 0; pid.maxValue = 50; pid.minValue = -50; // Must have TPS & PPS initialized for ETB setup Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0.0f, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f, true); // Limit of 5%, 1 second engineConfiguration->etbJamIntegratorLimit = 5; engineConfiguration->etbJamTimeout = 1; EtbController etb; etb.init(DC_Throttle1, nullptr, &pid, nullptr, true); timeNowUs = 0; // Reset timer while under integrator limit EXPECT_EQ(etb.getPidState().iTerm, 0); etb.checkOutput(0); EXPECT_EQ(etb.jamTimer, 0); EXPECT_FALSE(etb.jamDetected); for (size_t i = 0; i < ETB_LOOP_FREQUENCY; i++) { // Error of 10, should accumulate 10 integrator per second etb.getClosedLoop(50, 40); } EXPECT_NEAR(etb.getPidState().iTerm, 10.0f, 1e-3); // Just under time limit, no jam yet timeNowUs = 0.9e6; etb.checkOutput(0); EXPECT_NEAR(etb.jamTimer, 0.9f, 1e-3); EXPECT_FALSE(etb.jamDetected); // Above the time limit, jam detected! timeNowUs = 1.1e6; etb.checkOutput(0); EXPECT_TRUE(etb.jamDetected); } TEST(etb, openLoopThrottle) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); // Redundant TPS & accelerator pedal required for init Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0, true); EtbController etb; etb.init(DC_Throttle1, nullptr, nullptr, nullptr, true); // Map [0, 100] -> [-50, 50] setLinearCurve(config->etbBiasBins, 0, 100); setLinearCurve(config->etbBiasValues, -50, 50); EXPECT_NEAR(-50, etb.getOpenLoop(0).value_or(-1), EPS4D); EXPECT_NEAR(-25, etb.getOpenLoop(25).value_or(-1), EPS4D); EXPECT_NEAR(0, etb.getOpenLoop(50).value_or(-1), EPS4D); EXPECT_NEAR(25, etb.getOpenLoop(75).value_or(-1), EPS4D); EXPECT_NEAR(50, etb.getOpenLoop(100).value_or(-1), EPS4D); } TEST(etb, openLoopNonThrottle) { EngineTestHelper eth(engine_type_e::TEST_ENGINE); // Redundant TPS & accelerator pedal required for init Sensor::setMockValue(SensorType::Tps1Primary, 0); Sensor::setMockValue(SensorType::Tps1, 0, true); Sensor::setMockValue(SensorType::AcceleratorPedal, 0, true); EtbController etb; etb.init(DC_Wastegate, nullptr, nullptr, nullptr, false); // Map [0, 100] -> [-50, 50] setLinearCurve(config->etbBiasBins, 0, 100); setLinearCurve(config->etbBiasValues, -50, 50); // Should all return 0, as non-throttles don't use open loop table EXPECT_EQ(0, etb.getOpenLoop(0).value_or(-1)); EXPECT_EQ(0, etb.getOpenLoop(25).value_or(-1)); EXPECT_EQ(0, etb.getOpenLoop(50).value_or(-1)); EXPECT_EQ(0, etb.getOpenLoop(75).value_or(-1)); EXPECT_EQ(0, etb.getOpenLoop(100).value_or(-1)); }