#include "pch.h" // Decode what OBD code we should use for a particular [sensor, code] problem static ObdCode getCode(SensorType type, UnexpectedCode code) { switch (type) { case SensorType::Tps1: case SensorType::Tps1Primary: switch (code) { case UnexpectedCode::Timeout: return ObdCode::OBD_TPS1_Primary_Timeout; case UnexpectedCode::Low: return ObdCode::OBD_TPS1_Primary_Low; case UnexpectedCode::High: return ObdCode::OBD_TPS1_Primary_High; case UnexpectedCode::Inconsistent: return ObdCode::OBD_TPS1_Correlation; default: break; } break; case SensorType::Tps1Secondary: switch (code) { case UnexpectedCode::Timeout: return ObdCode::OBD_TPS1_Secondary_Timeout; case UnexpectedCode::Low: return ObdCode::OBD_TPS1_Secondary_Low; case UnexpectedCode::High: return ObdCode::OBD_TPS1_Secondary_High; default: break; } break; case SensorType::Tps2: case SensorType::Tps2Primary: switch (code) { case UnexpectedCode::Timeout: return ObdCode::OBD_TPS2_Primary_Timeout; case UnexpectedCode::Low: return ObdCode::OBD_TPS2_Primary_Low; case UnexpectedCode::High: return ObdCode::OBD_TPS2_Primary_High; case UnexpectedCode::Inconsistent: return ObdCode::OBD_TPS2_Correlation; default: break; } break; case SensorType::Tps2Secondary: switch (code) { case UnexpectedCode::Timeout: return ObdCode::OBD_TPS2_Secondary_Timeout; case UnexpectedCode::Low: return ObdCode::OBD_TPS2_Secondary_Low; case UnexpectedCode::High: return ObdCode::OBD_TPS2_Secondary_High; default: break; } break; case SensorType::AcceleratorPedal: case SensorType::AcceleratorPedalPrimary: switch (code) { case UnexpectedCode::Timeout: return ObdCode::OBD_PPS_Primary_Timeout; case UnexpectedCode::Low: return ObdCode::OBD_PPS_Primary_Low; case UnexpectedCode::High: return ObdCode::OBD_PPS_Primary_High; case UnexpectedCode::Inconsistent: return ObdCode::OBD_PPS_Correlation; default: break; } break; case SensorType::AcceleratorPedalSecondary: switch (code) { case UnexpectedCode::Timeout: return ObdCode::OBD_PPS_Secondary_Timeout; case UnexpectedCode::Low: return ObdCode::OBD_PPS_Secondary_Low; case UnexpectedCode::High: return ObdCode::OBD_PPS_Secondary_High; default: break; } break; case SensorType::Map: switch (code) { case UnexpectedCode::Timeout: return ObdCode::OBD_Map_Timeout; case UnexpectedCode::Low: return ObdCode::OBD_Map_Low; case UnexpectedCode::High: return ObdCode::OBD_Map_High; default: break; } break; case SensorType::Clt: switch (code) { case UnexpectedCode::Timeout: return ObdCode::OBD_Clt_Timeout; case UnexpectedCode::Low: return ObdCode::OBD_Clt_Low; case UnexpectedCode::High: return ObdCode::OBD_Clt_High; default: break; } break; case SensorType::Iat: switch (code) { case UnexpectedCode::Timeout: return ObdCode::OBD_Iat_Timeout; case UnexpectedCode::Low: return ObdCode::OBD_Iat_Low; case UnexpectedCode::High: return ObdCode::OBD_Iat_High; default: break; } break; case SensorType::FuelEthanolPercent: switch (code) { case UnexpectedCode::Timeout: return ObdCode::OBD_FlexSensor_Timeout; case UnexpectedCode::Low: return ObdCode::OBD_FlexSensor_Low; case UnexpectedCode::High: return ObdCode::OBD_FlexSensor_High; default: break; } break; default: break; } return ObdCode::None; } inline const char* describeUnexpected(UnexpectedCode code) { switch (code) { case UnexpectedCode::Timeout: return "has timed out"; case UnexpectedCode::High: return "input too high"; case UnexpectedCode::Low: return "input too low"; case UnexpectedCode::Inconsistent: return "is inconsistent"; case UnexpectedCode::Configuration: return "is misconfigured"; case UnexpectedCode::Unknown: default: return "unknown"; } } static void check(SensorType type) { // Don't check sensors we don't have if (!Sensor::hasSensor(type)) { return; } auto result = Sensor::get(type); // If the sensor is OK, nothing to check. if (result) { return; } ObdCode code = getCode(type, result.Code); if (code != ObdCode::None) { warning(code, "Sensor fault: %s %s", Sensor::getSensorName(type), describeUnexpected(result.Code)); } } static ObdCode getCodeForInjector(int idx, brain_pin_diag_e diag) { if (idx < 0 || idx >= 12) { return ObdCode::None; } // TODO: do something more intelligent with `diag`? UNUSED(diag); return (ObdCode)((int)ObdCode::OBD_Injector_Circuit_1 + idx); } static ObdCode getCodeForIgnition(int idx, brain_pin_diag_e diag) { if (idx < 0 || idx >= 12) { return ObdCode::None; } // TODO: do something more intelligent with `diag`? UNUSED(diag); return (ObdCode)((int)ObdCode::OBD_Ignition_Circuit_1 + idx); } void SensorChecker::onSlowCallback() { // Don't check when the ignition is off, or when it was just turned on (let things stabilize) // TODO: also inhibit checking if we just did a flash burn, since that blocks the ECU for a few seconds. bool shouldCheck = m_ignitionIsOn && m_timeSinceIgnOff.hasElapsedSec(5); m_analogSensorsShouldWork = shouldCheck; if (!shouldCheck) { return; } // Check sensors check(SensorType::Tps1Primary); check(SensorType::Tps1Secondary); check(SensorType::Tps1); check(SensorType::Tps2Primary); check(SensorType::Tps2Secondary); check(SensorType::Tps2); check(SensorType::AcceleratorPedalPrimary); check(SensorType::AcceleratorPedalSecondary); check(SensorType::AcceleratorPedal); check(SensorType::Map); check(SensorType::Map2); check(SensorType::Clt); check(SensorType::Iat); check(SensorType::FuelEthanolPercent); // only bother checking these if we have GPIO chips actually capable of reporting an error #if BOARD_EXT_GPIOCHIPS > 0 && EFI_PROD_CODE // Check injectors bool withInjectorIssues = false; for (size_t i = 0; i < efi::size(enginePins.injectors); i++) { InjectorOutputPin& pin = enginePins.injectors[i]; // Skip not-configured pins if (!isBrainPinValid(pin.brainPin)) { continue; } auto diag = pin.getDiag(); if (diag != PIN_OK && diag != PIN_INVALID) { withInjectorIssues = true; auto code = getCodeForInjector(i + 1, diag); char description[32]; pinDiag2string(description, efi::size(description), diag); warning(code, "Injector %d fault: %s", i, description); } } engine->fuelComputer.injectorHwIssue = withInjectorIssues; // Check ignition for (size_t i = 0; i < efi::size(enginePins.injectors); i++) { IgnitionOutputPin& pin = enginePins.coils[i]; // Skip not-configured pins if (!isBrainPinValid(pin.brainPin)) { continue; } auto diag = pin.getDiag(); if (diag != PIN_OK && diag != PIN_INVALID) { auto code = getCodeForIgnition(i + 1, diag); char description[32]; pinDiag2string(description, efi::size(description), diag); warning(code, "Ignition %d fault: %s", i, description); } } #endif // BOARD_EXT_GPIOCHIPS > 0 } void SensorChecker::onIgnitionStateChanged(bool ignitionOn) { m_ignitionIsOn = ignitionOn; if (!ignitionOn) { // timer keeps track of how long since the state was turned to on (ie, how long ago was it last off) m_timeSinceIgnOff.reset(); } }