/* * engine2.cpp * * @date Jan 5, 2019 * @author Andrey Belomutskiy, (c) 2012-2020 */ // todo: move this code to more proper locations #include "pch.h" #include "speed_density.h" #include "fuel_math.h" #include "advance_map.h" #include "aux_valves.h" #include "closed_loop_fuel.h" #include "launch_control.h" #include "injector_model.h" #include "tunerstudio.h" #if ! EFI_UNIT_TEST #include "status_loop.h" #endif WarningCodeState::WarningCodeState() { clear(); } void WarningCodeState::clear() { warningCounter = 0; lastErrorCode = ObdCode::None; recentWarnings.clear(); } void WarningCodeState::addWarningCode(ObdCode code) { warningCounter++; lastErrorCode = code; warning_t* existing = recentWarnings.find(code); if (!existing) { chibios_rt::CriticalSectionLocker csl; // Add the code to the list existing = recentWarnings.add(warning_t(code)); } if (existing) { // Reset the timer on the code to now existing->LastTriggered.reset(); } // Reset the "any warning" timer too timeSinceLastWarning.reset(); } /** * @param forIndicator if we want to retrieving value for TS indicator, this case a minimal period is applued */ bool WarningCodeState::isWarningNow() const { int period = maxI(3, engineConfiguration->warningPeriod); return !timeSinceLastWarning.hasElapsedSec(period); } // Check whether a particular warning is active bool WarningCodeState::isWarningNow(ObdCode code) const { warning_t* warn = recentWarnings.find(code); // No warning found at all if (!warn) { return false; } // If the warning is old, it is not active return !warn->LastTriggered.hasElapsedSec(maxI(3, engineConfiguration->warningPeriod)); } EngineState::EngineState() { timeSinceLastTChargeK.reset(getTimeNowNt()); } void EngineState::updateSlowSensors() { } void EngineState::updateSparkSkip() { #if EFI_LAUNCH_CONTROL engine->softSparkLimiter.updateTargetSkipRatio(luaSoftSparkSkip, tractionControlSparkSkip); engine->hardSparkLimiter.updateTargetSkipRatio( luaHardSparkSkip, tractionControlSparkSkip, /* * We are applying launch controller spark skip ratio only for hard skip limiter (see * https://github.com/rusefi/rusefi/issues/6566#issuecomment-2153149902). */ engine->launchController.getSparkSkipRatio() + engine->shiftTorqueReductionController.getSparkSkipRatio() ); #endif // EFI_LAUNCH_CONTROL } #define MAKE_HUMAN_READABLE_ADVANCE(advance) (advance > getEngineState()->engineCycle / 2 ? advance - getEngineState()->engineCycle : advance) void EngineState::periodicFastCallback() { ScopePerf perf(PE::EngineStatePeriodicFastCallback); #if EFI_SHAFT_POSITION_INPUT if (!engine->slowCallBackWasInvoked) { warning(ObdCode::CUSTOM_SLOW_NOT_INVOKED, "Slow not invoked yet"); } efitick_t nowNt = getTimeNowNt(); if (engine->rpmCalculator.isCranking()) { crankingTimer.reset(nowNt); } engine->fuelComputer.running.timeSinceCrankingInSecs = crankingTimer.getElapsedSeconds(nowNt); #if EFI_AUX_VALVES recalculateAuxValveTiming(); #endif //EFI_AUX_VALVES float rpm = Sensor::getOrZero(SensorType::Rpm); engine->ignitionState.sparkDwell = engine->ignitionState.getSparkDwell(rpm); engine->ignitionState.dwellDurationAngle = std::isnan(rpm) ? NAN : engine->ignitionState.sparkDwell / getOneDegreeTimeMs(rpm); // todo: move this into slow callback, no reason for IAT corr to be here engine->fuelComputer.running.intakeTemperatureCoefficient = getIatFuelCorrection(); // todo: move this into slow callback, no reason for CLT corr to be here engine->fuelComputer.running.coolantTemperatureCoefficient = getCltFuelCorrection(); engine->module()->update(); // should be called before getInjectionMass() and getLimitingTimingRetard() getLimpManager()->updateRevLimit(rpm); // post-cranking fuel enrichment. float m_postCrankingFactor = interpolate3d( config->postCrankingFactor, config->postCrankingCLTBins, Sensor::getOrZero(SensorType::Clt), config->postCrankingDurationBins, engine->rpmCalculator.getRevolutionCounterSinceStart() ); // for compatibility reasons, apply only if the factor is greater than unity (only allow adding fuel) // if the engine run time is past the last bin, disable ASE in case the table is filled with values more than 1.0, helps with compatibility if ((m_postCrankingFactor < 1.0f) || (engine->rpmCalculator.getRevolutionCounterSinceStart() > config->postCrankingDurationBins[efi::size(config->postCrankingDurationBins)-1])) { m_postCrankingFactor = 1.0f; } engine->fuelComputer.running.postCrankingFuelCorrection = m_postCrankingFactor; engine->ignitionState.cltTimingCorrection = getCltTimingCorrection(); baroCorrection = getBaroCorrection(); auto tps = Sensor::get(SensorType::Tps1); updateTChargeK(rpm, tps.value_or(0)); float untrimmedInjectionMass = getInjectionMass(rpm) * engine->engineState.lua.fuelMult + engine->engineState.lua.fuelAdd; auto clResult = fuelClosedLoopCorrection(); injectionStage2Fraction = getStage2InjectionFraction(rpm, engine->fuelComputer.afrTableYAxis); float stage2InjectionMass = untrimmedInjectionMass * injectionStage2Fraction; float stage1InjectionMass = untrimmedInjectionMass - stage2InjectionMass; // Store the pre-wall wetting injection duration for scheduling purposes only, not the actual injection duration engine->engineState.injectionDuration = engine->module()->getInjectionDuration(stage1InjectionMass); engine->engineState.injectionDurationStage2 = engineConfiguration->enableStagedInjection ? engine->module()->getInjectionDuration(stage2InjectionMass) : 0; float fuelLoad = getFuelingLoad(); injectionOffset = getInjectionOffset(rpm, fuelLoad); engine->lambdaMonitor.update(rpm, fuelLoad); #if EFI_LAUNCH_CONTROL engine->launchController.update(); engine->shiftTorqueReductionController.update(); #endif //EFI_LAUNCH_CONTROL float l_ignitionLoad = getIgnitionLoad(); float baseAdvance = getWrappedAdvance(rpm, l_ignitionLoad); float corrections = engineConfiguration->timingMode == TM_DYNAMIC ? // Pull any extra timing for knock retard - engine->module()->getKnockRetard() // Degrees of timing REMOVED from actual timing during soft RPM limit window - getLimpManager()->getLimitingTimingRetard() : 0; float correctedIgnitionAdvance = baseAdvance + corrections; // these fields are scaled_channel so let's only use for observability, with a local variables holding value while it matters locally engine->ignitionState.baseIgnitionAdvance = MAKE_HUMAN_READABLE_ADVANCE(baseAdvance); engine->ignitionState.correctedIgnitionAdvance = MAKE_HUMAN_READABLE_ADVANCE(correctedIgnitionAdvance); // compute per-bank fueling for (size_t i = 0; i < STFT_BANK_COUNT; i++) { float corr = clResult.banks[i]; // todo: move to engine_state.txt and get rid of fuelPidCorrection in output_channels.txt? engine->engineState.stftCorrection[i] = corr; } // Now apply that to per-cylinder fueling and timing for (size_t i = 0; i < engineConfiguration->cylindersCount; i++) { uint8_t bankIndex = engineConfiguration->cylinderBankSelect[i]; auto bankTrim = engine->engineState.stftCorrection[bankIndex]; auto cylinderTrim = getCylinderFuelTrim(i, rpm, fuelLoad); auto knockTrim = engine->module()->getFuelTrimMultiplier(); // Apply both per-bank and per-cylinder trims engine->engineState.injectionMass[i] = untrimmedInjectionMass * bankTrim * cylinderTrim * knockTrim; // todo: is it OK to apply cylinder trim with FIXED timing? timingAdvance[i] = correctedIgnitionAdvance + getCylinderIgnitionTrim(i, rpm, l_ignitionLoad); } shouldUpdateInjectionTiming = getInjectorDutyCycle(rpm) < 90; // TODO: calculate me from a table! trailingSparkAngle = engineConfiguration->trailingSparkAngle; multispark.count = getMultiSparkCount(rpm); #if EFI_ANTILAG_SYSTEM engine->antilagController.update(); #endif //EFI_ANTILAG_SYSTEM #endif // EFI_SHAFT_POSITION_INPUT } #if EFI_ENGINE_CONTROL void EngineState::updateTChargeK(float rpm, float tps) { float newTCharge = engine->fuelComputer.getTCharge(rpm, tps); if (!std::isnan(newTCharge)) { // control the rate of change or just fill with the initial value efitick_t nowNt = getTimeNowNt(); float secsPassed = timeSinceLastTChargeK.getElapsedSeconds(nowNt); sd.tCharge = (sd.tChargeK == 0) ? newTCharge : limitRateOfChange(newTCharge, sd.tCharge, engineConfiguration->tChargeAirIncrLimit, engineConfiguration->tChargeAirDecrLimit, secsPassed); sd.tChargeK = convertCelsiusToKelvin(sd.tCharge); timeSinceLastTChargeK.reset(nowNt); } } #endif void TriggerConfiguration::update() { VerboseTriggerSynchDetails = isVerboseTriggerSynchDetails(); TriggerType = getType(); } trigger_config_s PrimaryTriggerConfiguration::getType() const { return engineConfiguration->trigger; } bool PrimaryTriggerConfiguration::isVerboseTriggerSynchDetails() const { return engineConfiguration->verboseTriggerSynchDetails; } trigger_config_s VvtTriggerConfiguration::getType() const { // Convert from VVT type to trigger_config_s return { getVvtTriggerType(engineConfiguration->vvtMode[index]), 0, 0 }; } bool VvtTriggerConfiguration::isVerboseTriggerSynchDetails() const { return engineConfiguration->verboseVVTDecoding; } bool isLockedFromUser() { int lock = engineConfiguration->tuneHidingKey; bool isLocked = lock > 0; if (isLocked) { criticalError("Tune is password protected. Please use console to unlock tune."); } return isLocked; } void unlockEcu(int password) { if (password != engineConfiguration->tuneHidingKey) { efiPrintf("Nope rebooting..."); #if EFI_PROD_CODE scheduleReboot(); #endif // EFI_PROD_CODE } else { efiPrintf("Unlocked! Burning..."); engineConfiguration->tuneHidingKey = 0; requestBurn(); } }