add i2c trait
This commit is contained in:
parent
88946840d1
commit
492f7aeea6
2 changed files with 411 additions and 0 deletions
409
embassy-traits/src/i2c.rs
Normal file
409
embassy-traits/src/i2c.rs
Normal file
|
@ -0,0 +1,409 @@
|
||||||
|
//! Blocking I2C API
|
||||||
|
//!
|
||||||
|
//! This API supports 7-bit and 10-bit addresses. Traits feature an `AddressMode`
|
||||||
|
//! marker type parameter. Two implementation of the `AddressMode` exist:
|
||||||
|
//! `SevenBitAddress` and `TenBitAddress`.
|
||||||
|
//!
|
||||||
|
//! Through this marker types it is possible to implement each address mode for
|
||||||
|
//! the traits independently in `embedded-hal` implementations and device drivers
|
||||||
|
//! can depend only on the mode that they support.
|
||||||
|
//!
|
||||||
|
//! Additionally, the I2C 10-bit address mode has been developed to be fully
|
||||||
|
//! backwards compatible with the 7-bit address mode. This allows for a
|
||||||
|
//! software-emulated 10-bit addressing implementation if the address mode
|
||||||
|
//! is not supported by the hardware.
|
||||||
|
//!
|
||||||
|
//! Since 7-bit addressing is the mode of the majority of I2C devices,
|
||||||
|
//! `SevenBitAddress` has been set as default mode and thus can be omitted if desired.
|
||||||
|
//!
|
||||||
|
//! ## Examples
|
||||||
|
//!
|
||||||
|
//! ### `embedded-hal` implementation for an MCU
|
||||||
|
//! Here is an example of an embedded-hal implementation of the `Write` trait
|
||||||
|
//! for both modes:
|
||||||
|
//! ```
|
||||||
|
//! # use embedded_hal::blocking::i2c::{SevenBitAddress, TenBitAddress, Write};
|
||||||
|
//! /// I2C0 hardware peripheral which supports both 7-bit and 10-bit addressing.
|
||||||
|
//! pub struct I2c0;
|
||||||
|
//!
|
||||||
|
//! impl Write<SevenBitAddress> for I2c0
|
||||||
|
//! {
|
||||||
|
//! # type Error = ();
|
||||||
|
//! #
|
||||||
|
//! fn try_write(&mut self, addr: u8, output: &[u8]) -> Result<(), Self::Error> {
|
||||||
|
//! // ...
|
||||||
|
//! # Ok(())
|
||||||
|
//! }
|
||||||
|
//! }
|
||||||
|
//!
|
||||||
|
//! impl Write<TenBitAddress> for I2c0
|
||||||
|
//! {
|
||||||
|
//! # type Error = ();
|
||||||
|
//! #
|
||||||
|
//! fn try_write(&mut self, addr: u16, output: &[u8]) -> Result<(), Self::Error> {
|
||||||
|
//! // ...
|
||||||
|
//! # Ok(())
|
||||||
|
//! }
|
||||||
|
//! }
|
||||||
|
//! ```
|
||||||
|
//!
|
||||||
|
//! ### Device driver compatible only with 7-bit addresses
|
||||||
|
//!
|
||||||
|
//! For demonstration purposes the address mode parameter has been omitted in this example.
|
||||||
|
//!
|
||||||
|
//! ```
|
||||||
|
//! # use embedded_hal::blocking::i2c::WriteRead;
|
||||||
|
//! const ADDR: u8 = 0x15;
|
||||||
|
//! # const TEMP_REGISTER: u8 = 0x1;
|
||||||
|
//! pub struct TemperatureSensorDriver<I2C> {
|
||||||
|
//! i2c: I2C,
|
||||||
|
//! }
|
||||||
|
//!
|
||||||
|
//! impl<I2C, E> TemperatureSensorDriver<I2C>
|
||||||
|
//! where
|
||||||
|
//! I2C: WriteRead<Error = E>,
|
||||||
|
//! {
|
||||||
|
//! pub fn read_temperature(&mut self) -> Result<u8, E> {
|
||||||
|
//! let mut temp = [0];
|
||||||
|
//! self.i2c
|
||||||
|
//! .try_write_read(ADDR, &[TEMP_REGISTER], &mut temp)
|
||||||
|
//! .and(Ok(temp[0]))
|
||||||
|
//! }
|
||||||
|
//! }
|
||||||
|
//! ```
|
||||||
|
//!
|
||||||
|
//! ### Device driver compatible only with 10-bit addresses
|
||||||
|
//!
|
||||||
|
//! ```
|
||||||
|
//! # use embedded_hal::blocking::i2c::{TenBitAddress, WriteRead};
|
||||||
|
//! const ADDR: u16 = 0x158;
|
||||||
|
//! # const TEMP_REGISTER: u8 = 0x1;
|
||||||
|
//! pub struct TemperatureSensorDriver<I2C> {
|
||||||
|
//! i2c: I2C,
|
||||||
|
//! }
|
||||||
|
//!
|
||||||
|
//! impl<I2C, E> TemperatureSensorDriver<I2C>
|
||||||
|
//! where
|
||||||
|
//! I2C: WriteRead<TenBitAddress, Error = E>,
|
||||||
|
//! {
|
||||||
|
//! pub fn read_temperature(&mut self) -> Result<u8, E> {
|
||||||
|
//! let mut temp = [0];
|
||||||
|
//! self.i2c
|
||||||
|
//! .try_write_read(ADDR, &[TEMP_REGISTER], &mut temp)
|
||||||
|
//! .and(Ok(temp[0]))
|
||||||
|
//! }
|
||||||
|
//! }
|
||||||
|
//! ```
|
||||||
|
|
||||||
|
use core::future::Future;
|
||||||
|
|
||||||
|
mod private {
|
||||||
|
pub trait Sealed {}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Address mode (7-bit / 10-bit)
|
||||||
|
///
|
||||||
|
/// Note: This trait is sealed and should not be implemented outside of this crate.
|
||||||
|
pub trait AddressMode: private::Sealed {}
|
||||||
|
|
||||||
|
/// 7-bit address mode type
|
||||||
|
pub type SevenBitAddress = u8;
|
||||||
|
|
||||||
|
/// 10-bit address mode type
|
||||||
|
pub type TenBitAddress = u16;
|
||||||
|
|
||||||
|
impl private::Sealed for SevenBitAddress {}
|
||||||
|
impl private::Sealed for TenBitAddress {}
|
||||||
|
|
||||||
|
impl AddressMode for SevenBitAddress {}
|
||||||
|
|
||||||
|
impl AddressMode for TenBitAddress {}
|
||||||
|
|
||||||
|
/// Blocking read
|
||||||
|
pub trait Read<A: AddressMode = SevenBitAddress> {
|
||||||
|
/// Error type
|
||||||
|
type Error;
|
||||||
|
|
||||||
|
type ReadFuture<'a>: Future<Output = Result<(), Self::Error>> + 'a;
|
||||||
|
|
||||||
|
/// Reads enough bytes from slave with `address` to fill `buffer`
|
||||||
|
///
|
||||||
|
/// # I2C Events (contract)
|
||||||
|
///
|
||||||
|
/// ``` text
|
||||||
|
/// Master: ST SAD+R MAK MAK ... NMAK SP
|
||||||
|
/// Slave: SAK B0 B1 ... BN
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// Where
|
||||||
|
///
|
||||||
|
/// - `ST` = start condition
|
||||||
|
/// - `SAD+R` = slave address followed by bit 1 to indicate reading
|
||||||
|
/// - `SAK` = slave acknowledge
|
||||||
|
/// - `Bi` = ith byte of data
|
||||||
|
/// - `MAK` = master acknowledge
|
||||||
|
/// - `NMAK` = master no acknowledge
|
||||||
|
/// - `SP` = stop condition
|
||||||
|
fn read<'a>(&mut self, address: A, buffer: &mut [u8]) -> Self::ReadFuture<'a>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Blocking write
|
||||||
|
pub trait Write<A: AddressMode = SevenBitAddress> {
|
||||||
|
/// Error type
|
||||||
|
type Error;
|
||||||
|
|
||||||
|
type WriteFuture<'a>: Future<Output = Result<(), Self::Error>> + 'a;
|
||||||
|
|
||||||
|
/// Sends bytes to slave with address `address`
|
||||||
|
///
|
||||||
|
/// # I2C Events (contract)
|
||||||
|
///
|
||||||
|
/// ``` text
|
||||||
|
/// Master: ST SAD+W B0 B1 ... BN SP
|
||||||
|
/// Slave: SAK SAK SAK ... SAK
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// Where
|
||||||
|
///
|
||||||
|
/// - `ST` = start condition
|
||||||
|
/// - `SAD+W` = slave address followed by bit 0 to indicate writing
|
||||||
|
/// - `SAK` = slave acknowledge
|
||||||
|
/// - `Bi` = ith byte of data
|
||||||
|
/// - `SP` = stop condition
|
||||||
|
fn write<'a>(&mut self, address: A, bytes: &[u8]) -> Self::WriteFuture<'a>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Blocking write (iterator version)
|
||||||
|
pub trait WriteIter<A: AddressMode = SevenBitAddress> {
|
||||||
|
/// Error type
|
||||||
|
type Error;
|
||||||
|
|
||||||
|
type WriteIterFuture<'a>: Future<Output = Result<(), Self::Error>> + 'a;
|
||||||
|
|
||||||
|
/// Sends bytes to slave with address `address`
|
||||||
|
///
|
||||||
|
/// # I2C Events (contract)
|
||||||
|
///
|
||||||
|
/// Same as `Write`
|
||||||
|
fn write_iter<'a, B>(&mut self, address: A, bytes: B) -> Self::WriteIterFuture<'a>
|
||||||
|
where
|
||||||
|
B: IntoIterator<Item = u8>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Blocking write + read
|
||||||
|
pub trait WriteRead<A: AddressMode = SevenBitAddress> {
|
||||||
|
/// Error type
|
||||||
|
type Error;
|
||||||
|
|
||||||
|
type WriteReadFuture<'a>: Future<Output = Result<(), Self::Error>> + 'a;
|
||||||
|
|
||||||
|
/// Sends bytes to slave with address `address` and then reads enough bytes to fill `buffer` *in a
|
||||||
|
/// single transaction*
|
||||||
|
///
|
||||||
|
/// # I2C Events (contract)
|
||||||
|
///
|
||||||
|
/// ``` text
|
||||||
|
/// Master: ST SAD+W O0 O1 ... OM SR SAD+R MAK MAK ... NMAK SP
|
||||||
|
/// Slave: SAK SAK SAK ... SAK SAK I0 I1 ... IN
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// Where
|
||||||
|
///
|
||||||
|
/// - `ST` = start condition
|
||||||
|
/// - `SAD+W` = slave address followed by bit 0 to indicate writing
|
||||||
|
/// - `SAK` = slave acknowledge
|
||||||
|
/// - `Oi` = ith outgoing byte of data
|
||||||
|
/// - `SR` = repeated start condition
|
||||||
|
/// - `SAD+R` = slave address followed by bit 1 to indicate reading
|
||||||
|
/// - `Ii` = ith incoming byte of data
|
||||||
|
/// - `MAK` = master acknowledge
|
||||||
|
/// - `NMAK` = master no acknowledge
|
||||||
|
/// - `SP` = stop condition
|
||||||
|
fn write_read<'a>(
|
||||||
|
&mut self,
|
||||||
|
address: A,
|
||||||
|
bytes: &[u8],
|
||||||
|
buffer: &mut [u8],
|
||||||
|
) -> Self::WriteReadFuture<'a>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Blocking write (iterator version) + read
|
||||||
|
pub trait WriteIterRead<A: AddressMode = SevenBitAddress> {
|
||||||
|
/// Error type
|
||||||
|
type Error;
|
||||||
|
|
||||||
|
type WriteIterReadFuture<'a>: Future<Output = Result<(), Self::Error>> + 'a;
|
||||||
|
|
||||||
|
/// Sends bytes to slave with address `address` and then reads enough bytes to fill `buffer` *in a
|
||||||
|
/// single transaction*
|
||||||
|
///
|
||||||
|
/// # I2C Events (contract)
|
||||||
|
///
|
||||||
|
/// Same as the `WriteRead` trait
|
||||||
|
fn write_iter_read<'a, B>(
|
||||||
|
&mut self,
|
||||||
|
address: A,
|
||||||
|
bytes: B,
|
||||||
|
buffer: &mut [u8],
|
||||||
|
) -> Self::WriteIterReadFuture<'a>
|
||||||
|
where
|
||||||
|
B: IntoIterator<Item = u8>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Transactional I2C operation.
|
||||||
|
///
|
||||||
|
/// Several operations can be combined as part of a transaction.
|
||||||
|
#[derive(Debug, PartialEq)]
|
||||||
|
pub enum Operation<'a> {
|
||||||
|
/// Read data into the provided buffer
|
||||||
|
Read(&'a mut [u8]),
|
||||||
|
/// Write data from the provided buffer
|
||||||
|
Write(&'a [u8]),
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Transactional I2C interface.
|
||||||
|
///
|
||||||
|
/// This allows combining operations within an I2C transaction.
|
||||||
|
pub trait Transactional<A: AddressMode = SevenBitAddress> {
|
||||||
|
/// Error type
|
||||||
|
type Error;
|
||||||
|
|
||||||
|
type TransactionalFuture<'a>: Future<Output = Result<(), Self::Error>> + 'a;
|
||||||
|
|
||||||
|
/// Execute the provided operations on the I2C bus.
|
||||||
|
///
|
||||||
|
/// Transaction contract:
|
||||||
|
/// - Before executing the first operation an ST is sent automatically. This is followed by SAD+R/W as appropriate.
|
||||||
|
/// - Data from adjacent operations of the same type are sent after each other without an SP or SR.
|
||||||
|
/// - Between adjacent operations of a different type an SR and SAD+R/W is sent.
|
||||||
|
/// - After executing the last operation an SP is sent automatically.
|
||||||
|
/// - If the last operation is a `Read` the master does not send an acknowledge for the last byte.
|
||||||
|
///
|
||||||
|
/// - `ST` = start condition
|
||||||
|
/// - `SAD+R/W` = slave address followed by bit 1 to indicate reading or 0 to indicate writing
|
||||||
|
/// - `SR` = repeated start condition
|
||||||
|
/// - `SP` = stop condition
|
||||||
|
fn exec<'a>(
|
||||||
|
&mut self,
|
||||||
|
address: A,
|
||||||
|
operations: &mut [Operation<'a>],
|
||||||
|
) -> Self::TransactionalFuture<'a>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Transactional I2C interface (iterator version).
|
||||||
|
///
|
||||||
|
/// This allows combining operation within an I2C transaction.
|
||||||
|
pub trait TransactionalIter<A: AddressMode = SevenBitAddress> {
|
||||||
|
/// Error type
|
||||||
|
type Error;
|
||||||
|
|
||||||
|
type TransactionalIterFuture<'a>: Future<Output = Result<(), Self::Error>> + 'a;
|
||||||
|
|
||||||
|
/// Execute the provided operations on the I2C bus (iterator version).
|
||||||
|
///
|
||||||
|
/// Transaction contract:
|
||||||
|
/// - Before executing the first operation an ST is sent automatically. This is followed by SAD+R/W as appropriate.
|
||||||
|
/// - Data from adjacent operations of the same type are sent after each other without an SP or SR.
|
||||||
|
/// - Between adjacent operations of a different type an SR and SAD+R/W is sent.
|
||||||
|
/// - After executing the last operation an SP is sent automatically.
|
||||||
|
/// - If the last operation is a `Read` the master does not send an acknowledge for the last byte.
|
||||||
|
///
|
||||||
|
/// - `ST` = start condition
|
||||||
|
/// - `SAD+R/W` = slave address followed by bit 1 to indicate reading or 0 to indicate writing
|
||||||
|
/// - `SR` = repeated start condition
|
||||||
|
/// - `SP` = stop condition
|
||||||
|
fn exec_iter<'a, O>(&mut self, address: A, operations: O) -> Self::TransactionalIterFuture<'a>
|
||||||
|
where
|
||||||
|
O: IntoIterator<Item = Operation<'a>>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Default implementation of `blocking::i2c::Write`, `blocking::i2c::Read` and
|
||||||
|
/// `blocking::i2c::WriteRead` traits for `blocking::i2c::Transactional` implementers.
|
||||||
|
///
|
||||||
|
/// If you implement `blocking::i2c::Transactional` for your I2C peripheral,
|
||||||
|
/// you can use this default implementation so that you do not need to implement
|
||||||
|
/// the `blocking::i2c::Write`, `blocking::i2c::Read` and `blocking::i2c::WriteRead`
|
||||||
|
/// traits as well.
|
||||||
|
/// ```
|
||||||
|
/// use embedded_hal::blocking::i2c;
|
||||||
|
///
|
||||||
|
/// struct I2c1;
|
||||||
|
///
|
||||||
|
/// impl i2c::Transactional<i2c::SevenBitAddress> for I2c1 {
|
||||||
|
/// # type Error = ();
|
||||||
|
/// fn try_exec<'a>(
|
||||||
|
/// &mut self,
|
||||||
|
/// address: i2c::SevenBitAddress,
|
||||||
|
/// operations: &mut [i2c::Operation<'a>],
|
||||||
|
/// ) -> Result<(), Self::Error> {
|
||||||
|
/// // ...
|
||||||
|
/// # Ok(())
|
||||||
|
/// }
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// // This is all you need to do:
|
||||||
|
/// impl i2c::transactional::Default<i2c::SevenBitAddress> for I2c1 {};
|
||||||
|
///
|
||||||
|
/// // Then you can use `Write` and so on:
|
||||||
|
/// use i2c::Write;
|
||||||
|
///
|
||||||
|
/// let mut i2c1 = I2c1{};
|
||||||
|
/// i2c1.try_write(0x01, &[0xAB, 0xCD]).unwrap();
|
||||||
|
/// ```
|
||||||
|
pub mod transactional {
|
||||||
|
use core::future::Future;
|
||||||
|
|
||||||
|
use super::{AddressMode, Operation, Read, Transactional, Write, WriteRead};
|
||||||
|
|
||||||
|
/// Default implementation of `blocking::i2c::Write`, `blocking::i2c::Read` and
|
||||||
|
/// `blocking::i2c::WriteRead` traits for `blocking::i2c::Transactional` implementers.
|
||||||
|
pub trait Default<A: AddressMode>: Transactional<A> {}
|
||||||
|
|
||||||
|
// impl<A, E, S> Write<A> for S
|
||||||
|
// where
|
||||||
|
// A: AddressMode + 'static,
|
||||||
|
// S: self::Default<A> + Transactional<A, Error = E> + 'static,
|
||||||
|
// E: 'static,
|
||||||
|
// {
|
||||||
|
// type Error = E;
|
||||||
|
//
|
||||||
|
// type WriteFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a;
|
||||||
|
//
|
||||||
|
// fn write<'a>(&mut self, address: A, bytes: &[u8]) -> Self::WriteFuture<'a> {
|
||||||
|
// self.exec(address, &mut [Operation::Write(bytes)])
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
/*
|
||||||
|
impl<A, E, S> Read<A> for S
|
||||||
|
where
|
||||||
|
A: AddressMode,
|
||||||
|
S: self::Default<A> + Transactional<A, Error = E>,
|
||||||
|
{
|
||||||
|
type Error = E;
|
||||||
|
|
||||||
|
fn read(&mut self, address: A, buffer: &mut [u8]) -> Result<(), Self::Error> {
|
||||||
|
self.exec(address, &mut [Operation::Read(buffer)])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<A, E, S> WriteRead<A> for S
|
||||||
|
where
|
||||||
|
A: AddressMode,
|
||||||
|
S: self::Default<A> + Transactional<A, Error = E>,
|
||||||
|
{
|
||||||
|
type Error = E;
|
||||||
|
|
||||||
|
fn write_read(
|
||||||
|
&mut self,
|
||||||
|
address: A,
|
||||||
|
bytes: &[u8],
|
||||||
|
buffer: &mut [u8],
|
||||||
|
) -> Result<(), Self::Error> {
|
||||||
|
self.exec(
|
||||||
|
address,
|
||||||
|
&mut [Operation::Write(bytes), Operation::Read(buffer)],
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
}
|
|
@ -4,8 +4,10 @@
|
||||||
#![feature(const_fn_fn_ptr_basics)]
|
#![feature(const_fn_fn_ptr_basics)]
|
||||||
#![feature(const_option)]
|
#![feature(const_option)]
|
||||||
#![allow(incomplete_features)]
|
#![allow(incomplete_features)]
|
||||||
|
#![feature(type_alias_impl_trait)]
|
||||||
|
|
||||||
pub mod delay;
|
pub mod delay;
|
||||||
pub mod flash;
|
pub mod flash;
|
||||||
pub mod gpio;
|
pub mod gpio;
|
||||||
|
pub mod i2c;
|
||||||
pub mod uart;
|
pub mod uart;
|
||||||
|
|
Loading…
Reference in a new issue