rusefi-full/firmware/controllers/limp_manager.h

127 lines
2.6 KiB
C++

#pragma once
#include "rusefi_types.h"
#include <cstdint>
// Keep this list in sync with fuelIgnCutCodeList in rusefi.input!
enum class ClearReason : uint8_t {
None, // 0
Fatal,
Settings,
HardLimit, // 3
FaultRevLimit,
BoostCut, // 5
OilPressure,
StopRequested, // 7
EtbProblem, // 8
LaunchCut, // 9
InjectorDutyCycle, // 10
FloodClear, // 11
EnginePhase, // 12
KickStart, // 13
// Keep this list in sync with fuelIgnCutCodeList in rusefi.input!
};
// Only allows clearing the value, but never resetting it.
class Clearable {
public:
Clearable() : m_value(true) {}
Clearable(bool value) : m_value(value) {
if (!m_value) {
clearReason = ClearReason::Settings;
}
}
void clear(ClearReason clearReason) {
if (m_value) {
m_value = false;
this->clearReason = clearReason;
}
}
operator bool() const {
return m_value;
}
ClearReason clearReason = ClearReason::None;
private:
bool m_value = true;
};
struct LimpState {
const bool value;
const ClearReason reason;
// Implicit conversion operator to bool, so you can do things like if (myResult) { ... }
constexpr explicit operator bool() const {
return value;
}
};
class Hysteresis {
public:
// returns true if value > rising, false if value < falling, previous if falling < value < rising.
bool test(float value, float rising, float falling) {
if (value > rising) {
m_state = true;
} else if (value < falling) {
m_state = false;
}
return m_state;
}
private:
bool m_state = false;
};
class LimpManager {
public:
// This is called from periodicFastCallback to update internal state
void updateState(int rpm, efitick_t nowNt);
// Other subsystems call these APIs to determine their behavior
bool allowElectronicThrottle() const;
LimpState allowInjection() const;
LimpState allowIgnition() const;
bool allowTriggerInput() const;
// Other subsystems call these APIs to indicate a problem has occured
void etbProblem();
void fatalError();
void stopEngine();
bool isEngineStop(efitick_t nowNt) const;
float getTimeSinceEngineStop(efitick_t nowNt) const;
private:
void setFaultRevLimit(int limit);
Hysteresis m_revLimitHysteresis;
Hysteresis m_boostCutHysteresis;
Hysteresis m_injectorDutyCutHysteresis;
// Start with no fault rev limit
int32_t m_faultRevLimit = INT32_MAX;
Clearable m_allowEtb;
Clearable m_allowInjection;
Clearable m_allowIgnition;
Clearable m_allowTriggerInput;
Clearable m_transientAllowInjection = true;
Clearable m_transientAllowIgnition = true;
bool m_hadOilPressureAfterStart = false;
Timer m_engineStopTimer;
};
LimpManager * getLimpManager();