/* * @file test_nissan_vq_vvt.cpp * * Created on: Jul 2, 2021 * @author Andrey Belomutskiy, (c) 2012-2021 */ #include "pch.h" #include "trigger_nissan.h" #include "nissan_vq.h" class TriggerCallback { public: Engine *engine; int toothIndex; TriggerWaveform *form; bool isVvt; int vvtBankIndex; scheduling_s sched; }; static void func(TriggerCallback *callback) { int formIndex = callback->toothIndex % callback->form->getSize(); Engine *engine = callback->engine; EXPAND_Engine; int value = callback->form->wave.getChannelState(0, formIndex); efitick_t nowNt = getTimeNowNt(); if (callback->isVvt) { trigger_value_e v = value ? TV_RISE : TV_FALL; hwHandleVvtCamSignal(v, nowNt, callback->vvtBankIndex * CAMS_PER_BANK PASS_ENGINE_PARAMETER_SUFFIX); } else { handleShaftSignal(0, value, nowNt PASS_ENGINE_PARAMETER_SUFFIX); } } static void scheduleTriggerEvents(TriggerWaveform *shape, float timeScale, int count, bool isVvt, int vvtBankIndex, int vvtOffset, std::vector>& ptrs DECLARE_ENGINE_PARAMETER_SUFFIX) { int totalIndex = 0; /** * yet another approach to trigger testing: let's schedule a huge list of events from heap * and then execute those one */ for (int r = 0; r < count; r++) { for (size_t i = 0; i < shape->getSize(); i++) { float angle = vvtOffset + shape->getAngle(totalIndex); std::shared_ptr param = std::make_shared(); ptrs.push_back(param); param->engine = engine; param->toothIndex = totalIndex; param->form = shape; param->isVvt = isVvt; param->vvtBankIndex = vvtBankIndex; engine->executor.scheduleByTimestamp("test", ¶m->sched, timeScale * 1000 * angle, { func, param.get() }); totalIndex++; } } } TEST(nissan, vq_vvt) { // hold a reference to the heap allocated scheduling events until the test is done std::vector> ptrs; WITH_ENGINE_TEST_HELPER (HELLEN_121_NISSAN_6_CYL); engineConfiguration->isIgnitionEnabled = false; engineConfiguration->isInjectionEnabled = false; int cyclesCount = 48; { static TriggerWaveform crank; initializeNissanVQ35crank(&crank); scheduleTriggerEvents(&crank, /* timeScale */ 1, cyclesCount, false, -1, 0, ptrs PASS_ENGINE_PARAMETER_SUFFIX); } float vvtTimeScale = 1; angle_t testVvtOffset = 13; { static TriggerWaveform vvt; initializeNissanVQvvt(&vvt); scheduleTriggerEvents(&vvt, /* timeScale */ vvtTimeScale, cyclesCount / 6, true, /* vvtBankIndex */ 0, /* vvtOffset */ testVvtOffset, ptrs PASS_ENGINE_PARAMETER_SUFFIX); } { static TriggerWaveform vvt; initializeNissanVQvvt(&vvt); scheduleTriggerEvents(&vvt, /* timeScale */ vvtTimeScale, cyclesCount / 6, true, /* vvtBankIndex */1, /* vvtOffset */ testVvtOffset + NISSAN_VQ_CAM_OFFSET, ptrs PASS_ENGINE_PARAMETER_SUFFIX); } eth.executeUntil(1473000); ASSERT_EQ(0, GET_RPM()); eth.executeUntil(1475000); ASSERT_EQ(167, GET_RPM()); TriggerCentral *tc = &engine->triggerCentral; eth.executeUntil(3593000); ASSERT_TRUE(tc->vvtState[0][0].getShaftSynchronized()); scheduling_s *head; while ((head = engine->executor.getHead()) != nullptr) { eth.setTimeAndInvokeEventsUs(head->momentX); ASSERT_TRUE(tc->vvtState[0][0].getShaftSynchronized()); // let's celebrate that vvtPosition stays the same ASSERT_NEAR(-testVvtOffset, tc->vvtPosition[0][0], EPS2D); } ASSERT_TRUE(tc->vvtState[1][0].getShaftSynchronized()); ASSERT_NEAR(-testVvtOffset, tc->vvtPosition[0][0], EPS2D); ASSERT_NEAR(-testVvtOffset, tc->vvtPosition[1][0], EPS2D); EXPECT_EQ(0, eth.recentWarnings()->getCount()); }