scsir 0.3.0

A simple library for issuing SCSI commands
Documentation
use modular_bitfield_msb::prelude::*;

use crate::command::get_array;

use super::{GeneralParameter, LogParameter, ParameterHeader};

pub const ENVIRONMENTAL_LIMITS_PAGE_CODE: u8 = 0x0D;
pub const ENVIRONMENTAL_LIMITS_SUBPAGE_CODE: u8 = 0x02;

pub enum EnvironmentalLimitsParameter {
    TemperatureLimits(TemperatureLimits),
    RelativeHumidityLimits(RelativeHumidityLimits),
    Other(GeneralParameter),
}

#[bitfield]
#[derive(Clone, Copy, Debug)]
pub struct TemperatureLimits {
    pub header: ParameterHeader,
    pub high_critical_temperature_limit_trigger: B8,
    pub high_critical_temperature_limit_reset: B8,
    pub low_critical_temperature_limit_reset: B8,
    pub low_critical_temperature_limit_trigger: B8,
    pub high_operating_temperature_limit_trigger: B8,
    pub high_operating_temperature_limit_reset: B8,
    pub low_operating_temperature_limit_reset: B8,
    pub low_operating_temperature_limit_trigger: B8,
}

#[bitfield]
#[derive(Clone, Copy, Debug)]
pub struct RelativeHumidityLimits {
    pub header: ParameterHeader,
    pub high_critical_relative_humidity_limit_trigger: B8,
    pub high_critical_relative_humidity_limit_reset: B8,
    pub low_critical_relative_humidity_limit_reset: B8,
    pub low_critical_relative_humidity_limit_trigger: B8,
    pub high_operating_relative_humidity_limit_trigger: B8,
    pub high_operating_relative_humidity_limit_reset: B8,
    pub low_operating_relative_humidity_limit_reset: B8,
    pub low_operating_relative_humidity_limit_trigger: B8,
}

impl LogParameter for EnvironmentalLimitsParameter {
    fn new() -> Self {
        Self::Other(GeneralParameter::new())
    }

    fn from_bytes(bytes: &[u8]) -> (Self, &[u8]) {
        let (array, _) = get_array(bytes);
        let header = ParameterHeader::from_bytes(array);
        let result = match header.parameter_code() {
            0x0000..=0x00FF => {
                let (array, left) = get_array(bytes);
                let parameter = EnvironmentalLimitsParameter::TemperatureLimits(
                    TemperatureLimits::from_bytes(array),
                );
                (parameter, left)
            }
            0x0100..=0x01FF => {
                let (array, left) = get_array(bytes);
                let parameter = EnvironmentalLimitsParameter::RelativeHumidityLimits(
                    RelativeHumidityLimits::from_bytes(array),
                );
                (parameter, left)
            }
            _ => {
                let (parameter, left) = GeneralParameter::from_bytes(bytes);
                (EnvironmentalLimitsParameter::Other(parameter), left)
            }
        };

        result
    }

    fn to_bytes(&self) -> Vec<u8> {
        match self {
            EnvironmentalLimitsParameter::TemperatureLimits(p) => p.bytes.to_vec(),
            EnvironmentalLimitsParameter::RelativeHumidityLimits(p) => p.bytes.to_vec(),
            EnvironmentalLimitsParameter::Other(p) => p.to_bytes(),
        }
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use std::mem::size_of;

    const TEMPERATURE_LIMITS_LENGTH: usize = 12;
    const RELATIVE_HUMIDITY_LIMITS_LENGTH: usize = 12;

    #[test]
    fn layout_test() {
        assert_eq!(
            size_of::<TemperatureLimits>(),
            TEMPERATURE_LIMITS_LENGTH,
            concat!("Size of: ", stringify!(TemperatureLimits))
        );

        assert_eq!(
            size_of::<RelativeHumidityLimits>(),
            RELATIVE_HUMIDITY_LIMITS_LENGTH,
            concat!("Size of: ", stringify!(RelativeHumidityLimits))
        );
    }
}