From b3ca672f70c71194da40a7670549a8264c13d971 Mon Sep 17 00:00:00 2001 From: Markus Reiter Date: Fri, 12 Jan 2024 20:24:51 +0100 Subject: Update to `embedded-hal` 1.0. --- src/channel.rs | 98 +++++++++++++++++++++++++++++++++ src/channels.rs | 121 ----------------------------------------- src/construction.rs | 3 +- src/conversion.rs | 2 +- src/devices/common.rs | 9 ++- src/devices/mode/continuous.rs | 13 ++--- src/devices/mode/oneshot.rs | 62 ++++++++++----------- src/interface.rs | 7 ++- src/lib.rs | 9 +-- src/types.rs | 3 +- 10 files changed, 149 insertions(+), 178 deletions(-) create mode 100644 src/channel.rs delete mode 100644 src/channels.rs (limited to 'src') diff --git a/src/channel.rs b/src/channel.rs new file mode 100644 index 0000000..df23c39 --- /dev/null +++ b/src/channel.rs @@ -0,0 +1,98 @@ +//! ADC input channels +use crate::{ic, Ads1x1x, BitFlags as BF, Config}; + +mod private { + pub trait Sealed {} +} + +/// Marker type for an ADC input channel. +pub trait ChannelId: private::Sealed { + /// Get the channel. + fn channel_id() -> ChannelSelection; +} + +macro_rules! impl_channels { + ($(#[doc = $doc:expr] $CH:ident => [$($IC:ident),+]),+ $(,)?) => { + #[derive(Debug, Clone, Copy)] + /// ADC input channel selection. + pub enum ChannelSelection { + $( + #[doc = $doc] + $CH, + )+ + } + + $( + #[doc = $doc] + pub struct $CH; + + impl private::Sealed for $CH {} + + $( + impl ChannelId> for $CH { + fn channel_id() -> ChannelSelection { + ChannelSelection::$CH + } + } + )+ + )+ + }; +} + +impl_channels!( + /// Measure signal on input channel 0 differentially to signal on input channel 1. + DifferentialA0A1 => [Ads1013, Ads1014, Ads1015, Ads1113, Ads1114, Ads1115], + /// Measure signal on input channel 0 differentially to signal on input channel 3. + DifferentialA0A3 => [Ads1015, Ads1115], + /// Measure signal on input channel 1 differentially to signal on input channel 3. + DifferentialA1A3 => [Ads1015, Ads1115], + /// Measure signal on input channel 3 differentially to signal on input channel 3. + DifferentialA2A3 => [Ads1015, Ads1115], + /// Measure single-ended signal on input channel 0. + SingleA0 => [Ads1015, Ads1115], + /// Measure single-ended signal on input channel 1. + SingleA1 => [Ads1015, Ads1115], + /// Measure single-ended signal on input channel 2. + SingleA2 => [Ads1015, Ads1115], + /// Measure single-ended signal on input channel 3. + SingleA3 => [Ads1015, Ads1115] +); + +impl Config { + pub(crate) fn with_mux_bits(&self, ch: ChannelSelection) -> Self { + match ch { + ChannelSelection::DifferentialA0A1 => self + .with_low(BF::MUX2) + .with_low(BF::MUX1) + .with_low(BF::MUX0), + ChannelSelection::DifferentialA0A3 => self + .with_low(BF::MUX2) + .with_low(BF::MUX1) + .with_high(BF::MUX0), + ChannelSelection::DifferentialA1A3 => self + .with_low(BF::MUX2) + .with_high(BF::MUX1) + .with_low(BF::MUX0), + ChannelSelection::DifferentialA2A3 => self + .with_low(BF::MUX2) + .with_high(BF::MUX1) + .with_high(BF::MUX0), + ChannelSelection::SingleA0 => self + .with_high(BF::MUX2) + .with_low(BF::MUX1) + .with_low(BF::MUX0), + ChannelSelection::SingleA1 => self + .with_high(BF::MUX2) + .with_low(BF::MUX1) + .with_high(BF::MUX0), + ChannelSelection::SingleA2 => self + .with_high(BF::MUX2) + .with_high(BF::MUX1) + .with_low(BF::MUX0), + ChannelSelection::SingleA3 => self + .with_high(BF::MUX2) + .with_high(BF::MUX1) + .with_high(BF::MUX0), + } + } +} diff --git a/src/channels.rs b/src/channels.rs deleted file mode 100644 index a4c806c..0000000 --- a/src/channels.rs +++ /dev/null @@ -1,121 +0,0 @@ -//! ADC input channels -use crate::{ic, Ads1x1x, BitFlags as BF, Config}; -use embedded_hal::adc; - -/// ADC input channel selection -#[allow(dead_code)] -pub mod channel { - /// Measure single-ended signal on input channel 0 - pub struct SingleA0; - /// Measure single-ended signal on input channel 1 - pub struct SingleA1; - /// Measure single-ended signal on input channel 2 - pub struct SingleA2; - /// Measure single-ended signal on input channel 3 - pub struct SingleA3; - /// Measure signal on input channel 0 differentially to signal on input channel 1 - pub struct DifferentialA0A1; - /// Measure signal on input channel 0 differentially to signal on input channel 3 - pub struct DifferentialA0A3; - /// Measure signal on input channel 1 differentially to signal on input channel 3 - pub struct DifferentialA1A3; - /// Measure signal on input channel 3 differentially to signal on input channel 3 - pub struct DifferentialA2A3; -} - -/// ADC input channel selection -#[derive(Debug, Clone, Copy)] -pub enum ChannelSelection { - /// Measure single-ended signal on input channel 0 - SingleA0, - /// Measure single-ended signal on input channel 1 - SingleA1, - /// Measure single-ended signal on input channel 2 - SingleA2, - /// Measure single-ended signal on input channel 3 - SingleA3, - /// Measure signal on input channel 0 differentially to signal on input channel 1 - DifferentialA0A1, - /// Measure signal on input channel 0 differentially to signal on input channel 3 - DifferentialA0A3, - /// Measure signal on input channel 1 differentially to signal on input channel 3 - DifferentialA1A3, - /// Measure signal on input channel 2 differentially to signal on input channel 3 - DifferentialA2A3, -} - -macro_rules! impl_channel { - ( $IC:ident, $CH:ident ) => { - impl adc::Channel> for channel::$CH { - type ID = ChannelSelection; - - fn channel() -> Self::ID { - ChannelSelection::$CH - } - } - }; -} - -impl_channel!(Ads1013, DifferentialA0A1); -impl_channel!(Ads1113, DifferentialA0A1); - -impl_channel!(Ads1014, DifferentialA0A1); -impl_channel!(Ads1114, DifferentialA0A1); - -impl_channel!(Ads1015, DifferentialA0A1); -impl_channel!(Ads1015, DifferentialA0A3); -impl_channel!(Ads1015, DifferentialA1A3); -impl_channel!(Ads1015, DifferentialA2A3); -impl_channel!(Ads1015, SingleA0); -impl_channel!(Ads1015, SingleA1); -impl_channel!(Ads1015, SingleA2); -impl_channel!(Ads1015, SingleA3); - -impl_channel!(Ads1115, DifferentialA0A1); -impl_channel!(Ads1115, DifferentialA0A3); -impl_channel!(Ads1115, DifferentialA1A3); -impl_channel!(Ads1115, DifferentialA2A3); -impl_channel!(Ads1115, SingleA0); -impl_channel!(Ads1115, SingleA1); -impl_channel!(Ads1115, SingleA2); -impl_channel!(Ads1115, SingleA3); - -impl Config { - pub(crate) fn with_mux_bits(&self, ch: ChannelSelection) -> Self { - use self::ChannelSelection as CS; - match ch { - CS::DifferentialA0A1 => self - .with_low(BF::MUX2) - .with_low(BF::MUX1) - .with_low(BF::MUX0), - CS::DifferentialA0A3 => self - .with_low(BF::MUX2) - .with_low(BF::MUX1) - .with_high(BF::MUX0), - CS::DifferentialA1A3 => self - .with_low(BF::MUX2) - .with_high(BF::MUX1) - .with_low(BF::MUX0), - CS::DifferentialA2A3 => self - .with_low(BF::MUX2) - .with_high(BF::MUX1) - .with_high(BF::MUX0), - CS::SingleA0 => self - .with_high(BF::MUX2) - .with_low(BF::MUX1) - .with_low(BF::MUX0), - CS::SingleA1 => self - .with_high(BF::MUX2) - .with_low(BF::MUX1) - .with_high(BF::MUX0), - CS::SingleA2 => self - .with_high(BF::MUX2) - .with_high(BF::MUX1) - .with_low(BF::MUX0), - CS::SingleA3 => self - .with_high(BF::MUX2) - .with_high(BF::MUX1) - .with_high(BF::MUX0), - } - } -} diff --git a/src/construction.rs b/src/construction.rs index e217c85..d6463df 100644 --- a/src/construction.rs +++ b/src/construction.rs @@ -5,13 +5,12 @@ use crate::{ DEVICE_BASE_ADDRESS, }; use core::marker::PhantomData; -use embedded_hal::blocking; macro_rules! impl_new_destroy { ( $IC:ident, $create:ident, $destroy:ident, $conv:ty ) => { impl Ads1x1x, ic::$IC, $conv, mode::OneShot> where - I2C: blocking::i2c::Write + blocking::i2c::WriteRead, + I2C: embedded_hal::i2c::I2c, { /// Create a new instance of the device in OneShot mode. pub fn $create(i2c: I2C, address: SlaveAddr) -> Self { diff --git a/src/conversion.rs b/src/conversion.rs index d6aea82..142de0e 100644 --- a/src/conversion.rs +++ b/src/conversion.rs @@ -7,7 +7,7 @@ pub trait ConvertThreshold: private::Sealed { impl ConvertThreshold for ic::Resolution12Bit { fn convert_threshold(value: i16) -> Result> { - if value < -2048 || value > 2047 { + if !(-2048..=2047).contains(&value) { return Err(Error::InvalidInputData); } Ok((value << 4) as u16) diff --git a/src/devices/common.rs b/src/devices/common.rs index 2234d93..f1509d3 100644 --- a/src/devices/common.rs +++ b/src/devices/common.rs @@ -7,11 +7,10 @@ where DI: interface::WriteData + interface::ReadData, { pub(super) fn set_operating_mode(&mut self, mode: OperatingMode) -> Result<(), Error> { - let config; - match mode { - OperatingMode::OneShot => config = self.config.with_high(BitFlags::OP_MODE), - OperatingMode::Continuous => config = self.config.with_low(BitFlags::OP_MODE), - } + let config = match mode { + OperatingMode::OneShot => self.config.with_high(BitFlags::OP_MODE), + OperatingMode::Continuous => self.config.with_low(BitFlags::OP_MODE), + }; self.iface.write_register(Register::CONFIG, config.bits)?; self.config = config; Ok(()) diff --git a/src/devices/mode/continuous.rs b/src/devices/mode/continuous.rs index 64aa179..b3b4807 100644 --- a/src/devices/mode/continuous.rs +++ b/src/devices/mode/continuous.rs @@ -1,11 +1,10 @@ //! Continuous measurement mode use crate::{ - channels::ChannelSelection, conversion, devices::OperatingMode, interface, mode, Ads1x1x, - Error, ModeChangeError, Register, + conversion, devices::OperatingMode, interface, mode, Ads1x1x, ChannelId, Error, + ModeChangeError, Register, }; use core::marker::PhantomData; -use embedded_hal::adc; impl Ads1x1x where @@ -41,11 +40,9 @@ where /// Note that when changing the channel in continuous conversion mode, the /// ongoing conversion will be completed. /// The following conversions will use the new channel configuration. - pub fn select_channel(&mut self, _channel: &mut CH) -> Result<(), Error> - where - CH: adc::Channel, ID = ChannelSelection>, - { - let config = self.config.with_mux_bits(CH::channel()); + #[allow(unused_variables)] + pub fn select_channel>(&mut self, channel: CH) -> Result<(), Error> { + let config = self.config.with_mux_bits(CH::channel_id()); self.iface.write_register(Register::CONFIG, config.bits)?; self.config = config; Ok(()) diff --git a/src/devices/mode/oneshot.rs b/src/devices/mode/oneshot.rs index fa4d3b8..0663e4b 100644 --- a/src/devices/mode/oneshot.rs +++ b/src/devices/mode/oneshot.rs @@ -1,10 +1,9 @@ //! Common functions use crate::{ - conversion, devices::OperatingMode, interface, mode, Ads1x1x, BitFlags, ChannelSelection, - Config, DynamicOneShot, Error, ModeChangeError, Register, + conversion, devices::OperatingMode, interface, mode, Ads1x1x, BitFlags, ChannelId, + ChannelSelection, Config, DynamicOneShot, Error, ModeChangeError, Register, }; use core::marker::PhantomData; -use embedded_hal::adc; impl Ads1x1x where @@ -35,14 +34,35 @@ where } } -impl adc::OneShot, i16, CH> - for Ads1x1x +impl Ads1x1x where DI: interface::ReadData + interface::WriteData, CONV: conversion::ConvertMeasurement, - CH: adc::Channel, ID = ChannelSelection>, { - type Error = Error; + fn read_inner(&mut self, channel: ChannelSelection) -> nb::Result> { + if self + .is_measurement_in_progress() + .map_err(nb::Error::Other)? + { + return Err(nb::Error::WouldBlock); + } + let config = self.config.with_mux_bits(channel); + let same_channel = self.config == config; + if self.a_conversion_was_started && same_channel { + // result is ready + let value = self + .iface + .read_register(Register::CONVERSION) + .map_err(nb::Error::Other)?; + self.a_conversion_was_started = false; + return Ok(CONV::convert_measurement(value)); + } + self.trigger_measurement(&config) + .map_err(nb::Error::Other)?; + self.config = config; + self.a_conversion_was_started = true; + Err(nb::Error::WouldBlock) + } /// Request that the ADC begin a conversion on the specified channel. /// @@ -57,8 +77,9 @@ where /// In case a measurement was requested and after is it is finished a /// measurement on a different channel is requested, a new measurement on /// using the new channel selection is triggered. - fn read(&mut self, _channel: &mut CH) -> nb::Result { - ::read(self, CH::channel()) + #[allow(unused_variables)] + pub fn read>(&mut self, channel: CH) -> nb::Result> { + self.read_inner(CH::channel_id()) } } @@ -70,27 +91,6 @@ where type Error = Error; fn read(&mut self, channel: ChannelSelection) -> nb::Result { - if self - .is_measurement_in_progress() - .map_err(nb::Error::Other)? - { - return Err(nb::Error::WouldBlock); - } - let same_channel = self.config == self.config.with_mux_bits(channel); - if self.a_conversion_was_started && same_channel { - // result is ready - let value = self - .iface - .read_register(Register::CONVERSION) - .map_err(nb::Error::Other)?; - self.a_conversion_was_started = false; - return Ok(CONV::convert_measurement(value)); - } - let config = self.config.with_mux_bits(channel); - self.trigger_measurement(&config) - .map_err(nb::Error::Other)?; - self.config = config; - self.a_conversion_was_started = true; - Err(nb::Error::WouldBlock) + self.read_inner(channel) } } diff --git a/src/interface.rs b/src/interface.rs index 960c7ef..0e665a3 100644 --- a/src/interface.rs +++ b/src/interface.rs @@ -1,7 +1,6 @@ //! I2C interface use crate::{private, Error}; -use embedded_hal::blocking; /// I2C interface #[derive(Debug, Default)] @@ -14,13 +13,14 @@ pub struct I2cInterface { pub trait WriteData: private::Sealed { /// Error type type Error; + /// Write to an u16 register fn write_register(&mut self, register: u8, data: u16) -> Result<(), Error>; } impl WriteData for I2cInterface where - I2C: blocking::i2c::Write, + I2C: embedded_hal::i2c::I2c, { type Error = E; fn write_register(&mut self, register: u8, data: u16) -> Result<(), Error> { @@ -33,13 +33,14 @@ where pub trait ReadData: private::Sealed { /// Error type type Error; + /// Read an u16 register fn read_register(&mut self, register: u8) -> Result>; } impl ReadData for I2cInterface where - I2C: blocking::i2c::WriteRead, + I2C: embedded_hal::i2c::I2c, { type Error = E; fn read_register(&mut self, register: u8) -> Result> { diff --git a/src/lib.rs b/src/lib.rs index e5405d1..a9046ed 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -129,13 +129,12 @@ //! ### Make a one-shot measurement //! ```no_run //! use ads1x1x::{channel, Ads1x1x, SlaveAddr}; -//! use embedded_hal::adc::OneShot; //! use linux_embedded_hal::I2cdev; //! use nb::block; //! //! let dev = I2cdev::new("/dev/i2c-1").unwrap(); //! let mut adc = Ads1x1x::new_ads1013(dev, SlaveAddr::default()); -//! let measurement = block!(adc.read(&mut channel::DifferentialA0A1)).unwrap(); +//! let measurement = block!(adc.read(channel::DifferentialA0A1)).unwrap(); //! println!("Measurement: {}", measurement); //! let _dev = adc.destroy_ads1013(); // get I2C device back //! ``` @@ -201,8 +200,6 @@ //! adc.set_high_threshold_raw(1500).unwrap(); //! adc.set_comparator_latching(ComparatorLatching::Latching).unwrap(); //! ``` - -#![doc(html_root_url = "https://docs.rs/ads1x1x/0.2.2")] #![deny(unsafe_code)] #![deny(missing_docs)] #![no_std] @@ -237,8 +234,8 @@ impl BitFlags { const COMP_QUE0: u16 = 0b0000_0000_0000_0001; } -mod channels; -pub use crate::channels::{channel, ChannelSelection}; +pub mod channel; +pub use channel::{ChannelId, ChannelSelection}; mod construction; mod conversion; pub use crate::conversion::{ConvertMeasurement, ConvertThreshold}; diff --git a/src/types.rs b/src/types.rs index 957825e..e4402fe 100644 --- a/src/types.rs +++ b/src/types.rs @@ -1,8 +1,9 @@ //! Type definitions. -use crate::{channels::ChannelSelection, private}; use core::marker::PhantomData; +use crate::{private, ChannelSelection}; + /// Errors in this crate #[derive(Debug)] pub enum Error { -- cgit v1.2.3-54-g00ecf