/** * @file rpm_calculator.h * @brief Shaft position sensor(s) decoder header * * @date Jan 1, 2013 * @author Andrey Belomutskiy, (c) 2012-2017 */ #ifndef RPM_REPORTER_H_ #define RPM_REPORTER_H_ #include #include "engine_configuration.h" #include "scheduler.h" #define TOP_DEAD_CENTER_MESSAGE "r" #define WC_DOWN "d" #define WC_UP "u" /** * See also 'CRANK1' in java dev console code */ #define CRANK1 "t1" #define CRANK2 "t2" #define CRANK3 "t3" #define VVT_NAME "VVT" #define NOISY_RPM -1 #define UNREALISTIC_RPM 30000 #ifndef RPM_LOW_THRESHOLD // no idea what is the best value, 25 is as good as any other guess #define RPM_LOW_THRESHOLD 25 #endif #ifdef __cplusplus typedef enum { /** * The engine is not spinning, RPM=0 */ STOPPED, /** * The engine is spinning up (reliable RPM is not detected yet). * In this state, rpmValue is >= 0 (can be zero). */ SPINNING_UP, /** * The engine is cranking (0 < RPM < cranking.rpm) */ CRANKING, /** * The engine is running (RPM >= cranking.rpm) */ RUNNING, } spinning_state_e; class Engine; #define GET_RPM() ( ENGINE(rpmCalculator.rpmValue) ) class RpmCalculator { public: #if !EFI_PROD_CODE int mockRpm; #endif /* EFI_PROD_CODE */ RpmCalculator(); /** * Returns true if the engine is not spinning (RPM==0) */ bool isStopped(DECLARE_ENGINE_PARAMETER_SIGNATURE); /** * Returns true if the engine is spinning up */ bool isSpinningUp(DECLARE_ENGINE_PARAMETER_SIGNATURE); /** * Returns true if the engine is cranking OR spinning up */ bool isCranking(DECLARE_ENGINE_PARAMETER_SIGNATURE); /** * Returns true if the engine is running and not cranking */ bool isRunning(DECLARE_ENGINE_PARAMETER_SIGNATURE); bool checkIfSpinning(DECLARE_ENGINE_PARAMETER_SIGNATURE); /** * This accessor is used in unit-tests. */ spinning_state_e getState(void); /** * Should be called on every trigger event when the engine is just starting to spin up. */ void setSpinningUp(efitime_t nowNt DECLARE_ENGINE_PARAMETER_SUFFIX); /** * Called if the synchronization is lost due to a trigger timeout. */ void setStopSpinning(DECLARE_ENGINE_PARAMETER_SIGNATURE); int getRpm(DECLARE_ENGINE_PARAMETER_SIGNATURE); /** * This method is invoked once per engine cycle right after we calculate new RPM value */ void onNewEngineCycle(); uint32_t getRevolutionCounter(void); void setRpmValue(int value DECLARE_ENGINE_PARAMETER_SUFFIX); /** * The same as setRpmValue() but without state change. * We need this to be public because of calling rpmState->assignRpmValue() from rpmShaftPositionCallback() */ void assignRpmValue(int value DECLARE_ENGINE_PARAMETER_SUFFIX); uint32_t getRevolutionCounterSinceStart(void); /** * RPM rate of change between current RPM and RPM measured during previous engine cycle * see also SC_RPM_ACCEL */ float getRpmAcceleration(); /** * This is public because sometimes we cannot afford to call isRunning() and the value is good enough * Zero if engine is not running */ volatile int rpmValue = 0; /** * this is RPM on previous engine cycle. */ int previousRpmValue = 0; /** * This is a performance optimization: let's pre-calculate this each time RPM changes * NaN while engine is not spinning */ volatile floatus_t oneDegreeUs = NAN; volatile efitime_t lastRpmEventTimeNt = 0; private: /** * Should be called once we've realized engine is not spinning any more. */ void setStopped(DECLARE_ENGINE_PARAMETER_SIGNATURE); /** * This counter is incremented with each revolution of one of the shafts. Could be * crankshaft could be camshaft. */ volatile uint32_t revolutionCounterSinceBoot = 0; /** * Same as the above, but since the engine started spinning */ volatile uint32_t revolutionCounterSinceStart = 0; spinning_state_e state = STOPPED; /** * True if the engine is spinning (regardless of its state), i.e. if shaft position changes. * Needed by spinning-up logic. */ bool isSpinning = false; }; /** * @brief Current RPM */ #define getRpmE(engine) (engine)->rpmCalculator.getRpm(PASS_ENGINE_PARAMETER_SIGNATURE) void rpmShaftPositionCallback(trigger_event_e ckpSignalType, uint32_t index DECLARE_ENGINE_PARAMETER_SUFFIX); /** * @brief Initialize RPM calculator */ void initRpmCalculator(Logging *sharedLogger, Engine *engine); float getCrankshaftAngleNt(efitime_t timeNt DECLARE_ENGINE_PARAMETER_SUFFIX); #endif int getRevolutionCounter(void); #define isValidRpm(rpm) ((rpm) > 0 && (rpm) < UNREALISTIC_RPM) #if EFI_ENGINE_SNIFFER #define addEngineSnifferEvent(name, msg) if (ENGINE(isEngineChartEnabled)) { waveChart.addEvent3((name), (msg)); } #else #define addEngineSnifferEvent(n, msg) {} #endif /* EFI_ENGINE_SNIFFER */ void scheduleByAngle(int rpm, scheduling_s *timer, angle_t angle, schfunc_t callback, void *param, RpmCalculator *calc DECLARE_ENGINE_PARAMETER_SUFFIX); #endif /* RPM_REPORTER_H_ */