/** * @file test_trigger_noiseless.cpp * * @date Apr 20, 2018 */ #include "pch.h" #include "trigger_decoder.h" #include "event_queue.h" #include "trigger_central.h" #include "main_trigger_callback.h" #include "advance_map.h" #include "speed_density.h" #include "fuel_math.h" #include "spark_logic.h" #include "trigger_universal.h" extern bool printTriggerDebug; static void fireEvent(EngineTestHelper *eth, bool isRise) { // mostly we fire only rise events (useOnlyRisingEdgeForTrigger=true). // but for noise filtering, both edges should be processed, so we fire falling events too if (isRise) eth->firePrimaryTriggerRise(); else if (eth->engine.engineConfiguration->useNoiselessTriggerDecoder) eth->firePrimaryTriggerFall(); } /* ________ __ _ _ __ * __|_|_|__| OR | | | |________ * spikes signal spikes signal */ static void noisyPulse(EngineTestHelper *eth, int idx, int durationUs, bool isRise, int noiseIdx, int durationNoiseUs, int offsetNoiseUs, int numSpikes) { // skip some time at the beginning eth->moveTimeForwardUs(offsetNoiseUs); durationUs -= offsetNoiseUs; // add noise spikes if (idx == noiseIdx) { // calculate the distance between noise spikes (evenly spaced) int noiseIntervalUs = (durationUs - durationNoiseUs * numSpikes) / numSpikes; for (int i = 0; i < numSpikes; i++) { // start spike fireEvent(eth, isRise); eth->moveTimeForwardUs(durationNoiseUs); durationUs -= durationNoiseUs; // end spike fireEvent(eth, !isRise); // add space between spikes eth->moveTimeForwardUs(noiseIntervalUs); durationUs -= noiseIntervalUs; } } // add the rest of pulse period eth->moveTimeForwardUs(durationUs); fireEvent(eth, isRise); } static void fireNoisyCycle60_2(EngineTestHelper *eth, int numCycles, int durationUs, int noiseIdx, int durationNoiseUs, int offsetNoiseUs, int numSpikes) { int idx = 0; for (int cycles = 0; cycles < numCycles; cycles++) { int count = 60 - 2 - 1; for (int i = 0; i < count; i++) { // rising noisyPulse(eth, idx++, durationUs, true, noiseIdx, durationNoiseUs, offsetNoiseUs, numSpikes); // falling noisyPulse(eth, idx++, durationUs, false, noiseIdx, durationNoiseUs, offsetNoiseUs, numSpikes); } // skip 2 teeth (durationUs * 4) // and add 1st rising teeth of the next cycle noisyPulse(eth, idx++, (durationUs * 4) + durationUs, true, noiseIdx, durationNoiseUs, offsetNoiseUs, numSpikes); // falling noisyPulse(eth, idx++, durationUs, false, noiseIdx, durationNoiseUs, offsetNoiseUs, numSpikes); } } static void resetTrigger(EngineTestHelper ð) { eth.applyTriggerWaveform(); eth.engine.triggerCentral.noiseFilter.resetAccumSignalData(); // reset error counter eth.engine.triggerCentral.triggerState.totalTriggerErrorCounter = 0; } static void testNoiselessDecoderProcedure(EngineTestHelper ð, int errorToleranceCnt DECLARE_ENGINE_PARAMETER_SUFFIX) { printf("*** (bc->useNoiselessTriggerDecoder = %s)\r\n", CONFIG(useNoiselessTriggerDecoder) ? "true" : "false"); resetTrigger(eth); // first, no noise fireNoisyCycle60_2(ð, 2, 1000, -1, 0, 0, 0); // should be no errors anyway ASSERT_EQ( 0, engine->triggerCentral.triggerState.totalTriggerErrorCounter) << "testNoiselessDecoderProcedure totalTriggerErrorCounter"; // check if we're imitating the 60-2 signal correctly ASSERT_EQ( 0, eth.engine.triggerCentral.triggerState.getCurrentIndex()) << "index #1"; // check rpm (60secs / (1000us * 60teeth)) = 1000rpm ASSERT_EQ( 1000, GET_RPM()) << "testNoiselessDecoder RPM"; // add noise1 - 1 spike in the middle of the 2nd rising pulse fireNoisyCycle60_2(ð, 2, 1000, 2, 10, 500, 1); assertEqualsM("testNoiselessDecoder noise#1", errorToleranceCnt, engine->triggerCentral.triggerState.totalTriggerErrorCounter); resetTrigger(eth); // add noise2 - 1 spike in the middle of the 2nd falling pulse fireNoisyCycle60_2(ð, 2, 1000, 3, 10, 500, 1); //assertEqualsM("noise#2", errorToleranceCnt, engine->triggerCentral.triggerState.totalTriggerErrorCounter); resetTrigger(eth); // add noise3 - in the middle of the sync.gap, // so that we cannot tell for sure if it's a start of another 'extra' tooth or just a noise inside the gap, // that's why we used expectedEventCount[] in our filtering algo to make a prediction about gap fireNoisyCycle60_2(ð, 2, 1000, 114, 10, 1500, 1); // so everything runs smoothly! assertEqualsM("noise#3", errorToleranceCnt, engine->triggerCentral.triggerState.totalTriggerErrorCounter); resetTrigger(eth); // add noise4 - too close to the start of the real next signal, so the noise spike is accepted as a signal // but when the real signal comes shortly afterwards, it we be treated as a noise spike, fireNoisyCycle60_2(ð, 2, 1000, 4, 10, 980, 1); // and we won't get out of sync! assertEqualsM("noise#4", errorToleranceCnt, engine->triggerCentral.triggerState.totalTriggerErrorCounter); resetTrigger(eth); // add noise5 - one very long 333us noise spike fireNoisyCycle60_2(ð, 2, 1000, 4, 333, 10, 1); // still ok assertEqualsM("noise#5", errorToleranceCnt, engine->triggerCentral.triggerState.totalTriggerErrorCounter); resetTrigger(eth); // add noise6 - 10 short spikes across the entire signal pulse const int failProofNumSpikes = 10; fireNoisyCycle60_2(ð, 2, 1000, 4, 5, 10, failProofNumSpikes); // we barely survived this time assertEqualsM("testNoiselessDecoder noise#6", errorToleranceCnt, engine->triggerCentral.triggerState.totalTriggerErrorCounter); resetTrigger(eth); // add noise7 - 34 short spikes across the entire signal pulse fireNoisyCycle60_2(ð, 2, 1000, 2, 10, 10, failProofNumSpikes + 1); // alas, this is a hard case even for noiseless decoder, and it fails... // but still we're close to 33% signal-noise ratio threshold - not bad! // so here's an error anyway! ASSERT_EQ( 1, engine->triggerCentral.triggerState.totalTriggerErrorCounter) << "testNoiselessDecoder noise#7_fail_test"; } TEST(big, testNoiselessDecoder) { printf("====================================================================================== testNoiselessDecoder\r\n"); WITH_ENGINE_TEST_HELPER(TEST_ENGINE); engineConfiguration->ignitionMode = IM_WASTED_SPARK; engineConfiguration->useOnlyRisingEdgeForTrigger = true; // we'll test on 60-2 wheel eth.setTriggerType(TT_TOOTHED_WHEEL_60_2 PASS_ENGINE_PARAMETER_SUFFIX); ASSERT_EQ(0, engine->triggerCentral.triggerState.totalTriggerErrorCounter); ASSERT_EQ( 0, GET_RPM()) << "testNoiselessDecoder RPM"; //printTriggerDebug = true; #if 0 // try normal trigger mode, no noise filtering CONFIG(useNoiselessTriggerDecoder) = false; // for test validation, it should be 1 trigger error testNoiselessDecoderProcedure(eth, 1 PASS_ENGINE_PARAMETER_SUFFIX); #endif // now enable our noise filtering algo CONFIG(useNoiselessTriggerDecoder) = true; // should be 0 errors! testNoiselessDecoderProcedure(eth, 0 PASS_ENGINE_PARAMETER_SUFFIX); //printTriggerDebug = false; }