From fce3d8546bd619964ec5dd1594cd93e744a99521 Mon Sep 17 00:00:00 2001 From: Diego Barrios Romero Date: Sun, 11 Nov 2018 17:32:52 +0100 Subject: Implement value conversions over type parameter --- src/conversion.rs | 92 +++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 92 insertions(+) create mode 100644 src/conversion.rs (limited to 'src/conversion.rs') diff --git a/src/conversion.rs b/src/conversion.rs new file mode 100644 index 0000000..20d04c7 --- /dev/null +++ b/src/conversion.rs @@ -0,0 +1,92 @@ + +use { ic, private, Error }; + +#[doc(hidden)] +pub trait ConvertThreshold : private::Sealed { + fn convert_threshold(value: i16) -> Result>; +} + +impl ConvertThreshold for ic::Resolution12Bit { + fn convert_threshold(value: i16) -> Result> { + if value < -2048 || value > 2047 { + return Err(Error::InvalidInputData); + } + Ok((value << 4) as u16) + } +} + +impl ConvertThreshold for ic::Resolution16Bit { + fn convert_threshold(value: i16) -> Result> { + Ok(value as u16) + } +} + +#[doc(hidden)] +pub trait ConvertMeasurement : private::Sealed { + fn convert_measurement(register_data: u16) -> i16; +} + +impl ConvertMeasurement for ic::Resolution12Bit { + fn convert_measurement(register_data: u16) -> i16 { + let value = register_data; + let is_negative = (value & 0b1000_0000_0000_0000) != 0; + if is_negative { + let value = 0b1111_0000_0000_0000 | (value >> 4); + value as i16 + } + else { + (value >> 4) as i16 + } + } +} + +impl ConvertMeasurement for ic::Resolution16Bit { + fn convert_measurement(register_data: u16) -> i16 { + register_data as i16 + } +} + + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn convert_measurement_12_bits() { + assert_eq!( 0, ic::Resolution12Bit::convert_measurement(0)); + assert_eq!( 2047, ic::Resolution12Bit::convert_measurement(0x7FFF)); + assert_eq!(-2048, ic::Resolution12Bit::convert_measurement(0x8000)); + assert_eq!( -1, ic::Resolution12Bit::convert_measurement(0xFFFF)); + } + + #[test] + fn convert_measurement_16_bits() { + assert_eq!( 0, ic::Resolution16Bit::convert_measurement(0)); + assert_eq!( 32767, ic::Resolution16Bit::convert_measurement(0x7FFF)); + assert_eq!(-32768, ic::Resolution16Bit::convert_measurement(0x8000)); + assert_eq!( -1, ic::Resolution16Bit::convert_measurement(0xFFFF)); + } + + fn assert_invalid_input_data(result: Result>) { + match result { + Err(Error::InvalidInputData) => (), + _ => panic!("InvalidInputData error was not returned.") + } + } + + #[test] + fn convert_threshold_12_bits() { + assert_invalid_input_data::<()>(ic::Resolution12Bit::convert_threshold(2048)); + assert_invalid_input_data::<()>(ic::Resolution12Bit::convert_threshold(-2049)); + assert_eq!( 0, >::convert_threshold(0).unwrap()); + assert_eq!(0x7FF0, >::convert_threshold(2047).unwrap()); + assert_eq!(0x8000, >::convert_threshold(-2048).unwrap()); + assert_eq!(0xFFF0, >::convert_threshold(-1).unwrap()); + } + + #[test] + fn convert_threshold_16_bits() { + assert_eq!(0x7FFF, >::convert_threshold(32767).unwrap()); + assert_eq!(0x8000, >::convert_threshold(-32768).unwrap()); + } +} -- cgit v1.2.3-54-g00ecf