/* * @file test_etb.cpp * * @date Dec 13, 2019 * @author Andrey Belomutskiy, (c) 2012-2020 */ #include "engine_test_helper.h" #include "electronic_throttle.h" #include "dc_motor.h" #include "engine_controller.h" #include "sensor.h" #include "idle_thread.h" #include "mocks.h" using ::testing::_; using ::testing::Ne; using ::testing::StrictMock; TEST(etb, initializationNoPedal) { StrictMock mocks[ETB_COUNT]; WITH_ENGINE_TEST_HELPER(TEST_ENGINE); for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &mocks[i]; } // We expect no throttle init stuff to be called - lack of pedal should disable ETB doInitElectronicThrottle(PASS_ENGINE_PARAMETER_SIGNATURE); } TEST(etb, initializationSingleThrottle) { StrictMock mocks[ETB_COUNT]; WITH_ENGINE_TEST_HELPER(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); Sensor::setMockValue(SensorType::AcceleratorPedalPrimary, 0); // Expect mock0 to be init with TPS 1, index 0, and PID params EXPECT_CALL(mocks[0], init(SensorType::Tps1, _, 0, &engineConfiguration->etb, Ne(nullptr))); EXPECT_CALL(mocks[0], reset); EXPECT_CALL(mocks[0], start); // We do not expect throttle #2 to be initialized doInitElectronicThrottle(PASS_ENGINE_PARAMETER_SIGNATURE); } TEST(etb, initializationDualThrottle) { StrictMock mocks[ETB_COUNT]; WITH_ENGINE_TEST_HELPER(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); Sensor::setMockValue(SensorType::AcceleratorPedalPrimary, 0); // The presence of a second TPS indicates dual throttle Sensor::setMockValue(SensorType::Tps2, 25.0f); // Expect mock0 to be init with TPS 1, index 0, and PID params EXPECT_CALL(mocks[0], init(SensorType::Tps1, _, 0, &engineConfiguration->etb, Ne(nullptr))); EXPECT_CALL(mocks[0], reset); EXPECT_CALL(mocks[0], start); // Expect mock1 to be init with TPS 2, index 1, and PID params EXPECT_CALL(mocks[1], init(SensorType::Tps2, _, 1, &engineConfiguration->etb, Ne(nullptr))); EXPECT_CALL(mocks[1], reset); EXPECT_CALL(mocks[1], start); doInitElectronicThrottle(PASS_ENGINE_PARAMETER_SIGNATURE); } TEST(etb, initializationDcMotorIdleValveMode) { StrictMock mocks[ETB_COUNT]; WITH_ENGINE_TEST_HELPER(TEST_ENGINE); // Enable VW idle mode engineConfiguration->dcMotorIdleValve = true; for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &mocks[i]; EXPECT_CALL(mocks[i], setIdlePosition(33.0f)); } // No accelerator pedal configured - this mode doesn't use it // Expect mock0 to be init with TPS 2, index 0, and PID params EXPECT_CALL(mocks[0], init(SensorType::Tps2, _, 0, &engineConfiguration->etb, Ne(nullptr))); EXPECT_CALL(mocks[0], reset); EXPECT_CALL(mocks[0], start); // We do not expect throttle #2 to be initialized doInitElectronicThrottle(PASS_ENGINE_PARAMETER_SIGNATURE); applyIACposition(33.0f PASS_ENGINE_PARAMETER_SUFFIX); } TEST(etb, idlePlumbing) { StrictMock mocks[ETB_COUNT]; WITH_ENGINE_TEST_HELPER(TEST_ENGINE); engineConfiguration->useETBforIdleControl = true; Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f); for (int i = 0; i < ETB_COUNT; i++) { engine->etbControllers[i] = &mocks[i]; EXPECT_CALL(mocks[i], setIdlePosition(33.0f)); } applyIACposition(33.0f PASS_ENGINE_PARAMETER_SUFFIX); } TEST(etb, testSetpointOnlyPedal) { WITH_ENGINE_TEST_HELPER(TEST_ENGINE); // Don't use ETB for idle, we aren't testing that yet - just pedal table for now engineConfiguration->useETBforIdleControl = false; EtbController etb; INJECT_ENGINE_REFERENCE(&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); etb.init(SensorType::Invalid, nullptr, 0, nullptr, &pedalMap); // Check endpoints and midpoint Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f); EXPECT_EQ(0, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f); EXPECT_EQ(50, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 100.0f); EXPECT_EQ(100, etb.getSetpoint().value_or(-1)); // Test some floating point pedal/output values Sensor::setMockValue(SensorType::AcceleratorPedal, 50.8302f); EXPECT_NEAR(50.8302, etb.getSetpoint().value_or(-1), EPS4D); Sensor::setMockValue(SensorType::AcceleratorPedal, 51.6605f); 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); EXPECT_EQ(0, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 105); EXPECT_EQ(100, etb.getSetpoint().value_or(-1)); // Check that ETB idle does NOT work - it's disabled etb.setIdlePosition(50); Sensor::setMockValue(SensorType::AcceleratorPedal, 0); EXPECT_EQ(0, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 20); EXPECT_EQ(20, etb.getSetpoint().value_or(-1)); // Test invalid pedal position - should give 0 position Sensor::resetMockValue(SensorType::AcceleratorPedal); EXPECT_EQ(0, etb.getSetpoint().value_or(-1)); } TEST(etb, setpointIdle) { WITH_ENGINE_TEST_HELPER(TEST_ENGINE); // Use ETB for idle, but don't give it any range (yet) engineConfiguration->useETBforIdleControl = true; engineConfiguration->etbIdleThrottleRange = 0; EtbController etb; INJECT_ENGINE_REFERENCE(&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(SensorType::Invalid, nullptr, 0, nullptr, &pedalMap); // No idle range, should just pass pedal Sensor::setMockValue(SensorType::AcceleratorPedal, 0.0f); EXPECT_EQ(0, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f); 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); EXPECT_FLOAT_EQ(5, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f); EXPECT_FLOAT_EQ(52.5, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 100.0f); 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); EXPECT_FLOAT_EQ(10, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f); EXPECT_FLOAT_EQ(55, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 100.0f); 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); EXPECT_FLOAT_EQ(10, etb.getSetpoint().value_or(-1)); Sensor::setMockValue(SensorType::AcceleratorPedal, 50.0f); EXPECT_FLOAT_EQ(55, etb.getSetpoint().value_or(-1)); } TEST(etb, idleVolkswagenMode) { WITH_ENGINE_TEST_HELPER(TEST_ENGINE); // In this mode the idle position should be passed thru as the setpoint directly engineConfiguration->dcMotorIdleValve = true; EtbController etb; INJECT_ENGINE_REFERENCE(&etb); 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, 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); Sensor::setMockValue(SensorType::Tps2, 75.0f); // Test first throttle { EtbController etb; etb.init(SensorType::Tps1, nullptr, 0, nullptr, nullptr); EXPECT_EQ(etb.observePlant().Value, 25.0f); } // Test second throttle { EtbController etb; etb.init(SensorType::Tps2, nullptr, 1, nullptr, nullptr); EXPECT_EQ(etb.observePlant().Value, 75.0f); } } TEST(etb, setOutputInvalid) { WITH_ENGINE_TEST_HELPER(TEST_ENGINE); StrictMock motor; EtbController etb; INJECT_ENGINE_REFERENCE(&etb); etb.init(SensorType::Invalid, &motor, 0, nullptr, nullptr); // Should be disabled in case of unexpected EXPECT_CALL(motor, disable()); etb.setOutput(unexpected); } TEST(etb, setOutputValid) { WITH_ENGINE_TEST_HELPER(TEST_ENGINE); StrictMock motor; EtbController etb; INJECT_ENGINE_REFERENCE(&etb); etb.init(SensorType::Invalid, &motor, 0, nullptr, nullptr); // 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) { WITH_ENGINE_TEST_HELPER(TEST_ENGINE); StrictMock motor; EtbController etb; INJECT_ENGINE_REFERENCE(&etb); etb.init(SensorType::Invalid, &motor, 0, nullptr, nullptr); // 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) { WITH_ENGINE_TEST_HELPER(TEST_ENGINE); StrictMock motor; EtbController etb; INJECT_ENGINE_REFERENCE(&etb); etb.init(SensorType::Invalid, &motor, 0, nullptr, nullptr); // 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) { WITH_ENGINE_TEST_HELPER(TEST_ENGINE); StrictMock motor; EtbController etb; INJECT_ENGINE_REFERENCE(&etb); etb.init(SensorType::Invalid, &motor, 0, nullptr, nullptr); // 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) { WITH_ENGINE_TEST_HELPER(TEST_ENGINE); StrictMock motor; EtbController etb; INJECT_ENGINE_REFERENCE(&etb); etb.init(SensorType::Invalid, &motor, 0, nullptr, nullptr); // 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, closedLoopPid) { pid_s pid = {}; pid.pFactor = 5; pid.maxValue = 75; pid.minValue = -60; EtbController etb; etb.init(SensorType::Invalid, nullptr, 0, &pid, nullptr); // Disable autotune for now Engine e; e.etbAutoTune = false; etb.engine = &e; // 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); }