/** * @file can_verbose.cpp * * TODO: change 'verbose' into 'broadcast'? * * If you edit this file, please update rusEFI_CAN_verbose.dbc! * Kvaser Database Editor works well for this task, and is free. * * @author Matthew Kennedy, (c) 2020 */ #include "pch.h" #if EFI_CAN_SUPPORT #include "efi_scaled_channel.h" #include "can_msg_tx.h" #include "can.h" #include "fuel_math.h" #include "spark_logic.h" struct Status { uint16_t warningCounter; uint16_t lastErrorCode; uint8_t revLimit : 1; uint8_t mainRelay : 1; uint8_t fuelPump : 1; uint8_t checkEngine : 1; uint8_t o2Heater : 1; uint8_t lambdaProtectActive : 1; uint8_t fan : 1; uint8_t fan2 : 1; uint8_t gear; uint16_t distanceTraveled; }; static void populateFrame(Status& msg) { msg.warningCounter = engine->engineState.warnings.warningCounter; msg.lastErrorCode = static_cast(engine->engineState.warnings.lastErrorCode); msg.revLimit = Sensor::getOrZero(SensorType::Rpm) > engineConfiguration->rpmHardLimit; msg.mainRelay = enginePins.mainRelay.getLogicValue(); msg.fuelPump = enginePins.fuelPumpRelay.getLogicValue(); msg.checkEngine = enginePins.checkEnginePin.getLogicValue(); msg.o2Heater = enginePins.o2heater.getLogicValue(); #if EFI_ENGINE_CONTROL msg.lambdaProtectActive = engine->lambdaMonitor.isCut(); #endif // EFI_ENGINE_CONTROL msg.fan = enginePins.fanRelay.getLogicValue(); msg.fan2 = enginePins.fanRelay2.getLogicValue(); msg.gear = Sensor::getOrZero(SensorType::DetectedGear); // scale to units of 0.1km msg.distanceTraveled = engine->module()->getDistanceMeters() / 100; } struct Speeds { uint16_t rpm; scaled_angle timing; scaled_channel injDuty; scaled_channel coilDuty; scaled_channel vssKph; uint8_t EthanolPercent; }; static void populateFrame(Speeds& msg) { auto rpm = Sensor::getOrZero(SensorType::Rpm); msg.rpm = rpm; auto timing = engine->engineState.timingAdvance[0]; msg.timing = timing > 360 ? timing - 720 : timing; #if EFI_ENGINE_CONTROL msg.injDuty = getInjectorDutyCycle(rpm); msg.coilDuty = getCoilDutyCycle(rpm); #endif // EFI_ENGINE_CONTROL msg.vssKph = Sensor::getOrZero(SensorType::VehicleSpeed); msg.EthanolPercent = Sensor::getOrZero(SensorType::FuelEthanolPercent); } struct PedalAndTps { scaled_percent pedal; scaled_percent tps1; scaled_percent tps2; scaled_percent wastegate; }; static void populateFrame(PedalAndTps& msg) { msg.pedal = Sensor::get(SensorType::AcceleratorPedal).value_or(-1); msg.tps1 = Sensor::get(SensorType::Tps1).value_or(-1); msg.tps2 = Sensor::get(SensorType::Tps2).value_or(-1); msg.wastegate = Sensor::get(SensorType::WastegatePosition).value_or(-1); } struct Sensors1 { scaled_pressure map; scaled_channel clt; scaled_channel iat; scaled_channel aux1; scaled_channel aux2; scaled_channel mcuTemp; scaled_channel fuelLevel; }; static void populateFrame(Sensors1& msg) { msg.map = Sensor::getOrZero(SensorType::Map); msg.clt = Sensor::getOrZero(SensorType::Clt) + PACK_ADD_TEMPERATURE; msg.iat = Sensor::getOrZero(SensorType::Iat) + PACK_ADD_TEMPERATURE; msg.aux1 = Sensor::getOrZero(SensorType::AuxTemp1) + PACK_ADD_TEMPERATURE; msg.aux2 = Sensor::getOrZero(SensorType::AuxTemp2) + PACK_ADD_TEMPERATURE; #if HAL_USE_ADC msg.mcuTemp = getMCUInternalTemperature() + PACK_ADD_TEMPERATURE; #endif msg.fuelLevel = Sensor::getOrZero(SensorType::FuelLevel); } struct Sensors2 { uint8_t pad[2]; scaled_pressure oilPressure; uint8_t oilTemp; uint8_t fuelTemp; scaled_voltage vbatt; }; static void populateFrame(Sensors2& msg) { msg.oilPressure = Sensor::get(SensorType::OilPressure).value_or(-1); msg.oilTemp = Sensor::getOrZero(SensorType::OilTemperature) + PACK_ADD_TEMPERATURE; msg.fuelTemp = Sensor::getOrZero(SensorType::FuelTemperature) + PACK_ADD_TEMPERATURE; msg.vbatt = Sensor::getOrZero(SensorType::BatteryVoltage); } struct Fueling { scaled_channel cylAirmass; scaled_channel estAirflow; scaled_ms fuel_pulse; uint16_t knockCount; }; static void populateFrame(Fueling& msg) { #if EFI_ENGINE_CONTROL msg.cylAirmass = engine->fuelComputer.sdAirMassInOneCylinder; msg.estAirflow = engine->engineState.airflowEstimate; msg.fuel_pulse = (float)engine->outputChannels.actualLastInjection; msg.knockCount = engine->module()->getKnockCount(); #endif // EFI_ENGINE_CONTROL } struct Fueling2 { scaled_channel fuelConsumedGram; scaled_channel fuelFlowRate; scaled_percent fuelTrim[2]; }; static void populateFrame(Fueling2& msg) { msg.fuelConsumedGram = engine->module()->getConsumedGrams(); msg.fuelFlowRate = engine->module()->getConsumptionGramPerSecond(); for (size_t i = 0; i < 2; i++) { msg.fuelTrim[i] = 100.0f * (engine->stftCorrection[i] - 1.0f); } } struct Fueling3 { scaled_channel Lambda; scaled_channel Lambda2; scaled_channel FuelPressureLow; scaled_channel FuelPressureHigh; }; static void populateFrame(Fueling3& msg) { msg.Lambda = Sensor::getOrZero(SensorType::Lambda1); msg.Lambda2 = Sensor::getOrZero(SensorType::Lambda2); msg.FuelPressureLow = Sensor::getOrZero(SensorType::FuelPressureLow); msg.FuelPressureHigh = KPA2BAR(Sensor::getOrZero(SensorType::FuelPressureHigh)); } struct Cams { int8_t Bank1IntakeActual; int8_t Bank1IntakeTarget; int8_t Bank1ExhaustActual; int8_t Bank1ExhaustTarget; int8_t Bank2IntakeActual; int8_t Bank2IntakeTarget; int8_t Bank2ExhaustActual; int8_t Bank2ExhaustTarget; }; static void populateFrame(Cams& msg) { #if EFI_SHAFT_POSITION_INPUT msg.Bank1IntakeActual = engine->triggerCentral.getVVTPosition(0, 0); msg.Bank1ExhaustActual = engine->triggerCentral.getVVTPosition(0, 1); msg.Bank2IntakeActual = engine->triggerCentral.getVVTPosition(1, 0); msg.Bank2ExhaustActual = engine->triggerCentral.getVVTPosition(1, 1); #endif // EFI_SHAFT_POSITION_INPUT // TODO: maybe don't rely on outputChannels here msg.Bank1IntakeTarget = engine->outputChannels.vvtTargets[0]; msg.Bank1ExhaustTarget = engine->outputChannels.vvtTargets[1]; msg.Bank2IntakeTarget = engine->outputChannels.vvtTargets[2]; msg.Bank2ExhaustTarget = engine->outputChannels.vvtTargets[3]; } void sendCanVerbose() { auto base = engineConfiguration->verboseCanBaseAddress; auto isExt = engineConfiguration->rusefiVerbose29b; auto canChannel = engineConfiguration->canBroadcastUseChannelTwo; transmitStruct (CanCategory::VERBOSE, base + 0, isExt, canChannel); transmitStruct (CanCategory::VERBOSE, base + 1, isExt, canChannel); transmitStruct (CanCategory::VERBOSE, base + CAN_PEDAL_TPS_OFFSET, isExt, canChannel); transmitStruct (CanCategory::VERBOSE, base + CAN_SENSOR_1_OFFSET, isExt, canChannel); transmitStruct (CanCategory::VERBOSE, base + 4, isExt, canChannel); transmitStruct (CanCategory::VERBOSE, base + 5, isExt, canChannel); transmitStruct (CanCategory::VERBOSE, base + 6, isExt, canChannel); transmitStruct (CanCategory::VERBOSE, base + 7, isExt, canChannel); transmitStruct (CanCategory::VERBOSE, base + 8, isExt, canChannel); } #endif // EFI_CAN_SUPPORT