ecu_diagnostics 0.105.0

A rust crate for ECU diagnostic servers and communication APIs
//! OBD2 service 09 (Request vehicle information)

use crate::dynamic_diag::DynamicDiagSession;
use crate::obd2::decode_pid_response;
use crate::{DiagError, DiagServerResult};
use automotive_diag::obd2::{Obd2Command, Service09Pid, Service09PidByte};
use automotive_diag::ByteWrapper::{Extended, Standard};

#[derive(Debug)]
/// Service 09 wrapper for OBD
pub struct Service09<'a> {
    server: &'a DynamicDiagSession,
    support_list: Vec<bool>,
}

impl DynamicDiagSession {
    /// Initializes the service 09 wrapper. Automatically query's the ECU
    /// on init for supported PIDs
    pub fn obd_init_service_09(&self) -> DiagServerResult<Service09> {
        // Query supported pids
        let pid_support_list = self.send_command_with_response(Obd2Command::Service09, &[0x00])?;
        Ok(Service09 {
            server: self,
            support_list: decode_pid_response(&pid_support_list[2..]),
        })
    }
}

impl<'a> Service09<'a> {
    /// Returns a list of supported PIDs supported by the ECU for service 01
    pub fn get_supported_sids(&self) -> Vec<Service09PidByte> {
        let mut r = Vec::new();
        for (pid, supported) in self.support_list.iter().enumerate() {
            // Remember +1 as pid 0x00 is supported (Requested supported IDs
            if *supported {
                r.push(match pid + 1 {
                    0x01 => Standard(Service09Pid::VinMsgCount),
                    0x02 => Standard(Service09Pid::Vin),
                    0x03 => Standard(Service09Pid::CalibrationIDMsgCount),
                    0x04 => Standard(Service09Pid::CalibrationID),
                    0x05 => Standard(Service09Pid::CvnMsgCount),
                    0x06 => Standard(Service09Pid::Cvn),
                    0x08 => Standard(Service09Pid::InUsePerfTracking),
                    0x09 => Standard(Service09Pid::EcuNameMsgCount),
                    0x0A => Standard(Service09Pid::EcuName),
                    x => Extended(x as u8),
                })
            }
        }
        r
    }

    /// Reads the ECU's stored VIN
    pub fn read_vin(&self) -> DiagServerResult<String> {
        if !self.support_list[0x01] {
            return Err(DiagError::NotSupported); // Unsupported request
        }
        let resp = self
            .server
            .send_command_with_response(Obd2Command::Service09, &[0x02])?;
        Ok(
            String::from_utf8_lossy(resp.get(3..).ok_or(DiagError::InvalidResponseLength)?)
                .into_owned(),
        )
    }

    /// Reads the vehicles stored calibration ID (More than 1 may be returned)
    pub fn read_calibration_id(&self) -> DiagServerResult<Vec<String>> {
        if !self.support_list[0x03] {
            return Err(DiagError::NotSupported); // Unsupported request
        }
        let mut resp = self
            .server
            .send_command_with_response(Obd2Command::Service09, &[0x04])?;
        resp.drain(0..3);
        return Ok(resp
            .chunks(16)
            .map(|c| String::from_utf8_lossy(c).to_string())
            .collect());
    }

    /// Reads the vehicles stored calibration verification number (More than 1 may be returned)
    pub fn read_cvn(&self) -> DiagServerResult<Vec<String>> {
        if !self.support_list[0x05] {
            return Err(DiagError::NotSupported); // Unsupported request
        }
        let mut resp = self
            .server
            .send_command_with_response(Obd2Command::Service09, &[0x06])?;
        resp.drain(0..3);
        return Ok(resp
            .chunks(4)
            .map(|c| format!("{:02X}{:02X}{:02X}{:02X}", c[0], c[1], c[2], c[3]))
            .collect());
    }
}