rusefi/firmware/controllers/algo/engine.h

469 lines
12 KiB
C
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file engine.h
*
* @date May 21, 2014
2020-01-07 21:02:40 -08:00
* @author Andrey Belomutskiy, (c) 2012-2020
2015-07-10 06:01:56 -07:00
*/
2019-09-19 21:17:00 -07:00
#pragma once
2015-07-10 06:01:56 -07:00
2019-01-27 21:44:30 -08:00
#include "globalaccess.h"
#include "engine_module.h"
2019-05-27 16:05:59 -07:00
#include "engine_state.h"
2015-07-10 06:01:56 -07:00
#include "rpm_calculator.h"
#include "event_registry.h"
#include "table_helper.h"
#include "listener_array.h"
#include "accel_enrichment.h"
2015-09-13 14:02:44 -07:00
#include "trigger_central.h"
#include "local_version_holder.h"
#include "buttonshift.h"
#include "gear_controller.h"
#include "high_pressure_fuel_pump.h"
#include "limp_manager.h"
#include "pin_repository.h"
#include "ac_control.h"
#include "knock_logic.h"
2021-10-16 20:55:29 -07:00
#include "idle_state_generated.h"
#include "idle_thread.h"
#include "injector_model.h"
2021-11-15 15:57:12 -08:00
#include "launch_control.h"
#include "trigger_scheduler.h"
#include "fuel_pump.h"
#include "main_relay.h"
#include "ac_control.h"
#include "type_list.h"
2021-11-24 09:59:57 -08:00
#include "boost_control.h"
#include "ignition_controller.h"
#include "alternator_controller.h"
#include "dfco.h"
2015-07-10 06:01:56 -07:00
#ifndef EFI_UNIT_TEST
#error EFI_UNIT_TEST must be defined!
#endif
#ifndef EFI_SIMULATOR
#error EFI_SIMULATOR must be defined!
#endif
#ifndef EFI_PROD_CODE
#error EFI_PROD_CODE must be defined!
#endif
#if EFI_SIGNAL_EXECUTOR_ONE_TIMER
// PROD real firmware uses this implementation
2019-03-29 06:11:13 -07:00
#include "single_timer_executor.h"
#endif /* EFI_SIGNAL_EXECUTOR_ONE_TIMER */
#if EFI_SIGNAL_EXECUTOR_SLEEP
#include "signal_executor_sleep.h"
#endif /* EFI_SIGNAL_EXECUTOR_SLEEP */
#if EFI_UNIT_TEST
#include "global_execution_queue.h"
#endif /* EFI_UNIT_TEST */
#define FAST_CALLBACK_PERIOD_MS 5
#define SLOW_CALLBACK_PERIOD_MS 50
2016-11-30 19:06:43 -08:00
2015-07-10 06:01:56 -07:00
class RpmCalculator;
2021-05-14 04:17:22 -07:00
struct AirmassModelBase;
2015-07-10 06:01:56 -07:00
#define MAF_DECODING_CACHE_SIZE 256
#define MAF_DECODING_CACHE_MULT (MAF_DECODING_CACHE_SIZE / 5.0)
/**
* I am not sure if this needs to be configurable.
*
* Also technically the whole feature might be implemented as cranking fuel coefficient curve by TPS.
*/
// todo: not great location for these
#define CLEANUP_MODE_TPS 90
#define STEPPER_PARKING_TPS CLEANUP_MODE_TPS
2019-11-23 19:55:21 -08:00
#define CYCLE_ALTERNATION 2
class IEtbController;
2020-12-11 09:50:10 -08:00
struct IFuelComputer;
2020-12-17 14:46:51 -08:00
struct IIdleController;
class PrimaryTriggerConfiguration final : public TriggerConfiguration {
2020-08-23 22:21:42 -07:00
public:
PrimaryTriggerConfiguration() : TriggerConfiguration("TRG ") {}
protected:
bool isUseOnlyRisingEdgeForTrigger() const override;
bool isVerboseTriggerSynchDetails() const override;
trigger_type_e getType() const override;
2020-08-23 22:21:42 -07:00
};
class VvtTriggerConfiguration final : public TriggerConfiguration {
2020-08-26 20:35:11 -07:00
public:
2021-07-21 19:21:59 -07:00
const int index;
2021-07-21 19:14:21 -07:00
2021-07-21 19:28:35 -07:00
VvtTriggerConfiguration(const char * prefix, const int index) : TriggerConfiguration(prefix), index(index) {
2021-07-21 19:14:21 -07:00
}
protected:
bool isUseOnlyRisingEdgeForTrigger() const override;
bool isVerboseTriggerSynchDetails() const override;
trigger_type_e getType() const override;
2020-08-26 20:35:11 -07:00
};
class PrimeController : public EngineModule {
public:
void onIgnitionStateChanged(bool ignitionOn) override;
floatms_t getPrimeDuration() const;
void onPrimeStart();
void onPrimeEnd();
bool isPriming() const {
return m_isPriming;
}
private:
scheduling_s m_start;
scheduling_s m_end;
bool m_isPriming = false;
static void onPrimeStartAdapter(PrimeController* instance) {
instance->onPrimeStart();
}
static void onPrimeEndAdapter(PrimeController* instance) {
instance->onPrimeEnd();
}
};
2021-08-03 16:48:24 -07:00
class Engine final : public TriggerStateListener {
2015-07-10 06:01:56 -07:00
public:
2016-12-18 07:02:38 -08:00
Engine();
// todo: technical debt: enableOverdwellProtection #3553
bool enableOverdwellProtection = true;
TunerStudioOutputChannels outputChannels;
/**
* Sometimes for instance during shutdown we need to completely supress CAN TX
*/
bool allowCanTx = true;
// used by HW CI
bool isPwmEnabled = true;
const char *prevOutputName = nullptr;
2021-12-05 18:20:54 -08:00
/**
* ELM327 cannot handle both RX and TX at the same time, we have to stay quite once first ISO/TP packet was detected
* this is a pretty temporary hack only while we are trying ELM327, long term ISO/TP and rusEFI broadcast should find a way to coexists
*/
bool pauseCANdueToSerial = false;
PinRepository pinRepository;
2020-04-21 05:18:32 -07:00
IEtbController *etbControllers[ETB_COUNT] = {nullptr};
IFuelComputer *fuelComputer = nullptr;
type_list<
Mockable<InjectorModel>,
#if EFI_IDLE_CONTROL
Mockable<IdleController>,
#endif // EFI_IDLE_CONTROL
TriggerScheduler,
#if EFI_HPFP && EFI_ENGINE_CONTROL
HpfpController,
#endif // EFI_HPFP && EFI_ENGINE_CONTROL
#if EFI_ALTERNATOR_CONTROL
AlternatorController,
#endif /* EFI_ALTERNATOR_CONTROL */
FuelPumpController,
MainRelayController,
IgnitionController,
AcController,
PrimeController,
DfcoController,
Mockable<WallFuelController>,
EngineModule // dummy placeholder so the previous entries can all have commas
> engineModules;
/**
* Slightly shorter helper function to keep the code looking clean.
*/
template<typename get_t>
auto & module() {
return engineModules.get<get_t>();
}
2020-01-21 22:22:44 -08:00
cyclic_buffer<int> triggerErrorDetection;
GearControllerBase *gearController;
#if EFI_LAUNCH_CONTROL
2021-11-15 15:57:12 -08:00
LaunchControlBase launchController;
2021-11-16 10:15:12 -08:00
SoftSparkLimiter softSparkLimiter;
2022-02-06 15:46:09 -08:00
#endif // EFI_LAUNCH_CONTROL
2021-11-24 09:59:57 -08:00
#if EFI_BOOST_CONTROL
BoostController boostController;
#endif // EFI_BOOST_CONTROL
efitick_t mostRecentSparkEvent;
efitick_t mostRecentTimeBetweenSparkEvents;
efitick_t mostRecentIgnitionEvent;
efitick_t mostRecentTimeBetweenIgnitionEvents;
2020-08-23 22:21:42 -07:00
PrimaryTriggerConfiguration primaryTriggerConfiguration;
2021-07-21 20:06:35 -07:00
#if CAMS_PER_BANK == 1
VvtTriggerConfiguration vvtTriggerConfiguration[CAMS_PER_BANK] = {{"VVT1 ", 0}};
2021-07-21 19:38:00 -07:00
#else
2021-07-21 19:14:21 -07:00
VvtTriggerConfiguration vvtTriggerConfiguration[CAMS_PER_BANK] = {{"VVT1 ", 0}, {"VVT2 ", 1}};
2021-07-21 19:38:00 -07:00
#endif
efitick_t startStopStateLastPushTime = 0;
2020-08-23 22:21:42 -07:00
#if EFI_SHAFT_POSITION_INPUT
2020-01-26 03:12:01 -08:00
void OnTriggerStateDecodingError();
void OnTriggerStateProperState(efitick_t nowNt) override;
2020-01-22 10:25:35 -08:00
void OnTriggerSyncronization(bool wasSynchronized) override;
2020-01-26 00:33:45 -08:00
void OnTriggerInvalidIndex(int currentIndex) override;
2020-01-26 09:02:54 -08:00
void OnTriggerSynchronizationLost() override;
#endif
void setConfig();
injection_mode_e getCurrentInjectionMode();
2016-12-18 07:02:38 -08:00
LocalVersionHolder versionForConfigurationListeners;
LocalVersionHolder auxParametersVersion;
operation_mode_e getOperationMode();
2019-12-02 21:29:12 -08:00
AuxActor auxValves[AUX_DIGITAL_VALVE_COUNT][2];
2019-11-23 19:55:21 -08:00
2019-12-23 18:58:06 -08:00
#if EFI_UNIT_TEST
bool needTdcCallback = true;
2019-12-23 18:58:06 -08:00
#endif /* EFI_UNIT_TEST */
2019-10-07 22:36:03 -07:00
/**
* By the way 32-bit value should hold at least 400 hours of events at 6K RPM x 12 events per revolution
*/
int globalSparkIdCounter = 0;
2019-11-11 19:19:35 -08:00
// this is useful at least for real hardware integration testing - maybe a proper solution would be to simply
// GND input pins instead of leaving them floating
bool hwTriggerInputEnabled = true;
int getGlobalConfigurationVersion(void) const;
// a pointer with interface type would make this code nicer but would carry extra runtime
// cost to resolve pointer, we use instances as a micro optimization
#if EFI_SIGNAL_EXECUTOR_ONE_TIMER
SingleTimerExecutor executor;
#endif
#if EFI_SIGNAL_EXECUTOR_SLEEP
SleepExecutor executor;
#endif
#if EFI_UNIT_TEST
TestExecutor executor;
#endif
2016-01-25 09:01:30 -08:00
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_CONTROL
2016-12-18 09:03:48 -08:00
FuelSchedule injectionEvents;
IgnitionEventList ignitionEvents;
scheduling_s tdcScheduler[2];
2016-12-18 09:03:48 -08:00
#endif /* EFI_ENGINE_CONTROL */
2016-02-06 07:01:34 -08:00
2019-01-19 19:31:55 -08:00
bool etbAutoTune = false;
2016-01-30 19:03:36 -08:00
/**
* this is based on isEngineChartEnabled and engineSnifferRpmThreshold settings
*/
2019-01-19 19:31:55 -08:00
bool isEngineChartEnabled = false;
bool tdcMarkEnabled = true; // used by unit tests only
2016-01-30 19:03:36 -08:00
/**
* this is based on sensorChartMode and sensorSnifferRpmThreshold settings
*/
2019-01-19 19:31:55 -08:00
sensor_chart_e sensorChartMode = SC_OFF;
2016-01-26 20:01:44 -08:00
bool slowCallBackWasInvoked = false;
2017-05-11 05:32:08 -07:00
2017-05-01 19:33:20 -07:00
/**
2017-05-04 14:03:23 -07:00
* remote telemetry: if not zero, time to stop flashing 'CALL FROM PIT STOP' light
* todo: looks like there is a bug here? 64 bit storage an 32 bit time logic? anyway this feature is mostly a dream at this point
2017-05-01 19:33:20 -07:00
*/
efitimems64_t callFromPitStopEndTime = 0;
2017-05-01 19:33:20 -07:00
2015-07-10 06:01:56 -07:00
RpmCalculator rpmCalculator;
2020-03-23 07:32:41 -07:00
bool startStopState = false;
int startStopStateToggleCounter = 0;
/**
* this is needed by getTimeIgnitionSeconds() and checkShutdown()
*/
efitick_t ignitionOnTimeNt = 0;
/**
2019-05-02 14:52:48 -07:00
* This counter is incremented every time user adjusts ECU parameters online (either via rusEfi console or other
* tuning software)
*/
volatile int globalConfigurationVersion = 0;
2016-01-30 19:03:36 -08:00
/**
* always 360 or 720, never zero
*/
2016-01-14 21:01:42 -08:00
angle_t engineCycle;
TpsAccelEnrichment tpsAccelEnrichment;
2015-07-10 06:01:56 -07:00
2015-09-13 14:02:44 -07:00
TriggerCentral triggerCentral;
2015-07-10 06:01:56 -07:00
/**
2016-10-12 22:02:02 -07:00
* Each individual fuel injection duration for current engine cycle, without wall wetting
* including everything including injector lag, both cranking and running
2017-11-06 19:29:39 -08:00
* @see getInjectionDuration()
2015-07-10 06:01:56 -07:00
*/
2019-01-19 19:31:55 -08:00
floatms_t injectionDuration = 0;
2015-07-10 06:01:56 -07:00
// Per-injection fuel mass, including TPS accel enrich
float injectionMass[MAX_CYLINDER_COUNT] = {0};
float stftCorrection[STFT_BANK_COUNT] = {0};
// Stores the actual pulse duration of the last injection for every cylinder
floatms_t actualLastInjection[MAX_CYLINDER_COUNT] = {0};
2015-08-23 20:02:37 -07:00
2020-05-05 05:01:40 -07:00
// Standard cylinder air charge - 100% VE at standard temperature, grams per cylinder
float standardAirCharge = 0;
void periodicFastCallback();
void periodicSlowCallback();
void updateSlowSensors();
void updateSwitchInputs();
2022-04-02 16:27:18 -07:00
void updateTriggerWaveform();
2015-07-10 06:01:56 -07:00
2019-01-19 19:31:55 -08:00
bool clutchUpState = false;
bool clutchDownState = false;
bool brakePedalState = false;
2019-09-08 18:05:03 -07:00
// todo: extract some helper which would contain boolean state and most recent toggle time?
bool acSwitchState = false;
2019-09-08 18:05:03 -07:00
efitimeus_t acSwitchLastChangeTime = 0;
2015-07-10 06:01:56 -07:00
2019-01-19 19:31:55 -08:00
bool isRunningPwmTest = false;
2015-07-10 06:01:56 -07:00
/**
* are we running any kind of functional test? this affect
* some areas
*/
bool isFunctionalTestMode = false;
2020-09-19 01:38:59 -07:00
/**
* See also triggerSimulatorFrequency
*/
bool directSelfStimulation = false;
2019-12-21 17:43:11 -08:00
void resetEngineSnifferIfInTestMode();
2015-07-10 06:01:56 -07:00
/**
* pre-calculated reference to which output pin should be used for
* given sequence index within engine cycle
2016-01-16 14:02:38 -08:00
* todo: update documentation
2015-07-10 06:01:56 -07:00
*/
int ignitionPin[MAX_CYLINDER_COUNT];
2015-07-10 06:01:56 -07:00
2017-11-20 12:01:48 -08:00
/**
* this is invoked each time we register a trigger tooth signal
*/
2021-03-15 07:23:19 -07:00
void onTriggerSignalEvent();
2015-07-10 06:01:56 -07:00
EngineState engineState;
2021-10-16 21:27:12 -07:00
/**
* idle blip is a development tool: alternator PID research for instance have benefited from a repetitive change of RPM
*/
percent_t blipIdlePosition;
efitimeus_t timeToStopBlip = 0;
efitimeus_t timeToStopIdleTest = 0;
2021-10-16 20:55:29 -07:00
SensorsState sensors;
2021-01-10 20:46:50 -08:00
efitick_t mainRelayBenchStartNt = 0;
2015-07-10 06:01:56 -07:00
/**
* value of 'triggerShape.getLength()'
* pre-calculating this value is a performance optimization
*/
2019-01-19 19:31:55 -08:00
uint32_t engineCycleEventCount = 0;
2015-07-10 06:01:56 -07:00
void preCalculate();
2015-07-10 06:01:56 -07:00
void watchdog();
/**
* Needed by EFI_MAIN_RELAY_CONTROL to shut down the engine correctly.
* This method cancels shutdown if the ignition voltage is detected.
*/
void checkShutdown();
/**
* Allows to finish some long-term shutdown procedures (stepper motor parking etc.)
Called when the ignition switch is turned off (vBatt is too low).
Returns true if some operations are in progress on background.
*/
bool isInShutdownMode() const;
bool isInMainRelayBench();
2021-01-10 20:46:50 -08:00
/**
* The stepper does not work if the main relay is turned off (it requires +12V).
* Needed by the stepper motor code to detect if it works.
*/
bool isMainRelayEnabled() const;
/**
* Needed by EFI_MAIN_RELAY_CONTROL to handle fuel pump and shutdown timings correctly.
* This method returns the number of seconds since the ignition voltage is present.
* The return value is float for more FSIO flexibility.
*/
float getTimeIgnitionSeconds(void) const;
2015-07-10 06:01:56 -07:00
void onSparkFireKnockSense(uint8_t cylinderIndex, efitick_t nowNt);
KnockController knockController;
2020-07-26 20:33:18 -07:00
AirmassModelBase* mockAirmassModel = nullptr;
LimpManager limpManager;
2015-07-10 06:01:56 -07:00
private:
/**
* By the way:
* 'cranking' means engine is not stopped and the rpm are below crankingRpm
* 'running' means RPM are above crankingRpm
* 'spinning' means the engine is not stopped
*/
2019-01-19 19:31:55 -08:00
bool isSpinning = false;
void reset();
void injectEngineReferences();
2015-07-10 06:01:56 -07:00
};
trigger_type_e getVvtTriggerType(vvt_mode_e vvtMode);
void prepareShapes();
void applyNonPersistentConfiguration();
void prepareOutputSignals();
2018-02-03 07:48:35 -08:00
void validateConfiguration();
void doScheduleStopEngine();
#define HW_CHECK_RPM 200
// These externs aren't needed for unit tests - everything is injected instead
#if !EFI_UNIT_TEST
extern Engine ___engine;
static Engine * const engine = &___engine;
#else // EFI_UNIT_TEST
extern Engine *engine;
#endif // EFI_UNIT_TEST