/* * @file spark_logic.cpp * * @date Sep 15, 2016 * @author Andrey Belomutskiy, (c) 2012-2017 */ #include "engine_math.h" #include "utlist.h" #include "event_queue.h" #include "efilib2.h" EXTERN_ENGINE; static cyclic_buffer ignitionErrorDetection; static Logging *logger; static const char *prevSparkName = NULL; IgnitionEventList::IgnitionEventList() { isReady = false; } int isInjectionEnabled(engine_configuration_s *engineConfiguration) { // todo: is this worth a method? should this be inlined? return engineConfiguration->isInjectionEnabled; } int isIgnitionTimingError(void) { return ignitionErrorDetection.sum(6) > 4; } void prepareIgnitionSchedule(IgnitionEvent *event DECLARE_ENGINE_PARAMETER_S); static void turnSparkPinLow2(IgnitionEvent *event, IgnitionOutputPin *output) { #if SPARK_EXTREME_LOGGING || defined(__DOXYGEN__) scheduleMsg(logger, "spark goes low %d %s %d current=%d cnt=%d id=%d", getRevolutionCounter(), output->name, (int)getTimeNowUs(), output->currentLogicValue, output->outOfOrder, event->sparkId); #endif /* FUEL_MATH_EXTREME_LOGGING */ /** * there are two kinds of 'out-of-order' * 1) low goes before high, everything is fine after words * * 2) we have an un-matched low followed by legit pairs * */ output->signalFallSparkId = event->sparkId; if (!output->currentLogicValue) { warning(CUSTOM_OUT_OF_ORDER_COIL, "out-of-order coil off %s", output->name); output->outOfOrder = true; } turnPinLow(output); #if EFI_PROD_CODE || defined(__DOXYGEN__) if (CONFIG(dizzySparkOutputPin) != GPIO_UNASSIGNED) { turnPinLow(&enginePins.dizzyOutput); } #endif /* EFI_PROD_CODE */ } void turnSparkPinLow(IgnitionEvent *event) { for (int i = 0; i< MAX_OUTPUTS_FOR_IGNITION;i++) { IgnitionOutputPin *output = event->outputs[i]; if (output != NULL) { turnSparkPinLow2(event, output); } } #if EFI_UNIT_TEST || defined(__DOXYGEN__) Engine *engine = event->engine; EXPAND_Engine; #endif // now that we've just fired a coil let's prepare the new schedule for the next engine revolution prepareIgnitionSchedule(event PASS_ENGINE_PARAMETER); } static void turnSparkPinHigh2(IgnitionEvent *event, IgnitionOutputPin *output) { #if ! EFI_UNIT_TEST || defined(__DOXYGEN__) if (engine->rpmCalculator.rpmValue > 2 * engineConfiguration->cranking.rpm) { const char *outputName = output->name; if (prevSparkName == outputName && engineConfiguration->ignitionMode != IM_ONE_COIL) { warning(CUSTOM_OBD_SKIPPED_SPARK, "looks like skipped spark event %d %s", getRevolutionCounter(), outputName); } prevSparkName = outputName; } #endif /* EFI_UNIT_TEST */ #if SPARK_EXTREME_LOGGING || defined(__DOXYGEN__) scheduleMsg(logger, "spark goes high %d %s %d current=%d cnt=%d id=%d", getRevolutionCounter(), output->name, (int)getTimeNowUs(), output->currentLogicValue, output->outOfOrder, event->sparkId); #endif /* FUEL_MATH_EXTREME_LOGGING */ if (output->outOfOrder) { output->outOfOrder = false; if (output->signalFallSparkId == event->sparkId) { // let's save this coil if things do not look right return; } } turnPinHigh(output); #if EFI_PROD_CODE || defined(__DOXYGEN__) if (CONFIG(dizzySparkOutputPin) != GPIO_UNASSIGNED) { turnPinHigh(&enginePins.dizzyOutput); } #endif /* EFI_PROD_CODE */ } void turnSparkPinHigh(IgnitionEvent *event) { for (int i = 0; i< MAX_OUTPUTS_FOR_IGNITION;i++) { IgnitionOutputPin *output = event->outputs[i]; if (output != NULL) { turnSparkPinHigh2(event, output); } } } static int globalSparkIdCoutner = 0; static ALWAYS_INLINE void handleSparkEvent(bool limitedSpark, uint32_t trgEventIndex, IgnitionEvent *iEvent, int rpm DECLARE_ENGINE_PARAMETER_S) { const floatms_t dwellMs = ENGINE(engineState.sparkDwell); if (cisnan(dwellMs) || dwellMs <= 0) { warning(CUSTOM_DWELL, "invalid dwell to handle: %f at %d", dwellMs, rpm); return; } floatus_t chargeDelayUs = ENGINE(rpmCalculator.oneDegreeUs) * iEvent->dwellPosition.angleOffset; int isIgnitionError = chargeDelayUs < 0; ignitionErrorDetection.add(isIgnitionError); if (isIgnitionError) { #if EFI_PROD_CODE || defined(__DOXYGEN__) scheduleMsg(logger, "Negative spark delay=%f", chargeDelayUs); #endif chargeDelayUs = 0; return; } iEvent->sparkId = globalSparkIdCoutner++; /** * We are alternating two event lists in order to avoid a potential issue around revolution boundary * when an event is scheduled within the next revolution. */ scheduling_s * sUp = &iEvent->signalTimerUp; scheduling_s * sDown = &iEvent->signalTimerDown; /** * The start of charge is always within the current trigger event range, so just plain time-based scheduling */ if (!limitedSpark) { #if SPARK_EXTREME_LOGGING || defined(__DOXYGEN__) scheduleMsg(logger, "scheduling sparkUp ind=%d %d %s now=%d %d later id=%d", trgEventIndex, getRevolutionCounter(), iEvent->getOutputForLoggins()->name, (int)getTimeNowUs(), (int)chargeDelayUs, iEvent->sparkId); #endif /* FUEL_MATH_EXTREME_LOGGING */ /** * Note how we do not check if spark is limited or not while scheduling 'spark down' * This way we make sure that coil dwell started while spark was enabled would fire and not burn * the coil. */ scheduleTask(false, "spark up", sUp, chargeDelayUs, (schfunc_t) &turnSparkPinHigh, iEvent); } /** * Spark event is often happening during a later trigger event timeframe * TODO: improve precision */ TRIGGER_SHAPE(findTriggerPosition(&iEvent->sparkPosition, iEvent->advance PASS_ENGINE_PARAMETER)); #if EFI_UNIT_TEST || defined(__DOXYGEN__) printf("spark dwell@ %d/%d spark@ %d/%d id=%d\r\n", iEvent->dwellPosition.eventIndex, (int)iEvent->dwellPosition.angleOffset, iEvent->sparkPosition.eventIndex, (int)iEvent->sparkPosition.angleOffset, iEvent->sparkId); #endif if (iEvent->sparkPosition.eventIndex == trgEventIndex) { /** * Spark should be fired before the next trigger event - time-based delay is best precision possible */ float timeTillIgnitionUs = ENGINE(rpmCalculator.oneDegreeUs) * iEvent->sparkPosition.angleOffset; #if SPARK_EXTREME_LOGGING || defined(__DOXYGEN__) scheduleMsg(logger, "scheduling sparkDown ind=%d %d %s now=%d %d later id=%d", trgEventIndex, getRevolutionCounter(), iEvent->getOutputForLoggins()->name, (int)getTimeNowUs(), (int)timeTillIgnitionUs, iEvent->sparkId); #endif /* FUEL_MATH_EXTREME_LOGGING */ scheduleTask(false, "spark1 down", sDown, (int) timeTillIgnitionUs, (schfunc_t) &turnSparkPinLow, iEvent); } else { #if SPARK_EXTREME_LOGGING || defined(__DOXYGEN__) scheduleMsg(logger, "to queue sparkDown ind=%d %d %s %d for %d", trgEventIndex, getRevolutionCounter(), iEvent->getOutputForLoggins()->name, (int)getTimeNowUs(), iEvent->sparkPosition.eventIndex); #endif /* FUEL_MATH_EXTREME_LOGGING */ /** * Spark should be scheduled in relation to some future trigger event, this way we get better firing precision */ bool isPending = assertNotInList(ENGINE(iHead), iEvent); if (isPending) { #if SPARK_EXTREME_LOGGING || defined(__DOXYGEN__) scheduleMsg(logger, "not adding to queue sparkDown ind=%d %d %s %d", trgEventIndex, getRevolutionCounter(), iEvent->getOutputForLoggins()->name, (int)getTimeNowUs()); #endif /* FUEL_MATH_EXTREME_LOGGING */ return; } LL_APPEND(ENGINE(iHead), iEvent); } } #define assertPinAssigned(output) { \ if (!isPinAssigned(output)) { \ warning(CUSTOM_OBD_COIL_PIN_NOT_ASSIGNED, "no_pin_cl #%s", (output)->name); \ } \ } void prepareIgnitionSchedule(IgnitionEvent *event DECLARE_ENGINE_PARAMETER_S) { // todo: clean up this implementation? does not look too nice as is. // change of sign here from 'before TDC' to 'after TDC' const angle_t localAdvance = -ENGINE(engineState.timingAdvance) + ENGINE(angleExtra[event->cylinderIndex]) + CONFIG(timing_offset_cylinder[event->cylinderIndex]); const int index = ENGINE(ignitionPin[event->cylinderIndex]); const int coilIndex = ID2INDEX(getCylinderId(index PASS_ENGINE_PARAMETER)); IgnitionOutputPin *output = &enginePins.coils[coilIndex]; IgnitionOutputPin *secondOutput; if (CONFIG(ignitionMode) == IM_WASTED_SPARK && CONFIG(twoWireBatchIgnition)) { int secondIndex = index + CONFIG(specs.cylindersCount) / 2; int secondCoilIndex = ID2INDEX(getCylinderId(secondIndex PASS_ENGINE_PARAMETER)); secondOutput = &enginePins.coils[secondCoilIndex]; assertPinAssigned(secondOutput); } else { secondOutput = NULL; } angle_t dwellAngle = ENGINE(engineState.dwellAngle); assertPinAssigned(output); event->outputs[0] = output; event->outputs[1] = secondOutput; event->advance = localAdvance; TRIGGER_SHAPE(findTriggerPosition(&event->dwellPosition, localAdvance - dwellAngle PASS_ENGINE_PARAMETER)); #if FUEL_MATH_EXTREME_LOGGING || defined(__DOXYGEN__) printf("addIgnitionEvent %s ind=%d\n", output->name, event->dwellPosition.eventIndex); // scheduleMsg(logger, "addIgnitionEvent %s ind=%d", output->name, event->dwellPosition->eventIndex); #endif /* FUEL_MATH_EXTREME_LOGGING */ } static void initializeIgnitionActions(IgnitionEventList *list DECLARE_ENGINE_PARAMETER_S) { if (cisnan(ENGINE(engineState.timingAdvance))) { // error should already be reported // need to invalidate previous ignition schedule list->isReady = false; return; } efiAssertVoid(engineConfiguration->specs.cylindersCount > 0, "cylindersCount"); for (int cylinderIndex = 0; cylinderIndex < CONFIG(specs.cylindersCount); cylinderIndex++) { list->elements[cylinderIndex].cylinderIndex = cylinderIndex; #if EFI_UNIT_TEST list->elements[cylinderIndex].engine = engine; #endif prepareIgnitionSchedule(&list->elements[cylinderIndex] PASS_ENGINE_PARAMETER); } list->isReady = true; } static ALWAYS_INLINE void prepareIgnitionSchedule(int rpm DECLARE_ENGINE_PARAMETER_S) { engine->m.beforeIgnitionSch = GET_TIMESTAMP(); /** * TODO: warning. there is a bit of a hack here, todo: improve. * currently output signals/times signalTimerUp from the previous revolutions could be * still used because they have crossed the revolution boundary * but we are already re-purposing the output signals, but everything works because we * are not affecting that space in memory. todo: use two instances of 'ignitionSignals' */ float maxAllowedDwellAngle = (int) (getEngineCycle(engineConfiguration->operationMode) / 2); // the cast is about making Coverity happy if (engineConfiguration->ignitionMode == IM_ONE_COIL) { maxAllowedDwellAngle = getEngineCycle(engineConfiguration->operationMode) / engineConfiguration->specs.cylindersCount / 1.1; } if (engine->engineState.dwellAngle == 0) { warning(CUSTOM_ZERO_DWELL, "dwell is zero?"); } if (engine->engineState.dwellAngle > maxAllowedDwellAngle) { warning(CUSTOM_DWELL_TOO_LONG, "dwell angle too long: %f", engine->engineState.dwellAngle); } // todo: add some check for dwell overflow? like 4 times 6 ms while engine cycle is less then that IgnitionEventList *list = &engine->ignitionEvents; initializeIgnitionActions(list PASS_ENGINE_PARAMETER); engine->m.ignitionSchTime = GET_TIMESTAMP() - engine->m.beforeIgnitionSch; } void handleSpark(bool limitedSpark, uint32_t trgEventIndex, int rpm DECLARE_ENGINE_PARAMETER_S) { if (!isValidRpm(rpm) || !CONFIG(isIgnitionEnabled)) { // this might happen for instance in case of a single trigger event after a pause return; } if (!engine->ignitionEvents.isReady) { prepareIgnitionSchedule(rpm PASS_ENGINE_PARAMETER); } /** * Ignition schedule is defined once per revolution * See initializeIgnitionActions() */ IgnitionEvent *current, *tmp; LL_FOREACH_SAFE(ENGINE(iHead), current, tmp) { if (current->sparkPosition.eventIndex == trgEventIndex) { // time to fire a spark which was scheduled previously LL_DELETE(ENGINE(iHead), current); scheduling_s * sDown = ¤t->signalTimerDown; #if SPARK_EXTREME_LOGGING || defined(__DOXYGEN__) scheduleMsg(logger, "time to sparkDown ind=%d %d %s %d", trgEventIndex, getRevolutionCounter(), current->getOutputForLoggins()->name, (int)getTimeNowUs()); #endif /* FUEL_MATH_EXTREME_LOGGING */ float timeTillIgnitionUs = ENGINE(rpmCalculator.oneDegreeUs) * current->sparkPosition.angleOffset; scheduleTask(false, "spark 2down", sDown, (int) timeTillIgnitionUs, (schfunc_t) &turnSparkPinLow, current); } } // scheduleSimpleMsg(&logger, "eventId spark ", eventIndex); if (engine->ignitionEvents.isReady) { for (int i = 0; i < CONFIG(specs.cylindersCount); i++) { IgnitionEvent *event = &engine->ignitionEvents.elements[i]; if (event->dwellPosition.eventIndex != trgEventIndex) continue; handleSparkEvent(limitedSpark, trgEventIndex, event, rpm PASS_ENGINE_PARAMETER); } } } void initSparkLogic(Logging *sharedLogger) { logger = sharedLogger; } /** * Number of sparks per physical coil * @see getNumberOfInjections */ int getNumberOfSparks(ignition_mode_e mode DECLARE_ENGINE_PARAMETER_S) { switch (mode) { case IM_ONE_COIL: return engineConfiguration->specs.cylindersCount; case IM_INDIVIDUAL_COILS: return 1; case IM_WASTED_SPARK: return 2; default: firmwareError(CUSTOM_ERR_IGNITION_MODE, "Unexpected ignition_mode_e %d", mode); return 1; } } /** * @see getInjectorDutyCycle */ percent_t getCoilDutyCycle(int rpm DECLARE_ENGINE_PARAMETER_S) { floatms_t totalPerCycle = 1/**getInjectionDuration(rpm PASS_ENGINE_PARAMETER)*/ * getNumberOfSparks(engineConfiguration->ignitionMode PASS_ENGINE_PARAMETER); floatms_t engineCycleDuration = getCrankshaftRevolutionTimeMs(rpm) * (engineConfiguration->operationMode == TWO_STROKE ? 1 : 2); return 100 * totalPerCycle / engineCycleDuration; }