Update embedded-hal crates.
This commit is contained in:
		
							parent
							
								
									f3ec6080bf
								
							
						
					
					
						commit
						be37eee13d
					
				| @ -19,8 +19,8 @@ nightly = ["embedded-hal-async", "embedded-storage-async"] | ||||
| [dependencies] | ||||
| embassy-sync = { version = "0.1.0", path = "../embassy-sync" } | ||||
| embedded-hal-02 = { package = "embedded-hal", version = "0.2.6", features = ["unproven"] } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0", optional = true } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1", optional = true } | ||||
| embedded-storage = "0.3.0" | ||||
| embedded-storage-async = { version = "0.4.0", optional = true } | ||||
| nb = "1.0.0" | ||||
|  | ||||
| @ -36,27 +36,22 @@ where | ||||
|     E: embedded_hal_1::i2c::Error + 'static, | ||||
|     T: blocking::i2c::WriteRead<Error = E> + blocking::i2c::Read<Error = E> + blocking::i2c::Write<Error = E>, | ||||
| { | ||||
|     async fn read<'a>(&'a mut self, address: u8, buffer: &'a mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.read(address, buffer) | ||||
|     async fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.read(address, read) | ||||
|     } | ||||
| 
 | ||||
|     async fn write<'a>(&'a mut self, address: u8, bytes: &'a [u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.write(address, bytes) | ||||
|     async fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.write(address, write) | ||||
|     } | ||||
| 
 | ||||
|     async fn write_read<'a>( | ||||
|         &'a mut self, | ||||
|     async fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.write_read(address, write, read) | ||||
|     } | ||||
| 
 | ||||
|     async fn transaction( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         bytes: &'a [u8], | ||||
|         buffer: &'a mut [u8], | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         self.wrapped.write_read(address, bytes, buffer) | ||||
|     } | ||||
| 
 | ||||
|     async fn transaction<'a, 'b>( | ||||
|         &'a mut self, | ||||
|         address: u8, | ||||
|         operations: &'a mut [embedded_hal_async::i2c::Operation<'b>], | ||||
|         operations: &mut [embedded_hal_1::i2c::Operation<'_>], | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         let _ = address; | ||||
|         let _ = operations; | ||||
|  | ||||
| @ -1,7 +1,7 @@ | ||||
| #![cfg_attr(not(feature = "std"), no_std)] | ||||
| #![cfg_attr(
 | ||||
|     feature = "nightly", | ||||
|     feature(type_alias_impl_trait, async_fn_in_trait, impl_trait_projections) | ||||
|     feature(type_alias_impl_trait, async_fn_in_trait, impl_trait_projections, try_blocks) | ||||
| )] | ||||
| #![cfg_attr(feature = "nightly", allow(incomplete_features))] | ||||
| #![warn(missing_docs)] | ||||
|  | ||||
| @ -54,35 +54,35 @@ where | ||||
|     M: RawMutex + 'static, | ||||
|     BUS: i2c::I2c + 'static, | ||||
| { | ||||
|     async fn read<'a>(&'a mut self, address: u8, buffer: &'a mut [u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|     async fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.read(address, buffer).await.map_err(I2cDeviceError::I2c)?; | ||||
|         bus.read(address, read).await.map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     async fn write<'a>(&'a mut self, address: u8, bytes: &'a [u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|     async fn write(&mut self, address: u8, write: &[u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.write(address, bytes).await.map_err(I2cDeviceError::I2c)?; | ||||
|         bus.write(address, write).await.map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     async fn write_read<'a>( | ||||
|         &'a mut self, | ||||
|     async fn write_read( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         wr_buffer: &'a [u8], | ||||
|         rd_buffer: &'a mut [u8], | ||||
|         write: &[u8], | ||||
|         read: &mut [u8], | ||||
|     ) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.write_read(address, wr_buffer, rd_buffer) | ||||
|         bus.write_read(address, write, read) | ||||
|             .await | ||||
|             .map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     async fn transaction<'a, 'b>( | ||||
|         &'a mut self, | ||||
|     async fn transaction( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         operations: &'a mut [embedded_hal_async::i2c::Operation<'b>], | ||||
|         operations: &mut [embedded_hal_async::i2c::Operation<'_>], | ||||
|     ) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let _ = address; | ||||
|         let _ = operations; | ||||
| @ -121,25 +121,25 @@ where | ||||
|     M: RawMutex + 'static, | ||||
|     BUS: i2c::I2c + SetConfig + 'static, | ||||
| { | ||||
|     async fn read<'a>(&'a mut self, address: u8, buffer: &'a mut [u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|     async fn read(&mut self, address: u8, buffer: &mut [u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         bus.read(address, buffer).await.map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     async fn write<'a>(&'a mut self, address: u8, bytes: &'a [u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|     async fn write(&mut self, address: u8, bytes: &[u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         bus.write(address, bytes).await.map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     async fn write_read<'a>( | ||||
|         &'a mut self, | ||||
|     async fn write_read( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         wr_buffer: &'a [u8], | ||||
|         rd_buffer: &'a mut [u8], | ||||
|         wr_buffer: &[u8], | ||||
|         rd_buffer: &mut [u8], | ||||
|     ) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
| @ -149,11 +149,7 @@ where | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     async fn transaction<'a, 'b>( | ||||
|         &'a mut self, | ||||
|         address: u8, | ||||
|         operations: &'a mut [embedded_hal_async::i2c::Operation<'b>], | ||||
|     ) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|     async fn transaction(&mut self, address: u8, operations: &mut [i2c::Operation<'_>]) -> Result<(), Self::Error> { | ||||
|         let _ = address; | ||||
|         let _ = operations; | ||||
|         todo!() | ||||
|  | ||||
| @ -25,12 +25,11 @@ | ||||
| //! let spi_dev2 = SpiDevice::new(spi_bus, cs_pin2);
 | ||||
| //! let display2 = ST7735::new(spi_dev2, dc2, rst2, Default::default(), 160, 128);
 | ||||
| //! ```
 | ||||
| use core::future::Future; | ||||
| 
 | ||||
| use embassy_sync::blocking_mutex::raw::RawMutex; | ||||
| use embassy_sync::mutex::Mutex; | ||||
| use embedded_hal_1::digital::OutputPin; | ||||
| use embedded_hal_1::spi::ErrorType; | ||||
| use embedded_hal_1::spi::Operation; | ||||
| use embedded_hal_async::spi; | ||||
| 
 | ||||
| use crate::shared_bus::SpiDeviceError; | ||||
| @ -57,33 +56,92 @@ where | ||||
|     type Error = SpiDeviceError<BUS::Error, CS::Error>; | ||||
| } | ||||
| 
 | ||||
| unsafe impl<M, BUS, CS> spi::SpiDevice for SpiDevice<'_, M, BUS, CS> | ||||
| impl<M, BUS, CS> spi::SpiDeviceRead for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex + 'static, | ||||
|     BUS: spi::SpiBusFlush + 'static, | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBusRead, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     type Bus = BUS; | ||||
| 
 | ||||
|     async fn transaction<R, F, Fut>(&mut self, f: F) -> Result<R, Self::Error> | ||||
|     where | ||||
|         F: FnOnce(*mut Self::Bus) -> Fut, | ||||
|         Fut: Future<Output = Result<R, <Self::Bus as ErrorType>::Error>>, | ||||
|     { | ||||
|     async fn read_transaction(&mut self, operations: &mut [&mut [u8]]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         let f_res = f(&mut *bus).await; | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for buf in operations { | ||||
|                 bus.read(buf).await?; | ||||
|             } | ||||
|         }; | ||||
| 
 | ||||
|         // On failure, it's important to still flush and deassert CS.
 | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|         let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         Ok(f_res) | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<M, BUS, CS> spi::SpiDeviceWrite for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBusWrite, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     async fn write_transaction(&mut self, operations: &[&[u8]]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for buf in operations { | ||||
|                 bus.write(buf).await?; | ||||
|             } | ||||
|         }; | ||||
| 
 | ||||
|         // On failure, it's important to still flush and deassert CS.
 | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<M, BUS, CS> spi::SpiDevice for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBus, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     async fn transaction(&mut self, operations: &mut [spi::Operation<'_, u8>]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for op in operations { | ||||
|                 match op { | ||||
|                     Operation::Read(buf) => bus.read(buf).await?, | ||||
|                     Operation::Write(buf) => bus.write(buf).await?, | ||||
|                     Operation::Transfer(read, write) => bus.transfer(read, write).await?, | ||||
|                     Operation::TransferInPlace(buf) => bus.transfer_in_place(buf).await?, | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
| 
 | ||||
|         // On failure, it's important to still flush and deassert CS.
 | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -114,33 +172,94 @@ where | ||||
|     type Error = SpiDeviceError<BUS::Error, CS::Error>; | ||||
| } | ||||
| 
 | ||||
| unsafe impl<M, BUS, CS> spi::SpiDevice for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| impl<M, BUS, CS> spi::SpiDeviceWrite for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex + 'static, | ||||
|     BUS: spi::SpiBusFlush + SetConfig + 'static, | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBusWrite + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     type Bus = BUS; | ||||
| 
 | ||||
|     async fn transaction<R, F, Fut>(&mut self, f: F) -> Result<R, Self::Error> | ||||
|     where | ||||
|         F: FnOnce(*mut Self::Bus) -> Fut, | ||||
|         Fut: Future<Output = Result<R, <Self::Bus as ErrorType>::Error>>, | ||||
|     { | ||||
|     async fn write_transaction(&mut self, operations: &[&[u8]]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         let f_res = f(&mut *bus).await; | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for buf in operations { | ||||
|                 bus.write(buf).await?; | ||||
|             } | ||||
|         }; | ||||
| 
 | ||||
|         // On failure, it's important to still flush and deassert CS.
 | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|         let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         Ok(f_res) | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<M, BUS, CS> spi::SpiDeviceRead for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBusRead + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     async fn read_transaction(&mut self, operations: &mut [&mut [u8]]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for buf in operations { | ||||
|                 bus.read(buf).await?; | ||||
|             } | ||||
|         }; | ||||
| 
 | ||||
|         // On failure, it's important to still flush and deassert CS.
 | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<M, BUS, CS> spi::SpiDevice for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBus + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     async fn transaction(&mut self, operations: &mut [spi::Operation<'_, u8>]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for op in operations { | ||||
|                 match op { | ||||
|                     Operation::Read(buf) => bus.read(buf).await?, | ||||
|                     Operation::Write(buf) => bus.write(buf).await?, | ||||
|                     Operation::Transfer(read, write) => bus.transfer(read, write).await?, | ||||
|                     Operation::TransferInPlace(buf) => bus.transfer_in_place(buf).await?, | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
| 
 | ||||
|         // On failure, it's important to still flush and deassert CS.
 | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -72,34 +72,6 @@ where | ||||
|         let _ = operations; | ||||
|         todo!() | ||||
|     } | ||||
| 
 | ||||
|     fn write_iter<B: IntoIterator<Item = u8>>(&mut self, addr: u8, bytes: B) -> Result<(), Self::Error> { | ||||
|         let _ = addr; | ||||
|         let _ = bytes; | ||||
|         todo!() | ||||
|     } | ||||
| 
 | ||||
|     fn write_iter_read<B: IntoIterator<Item = u8>>( | ||||
|         &mut self, | ||||
|         addr: u8, | ||||
|         bytes: B, | ||||
|         buffer: &mut [u8], | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         let _ = addr; | ||||
|         let _ = bytes; | ||||
|         let _ = buffer; | ||||
|         todo!() | ||||
|     } | ||||
| 
 | ||||
|     fn transaction_iter<'a, O: IntoIterator<Item = Operation<'a>>>( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         operations: O, | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         let _ = address; | ||||
|         let _ = operations; | ||||
|         todo!() | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<'a, M, BUS, E> embedded_hal_02::blocking::i2c::Write for I2cDevice<'_, M, BUS> | ||||
| @ -204,32 +176,4 @@ where | ||||
|         let _ = operations; | ||||
|         todo!() | ||||
|     } | ||||
| 
 | ||||
|     fn write_iter<B: IntoIterator<Item = u8>>(&mut self, addr: u8, bytes: B) -> Result<(), Self::Error> { | ||||
|         let _ = addr; | ||||
|         let _ = bytes; | ||||
|         todo!() | ||||
|     } | ||||
| 
 | ||||
|     fn write_iter_read<B: IntoIterator<Item = u8>>( | ||||
|         &mut self, | ||||
|         addr: u8, | ||||
|         bytes: B, | ||||
|         buffer: &mut [u8], | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         let _ = addr; | ||||
|         let _ = bytes; | ||||
|         let _ = buffer; | ||||
|         todo!() | ||||
|     } | ||||
| 
 | ||||
|     fn transaction_iter<'a, O: IntoIterator<Item = Operation<'a>>>( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         operations: O, | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         let _ = address; | ||||
|         let _ = operations; | ||||
|         todo!() | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -23,8 +23,7 @@ use core::cell::RefCell; | ||||
| use embassy_sync::blocking_mutex::raw::RawMutex; | ||||
| use embassy_sync::blocking_mutex::Mutex; | ||||
| use embedded_hal_1::digital::OutputPin; | ||||
| use embedded_hal_1::spi; | ||||
| use embedded_hal_1::spi::SpiBusFlush; | ||||
| use embedded_hal_1::spi::{self, Operation, SpiBus, SpiBusRead, SpiBusWrite}; | ||||
| 
 | ||||
| use crate::shared_bus::SpiDeviceError; | ||||
| use crate::SetConfig; | ||||
| @ -50,30 +49,85 @@ where | ||||
|     type Error = SpiDeviceError<BUS::Error, CS::Error>; | ||||
| } | ||||
| 
 | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDevice for SpiDevice<'_, M, BUS, CS> | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDeviceRead for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBusFlush, | ||||
|     BUS: SpiBusRead, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     type Bus = BUS; | ||||
| 
 | ||||
|     fn transaction<R>(&mut self, f: impl FnOnce(&mut Self::Bus) -> Result<R, BUS::Error>) -> Result<R, Self::Error> { | ||||
|     fn read_transaction(&mut self, operations: &mut [&mut [u8]]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|             let f_res = f(&mut bus); | ||||
|             let op_res = operations.iter_mut().try_for_each(|buf| bus.read(buf)); | ||||
| 
 | ||||
|             // On failure, it's important to still flush and deassert CS.
 | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|             let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|             Ok(f_res) | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDeviceWrite for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBusWrite, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     fn write_transaction(&mut self, operations: &[&[u8]]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|             let op_res = operations.iter().try_for_each(|buf| bus.write(buf)); | ||||
| 
 | ||||
|             // On failure, it's important to still flush and deassert CS.
 | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDevice for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBus, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     fn transaction(&mut self, operations: &mut [Operation<'_, u8>]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|             let op_res = operations.iter_mut().try_for_each(|op| match op { | ||||
|                 Operation::Read(buf) => bus.read(buf), | ||||
|                 Operation::Write(buf) => bus.write(buf), | ||||
|                 Operation::Transfer(read, write) => bus.transfer(read, write), | ||||
|                 Operation::TransferInPlace(buf) => bus.transfer_in_place(buf), | ||||
|             }); | ||||
| 
 | ||||
|             // On failure, it's important to still flush and deassert CS.
 | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| @ -89,11 +143,11 @@ where | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|             let f_res = bus.transfer(words); | ||||
|             let op_res = bus.transfer(words); | ||||
|             let cs_res = self.cs.set_high(); | ||||
|             let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|             Ok(f_res) | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| @ -110,11 +164,11 @@ where | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|             let f_res = bus.write(words); | ||||
|             let op_res = bus.write(words); | ||||
|             let cs_res = self.cs.set_high(); | ||||
|             let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|             Ok(f_res) | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| @ -146,30 +200,85 @@ where | ||||
|     type Error = SpiDeviceError<BUS::Error, CS::Error>; | ||||
| } | ||||
| 
 | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDevice for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDeviceRead for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBusFlush + SetConfig, | ||||
|     BUS: SpiBusRead + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     type Bus = BUS; | ||||
| 
 | ||||
|     fn transaction<R>(&mut self, f: impl FnOnce(&mut Self::Bus) -> Result<R, BUS::Error>) -> Result<R, Self::Error> { | ||||
|     fn read_transaction(&mut self, operations: &mut [&mut [u8]]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             bus.set_config(&self.config); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|             let f_res = f(&mut bus); | ||||
|             let op_res = operations.iter_mut().try_for_each(|buf| bus.read(buf)); | ||||
| 
 | ||||
|             // On failure, it's important to still flush and deassert CS.
 | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|             let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|             Ok(f_res) | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDeviceWrite for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBusWrite + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     fn write_transaction(&mut self, operations: &[&[u8]]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             bus.set_config(&self.config); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|             let op_res = operations.iter().try_for_each(|buf| bus.write(buf)); | ||||
| 
 | ||||
|             // On failure, it's important to still flush and deassert CS.
 | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDevice for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBus + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     fn transaction(&mut self, operations: &mut [Operation<'_, u8>]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             bus.set_config(&self.config); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
| 
 | ||||
|             let op_res = operations.iter_mut().try_for_each(|op| match op { | ||||
|                 Operation::Read(buf) => bus.read(buf), | ||||
|                 Operation::Write(buf) => bus.write(buf), | ||||
|                 Operation::Transfer(read, write) => bus.transfer(read, write), | ||||
|                 Operation::TransferInPlace(buf) => bus.transfer_in_place(buf), | ||||
|             }); | ||||
| 
 | ||||
|             // On failure, it's important to still flush and deassert CS.
 | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -31,8 +31,8 @@ log = { version = "0.4.14", optional = true } | ||||
| embassy-time = { version = "0.1.0", path = "../embassy-time" } | ||||
| embassy-sync = { version = "0.1.0", path = "../embassy-sync" } | ||||
| embassy-stm32 = { version = "0.1.0", path = "../embassy-stm32", default-features = false, optional = true } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0" } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1" } | ||||
| embassy-hal-common = { version = "0.1.0", path = "../embassy-hal-common", default-features = false } | ||||
| futures = { version = "0.3.17", default-features = false, features = [ "async-await" ] } | ||||
| embedded-hal = { version = "0.2", features = ["unproven"] } | ||||
|  | ||||
| @ -87,8 +87,8 @@ embassy-embedded-hal = {version = "0.1.0", path = "../embassy-embedded-hal" } | ||||
| embassy-usb-driver = {version = "0.1.0", path = "../embassy-usb-driver", optional=true } | ||||
| 
 | ||||
| embedded-hal-02 = { package = "embedded-hal", version = "0.2.6", features = ["unproven"] } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9", optional = true} | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0", optional = true} | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10", optional = true} | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1", optional = true} | ||||
| embedded-io = { version = "0.4.0", features = ["async"], optional = true } | ||||
| 
 | ||||
| defmt = { version = "0.3", optional = true } | ||||
|  | ||||
| @ -846,20 +846,6 @@ mod eh1 { | ||||
|             self.blocking_write(address, buffer) | ||||
|         } | ||||
| 
 | ||||
|         fn write_iter<B>(&mut self, _address: u8, _bytes: B) -> Result<(), Self::Error> | ||||
|         where | ||||
|             B: IntoIterator<Item = u8>, | ||||
|         { | ||||
|             todo!(); | ||||
|         } | ||||
| 
 | ||||
|         fn write_iter_read<B>(&mut self, _address: u8, _bytes: B, _buffer: &mut [u8]) -> Result<(), Self::Error> | ||||
|         where | ||||
|             B: IntoIterator<Item = u8>, | ||||
|         { | ||||
|             todo!(); | ||||
|         } | ||||
| 
 | ||||
|         fn write_read(&mut self, address: u8, wr_buffer: &[u8], rd_buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, wr_buffer, rd_buffer) | ||||
|         } | ||||
| @ -871,13 +857,6 @@ mod eh1 { | ||||
|         ) -> Result<(), Self::Error> { | ||||
|             todo!(); | ||||
|         } | ||||
| 
 | ||||
|         fn transaction_iter<'a, O>(&mut self, _address: u8, _operations: O) -> Result<(), Self::Error> | ||||
|         where | ||||
|             O: IntoIterator<Item = embedded_hal_1::i2c::Operation<'a>>, | ||||
|         { | ||||
|             todo!(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -885,28 +864,22 @@ mod eh1 { | ||||
| mod eha { | ||||
|     use super::*; | ||||
|     impl<'d, T: Instance> embedded_hal_async::i2c::I2c for Twim<'d, T> { | ||||
|         async fn read<'a>(&'a mut self, address: u8, buffer: &'a mut [u8]) -> Result<(), Error> { | ||||
|             self.read(address, buffer).await | ||||
|         async fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.read(address, read).await | ||||
|         } | ||||
| 
 | ||||
|         async fn write<'a>(&'a mut self, address: u8, bytes: &'a [u8]) -> Result<(), Error> { | ||||
|             self.write(address, bytes).await | ||||
|         async fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.write(address, write).await | ||||
|         } | ||||
|         async fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.write_read(address, write, read).await | ||||
|         } | ||||
| 
 | ||||
|         async fn write_read<'a>( | ||||
|             &'a mut self, | ||||
|         async fn transaction( | ||||
|             &mut self, | ||||
|             address: u8, | ||||
|             wr_buffer: &'a [u8], | ||||
|             rd_buffer: &'a mut [u8], | ||||
|         ) -> Result<(), Error> { | ||||
|             self.write_read(address, wr_buffer, rd_buffer).await | ||||
|         } | ||||
| 
 | ||||
|         async fn transaction<'a, 'b>( | ||||
|             &'a mut self, | ||||
|             address: u8, | ||||
|             operations: &'a mut [embedded_hal_async::i2c::Operation<'b>], | ||||
|         ) -> Result<(), Error> { | ||||
|             operations: &mut [embedded_hal_1::i2c::Operation<'_>], | ||||
|         ) -> Result<(), Self::Error> { | ||||
|             let _ = address; | ||||
|             let _ = operations; | ||||
|             todo!() | ||||
|  | ||||
| @ -65,9 +65,9 @@ rp2040-pac2 = { git = "https://github.com/embassy-rs/rp2040-pac2", rev="017e3c90 | ||||
| #rp2040-pac2 = { path = "../../rp2040-pac2", features = ["rt"] } | ||||
| 
 | ||||
| embedded-hal-02 = { package = "embedded-hal", version = "0.2.6", features = ["unproven"] } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9", optional = true} | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0", optional = true} | ||||
| embedded-hal-nb = { version = "=1.0.0-alpha.1", optional = true} | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10", optional = true} | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1", optional = true} | ||||
| embedded-hal-nb = { version = "=1.0.0-alpha.2", optional = true} | ||||
| 
 | ||||
| paste = "1.0" | ||||
| pio-proc = {version= "0.2", optional = true} | ||||
|  | ||||
| @ -490,14 +490,14 @@ impl<'d, T: Instance + 'd, M: Mode> I2c<'d, T, M> { | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fn read_blocking_internal(&mut self, buffer: &mut [u8], restart: bool, send_stop: bool) -> Result<(), Error> { | ||||
|         if buffer.is_empty() { | ||||
|     fn read_blocking_internal(&mut self, read: &mut [u8], restart: bool, send_stop: bool) -> Result<(), Error> { | ||||
|         if read.is_empty() { | ||||
|             return Err(Error::InvalidReadBufferLength); | ||||
|         } | ||||
| 
 | ||||
|         let p = T::regs(); | ||||
|         let lastindex = buffer.len() - 1; | ||||
|         for (i, byte) in buffer.iter_mut().enumerate() { | ||||
|         let lastindex = read.len() - 1; | ||||
|         for (i, byte) in read.iter_mut().enumerate() { | ||||
|             let first = i == 0; | ||||
|             let last = i == lastindex; | ||||
| 
 | ||||
| @ -524,15 +524,15 @@ impl<'d, T: Instance + 'd, M: Mode> I2c<'d, T, M> { | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     fn write_blocking_internal(&mut self, bytes: &[u8], send_stop: bool) -> Result<(), Error> { | ||||
|         if bytes.is_empty() { | ||||
|     fn write_blocking_internal(&mut self, write: &[u8], send_stop: bool) -> Result<(), Error> { | ||||
|         if write.is_empty() { | ||||
|             return Err(Error::InvalidWriteBufferLength); | ||||
|         } | ||||
| 
 | ||||
|         let p = T::regs(); | ||||
| 
 | ||||
|         for (i, byte) in bytes.iter().enumerate() { | ||||
|             let last = i == bytes.len() - 1; | ||||
|         for (i, byte) in write.iter().enumerate() { | ||||
|             let last = i == write.len() - 1; | ||||
| 
 | ||||
|             // NOTE(unsafe) We have &mut self
 | ||||
|             unsafe { | ||||
| @ -572,21 +572,21 @@ impl<'d, T: Instance + 'd, M: Mode> I2c<'d, T, M> { | ||||
|     // Blocking public API
 | ||||
|     // =========================
 | ||||
| 
 | ||||
|     pub fn blocking_read(&mut self, address: u8, buffer: &mut [u8]) -> Result<(), Error> { | ||||
|     pub fn blocking_read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Error> { | ||||
|         Self::setup(address.into())?; | ||||
|         self.read_blocking_internal(buffer, true, true) | ||||
|         self.read_blocking_internal(read, true, true) | ||||
|         // Automatic Stop
 | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write(&mut self, address: u8, bytes: &[u8]) -> Result<(), Error> { | ||||
|     pub fn blocking_write(&mut self, address: u8, write: &[u8]) -> Result<(), Error> { | ||||
|         Self::setup(address.into())?; | ||||
|         self.write_blocking_internal(bytes, true) | ||||
|         self.write_blocking_internal(write, true) | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write_read(&mut self, address: u8, bytes: &[u8], buffer: &mut [u8]) -> Result<(), Error> { | ||||
|     pub fn blocking_write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Error> { | ||||
|         Self::setup(address.into())?; | ||||
|         self.write_blocking_internal(bytes, false)?; | ||||
|         self.read_blocking_internal(buffer, true, true) | ||||
|         self.write_blocking_internal(write, false)?; | ||||
|         self.read_blocking_internal(read, true, true) | ||||
|         // Automatic Stop
 | ||||
|     } | ||||
| } | ||||
| @ -644,48 +644,22 @@ mod eh1 { | ||||
|     } | ||||
| 
 | ||||
|     impl<'d, T: Instance, M: Mode> embedded_hal_1::i2c::I2c for I2c<'d, T, M> { | ||||
|         fn read(&mut self, address: u8, buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_read(address, buffer) | ||||
|         fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_read(address, read) | ||||
|         } | ||||
| 
 | ||||
|         fn write(&mut self, address: u8, buffer: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write(address, buffer) | ||||
|         fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write(address, write) | ||||
|         } | ||||
| 
 | ||||
|         fn write_iter<B>(&mut self, address: u8, bytes: B) -> Result<(), Self::Error> | ||||
|         where | ||||
|             B: IntoIterator<Item = u8>, | ||||
|         { | ||||
|             let mut peekable = bytes.into_iter().peekable(); | ||||
|             Self::setup(address.into())?; | ||||
| 
 | ||||
|             while let Some(tx) = peekable.next() { | ||||
|                 self.write_blocking_internal(&[tx], peekable.peek().is_none())?; | ||||
|             } | ||||
|             Ok(()) | ||||
|         fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, write, read) | ||||
|         } | ||||
| 
 | ||||
|         fn write_iter_read<B>(&mut self, address: u8, bytes: B, buffer: &mut [u8]) -> Result<(), Self::Error> | ||||
|         where | ||||
|             B: IntoIterator<Item = u8>, | ||||
|         { | ||||
|             let peekable = bytes.into_iter().peekable(); | ||||
|             Self::setup(address.into())?; | ||||
| 
 | ||||
|             for tx in peekable { | ||||
|                 self.write_blocking_internal(&[tx], false)? | ||||
|             } | ||||
|             self.read_blocking_internal(buffer, true, true) | ||||
|         } | ||||
| 
 | ||||
|         fn write_read(&mut self, address: u8, wr_buffer: &[u8], rd_buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, wr_buffer, rd_buffer) | ||||
|         } | ||||
| 
 | ||||
|         fn transaction<'a>( | ||||
|         fn transaction( | ||||
|             &mut self, | ||||
|             address: u8, | ||||
|             operations: &mut [embedded_hal_1::i2c::Operation<'a>], | ||||
|             operations: &mut [embedded_hal_1::i2c::Operation<'_>], | ||||
|         ) -> Result<(), Self::Error> { | ||||
|             Self::setup(address.into())?; | ||||
|             for i in 0..operations.len() { | ||||
| @ -697,22 +671,6 @@ mod eh1 { | ||||
|             } | ||||
|             Ok(()) | ||||
|         } | ||||
| 
 | ||||
|         fn transaction_iter<'a, O>(&mut self, address: u8, operations: O) -> Result<(), Self::Error> | ||||
|         where | ||||
|             O: IntoIterator<Item = embedded_hal_1::i2c::Operation<'a>>, | ||||
|         { | ||||
|             Self::setup(address.into())?; | ||||
|             let mut peekable = operations.into_iter().peekable(); | ||||
|             while let Some(operation) = peekable.next() { | ||||
|                 let last = peekable.peek().is_none(); | ||||
|                 match operation { | ||||
|                     embedded_hal_1::i2c::Operation::Read(buf) => self.read_blocking_internal(buf, false, last)?, | ||||
|                     embedded_hal_1::i2c::Operation::Write(buf) => self.write_blocking_internal(buf, last)?, | ||||
|                 } | ||||
|             } | ||||
|             Ok(()) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| #[cfg(all(feature = "unstable-traits", feature = "nightly"))] | ||||
| @ -727,36 +685,29 @@ mod nightly { | ||||
|         A: AddressMode + Into<u16> + 'static, | ||||
|         T: Instance + 'd, | ||||
|     { | ||||
|         async fn read<'a>(&'a mut self, address: A, read: &'a mut [u8]) -> Result<(), Self::Error> { | ||||
|         async fn read(&mut self, address: A, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             let addr: u16 = address.into(); | ||||
| 
 | ||||
|             Self::setup(addr)?; | ||||
|             self.read_async_internal(read, false, true).await | ||||
|         } | ||||
| 
 | ||||
|         async fn write<'a>(&'a mut self, address: A, write: &'a [u8]) -> Result<(), Self::Error> { | ||||
|         async fn write(&mut self, address: A, write: &[u8]) -> Result<(), Self::Error> { | ||||
|             let addr: u16 = address.into(); | ||||
| 
 | ||||
|             Self::setup(addr)?; | ||||
|             self.write_async_internal(write.iter().copied(), true).await | ||||
|         } | ||||
|         async fn write_read<'a>( | ||||
|             &'a mut self, | ||||
|             address: A, | ||||
|             write: &'a [u8], | ||||
|             read: &'a mut [u8], | ||||
|         ) -> Result<(), Self::Error> { | ||||
| 
 | ||||
|         async fn write_read(&mut self, address: A, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             let addr: u16 = address.into(); | ||||
| 
 | ||||
|             Self::setup(addr)?; | ||||
|             self.write_async_internal(write.iter().cloned(), false).await?; | ||||
|             self.read_async_internal(read, false, true).await | ||||
|         } | ||||
|         async fn transaction<'a, 'b>( | ||||
|             &'a mut self, | ||||
|             address: A, | ||||
|             operations: &'a mut [Operation<'b>], | ||||
|         ) -> Result<(), Self::Error> { | ||||
| 
 | ||||
|         async fn transaction(&mut self, address: A, operations: &mut [Operation<'_>]) -> Result<(), Self::Error> { | ||||
|             let addr: u16 = address.into(); | ||||
| 
 | ||||
|             let mut iterator = operations.iter_mut(); | ||||
|  | ||||
| @ -19,6 +19,7 @@ pub enum Error { | ||||
| } | ||||
| 
 | ||||
| #[non_exhaustive] | ||||
| #[derive(Clone)] | ||||
| pub struct Config { | ||||
|     pub frequency: u32, | ||||
|     pub phase: Phase, | ||||
|  | ||||
| @ -44,9 +44,9 @@ embassy-net-driver = { version = "0.1.0", path = "../embassy-net-driver" } | ||||
| embassy-usb-driver = {version = "0.1.0", path = "../embassy-usb-driver", optional = true } | ||||
| 
 | ||||
| embedded-hal-02 = { package = "embedded-hal", version = "0.2.6", features = ["unproven"] } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9", optional = true} | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0", optional = true} | ||||
| embedded-hal-nb = { version = "=1.0.0-alpha.1", optional = true} | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10", optional = true} | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1", optional = true} | ||||
| embedded-hal-nb = { version = "=1.0.0-alpha.2", optional = true} | ||||
| 
 | ||||
| embedded-storage = "0.3.0" | ||||
| 
 | ||||
|  | ||||
| @ -28,64 +28,64 @@ impl<'d, T: Instance, TXDMA, RXDMA> TimeoutI2c<'d, T, TXDMA, RXDMA> { | ||||
|     } | ||||
| 
 | ||||
|     /// Blocking read with a custom timeout
 | ||||
|     pub fn blocking_read_timeout(&mut self, addr: u8, buffer: &mut [u8], timeout: Duration) -> Result<(), Error> { | ||||
|         self.i2c.blocking_read_timeout(addr, buffer, timeout_fn(timeout)) | ||||
|     pub fn blocking_read_timeout(&mut self, addr: u8, read: &mut [u8], timeout: Duration) -> Result<(), Error> { | ||||
|         self.i2c.blocking_read_timeout(addr, read, timeout_fn(timeout)) | ||||
|     } | ||||
| 
 | ||||
|     /// Blocking read with default timeout, provided in [`TimeoutI2c::new()`]
 | ||||
|     pub fn blocking_read(&mut self, addr: u8, buffer: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_read_timeout(addr, buffer, self.timeout) | ||||
|     pub fn blocking_read(&mut self, addr: u8, read: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_read_timeout(addr, read, self.timeout) | ||||
|     } | ||||
| 
 | ||||
|     /// Blocking write with a custom timeout
 | ||||
|     pub fn blocking_write_timeout(&mut self, addr: u8, bytes: &[u8], timeout: Duration) -> Result<(), Error> { | ||||
|         self.i2c.blocking_write_timeout(addr, bytes, timeout_fn(timeout)) | ||||
|     pub fn blocking_write_timeout(&mut self, addr: u8, write: &[u8], timeout: Duration) -> Result<(), Error> { | ||||
|         self.i2c.blocking_write_timeout(addr, write, timeout_fn(timeout)) | ||||
|     } | ||||
| 
 | ||||
|     /// Blocking write with default timeout, provided in [`TimeoutI2c::new()`]
 | ||||
|     pub fn blocking_write(&mut self, addr: u8, bytes: &[u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_timeout(addr, bytes, self.timeout) | ||||
|     pub fn blocking_write(&mut self, addr: u8, write: &[u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_timeout(addr, write, self.timeout) | ||||
|     } | ||||
| 
 | ||||
|     /// Blocking write-read with a custom timeout
 | ||||
|     pub fn blocking_write_read_timeout( | ||||
|         &mut self, | ||||
|         addr: u8, | ||||
|         bytes: &[u8], | ||||
|         buffer: &mut [u8], | ||||
|         write: &[u8], | ||||
|         read: &mut [u8], | ||||
|         timeout: Duration, | ||||
|     ) -> Result<(), Error> { | ||||
|         self.i2c | ||||
|             .blocking_write_read_timeout(addr, bytes, buffer, timeout_fn(timeout)) | ||||
|             .blocking_write_read_timeout(addr, write, read, timeout_fn(timeout)) | ||||
|     } | ||||
| 
 | ||||
|     /// Blocking write-read with default timeout, provided in [`TimeoutI2c::new()`]
 | ||||
|     pub fn blocking_write_read(&mut self, addr: u8, bytes: &[u8], buffer: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_read_timeout(addr, bytes, buffer, self.timeout) | ||||
|     pub fn blocking_write_read(&mut self, addr: u8, write: &[u8], read: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_read_timeout(addr, write, read, self.timeout) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<'d, T: Instance, TXDMA, RXDMA> embedded_hal_02::blocking::i2c::Read for TimeoutI2c<'d, T, TXDMA, RXDMA> { | ||||
|     type Error = Error; | ||||
| 
 | ||||
|     fn read(&mut self, addr: u8, buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_read(addr, buffer) | ||||
|     fn read(&mut self, addr: u8, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_read(addr, read) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<'d, T: Instance, TXDMA, RXDMA> embedded_hal_02::blocking::i2c::Write for TimeoutI2c<'d, T, TXDMA, RXDMA> { | ||||
|     type Error = Error; | ||||
| 
 | ||||
|     fn write(&mut self, addr: u8, bytes: &[u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_write(addr, bytes) | ||||
|     fn write(&mut self, addr: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_write(addr, write) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<'d, T: Instance, TXDMA, RXDMA> embedded_hal_02::blocking::i2c::WriteRead for TimeoutI2c<'d, T, TXDMA, RXDMA> { | ||||
|     type Error = Error; | ||||
| 
 | ||||
|     fn write_read(&mut self, addr: u8, bytes: &[u8], buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_write_read(addr, bytes, buffer) | ||||
|     fn write_read(&mut self, addr: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_write_read(addr, write, read) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -98,45 +98,24 @@ mod eh1 { | ||||
|     } | ||||
| 
 | ||||
|     impl<'d, T: Instance, TXDMA, RXDMA> embedded_hal_1::i2c::I2c for TimeoutI2c<'d, T, TXDMA, RXDMA> { | ||||
|         fn read(&mut self, address: u8, buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_read(address, buffer) | ||||
|         fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_read(address, read) | ||||
|         } | ||||
| 
 | ||||
|         fn write(&mut self, address: u8, buffer: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write(address, buffer) | ||||
|         fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write(address, write) | ||||
|         } | ||||
| 
 | ||||
|         fn write_iter<B>(&mut self, _address: u8, _bytes: B) -> Result<(), Self::Error> | ||||
|         where | ||||
|             B: IntoIterator<Item = u8>, | ||||
|         { | ||||
|             todo!(); | ||||
|         fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, write, read) | ||||
|         } | ||||
| 
 | ||||
|         fn write_iter_read<B>(&mut self, _address: u8, _bytes: B, _buffer: &mut [u8]) -> Result<(), Self::Error> | ||||
|         where | ||||
|             B: IntoIterator<Item = u8>, | ||||
|         { | ||||
|             todo!(); | ||||
|         } | ||||
| 
 | ||||
|         fn write_read(&mut self, address: u8, wr_buffer: &[u8], rd_buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, wr_buffer, rd_buffer) | ||||
|         } | ||||
| 
 | ||||
|         fn transaction<'a>( | ||||
|         fn transaction( | ||||
|             &mut self, | ||||
|             _address: u8, | ||||
|             _operations: &mut [embedded_hal_1::i2c::Operation<'a>], | ||||
|             _operations: &mut [embedded_hal_1::i2c::Operation<'_>], | ||||
|         ) -> Result<(), Self::Error> { | ||||
|             todo!(); | ||||
|         } | ||||
| 
 | ||||
|         fn transaction_iter<'a, O>(&mut self, _address: u8, _operations: O) -> Result<(), Self::Error> | ||||
|         where | ||||
|             O: IntoIterator<Item = embedded_hal_1::i2c::Operation<'a>>, | ||||
|         { | ||||
|             todo!(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -307,18 +307,18 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_read(&mut self, addr: u8, buffer: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_read_timeout(addr, buffer, || Ok(())) | ||||
|     pub fn blocking_read(&mut self, addr: u8, read: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_read_timeout(addr, read, || Ok(())) | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write_timeout( | ||||
|         &mut self, | ||||
|         addr: u8, | ||||
|         bytes: &[u8], | ||||
|         write: &[u8], | ||||
|         check_timeout: impl Fn() -> Result<(), Error>, | ||||
|     ) -> Result<(), Error> { | ||||
|         unsafe { | ||||
|             self.write_bytes(addr, bytes, &check_timeout)?; | ||||
|             self.write_bytes(addr, write, &check_timeout)?; | ||||
|             // Send a STOP condition
 | ||||
|             T::regs().cr1().modify(|reg| reg.set_stop(true)); | ||||
|             // Wait for STOP condition to transmit.
 | ||||
| @ -331,49 +331,49 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write(&mut self, addr: u8, bytes: &[u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_timeout(addr, bytes, || Ok(())) | ||||
|     pub fn blocking_write(&mut self, addr: u8, write: &[u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_timeout(addr, write, || Ok(())) | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write_read_timeout( | ||||
|         &mut self, | ||||
|         addr: u8, | ||||
|         bytes: &[u8], | ||||
|         buffer: &mut [u8], | ||||
|         write: &[u8], | ||||
|         read: &mut [u8], | ||||
|         check_timeout: impl Fn() -> Result<(), Error>, | ||||
|     ) -> Result<(), Error> { | ||||
|         unsafe { self.write_bytes(addr, bytes, &check_timeout)? }; | ||||
|         self.blocking_read_timeout(addr, buffer, &check_timeout)?; | ||||
|         unsafe { self.write_bytes(addr, write, &check_timeout)? }; | ||||
|         self.blocking_read_timeout(addr, read, &check_timeout)?; | ||||
| 
 | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write_read(&mut self, addr: u8, bytes: &[u8], buffer: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_read_timeout(addr, bytes, buffer, || Ok(())) | ||||
|     pub fn blocking_write_read(&mut self, addr: u8, write: &[u8], read: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_read_timeout(addr, write, read, || Ok(())) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<'d, T: Instance> embedded_hal_02::blocking::i2c::Read for I2c<'d, T> { | ||||
|     type Error = Error; | ||||
| 
 | ||||
|     fn read(&mut self, addr: u8, buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_read(addr, buffer) | ||||
|     fn read(&mut self, addr: u8, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_read(addr, read) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<'d, T: Instance> embedded_hal_02::blocking::i2c::Write for I2c<'d, T> { | ||||
|     type Error = Error; | ||||
| 
 | ||||
|     fn write(&mut self, addr: u8, bytes: &[u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_write(addr, bytes) | ||||
|     fn write(&mut self, addr: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_write(addr, write) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<'d, T: Instance> embedded_hal_02::blocking::i2c::WriteRead for I2c<'d, T> { | ||||
|     type Error = Error; | ||||
| 
 | ||||
|     fn write_read(&mut self, addr: u8, bytes: &[u8], buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_write_read(addr, bytes, buffer) | ||||
|     fn write_read(&mut self, addr: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.blocking_write_read(addr, write, read) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -402,46 +402,25 @@ mod eh1 { | ||||
|     } | ||||
| 
 | ||||
|     impl<'d, T: Instance> embedded_hal_1::i2c::I2c for I2c<'d, T> { | ||||
|         fn read(&mut self, address: u8, buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_read(address, buffer) | ||||
|         fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_read(address, read) | ||||
|         } | ||||
| 
 | ||||
|         fn write(&mut self, address: u8, buffer: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write(address, buffer) | ||||
|         fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write(address, write) | ||||
|         } | ||||
| 
 | ||||
|         fn write_iter<B>(&mut self, _address: u8, _bytes: B) -> Result<(), Self::Error> | ||||
|         where | ||||
|             B: IntoIterator<Item = u8>, | ||||
|         { | ||||
|             todo!(); | ||||
|         fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, write, read) | ||||
|         } | ||||
| 
 | ||||
|         fn write_iter_read<B>(&mut self, _address: u8, _bytes: B, _buffer: &mut [u8]) -> Result<(), Self::Error> | ||||
|         where | ||||
|             B: IntoIterator<Item = u8>, | ||||
|         { | ||||
|             todo!(); | ||||
|         } | ||||
| 
 | ||||
|         fn write_read(&mut self, address: u8, wr_buffer: &[u8], rd_buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, wr_buffer, rd_buffer) | ||||
|         } | ||||
| 
 | ||||
|         fn transaction<'a>( | ||||
|         fn transaction( | ||||
|             &mut self, | ||||
|             _address: u8, | ||||
|             _operations: &mut [embedded_hal_1::i2c::Operation<'a>], | ||||
|             _operations: &mut [embedded_hal_1::i2c::Operation<'_>], | ||||
|         ) -> Result<(), Self::Error> { | ||||
|             todo!(); | ||||
|         } | ||||
| 
 | ||||
|         fn transaction_iter<'a, O>(&mut self, _address: u8, _operations: O) -> Result<(), Self::Error> | ||||
|         where | ||||
|             O: IntoIterator<Item = embedded_hal_1::i2c::Operation<'a>>, | ||||
|         { | ||||
|             todo!(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -345,12 +345,12 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|     fn read_internal( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         buffer: &mut [u8], | ||||
|         read: &mut [u8], | ||||
|         restart: bool, | ||||
|         check_timeout: impl Fn() -> Result<(), Error>, | ||||
|     ) -> Result<(), Error> { | ||||
|         let completed_chunks = buffer.len() / 255; | ||||
|         let total_chunks = if completed_chunks * 255 == buffer.len() { | ||||
|         let completed_chunks = read.len() / 255; | ||||
|         let total_chunks = if completed_chunks * 255 == read.len() { | ||||
|             completed_chunks | ||||
|         } else { | ||||
|             completed_chunks + 1 | ||||
| @ -360,7 +360,7 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|         unsafe { | ||||
|             Self::master_read( | ||||
|                 address, | ||||
|                 buffer.len().min(255), | ||||
|                 read.len().min(255), | ||||
|                 Stop::Automatic, | ||||
|                 last_chunk_idx != 0, | ||||
|                 restart, | ||||
| @ -368,7 +368,7 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|             )?; | ||||
|         } | ||||
| 
 | ||||
|         for (number, chunk) in buffer.chunks_mut(255).enumerate() { | ||||
|         for (number, chunk) in read.chunks_mut(255).enumerate() { | ||||
|             if number != 0 { | ||||
|                 // NOTE(unsafe) We have &mut self
 | ||||
|                 unsafe { | ||||
| @ -391,12 +391,12 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|     fn write_internal( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         bytes: &[u8], | ||||
|         write: &[u8], | ||||
|         send_stop: bool, | ||||
|         check_timeout: impl Fn() -> Result<(), Error>, | ||||
|     ) -> Result<(), Error> { | ||||
|         let completed_chunks = bytes.len() / 255; | ||||
|         let total_chunks = if completed_chunks * 255 == bytes.len() { | ||||
|         let completed_chunks = write.len() / 255; | ||||
|         let total_chunks = if completed_chunks * 255 == write.len() { | ||||
|             completed_chunks | ||||
|         } else { | ||||
|             completed_chunks + 1 | ||||
| @ -410,14 +410,14 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|         unsafe { | ||||
|             Self::master_write( | ||||
|                 address, | ||||
|                 bytes.len().min(255), | ||||
|                 write.len().min(255), | ||||
|                 Stop::Software, | ||||
|                 last_chunk_idx != 0, | ||||
|                 &check_timeout, | ||||
|             )?; | ||||
|         } | ||||
| 
 | ||||
|         for (number, chunk) in bytes.chunks(255).enumerate() { | ||||
|         for (number, chunk) in write.chunks(255).enumerate() { | ||||
|             if number != 0 { | ||||
|                 // NOTE(unsafe) We have &mut self
 | ||||
|                 unsafe { | ||||
| @ -448,7 +448,7 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|     async fn write_dma_internal( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         bytes: &[u8], | ||||
|         write: &[u8], | ||||
|         first_slice: bool, | ||||
|         last_slice: bool, | ||||
|         check_timeout: impl Fn() -> Result<(), Error>, | ||||
| @ -456,7 +456,7 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|     where | ||||
|         TXDMA: crate::i2c::TxDma<T>, | ||||
|     { | ||||
|         let total_len = bytes.len(); | ||||
|         let total_len = write.len(); | ||||
|         let completed_chunks = total_len / 255; | ||||
|         let total_chunks = if completed_chunks * 255 == total_len { | ||||
|             completed_chunks | ||||
| @ -476,7 +476,7 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
| 
 | ||||
|             let ch = &mut self.tx_dma; | ||||
|             let request = ch.request(); | ||||
|             crate::dma::write(ch, request, bytes, dst) | ||||
|             crate::dma::write(ch, request, write, dst) | ||||
|         }; | ||||
| 
 | ||||
|         let state = T::state(); | ||||
| @ -641,25 +641,25 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|     // =========================
 | ||||
|     //  Async public API
 | ||||
| 
 | ||||
|     pub async fn write(&mut self, address: u8, bytes: &[u8]) -> Result<(), Error> | ||||
|     pub async fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Error> | ||||
|     where | ||||
|         TXDMA: crate::i2c::TxDma<T>, | ||||
|     { | ||||
|         if bytes.is_empty() { | ||||
|             self.write_internal(address, bytes, true, || Ok(())) | ||||
|         if write.is_empty() { | ||||
|             self.write_internal(address, write, true, || Ok(())) | ||||
|         } else { | ||||
|             self.write_dma_internal(address, bytes, true, true, || Ok(())).await | ||||
|             self.write_dma_internal(address, write, true, true, || Ok(())).await | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     pub async fn write_vectored(&mut self, address: u8, bytes: &[&[u8]]) -> Result<(), Error> | ||||
|     pub async fn write_vectored(&mut self, address: u8, write: &[&[u8]]) -> Result<(), Error> | ||||
|     where | ||||
|         TXDMA: crate::i2c::TxDma<T>, | ||||
|     { | ||||
|         if bytes.is_empty() { | ||||
|         if write.is_empty() { | ||||
|             return Err(Error::ZeroLengthTransfer); | ||||
|         } | ||||
|         let mut iter = bytes.iter(); | ||||
|         let mut iter = write.iter(); | ||||
| 
 | ||||
|         let mut first = true; | ||||
|         let mut current = iter.next(); | ||||
| @ -685,21 +685,21 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     pub async fn write_read(&mut self, address: u8, bytes: &[u8], buffer: &mut [u8]) -> Result<(), Error> | ||||
|     pub async fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Error> | ||||
|     where | ||||
|         TXDMA: super::TxDma<T>, | ||||
|         RXDMA: super::RxDma<T>, | ||||
|     { | ||||
|         if bytes.is_empty() { | ||||
|             self.write_internal(address, bytes, false, || Ok(()))?; | ||||
|         if write.is_empty() { | ||||
|             self.write_internal(address, write, false, || Ok(()))?; | ||||
|         } else { | ||||
|             self.write_dma_internal(address, bytes, true, true, || Ok(())).await?; | ||||
|             self.write_dma_internal(address, write, true, true, || Ok(())).await?; | ||||
|         } | ||||
| 
 | ||||
|         if buffer.is_empty() { | ||||
|             self.read_internal(address, buffer, true, || Ok(()))?; | ||||
|         if read.is_empty() { | ||||
|             self.read_internal(address, read, true, || Ok(()))?; | ||||
|         } else { | ||||
|             self.read_dma_internal(address, buffer, true, || Ok(())).await?; | ||||
|             self.read_dma_internal(address, read, true, || Ok(())).await?; | ||||
|         } | ||||
| 
 | ||||
|         Ok(()) | ||||
| @ -711,57 +711,57 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|     pub fn blocking_read_timeout( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         buffer: &mut [u8], | ||||
|         read: &mut [u8], | ||||
|         check_timeout: impl Fn() -> Result<(), Error>, | ||||
|     ) -> Result<(), Error> { | ||||
|         self.read_internal(address, buffer, false, &check_timeout) | ||||
|         self.read_internal(address, read, false, &check_timeout) | ||||
|         // Automatic Stop
 | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_read(&mut self, address: u8, buffer: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_read_timeout(address, buffer, || Ok(())) | ||||
|     pub fn blocking_read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_read_timeout(address, read, || Ok(())) | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write_timeout( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         bytes: &[u8], | ||||
|         write: &[u8], | ||||
|         check_timeout: impl Fn() -> Result<(), Error>, | ||||
|     ) -> Result<(), Error> { | ||||
|         self.write_internal(address, bytes, true, &check_timeout) | ||||
|         self.write_internal(address, write, true, &check_timeout) | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write(&mut self, address: u8, bytes: &[u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_timeout(address, bytes, || Ok(())) | ||||
|     pub fn blocking_write(&mut self, address: u8, write: &[u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_timeout(address, write, || Ok(())) | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write_read_timeout( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         bytes: &[u8], | ||||
|         buffer: &mut [u8], | ||||
|         write: &[u8], | ||||
|         read: &mut [u8], | ||||
|         check_timeout: impl Fn() -> Result<(), Error>, | ||||
|     ) -> Result<(), Error> { | ||||
|         self.write_internal(address, bytes, false, &check_timeout)?; | ||||
|         self.read_internal(address, buffer, true, &check_timeout) | ||||
|         self.write_internal(address, write, false, &check_timeout)?; | ||||
|         self.read_internal(address, read, true, &check_timeout) | ||||
|         // Automatic Stop
 | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write_read(&mut self, address: u8, bytes: &[u8], buffer: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_read_timeout(address, bytes, buffer, || Ok(())) | ||||
|     pub fn blocking_write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Error> { | ||||
|         self.blocking_write_read_timeout(address, write, read, || Ok(())) | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write_vectored_timeout( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         bytes: &[&[u8]], | ||||
|         write: &[&[u8]], | ||||
|         check_timeout: impl Fn() -> Result<(), Error>, | ||||
|     ) -> Result<(), Error> { | ||||
|         if bytes.is_empty() { | ||||
|         if write.is_empty() { | ||||
|             return Err(Error::ZeroLengthTransfer); | ||||
|         } | ||||
|         let first_length = bytes[0].len(); | ||||
|         let last_slice_index = bytes.len() - 1; | ||||
|         let first_length = write[0].len(); | ||||
|         let last_slice_index = write.len() - 1; | ||||
| 
 | ||||
|         // NOTE(unsafe) We have &mut self
 | ||||
|         unsafe { | ||||
| @ -774,7 +774,7 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|             )?; | ||||
|         } | ||||
| 
 | ||||
|         for (idx, slice) in bytes.iter().enumerate() { | ||||
|         for (idx, slice) in write.iter().enumerate() { | ||||
|             let slice_len = slice.len(); | ||||
|             let completed_chunks = slice_len / 255; | ||||
|             let total_chunks = if completed_chunks * 255 == slice_len { | ||||
| @ -828,8 +828,8 @@ impl<'d, T: Instance, TXDMA, RXDMA> I2c<'d, T, TXDMA, RXDMA> { | ||||
|         Ok(()) | ||||
|     } | ||||
| 
 | ||||
|     pub fn blocking_write_vectored(&mut self, address: u8, bytes: &[&[u8]]) -> Result<(), Error> { | ||||
|         self.blocking_write_vectored_timeout(address, bytes, || Ok(())) | ||||
|     pub fn blocking_write_vectored(&mut self, address: u8, write: &[&[u8]]) -> Result<(), Error> { | ||||
|         self.blocking_write_vectored_timeout(address, write, || Ok(())) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -847,16 +847,16 @@ mod eh02 { | ||||
|     impl<'d, T: Instance> embedded_hal_02::blocking::i2c::Write for I2c<'d, T> { | ||||
|         type Error = Error; | ||||
| 
 | ||||
|         fn write(&mut self, address: u8, bytes: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write(address, bytes) | ||||
|         fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write(address, write) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     impl<'d, T: Instance> embedded_hal_02::blocking::i2c::WriteRead for I2c<'d, T> { | ||||
|         type Error = Error; | ||||
| 
 | ||||
|         fn write_read(&mut self, address: u8, bytes: &[u8], buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, bytes, buffer) | ||||
|         fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, write, read) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -1010,46 +1010,25 @@ mod eh1 { | ||||
|     } | ||||
| 
 | ||||
|     impl<'d, T: Instance> embedded_hal_1::i2c::I2c for I2c<'d, T, NoDma, NoDma> { | ||||
|         fn read(&mut self, address: u8, buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_read(address, buffer) | ||||
|         fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_read(address, read) | ||||
|         } | ||||
| 
 | ||||
|         fn write(&mut self, address: u8, buffer: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write(address, buffer) | ||||
|         fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write(address, write) | ||||
|         } | ||||
| 
 | ||||
|         fn write_iter<B>(&mut self, _address: u8, _bytes: B) -> Result<(), Self::Error> | ||||
|         where | ||||
|             B: IntoIterator<Item = u8>, | ||||
|         { | ||||
|             todo!(); | ||||
|         fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, write, read) | ||||
|         } | ||||
| 
 | ||||
|         fn write_iter_read<B>(&mut self, _address: u8, _bytes: B, _buffer: &mut [u8]) -> Result<(), Self::Error> | ||||
|         where | ||||
|             B: IntoIterator<Item = u8>, | ||||
|         { | ||||
|             todo!(); | ||||
|         } | ||||
| 
 | ||||
|         fn write_read(&mut self, address: u8, wr_buffer: &[u8], rd_buffer: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.blocking_write_read(address, wr_buffer, rd_buffer) | ||||
|         } | ||||
| 
 | ||||
|         fn transaction<'a>( | ||||
|         fn transaction( | ||||
|             &mut self, | ||||
|             _address: u8, | ||||
|             _operations: &mut [embedded_hal_1::i2c::Operation<'a>], | ||||
|             _operations: &mut [embedded_hal_1::i2c::Operation<'_>], | ||||
|         ) -> Result<(), Self::Error> { | ||||
|             todo!(); | ||||
|         } | ||||
| 
 | ||||
|         fn transaction_iter<'a, O>(&mut self, _address: u8, _operations: O) -> Result<(), Self::Error> | ||||
|         where | ||||
|             O: IntoIterator<Item = embedded_hal_1::i2c::Operation<'a>>, | ||||
|         { | ||||
|             todo!(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -1059,27 +1038,22 @@ mod eha { | ||||
|     use super::*; | ||||
| 
 | ||||
|     impl<'d, T: Instance, TXDMA: TxDma<T>, RXDMA: RxDma<T>> embedded_hal_async::i2c::I2c for I2c<'d, T, TXDMA, RXDMA> { | ||||
|         async fn read<'a>(&'a mut self, address: u8, read: &'a mut [u8]) -> Result<(), Self::Error> { | ||||
|         async fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.read(address, read).await | ||||
|         } | ||||
| 
 | ||||
|         async fn write<'a>(&'a mut self, address: u8, write: &'a [u8]) -> Result<(), Self::Error> { | ||||
|         async fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|             self.write(address, write).await | ||||
|         } | ||||
| 
 | ||||
|         async fn write_read<'a>( | ||||
|             &'a mut self, | ||||
|             address: u8, | ||||
|             write: &'a [u8], | ||||
|             read: &'a mut [u8], | ||||
|         ) -> Result<(), Self::Error> { | ||||
|         async fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|             self.write_read(address, write, read).await | ||||
|         } | ||||
| 
 | ||||
|         async fn transaction<'a, 'b>( | ||||
|             &'a mut self, | ||||
|         async fn transaction( | ||||
|             &mut self, | ||||
|             address: u8, | ||||
|             operations: &'a mut [embedded_hal_1::i2c::Operation<'b>], | ||||
|             operations: &mut [embedded_hal_1::i2c::Operation<'_>], | ||||
|         ) -> Result<(), Self::Error> { | ||||
|             let _ = address; | ||||
|             let _ = operations; | ||||
|  | ||||
| @ -152,8 +152,8 @@ defmt = { version = "0.3", optional = true } | ||||
| log = { version = "0.4.14", optional = true } | ||||
| 
 | ||||
| embedded-hal-02 = { package = "embedded-hal", version = "0.2.6" } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9", optional = true} | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0", optional = true} | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10", optional = true} | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1", optional = true} | ||||
| 
 | ||||
| futures-util = { version = "0.3.17", default-features = false } | ||||
| embassy-sync = { version = "0.1", path = "../embassy-sync" } | ||||
|  | ||||
| @ -19,14 +19,12 @@ mod eh1 { | ||||
|     use super::*; | ||||
| 
 | ||||
|     impl embedded_hal_1::delay::DelayUs for Delay { | ||||
|         type Error = core::convert::Infallible; | ||||
| 
 | ||||
|         fn delay_us(&mut self, us: u32) -> Result<(), Self::Error> { | ||||
|             Ok(block_for(Duration::from_micros(us as u64))) | ||||
|         fn delay_us(&mut self, us: u32) { | ||||
|             block_for(Duration::from_micros(us as u64)) | ||||
|         } | ||||
| 
 | ||||
|         fn delay_ms(&mut self, ms: u32) -> Result<(), Self::Error> { | ||||
|             Ok(block_for(Duration::from_millis(ms as u64))) | ||||
|         fn delay_ms(&mut self, ms: u32) { | ||||
|             block_for(Duration::from_millis(ms as u64)) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -37,14 +35,12 @@ mod eha { | ||||
|     use crate::Timer; | ||||
| 
 | ||||
|     impl embedded_hal_async::delay::DelayUs for Delay { | ||||
|         type Error = core::convert::Infallible; | ||||
| 
 | ||||
|         async fn delay_us(&mut self, micros: u32) -> Result<(), Self::Error> { | ||||
|             Ok(Timer::after(Duration::from_micros(micros as _)).await) | ||||
|         async fn delay_us(&mut self, micros: u32) { | ||||
|             Timer::after(Duration::from_micros(micros as _)).await | ||||
|         } | ||||
| 
 | ||||
|         async fn delay_ms(&mut self, millis: u32) -> Result<(), Self::Error> { | ||||
|             Ok(Timer::after(Duration::from_millis(millis as _)).await) | ||||
|         async fn delay_ms(&mut self, millis: u32) { | ||||
|             Timer::after(Duration::from_millis(millis as _)).await | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| [target.'cfg(all(target_arch = "arm", target_os = "none"))'] | ||||
| runner = "probe-run --chip RP2040" | ||||
| runner = "probe-rs-cli run --chip RP2040" | ||||
| 
 | ||||
| [build] | ||||
| target = "thumbv6m-none-eabi"        # Cortex-M0 and Cortex-M0+ | ||||
|  | ||||
| @ -6,6 +6,7 @@ license = "MIT OR Apache-2.0" | ||||
| 
 | ||||
| 
 | ||||
| [dependencies] | ||||
| embassy-embedded-hal = { version = "0.1.0", path = "../../embassy-embedded-hal", features = ["defmt"] } | ||||
| embassy-sync = { version = "0.1.0", path = "../../embassy-sync", features = ["defmt"] } | ||||
| embassy-executor = { version = "0.1.0", path = "../../embassy-executor", features = ["arch-cortex-m", "executor-thread", "defmt", "integrated-timers"] } | ||||
| embassy-time = { version = "0.1.0", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime"] } | ||||
| @ -30,8 +31,8 @@ display-interface = "0.4.1" | ||||
| byte-slice-cast = { version = "1.2.0", default-features = false } | ||||
| smart-leds = "0.3.0" | ||||
| 
 | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9" } | ||||
| embedded-hal-async = "0.2.0-alpha.0" | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10" } | ||||
| embedded-hal-async = "0.2.0-alpha.1" | ||||
| embedded-io = { version = "0.4.0", features = ["async", "defmt"] } | ||||
| embedded-storage = { version = "0.3" } | ||||
| static_cell = "1.0.0" | ||||
|  | ||||
| @ -5,10 +5,13 @@ | ||||
| use core::cell::RefCell; | ||||
| 
 | ||||
| use defmt::*; | ||||
| use embassy_embedded_hal::shared_bus::blocking::spi::SpiDeviceWithConfig; | ||||
| use embassy_executor::Spawner; | ||||
| use embassy_rp::gpio::{Level, Output}; | ||||
| use embassy_rp::spi; | ||||
| use embassy_rp::spi::{Blocking, Spi}; | ||||
| use embassy_sync::blocking_mutex::raw::NoopRawMutex; | ||||
| use embassy_sync::blocking_mutex::Mutex; | ||||
| use embassy_time::Delay; | ||||
| use embedded_graphics::image::{Image, ImageRawLE}; | ||||
| use embedded_graphics::mono_font::ascii::FONT_10X20; | ||||
| @ -21,10 +24,9 @@ use st7789::{Orientation, ST7789}; | ||||
| use {defmt_rtt as _, panic_probe as _}; | ||||
| 
 | ||||
| use crate::my_display_interface::SPIDeviceInterface; | ||||
| use crate::shared_spi::SpiDeviceWithCs; | ||||
| use crate::touch::Touch; | ||||
| 
 | ||||
| //const DISPLAY_FREQ: u32 = 64_000_000;
 | ||||
| const DISPLAY_FREQ: u32 = 64_000_000; | ||||
| const TOUCH_FREQ: u32 = 200_000; | ||||
| 
 | ||||
| #[embassy_executor::main] | ||||
| @ -43,16 +45,20 @@ async fn main(_spawner: Spawner) { | ||||
|     //let touch_irq = p.PIN_17;
 | ||||
| 
 | ||||
|     // create SPI
 | ||||
|     let mut config = spi::Config::default(); | ||||
|     config.frequency = TOUCH_FREQ; // use the lowest freq
 | ||||
|     config.phase = spi::Phase::CaptureOnSecondTransition; | ||||
|     config.polarity = spi::Polarity::IdleHigh; | ||||
|     let mut display_config = spi::Config::default(); | ||||
|     display_config.frequency = DISPLAY_FREQ; | ||||
|     display_config.phase = spi::Phase::CaptureOnSecondTransition; | ||||
|     display_config.polarity = spi::Polarity::IdleHigh; | ||||
|     let mut touch_config = spi::Config::default(); | ||||
|     touch_config.frequency = TOUCH_FREQ; | ||||
|     touch_config.phase = spi::Phase::CaptureOnSecondTransition; | ||||
|     touch_config.polarity = spi::Polarity::IdleHigh; | ||||
| 
 | ||||
|     let spi: Spi<'_, _, Blocking> = Spi::new_blocking(p.SPI1, clk, mosi, miso, config); | ||||
|     let spi_bus = RefCell::new(spi); | ||||
|     let spi: Spi<'_, _, Blocking> = Spi::new_blocking(p.SPI1, clk, mosi, miso, touch_config.clone()); | ||||
|     let spi_bus: Mutex<NoopRawMutex, _> = Mutex::new(RefCell::new(spi)); | ||||
| 
 | ||||
|     let display_spi = SpiDeviceWithCs::new(&spi_bus, Output::new(display_cs, Level::High)); | ||||
|     let touch_spi = SpiDeviceWithCs::new(&spi_bus, Output::new(touch_cs, Level::High)); | ||||
|     let display_spi = SpiDeviceWithConfig::new(&spi_bus, Output::new(display_cs, Level::High), display_config); | ||||
|     let touch_spi = SpiDeviceWithConfig::new(&spi_bus, Output::new(touch_cs, Level::High), touch_config); | ||||
| 
 | ||||
|     let mut touch = Touch::new(touch_spi); | ||||
| 
 | ||||
| @ -104,85 +110,9 @@ async fn main(_spawner: Spawner) { | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| mod shared_spi { | ||||
|     use core::cell::RefCell; | ||||
|     use core::fmt::Debug; | ||||
| 
 | ||||
|     use embedded_hal_1::digital::OutputPin; | ||||
|     use embedded_hal_1::spi; | ||||
|     use embedded_hal_1::spi::SpiDevice; | ||||
| 
 | ||||
|     #[derive(Copy, Clone, Eq, PartialEq, Debug)] | ||||
|     pub enum SpiDeviceWithCsError<BUS, CS> { | ||||
|         #[allow(unused)] // will probably use in the future when adding a flush() to SpiBus
 | ||||
|         Spi(BUS), | ||||
|         Cs(CS), | ||||
|     } | ||||
| 
 | ||||
|     impl<BUS, CS> spi::Error for SpiDeviceWithCsError<BUS, CS> | ||||
|     where | ||||
|         BUS: spi::Error + Debug, | ||||
|         CS: Debug, | ||||
|     { | ||||
|         fn kind(&self) -> spi::ErrorKind { | ||||
|             match self { | ||||
|                 Self::Spi(e) => e.kind(), | ||||
|                 Self::Cs(_) => spi::ErrorKind::Other, | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     pub struct SpiDeviceWithCs<'a, BUS, CS> { | ||||
|         bus: &'a RefCell<BUS>, | ||||
|         cs: CS, | ||||
|     } | ||||
| 
 | ||||
|     impl<'a, BUS, CS> SpiDeviceWithCs<'a, BUS, CS> { | ||||
|         pub fn new(bus: &'a RefCell<BUS>, cs: CS) -> Self { | ||||
|             Self { bus, cs } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     impl<'a, BUS, CS> spi::ErrorType for SpiDeviceWithCs<'a, BUS, CS> | ||||
|     where | ||||
|         BUS: spi::ErrorType, | ||||
|         CS: OutputPin, | ||||
|     { | ||||
|         type Error = SpiDeviceWithCsError<BUS::Error, CS::Error>; | ||||
|     } | ||||
| 
 | ||||
|     impl<'a, BUS, CS> SpiDevice for SpiDeviceWithCs<'a, BUS, CS> | ||||
|     where | ||||
|         BUS: spi::SpiBusFlush, | ||||
|         CS: OutputPin, | ||||
|     { | ||||
|         type Bus = BUS; | ||||
| 
 | ||||
|         fn transaction<R>( | ||||
|             &mut self, | ||||
|             f: impl FnOnce(&mut Self::Bus) -> Result<R, BUS::Error>, | ||||
|         ) -> Result<R, Self::Error> { | ||||
|             let mut bus = self.bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceWithCsError::Cs)?; | ||||
| 
 | ||||
|             let f_res = f(&mut bus); | ||||
| 
 | ||||
|             // On failure, it's important to still flush and deassert CS.
 | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
| 
 | ||||
|             let f_res = f_res.map_err(SpiDeviceWithCsError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceWithCsError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceWithCsError::Cs)?; | ||||
| 
 | ||||
|             Ok(f_res) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /// Driver for the XPT2046 resistive touchscreen sensor
 | ||||
| mod touch { | ||||
|     use embedded_hal_1::spi::{SpiBus, SpiBusRead, SpiBusWrite, SpiDevice}; | ||||
|     use embedded_hal_1::spi::{Operation, SpiDevice}; | ||||
| 
 | ||||
|     struct Calibration { | ||||
|         x1: i32, | ||||
| @ -209,7 +139,6 @@ mod touch { | ||||
|     impl<SPI> Touch<SPI> | ||||
|     where | ||||
|         SPI: SpiDevice, | ||||
|         SPI::Bus: SpiBus, | ||||
|     { | ||||
|         pub fn new(spi: SPI) -> Self { | ||||
|             Self { spi } | ||||
| @ -219,13 +148,12 @@ mod touch { | ||||
|             let mut x = [0; 2]; | ||||
|             let mut y = [0; 2]; | ||||
|             self.spi | ||||
|                 .transaction(|bus| { | ||||
|                     bus.write(&[0x90])?; | ||||
|                     bus.read(&mut x)?; | ||||
|                     bus.write(&[0xd0])?; | ||||
|                     bus.read(&mut y)?; | ||||
|                     Ok(()) | ||||
|                 }) | ||||
|                 .transaction(&mut [ | ||||
|                     Operation::Write(&[0x90]), | ||||
|                     Operation::Read(&mut x), | ||||
|                     Operation::Write(&[0xd0]), | ||||
|                     Operation::Read(&mut y), | ||||
|                 ]) | ||||
|                 .unwrap(); | ||||
| 
 | ||||
|             let x = (u16::from_be_bytes(x) >> 3) as i32; | ||||
| @ -247,7 +175,7 @@ mod touch { | ||||
| mod my_display_interface { | ||||
|     use display_interface::{DataFormat, DisplayError, WriteOnlyDataCommand}; | ||||
|     use embedded_hal_1::digital::OutputPin; | ||||
|     use embedded_hal_1::spi::{SpiBusWrite, SpiDevice}; | ||||
|     use embedded_hal_1::spi::SpiDeviceWrite; | ||||
| 
 | ||||
|     /// SPI display interface.
 | ||||
|     ///
 | ||||
| @ -259,8 +187,7 @@ mod my_display_interface { | ||||
| 
 | ||||
|     impl<SPI, DC> SPIDeviceInterface<SPI, DC> | ||||
|     where | ||||
|         SPI: SpiDevice, | ||||
|         SPI::Bus: SpiBusWrite, | ||||
|         SPI: SpiDeviceWrite, | ||||
|         DC: OutputPin, | ||||
|     { | ||||
|         /// Create new SPI interface for communciation with a display driver
 | ||||
| @ -271,42 +198,27 @@ mod my_display_interface { | ||||
| 
 | ||||
|     impl<SPI, DC> WriteOnlyDataCommand for SPIDeviceInterface<SPI, DC> | ||||
|     where | ||||
|         SPI: SpiDevice, | ||||
|         SPI::Bus: SpiBusWrite, | ||||
|         SPI: SpiDeviceWrite, | ||||
|         DC: OutputPin, | ||||
|     { | ||||
|         fn send_commands(&mut self, cmds: DataFormat<'_>) -> Result<(), DisplayError> { | ||||
|             let r = self.spi.transaction(|bus| { | ||||
|                 // 1 = data, 0 = command
 | ||||
|                 if let Err(_) = self.dc.set_low() { | ||||
|                     return Ok(Err(DisplayError::DCError)); | ||||
|                 } | ||||
|             // 1 = data, 0 = command
 | ||||
|             self.dc.set_low().map_err(|_| DisplayError::DCError)?; | ||||
| 
 | ||||
|                 // Send words over SPI
 | ||||
|                 send_u8(bus, cmds)?; | ||||
| 
 | ||||
|                 Ok(Ok(())) | ||||
|             }); | ||||
|             r.map_err(|_| DisplayError::BusWriteError)? | ||||
|             send_u8(&mut self.spi, cmds).map_err(|_| DisplayError::BusWriteError)?; | ||||
|             Ok(()) | ||||
|         } | ||||
| 
 | ||||
|         fn send_data(&mut self, buf: DataFormat<'_>) -> Result<(), DisplayError> { | ||||
|             let r = self.spi.transaction(|bus| { | ||||
|                 // 1 = data, 0 = command
 | ||||
|                 if let Err(_) = self.dc.set_high() { | ||||
|                     return Ok(Err(DisplayError::DCError)); | ||||
|                 } | ||||
|             // 1 = data, 0 = command
 | ||||
|             self.dc.set_high().map_err(|_| DisplayError::DCError)?; | ||||
| 
 | ||||
|                 // Send words over SPI
 | ||||
|                 send_u8(bus, buf)?; | ||||
| 
 | ||||
|                 Ok(Ok(())) | ||||
|             }); | ||||
|             r.map_err(|_| DisplayError::BusWriteError)? | ||||
|             send_u8(&mut self.spi, buf).map_err(|_| DisplayError::BusWriteError)?; | ||||
|             Ok(()) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fn send_u8<T: SpiBusWrite>(spi: &mut T, words: DataFormat<'_>) -> Result<(), T::Error> { | ||||
|     fn send_u8<T: SpiDeviceWrite>(spi: &mut T, words: DataFormat<'_>) -> Result<(), T::Error> { | ||||
|         match words { | ||||
|             DataFormat::U8(slice) => spi.write(slice), | ||||
|             DataFormat::U16(slice) => { | ||||
|  | ||||
| @ -8,7 +8,7 @@ license = "MIT OR Apache-2.0" | ||||
| embassy-sync = { version = "0.1.0", path = "../../embassy-sync", features = ["defmt"] } | ||||
| embassy-executor = { version = "0.1.0", path = "../../embassy-executor", features = ["arch-cortex-m", "executor-thread", "defmt", "integrated-timers"] } | ||||
| embassy-time = { version = "0.1.0", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime", "tick-hz-32_768"] } | ||||
| embassy-stm32 = { version = "0.1.0", path = "../../embassy-stm32", features = ["nightly", "defmt", "stm32f103c8", "unstable-pac", "memory-x", "time-driver-any"]  } | ||||
| embassy-stm32 = { version = "0.1.0", path = "../../embassy-stm32", features = ["nightly", "defmt", "stm32f103c8", "unstable-pac", "memory-x", "time-driver-any", "unstable-traits" ]  } | ||||
| embassy-usb = { version = "0.1.0", path = "../../embassy-usb", features = ["defmt"] } | ||||
| embassy-futures = { version = "0.1.0", path = "../../embassy-futures" } | ||||
| 
 | ||||
|  | ||||
| @ -19,8 +19,8 @@ defmt-rtt = "0.4" | ||||
| cortex-m = { version = "0.7.6", features = ["critical-section-single-core"] } | ||||
| cortex-m-rt = "0.7.0" | ||||
| embedded-hal = "0.2.6" | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0" } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1" } | ||||
| embedded-nal-async = "0.4.0" | ||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||
|  | ||||
| @ -19,8 +19,8 @@ defmt-rtt = "0.4" | ||||
| cortex-m = { version = "0.7.6", features = ["critical-section-single-core"] } | ||||
| cortex-m-rt = "0.7.0" | ||||
| embedded-hal = "0.2.6" | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0" } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1" } | ||||
| embedded-nal-async = "0.4.0" | ||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||
|  | ||||
| @ -18,8 +18,8 @@ defmt-rtt = "0.4" | ||||
| cortex-m = { version = "0.7.6", features = ["critical-section-single-core"] } | ||||
| cortex-m-rt = "0.7.0" | ||||
| embedded-hal = "0.2.6" | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0" } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1" } | ||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||
| heapless = { version = "0.7.5", default-features = false } | ||||
|  | ||||
| @ -17,8 +17,8 @@ defmt-rtt = "0.4" | ||||
| cortex-m = { version = "0.7.6" } | ||||
| cortex-m-rt = "0.7.0" | ||||
| embedded-hal = "0.2.6" | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0" } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1" } | ||||
| panic-probe = { version = "0.3.0", features = ["print-defmt"] } | ||||
| futures = { version = "0.3.17", default-features = false, features = ["async-await"] } | ||||
| embedded-io = { version = "0.4.0", features = ["async"] } | ||||
|  | ||||
| @ -25,8 +25,8 @@ defmt-rtt = "0.4" | ||||
| cortex-m = { version = "0.7.6", features = ["critical-section-single-core"] } | ||||
| cortex-m-rt = "0.7.0" | ||||
| embedded-hal = "0.2.6" | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.9" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.0" } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1" } | ||||
| panic-probe = { version = "0.3.0", features = ["print-defmt"] } | ||||
| 
 | ||||
| [profile.dev] | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user