/* * @file obd2.cpp * * ISO 15765-4 * http://en.wikipedia.org/wiki/OBD-II_PIDs * * May 2019 status: looks like this code was never tested on any real hardware? * @see obd2viaCAN.c * * @date Jun 9, 2015 * @author Andrey Belomutskiy, (c) 2012-2020 * * This file is part of rusEfi - see http://rusefi.com * * rusEfi is free software; you can redistribute it and/or modify it under the terms of * the GNU General Public License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * rusEfi is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without * even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along with this program. * If not, see . */ #include "global.h" #if EFI_CAN_SUPPORT #include "os_access.h" #include "engine.h" #include "obd2.h" #include "can_msg_tx.h" #include "vehicle_speed.h" #include "map.h" #include "maf.h" #include "tps.h" #include "engine_math.h" #include "fuel_math.h" #include "thermistors.h" EXTERN_ENGINE ; static LoggingWithStorage logger("obd2"); static const int16_t supportedPids0120[] = { PID_MONITOR_STATUS, PID_FUEL_SYSTEM_STATUS, PID_ENGINE_LOAD, PID_COOLANT_TEMP, PID_INTAKE_MAP, PID_RPM, PID_SPEED, PID_TIMING_ADVANCE, PID_INTAKE_TEMP, PID_THROTTLE, -1 }; static const int16_t supportedPids2140[] = { -1 }; static const int16_t supportedPids4160[] = { PID_FUEL_RATE, -1 }; static void obdSendPacket(int mode, int PID, int numBytes, uint32_t iValue) { CanTxMessage resp(OBD_TEST_RESPONSE); // write number of bytes resp[0] = (uint8_t)(2 + numBytes); // write 2 bytes of header resp[1] = (uint8_t)(0x40 + mode); resp[2] = (uint8_t)PID; // write N data bytes for (int i = 8 * (numBytes - 1), j = 3; i >= 0; i -= 8, j++) { resp[j] = (uint8_t)((iValue >> i) & 0xff); } } static void obdSendValue(int mode, int PID, int numBytes, float value) { efiAssertVoid(CUSTOM_ERR_6662, numBytes <= 2, "invalid numBytes"); int iValue = (int)efiRound(value, 1.0f); // clamp to uint8_t (0..255) or uint16_t (0..65535) iValue = maxI(minI(iValue, (numBytes == 1) ? 255 : 65535), 0); obdSendPacket(mode, PID, numBytes, iValue); } //#define MOCK_SUPPORTED_PIDS 0xffffffff static void obdWriteSupportedPids(int PID, int bitOffset, const int16_t *supportedPids) { uint32_t value = 0; // gather all 32 bit fields for (int i = 0; i < 32 && supportedPids[i] > 0; i++) value |= 1 << (31 + bitOffset - supportedPids[i]); #ifdef MOCK_SUPPORTED_PIDS // for OBD debug value = MOCK_SUPPORTED_PIDS; #endif scheduleMsg(&logger, "Write bitfields 0x%08x", value); obdSendPacket(1, PID, 4, value); } static void handleGetDataRequest(CANRxFrame *rx) { int pid = rx->data8[2]; switch (pid) { case PID_SUPPORTED_PIDS_REQUEST_01_20: scheduleMsg(&logger, "Got lookup request 01-20"); obdWriteSupportedPids(pid, 1, supportedPids0120); break; case PID_SUPPORTED_PIDS_REQUEST_21_40: scheduleMsg(&logger, "Got lookup request 21-40"); obdWriteSupportedPids(pid, 21, supportedPids2140); break; case PID_SUPPORTED_PIDS_REQUEST_41_60: scheduleMsg(&logger, "Got lookup request 41-60"); obdWriteSupportedPids(pid, 41, supportedPids4160); break; case PID_MONITOR_STATUS: scheduleMsg(&logger, "Got monitor status request"); obdSendPacket(1, pid, 4, 0); // todo: add statuses break; case PID_FUEL_SYSTEM_STATUS: scheduleMsg(&logger, "Got fuel system status request"); // todo: add statuses obdSendValue(1, pid, 2, (2<<8)|(0)); // 2 = "Closed loop, using oxygen sensor feedback to determine fuel mix" break; case PID_ENGINE_LOAD: scheduleMsg(&logger, "Got engine load request"); obdSendValue(1, pid, 1, getEngineLoadT(PASS_ENGINE_PARAMETER_SIGNATURE) * 2.55f); break; case PID_COOLANT_TEMP: scheduleMsg(&logger, "Got CLT request"); obdSendValue(1, pid, 1, getCoolantTemperature() + 40.0f); break; case PID_INTAKE_MAP: scheduleMsg(&logger, "Got MAP request"); obdSendValue(1, pid, 1, getMap(PASS_ENGINE_PARAMETER_SIGNATURE)); break; case PID_RPM: scheduleMsg(&logger, "Got RPM request"); obdSendValue(1, pid, 2, GET_RPM() * 4.0f); // rotation/min. (A*256+B)/4 break; case PID_SPEED: scheduleMsg(&logger, "Got speed request"); obdSendValue(1, pid, 1, getVehicleSpeed()); break; case PID_TIMING_ADVANCE: { scheduleMsg(&logger, "Got timing request"); float timing = engine->engineState.timingAdvance; timing = (timing > 360.0f) ? (timing - 720.0f) : timing; obdSendValue(1, pid, 1, (timing + 64.0f) * 2.0f); // angle before TDC. (A/2)-64 break; } case PID_INTAKE_TEMP: scheduleMsg(&logger, "Got IAT request"); obdSendValue(1, pid, 1, getIntakeAirTemperature() + 40.0f); break; case PID_INTAKE_MAF: scheduleMsg(&logger, "Got MAF request"); obdSendValue(1, pid, 2, getRealMaf(PASS_ENGINE_PARAMETER_SIGNATURE) * 100.0f); // grams/sec (A*256+B)/100 break; case PID_THROTTLE: scheduleMsg(&logger, "Got throttle request"); obdSendValue(1, pid, 1, getTPS(PASS_ENGINE_PARAMETER_SIGNATURE) * 2.55f); // (A*100/255) break; case PID_FUEL_RATE: scheduleMsg(&logger, "Got fuel rate request"); obdSendValue(1, pid, 2, engine->engineState.fuelConsumption.perSecondConsumption * 20.0f); // L/h. (A*256+B)/20 break; default: scheduleMsg(&logger, "Got unhandled request (PID 0x%02x)", pid); } } static void handleDtcRequest(int numCodes, int *dtcCode) { // TODO: this appears to be unfinished? // int numBytes = numCodes * 2; // // write CAN-TP Single Frame header? // txmsg.data8[0] = (uint8_t)((0 << 4) | numBytes); // for (int i = 0, j = 1; i < numCodes; i++) { // txmsg.data8[j++] = (uint8_t)((dtcCode[i] >> 8) & 0xff); // txmsg.data8[j++] = (uint8_t)(dtcCode[i] & 0xff); // } } #if HAL_USE_CAN void obdOnCanPacketRx(CANRxFrame *rx) { if (rx->SID != OBD_TEST_REQUEST) { return; } if (rx->data8[0] == 2 && rx->data8[1] == OBD_CURRENT_DATA) { handleGetDataRequest(rx); } else if (rx->data8[0] == 1 && rx->data8[1] == OBD_STORED_DIAGNOSTIC_TROUBLE_CODES) { scheduleMsg(&logger, "Got stored DTC request"); // todo: implement stored/pending difference? handleDtcRequest(1, &engine->engineState.warnings.lastErrorCode); } else if (rx->data8[0] == 1 && rx->data8[1] == OBD_PENDING_DIAGNOSTIC_TROUBLE_CODES) { scheduleMsg(&logger, "Got pending DTC request"); // todo: implement stored/pending difference? handleDtcRequest(1, &engine->engineState.warnings.lastErrorCode); } else { scheduleMsg(&logger, "Got unhandled OBD message"); } } #endif /* HAL_USE_CAN */ #endif /* EFI_CAN_SUPPORT */