unexpected contains information about why it failed (#4393)

* unexpected can contain information

* info printing

* sensors return appropriate error codes

* remove reliance on undefined behavior

* s
This commit is contained in:
Matthew Kennedy 2022-07-28 00:04:28 -07:00 committed by GitHub
parent 17b306c375
commit 14b39b7b0a
18 changed files with 122 additions and 84 deletions

View File

@ -5,16 +5,16 @@
#include "bench_test.h" #include "bench_test.h"
bool FanController::getState(bool acActive, bool lastState) { bool FanController::getState(bool acActive, bool lastState) {
auto [cltValid, clt] = Sensor::get(SensorType::Clt); auto clt = Sensor::get(SensorType::Clt);
cranking = engine->rpmCalculator.isCranking(); cranking = engine->rpmCalculator.isCranking();
notRunning = !engine->rpmCalculator.isRunning(); notRunning = !engine->rpmCalculator.isRunning();
disabledWhileEngineStopped = notRunning && disableWhenStopped(); disabledWhileEngineStopped = notRunning && disableWhenStopped();
brokenClt = !cltValid; brokenClt = !clt;
enabledForAc = enableWithAc() && acActive; enabledForAc = enableWithAc() && acActive;
hot = clt > getFanOnTemp(); hot = clt.Value > getFanOnTemp();
cold = clt < getFanOffTemp(); cold = clt.Value < getFanOffTemp();
if (cranking) { if (cranking) {
// Inhibit while cranking // Inhibit while cranking

View File

@ -55,10 +55,10 @@ static angle_t getRunningAdvance(int rpm, float engineLoad) {
engine->module<IdleController>()->isIdlingOrTaper()) { engine->module<IdleController>()->isIdlingOrTaper()) {
float idleAdvance = interpolate2d(rpm, config->idleAdvanceBins, config->idleAdvance); float idleAdvance = interpolate2d(rpm, config->idleAdvanceBins, config->idleAdvance);
auto [valid, tps] = Sensor::get(SensorType::DriverThrottleIntent); auto tps = Sensor::get(SensorType::DriverThrottleIntent);
if (valid) { if (tps) {
// interpolate between idle table and normal (running) table using TPS threshold // interpolate between idle table and normal (running) table using TPS threshold
advanceAngle = interpolateClamped(0.0f, idleAdvance, engineConfiguration->idlePidDeactivationTpsThreshold, advanceAngle, tps); advanceAngle = interpolateClamped(0.0f, idleAdvance, engineConfiguration->idlePidDeactivationTpsThreshold, advanceAngle, tps.Value);
} }
} }
@ -80,14 +80,14 @@ static angle_t getRunningAdvance(int rpm, float engineLoad) {
} }
angle_t getAdvanceCorrections(int rpm) { angle_t getAdvanceCorrections(int rpm) {
const auto [iatValid, iat] = Sensor::get(SensorType::Iat); auto iat = Sensor::get(SensorType::Iat);
if (!iatValid) { if (!iat) {
engine->engineState.timingIatCorrection = 0; engine->engineState.timingIatCorrection = 0;
} else { } else {
engine->engineState.timingIatCorrection = interpolate3d( engine->engineState.timingIatCorrection = interpolate3d(
config->ignitionIatCorrTable, config->ignitionIatCorrTable,
config->ignitionIatCorrLoadBins, iat, config->ignitionIatCorrLoadBins, iat.Value,
config->ignitionIatCorrRpmBins, rpm config->ignitionIatCorrRpmBins, rpm
); );
} }

View File

@ -5,21 +5,21 @@ bool DfcoController::getState() const {
return false; return false;
} }
const auto [tpsValid, tpsPos] = Sensor::get(SensorType::DriverThrottleIntent); const auto tps = Sensor::get(SensorType::DriverThrottleIntent);
const auto [cltValid, clt] = Sensor::get(SensorType::Clt); const auto clt = Sensor::get(SensorType::Clt);
const auto [mapValid, map] = Sensor::get(SensorType::Map); const auto map = Sensor::get(SensorType::Map);
// If some sensor is broken, inhibit DFCO // If some sensor is broken, inhibit DFCO
if (!tpsValid || !cltValid || !mapValid) { if (!tps || !clt || !map) {
return false; return false;
} }
float rpm = Sensor::getOrZero(SensorType::Rpm); float rpm = Sensor::getOrZero(SensorType::Rpm);
float vss = Sensor::getOrZero(SensorType::VehicleSpeed); float vss = Sensor::getOrZero(SensorType::VehicleSpeed);
bool mapActivate = map < engineConfiguration->coastingFuelCutMap; bool mapActivate = map.Value < engineConfiguration->coastingFuelCutMap;
bool tpsActivate = tpsPos < engineConfiguration->coastingFuelCutTps; bool tpsActivate = tps.Value < engineConfiguration->coastingFuelCutTps;
bool cltActivate = clt > engineConfiguration->coastingFuelCutClt; bool cltActivate = clt.Value > engineConfiguration->coastingFuelCutClt;
// True if throttle, MAP, and CLT are all acceptable for DFCO to occur // True if throttle, MAP, and CLT are all acceptable for DFCO to occur
bool dfcoAllowed = mapActivate && tpsActivate && cltActivate; bool dfcoAllowed = mapActivate && tpsActivate && cltActivate;

View File

@ -344,30 +344,30 @@ void initFuelMap() {
* @brief Engine warm-up fuel correction. * @brief Engine warm-up fuel correction.
*/ */
float getCltFuelCorrection() { float getCltFuelCorrection() {
const auto [valid, clt] = Sensor::get(SensorType::Clt); const auto clt = Sensor::get(SensorType::Clt);
if (!valid) if (!clt)
return 1; // this error should be already reported somewhere else, let's just handle it return 1; // this error should be already reported somewhere else, let's just handle it
return interpolate2d(clt, config->cltFuelCorrBins, config->cltFuelCorr); return interpolate2d(clt.Value, config->cltFuelCorrBins, config->cltFuelCorr);
} }
angle_t getCltTimingCorrection() { angle_t getCltTimingCorrection() {
const auto [valid, clt] = Sensor::get(SensorType::Clt); const auto clt = Sensor::get(SensorType::Clt);
if (!valid) if (!clt)
return 0; // this error should be already reported somewhere else, let's just handle it return 0; // this error should be already reported somewhere else, let's just handle it
return interpolate2d(clt, config->cltTimingBins, config->cltTimingExtra); return interpolate2d(clt.Value, config->cltTimingBins, config->cltTimingExtra);
} }
float getIatFuelCorrection() { float getIatFuelCorrection() {
const auto [valid, iat] = Sensor::get(SensorType::Iat); const auto iat = Sensor::get(SensorType::Iat);
if (!valid) if (!iat)
return 1; // this error should be already reported somewhere else, let's just handle it return 1; // this error should be already reported somewhere else, let's just handle it
return interpolate2d(iat, config->iatFuelCorrBins, config->iatFuelCorr); return interpolate2d(iat.Value, config->iatFuelCorrBins, config->iatFuelCorr);
} }
float getBaroCorrection() { float getBaroCorrection() {

View File

@ -81,17 +81,17 @@ void recalculateAuxValveTiming() {
return; return;
} }
auto [valid, tps] = Sensor::get(SensorType::DriverThrottleIntent); auto tps = Sensor::get(SensorType::DriverThrottleIntent);
if (!valid) { if (!tps) {
// error should be already reported by now // error should be already reported by now
return; return;
} }
engine->engineState.auxValveStart = interpolate2d(tps, engine->engineState.auxValveStart = interpolate2d(tps.Value,
config->scriptCurve1Bins, config->scriptCurve1Bins,
config->scriptCurve1); config->scriptCurve1);
engine->engineState.auxValveEnd = interpolate2d(tps, engine->engineState.auxValveEnd = interpolate2d(tps.Value,
config->scriptCurve2Bins, config->scriptCurve2Bins,
config->scriptCurve2); config->scriptCurve2);

View File

@ -81,11 +81,16 @@ float HpfpQuantity::calcPI(int rpm, float calc_fuel_percent) {
m_pressureTarget_kPa - (engineConfiguration->hpfpTargetDecay * m_pressureTarget_kPa - (engineConfiguration->hpfpTargetDecay *
(FAST_CALLBACK_PERIOD_MS / 1000.)), (FAST_CALLBACK_PERIOD_MS / 1000.)),
interpolate3d(engineConfiguration->hpfpTarget, interpolate3d(engineConfiguration->hpfpTarget,
engineConfiguration->hpfpTargetLoadBins, Sensor::get(SensorType::Map).Value, // TODO: allow other load axis, like we claim to engineConfiguration->hpfpTargetLoadBins, Sensor::get(SensorType::Map).value_or(0), // TODO: allow other load axis, like we claim to
engineConfiguration->hpfpTargetRpmBins, rpm)); engineConfiguration->hpfpTargetRpmBins, rpm));
auto fuelPressure = Sensor::get(SensorType::FuelPressureHigh);
if (!fuelPressure) {
return 0;
}
float pressureError_kPa = float pressureError_kPa =
m_pressureTarget_kPa - Sensor::get(SensorType::FuelPressureHigh).Value; m_pressureTarget_kPa - fuelPressure.Value;
float p_control_percent = pressureError_kPa * engineConfiguration->hpfpPidP; float p_control_percent = pressureError_kPa * engineConfiguration->hpfpPidP;
float i_factor_divisor = float i_factor_divisor =

View File

@ -12,13 +12,17 @@ public:
SensorResult convert(float frequency) const override { SensorResult convert(float frequency) const override {
// Sensor should only report 50-150hz, significantly outside that range indicates a problem // Sensor should only report 50-150hz, significantly outside that range indicates a problem
// it changes to 200hz+ to indicate methanol "contamination" // it changes to 200hz+ to indicate methanol "contamination"
if (frequency > 45 && frequency < 155) { if (frequency < 45) {
return UnexpectedCode::Low;
}
if (frequency > 155) {
return UnexpectedCode::High;
}
float flexPct = clampF(0, frequency - 50, 100); float flexPct = clampF(0, frequency - 50, 100);
return m_filter.filter(flexPct); return m_filter.filter(flexPct);
} else {
return unexpected;
}
} }
private: private:

View File

@ -16,11 +16,15 @@ void LinearFunc::configure(float in1, float out1, float in2, float out2, float m
SensorResult LinearFunc::convert(float inputValue) const { SensorResult LinearFunc::convert(float inputValue) const {
float result = m_a * inputValue + m_b; float result = m_a * inputValue + m_b;
// Bounds check // Bounds checks
bool isValid = result <= m_maxOutput && result >= m_minOutput; // Flipped error codes in case of m_a < 0 so that they indicate whether the input
// voltage is high/low, instead of the output high/low
if (result > m_maxOutput) {
return m_a > 0 ? UnexpectedCode::High : UnexpectedCode::Low;
}
if (!isValid) { if (result < m_minOutput) {
return unexpected; return m_a > 0 ? UnexpectedCode::Low : UnexpectedCode::High;
} }
return result; return result;

View File

@ -12,12 +12,12 @@ void ResistanceFunc::configure(float supplyVoltage, float pullupResistor) {
SensorResult ResistanceFunc::convert(float raw) const { SensorResult ResistanceFunc::convert(float raw) const {
// If the voltage is very low, the sensor is a dead short. // If the voltage is very low, the sensor is a dead short.
if (raw < 0.05f) { if (raw < 0.05f) {
return unexpected; return UnexpectedCode::Low;
} }
// If the voltage is very high (98% VCC), the sensor is open circuit. // If the voltage is very high (98% VCC), the sensor is open circuit.
if (raw > (m_supplyVoltage * 0.98f)) { if (raw > (m_supplyVoltage * 0.98f)) {
return unexpected; return UnexpectedCode::High;
} }
// Voltage is in a sensible range - convert // Voltage is in a sensible range - convert

View File

@ -12,7 +12,7 @@ SensorResult ThermistorFunc::convert(float ohms) const {
// This resistance should have already been validated - only // This resistance should have already been validated - only
// thing we can check is that it's non-negative // thing we can check is that it's non-negative
if (ohms <= 0) { if (ohms <= 0) {
return unexpected; return UnexpectedCode::Low;
} }
float lnR = logf(ohms); float lnR = logf(ohms);
@ -26,9 +26,14 @@ SensorResult ThermistorFunc::convert(float ohms) const {
float celsius = convertKelvinToCelcius(kelvin); float celsius = convertKelvinToCelcius(kelvin);
// bounds check result - please don't try to run rusEfi when colder than -50C // bounds check result - please don't try to run rusEfi when colder than -50C
// high end limit is required as this could be an oil temp sensor // high end limit is required as this could be an oil temp sensor on an
if (celsius < -50 || celsius > 250) { // air cooled engine
return unexpected; if (celsius < -50) {
return UnexpectedCode::Low;
}
if (celsius > 250) {
return UnexpectedCode::High;
} }
return celsius; return celsius;

View File

@ -7,7 +7,7 @@
void FunctionalSensor::postRawValue(float inputValue, efitick_t timestamp) { void FunctionalSensor::postRawValue(float inputValue, efitick_t timestamp) {
// If no function is set, this sensor isn't valid. // If no function is set, this sensor isn't valid.
if (!m_function) { if (!m_function) {
invalidate(); invalidate(UnexpectedCode::Configuration);
return; return;
} }
@ -21,6 +21,6 @@ void FunctionalSensor::postRawValue(float inputValue, efitick_t timestamp) {
if (r.Valid) { if (r.Valid) {
setValidValue(r.Value, timestamp); setValidValue(r.Value, timestamp);
} else { } else {
invalidate(); invalidate(r.Code);
} }
} }

View File

@ -23,7 +23,7 @@ SensorResult RedundantFordTps::get() const {
// If either result is invalid, return invalid. // If either result is invalid, return invalid.
if (!tps1 || !tps2) { if (!tps1 || !tps2) {
return unexpected; return UnexpectedCode::Inconsistent;
} }
// The "actual" position resolved by the second throttle - this tops out at m_secondaryMaximum instead of 100% // The "actual" position resolved by the second throttle - this tops out at m_secondaryMaximum instead of 100%
@ -37,7 +37,7 @@ SensorResult RedundantFordTps::get() const {
// Check that the resolved positions are close // Check that the resolved positions are close
float delta = absF(tps1.Value - tps2Actual); float delta = absF(tps1.Value - tps2Actual);
if (delta > m_maxDifference) { if (delta > m_maxDifference) {
return unexpected; return UnexpectedCode::Inconsistent;
} }
return (tps1.Value + tps2Actual) / 2; return (tps1.Value + tps2Actual) / 2;
@ -49,5 +49,5 @@ SensorResult RedundantFordTps::get() const {
} }
// Any other condition indicates an mismatch, and therefore an error // Any other condition indicates an mismatch, and therefore an error
return unexpected; return UnexpectedCode::Inconsistent;
} }

View File

@ -26,13 +26,13 @@ SensorResult RedundantSensor::get() const {
// If either result is invalid, return invalid. // If either result is invalid, return invalid.
if (!result1.Valid || !result2.Valid) { if (!result1.Valid || !result2.Valid) {
return unexpected; return UnexpectedCode::Inconsistent;
} }
// If both are valid, check that they're near one another // If both are valid, check that they're near one another
float delta = absF(result1.Value - result2.Value); float delta = absF(result1.Value - result2.Value);
if (delta > m_maxDifference) { if (delta > m_maxDifference) {
return unexpected; return UnexpectedCode::Inconsistent;
} }
// Both sensors are valid, and their readings are close. All is well. // Both sensors are valid, and their readings are close. All is well.

View File

@ -55,7 +55,7 @@ public:
if (s) { if (s) {
// If this sensor says it doesn't exist, return unexpected // If this sensor says it doesn't exist, return unexpected
if (!s->hasSensor()) { if (!s->hasSensor()) {
return unexpected; return UnexpectedCode::Configuration;
} }
// If we found the sensor, ask it for a result. // If we found the sensor, ask it for a result.
@ -63,7 +63,7 @@ public:
} }
// We've exhausted all valid ways to return something - sensor not found. // We've exhausted all valid ways to return something - sensor not found.
return unexpected; return UnexpectedCode::Configuration;
} }
void showInfo(const char* sensorName) const { void showInfo(const char* sensorName) const {
@ -155,7 +155,7 @@ void Sensor::unregister() {
// Check if this is a valid sensor entry // Check if this is a valid sensor entry
if (!entry) { if (!entry) {
return unexpected; return UnexpectedCode::Configuration;
} }
return entry->get(); return entry->get();

View File

@ -15,8 +15,8 @@ void ProxySensor::showInfo(const char* sensorName) const {
} }
void FunctionalSensor::showInfo(const char* sensorName) const { void FunctionalSensor::showInfo(const char* sensorName) const {
const auto [valid, value] = get(); const auto value = get();
efiPrintf("Sensor \"%s\": Raw value: %.2f Valid: %s Converted value %.2f", sensorName, m_rawValue, boolToString(valid), value); efiPrintf("Sensor \"%s\": Raw value: %.2f Valid: %s Converted value %.2f", sensorName, m_rawValue, boolToString(value.Valid), value.Value);
// now print out the underlying function's info // now print out the underlying function's info
if (auto func = m_function) { if (auto func = m_function) {
@ -28,8 +28,8 @@ void FunctionalSensor::showInfo(const char* sensorName) const {
#include "can_sensor.h" #include "can_sensor.h"
void CanSensorBase::showInfo(const char* sensorName) const { void CanSensorBase::showInfo(const char* sensorName) const {
const auto [valid, value] = get(); const auto value = get();
efiPrintf("CAN Sensor \"%s\": valid: %s value: %.2f", sensorName, boolToString(valid), value); efiPrintf("CAN Sensor \"%s\": valid: %s value: %.2f", sensorName, boolToString(value.Valid), value.Value);
} }
#endif // EFI_CAN_SUPPORT #endif // EFI_CAN_SUPPORT
@ -63,8 +63,8 @@ void Lps25Sensor::showInfo(const char* sensorName) const {
void LinearFunc::showInfo(float testRawValue) const { void LinearFunc::showInfo(float testRawValue) const {
efiPrintf(" Linear function slope: %.2f offset: %.2f min: %.1f max: %.1f", m_a, m_b, m_minOutput, m_maxOutput); efiPrintf(" Linear function slope: %.2f offset: %.2f min: %.1f max: %.1f", m_a, m_b, m_minOutput, m_maxOutput);
const auto [valid, value] = convert(testRawValue); const auto value = convert(testRawValue);
efiPrintf(" raw value %.2f converts to %.2f valid: %s", testRawValue, value, boolToString(valid)); efiPrintf(" raw value %.2f converts to %.2f valid: %s", testRawValue, value.Value, boolToString(value.Valid));
} }
void ResistanceFunc::showInfo(float testInputValue) const { void ResistanceFunc::showInfo(float testInputValue) const {
@ -73,8 +73,8 @@ void ResistanceFunc::showInfo(float testInputValue) const {
} }
void ThermistorFunc::showInfo(float testInputValue) const { void ThermistorFunc::showInfo(float testInputValue) const {
const auto [valid, value] = convert(testInputValue); const auto value = convert(testInputValue);
efiPrintf(" %.1f ohms -> valid: %s. %.1f deg C", testInputValue, boolToString(valid), value); efiPrintf(" %.1f ohms -> valid: %s. %.1f deg C", testInputValue, boolToString(value.Valid), value.Value);
} }
void IdentityFunction::showInfo(float /*testInputValue*/) const { void IdentityFunction::showInfo(float /*testInputValue*/) const {

View File

@ -30,51 +30,50 @@
class StoredValueSensor : public Sensor { class StoredValueSensor : public Sensor {
public: public:
SensorResult get() const final override { SensorResult get() const final override {
bool valid = m_isValid; auto result = m_result;
float value = m_value;
if (!valid) {
return unexpected;
}
// Timeouts are disabled, return last value // Timeouts are disabled, return last value
if (Sensor::s_inhibitSensorTimeouts) { if (Sensor::s_inhibitSensorTimeouts) {
return value; return result;
} }
if (m_timeoutPeriod != 0) { // zero m_timeoutPeriod means value lasts forever if (m_timeoutPeriod != 0) { // zero m_timeoutPeriod means value lasts forever
if (getTimeNowNt() - m_timeoutPeriod > m_lastUpdate) { if (getTimeNowNt() - m_timeoutPeriod > m_lastUpdate) {
return unexpected; return UnexpectedCode::Timeout;
} }
} }
return value; return result;
} }
StoredValueSensor(SensorType type, efitick_t timeoutNt) StoredValueSensor(SensorType type, efitick_t timeoutNt)
: Sensor(type) : Sensor(type)
, m_result(unexpected)
, m_timeoutPeriod(timeoutNt) , m_timeoutPeriod(timeoutNt)
{ {
} }
// Invalidate the stored value. // Invalidate the stored value.
void invalidate() { void invalidate() {
m_isValid = false; m_result = unexpected;
}
// Invalidate the stored value with an error code
void invalidate(UnexpectedCode why) {
m_result = why;
} }
// A new reading is available: set and validate a new value for the sensor. // A new reading is available: set and validate a new value for the sensor.
void setValidValue(float value, efitick_t timestamp) { void setValidValue(float value, efitick_t timestamp) {
// Set value before valid - so we don't briefly have the valid bit set on an invalid value // Set value before valid - so we don't briefly have the valid bit set on an invalid value
m_value = value; m_result = value;
m_isValid = true;
m_lastUpdate = timestamp; m_lastUpdate = timestamp;
} }
void showInfo(const char*) const override { } void showInfo(const char*) const override { }
private: private:
bool m_isValid = false; SensorResult m_result;
float m_value = 0.0f;
const efitick_t m_timeoutPeriod; const efitick_t m_timeoutPeriod;
efitick_t m_lastUpdate = 0; efitick_t m_lastUpdate = 0;

View File

@ -17,13 +17,35 @@
struct unexpected_t {}; struct unexpected_t {};
enum class UnexpectedCode : char {
Unknown = 0,
// Too much time has passed
Timeout,
// The decoded value was impossibly high/low
High,
Low,
// An inconsistency was detected using multiple sources of information
Inconsistent,
// A value is unavailable due to configuration
Configuration,
};
template <class TValue> template <class TValue>
struct expected { struct expected {
const bool Valid; bool Valid;
const TValue Value;
union {
TValue Value;
UnexpectedCode Code;
};
// Implicit constructor to construct in the invalid state // Implicit constructor to construct in the invalid state
constexpr expected(const unexpected_t&) : Valid(false), Value{} {} constexpr expected(const unexpected_t&) : Valid(false), Code{UnexpectedCode::Unknown} {}
constexpr expected(UnexpectedCode code) : Valid(false), Code{code} {}
// Implicit constructor to convert from TValue (for valid values, so an expected<T> behaves like a T) // Implicit constructor to convert from TValue (for valid values, so an expected<T> behaves like a T)
constexpr expected(TValue validValue) constexpr expected(TValue validValue)

View File

@ -64,6 +64,5 @@ TEST_F(SensorConverted, TestInvalid) {
{ {
auto s = Sensor::get(SensorType::Clt); auto s = Sensor::get(SensorType::Clt);
EXPECT_FALSE(s.Valid); EXPECT_FALSE(s.Valid);
EXPECT_FLOAT_EQ(s.Value, 0);
} }
} }