/* * @file tachometer.cpp * @brief This is about driving external analog tachometers * * This implementation produces one pulse per engine cycle * * todo: these is a bit of duplication with dizzySparkOutputPin * * @date Aug 18, 2015 * @author Andrey Belomutskiy, (c) 2012-2020 */ #include "tachometer.h" #include "trigger_central.h" #include "pwm_generator.h" EXTERN_ENGINE; static SimplePwm tachControl("tach"); static float tachFreq; static float duty; #if EFI_UNIT_TEST float getTachFreq(void) { return tachFreq; } float getTachDuty(void) { return duty; } #endif static void tachSignalCallback(trigger_event_e ckpSignalType, uint32_t index, efitick_t edgeTimestamp DECLARE_ENGINE_PARAMETER_SUFFIX) { // only process at index configured to avoid too much cpu time for index 0? if (index != (uint32_t)CONFIG(tachPulseTriggerIndex)) { return; } #if EFI_UNIT_TEST printf("tachSignalCallback(%d %d)\n", ckpSignalType, index); printf("Current RPM: %d\n",GET_RPM()); UNUSED(edgeTimestamp); #else UNUSED(ckpSignalType); UNUSED(edgeTimestamp); #endif // How many tach pulse periods do we have? int periods = CONFIG(tachPulsePerRev); if(periods == 0){ warning(CUSTOM_ERR_6709,"Check Tachometer Pulse per Rev!"); return; } // What is the angle per tach output period? float cycleTimeMs = 60000.0 / GET_RPM(); float periodTimeMs = cycleTimeMs / periods; tachFreq = 1000.0 / periodTimeMs; if (CONFIG(tachPulseDurationAsDutyCycle)) { // Simple case - duty explicitly set duty = CONFIG(tachPulseDuractionMs); } else { // Constant high-time mode - compute the correct duty cycle duty = CONFIG(tachPulseDuractionMs) / periodTimeMs; } // In case Freq is under 1Hz, we stop pwm to avoid warnings! if (tachFreq < 1.0) { tachFreq = NAN; } tachControl.setSimplePwmDutyCycle(duty); tachControl.setFrequency(tachFreq); } void initTachometer(DECLARE_ENGINE_PARAMETER_SIGNATURE) { if (CONFIG(tachOutputPin) == GPIO_UNASSIGNED) { return; } startSimplePwmExt(&tachControl, "Tachometer", &engine->executor, CONFIG(tachOutputPin), &enginePins.tachOut, NAN, 0.1, (pwm_gen_callback*)applyPinState); #if EFI_SHAFT_POSITION_INPUT addTriggerEventListener(tachSignalCallback, "tach", engine); #endif /* EFI_SHAFT_POSITION_INPUT */ }