aboutsummaryrefslogtreecommitdiff
path: root/embassy-embedded-hal/src
diff options
context:
space:
mode:
Diffstat (limited to 'embassy-embedded-hal/src')
-rw-r--r--embassy-embedded-hal/src/adapter/blocking_async.rs (renamed from embassy-embedded-hal/src/adapter.rs)2
-rw-r--r--embassy-embedded-hal/src/adapter/mod.rs7
-rw-r--r--embassy-embedded-hal/src/adapter/yielding_async.rs232
-rw-r--r--embassy-embedded-hal/src/flash.rs286
-rw-r--r--embassy-embedded-hal/src/lib.rs2
5 files changed, 527 insertions, 2 deletions
diff --git a/embassy-embedded-hal/src/adapter.rs b/embassy-embedded-hal/src/adapter/blocking_async.rs
index 171ff6c9f..b996d6a75 100644
--- a/embassy-embedded-hal/src/adapter.rs
+++ b/embassy-embedded-hal/src/adapter/blocking_async.rs
@@ -1,5 +1,3 @@
1//! Adapters between embedded-hal traits.
2
3use embedded_hal_02::{blocking, serial}; 1use embedded_hal_02::{blocking, serial};
4 2
5/// Wrapper that implements async traits using blocking implementations. 3/// Wrapper that implements async traits using blocking implementations.
diff --git a/embassy-embedded-hal/src/adapter/mod.rs b/embassy-embedded-hal/src/adapter/mod.rs
new file mode 100644
index 000000000..28da56137
--- /dev/null
+++ b/embassy-embedded-hal/src/adapter/mod.rs
@@ -0,0 +1,7 @@
1//! Adapters between embedded-hal traits.
2
3mod blocking_async;
4mod yielding_async;
5
6pub use blocking_async::BlockingAsync;
7pub use yielding_async::YieldingAsync;
diff --git a/embassy-embedded-hal/src/adapter/yielding_async.rs b/embassy-embedded-hal/src/adapter/yielding_async.rs
new file mode 100644
index 000000000..96d5cca8e
--- /dev/null
+++ b/embassy-embedded-hal/src/adapter/yielding_async.rs
@@ -0,0 +1,232 @@
1use embassy_futures::yield_now;
2
3/// Wrapper that yields for each operation to the wrapped instance
4///
5/// This can be used in combination with BlockingAsync<T> to enforce yields
6/// between long running blocking operations.
7pub struct YieldingAsync<T> {
8 wrapped: T,
9}
10
11impl<T> YieldingAsync<T> {
12 /// Create a new instance of a wrapper that yields after each operation.
13 pub fn new(wrapped: T) -> Self {
14 Self { wrapped }
15 }
16}
17
18//
19// I2C implementations
20//
21impl<T> embedded_hal_1::i2c::ErrorType for YieldingAsync<T>
22where
23 T: embedded_hal_1::i2c::ErrorType,
24{
25 type Error = T::Error;
26}
27
28impl<T> embedded_hal_async::i2c::I2c for YieldingAsync<T>
29where
30 T: embedded_hal_async::i2c::I2c,
31{
32 async fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Self::Error> {
33 self.wrapped.read(address, read).await?;
34 yield_now().await;
35 Ok(())
36 }
37
38 async fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Self::Error> {
39 self.wrapped.write(address, write).await?;
40 yield_now().await;
41 Ok(())
42 }
43
44 async fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> {
45 self.wrapped.write_read(address, write, read).await?;
46 yield_now().await;
47 Ok(())
48 }
49
50 async fn transaction(
51 &mut self,
52 address: u8,
53 operations: &mut [embedded_hal_1::i2c::Operation<'_>],
54 ) -> Result<(), Self::Error> {
55 self.wrapped.transaction(address, operations).await?;
56 yield_now().await;
57 Ok(())
58 }
59}
60
61//
62// SPI implementations
63//
64
65impl<T> embedded_hal_async::spi::ErrorType for YieldingAsync<T>
66where
67 T: embedded_hal_async::spi::ErrorType,
68{
69 type Error = T::Error;
70}
71
72impl<T> embedded_hal_async::spi::SpiBus<u8> for YieldingAsync<T>
73where
74 T: embedded_hal_async::spi::SpiBus,
75{
76 async fn transfer<'a>(&'a mut self, read: &'a mut [u8], write: &'a [u8]) -> Result<(), Self::Error> {
77 self.wrapped.transfer(read, write).await?;
78 yield_now().await;
79 Ok(())
80 }
81
82 async fn transfer_in_place<'a>(&'a mut self, words: &'a mut [u8]) -> Result<(), Self::Error> {
83 self.wrapped.transfer_in_place(words).await?;
84 yield_now().await;
85 Ok(())
86 }
87}
88
89impl<T> embedded_hal_async::spi::SpiBusFlush for YieldingAsync<T>
90where
91 T: embedded_hal_async::spi::SpiBusFlush,
92{
93 async fn flush(&mut self) -> Result<(), Self::Error> {
94 self.wrapped.flush().await?;
95 yield_now().await;
96 Ok(())
97 }
98}
99
100impl<T> embedded_hal_async::spi::SpiBusWrite<u8> for YieldingAsync<T>
101where
102 T: embedded_hal_async::spi::SpiBusWrite<u8>,
103{
104 async fn write(&mut self, data: &[u8]) -> Result<(), Self::Error> {
105 self.wrapped.write(data).await?;
106 yield_now().await;
107 Ok(())
108 }
109}
110
111impl<T> embedded_hal_async::spi::SpiBusRead<u8> for YieldingAsync<T>
112where
113 T: embedded_hal_async::spi::SpiBusRead<u8>,
114{
115 async fn read(&mut self, data: &mut [u8]) -> Result<(), Self::Error> {
116 self.wrapped.read(data).await?;
117 yield_now().await;
118 Ok(())
119 }
120}
121
122///
123/// NOR flash implementations
124///
125impl<T: embedded_storage::nor_flash::ErrorType> embedded_storage::nor_flash::ErrorType for YieldingAsync<T> {
126 type Error = T::Error;
127}
128
129impl<T: embedded_storage_async::nor_flash::ReadNorFlash> embedded_storage_async::nor_flash::ReadNorFlash
130 for YieldingAsync<T>
131{
132 const READ_SIZE: usize = T::READ_SIZE;
133
134 async fn read(&mut self, offset: u32, bytes: &mut [u8]) -> Result<(), Self::Error> {
135 self.wrapped.read(offset, bytes).await?;
136 Ok(())
137 }
138
139 fn capacity(&self) -> usize {
140 self.wrapped.capacity()
141 }
142}
143
144impl<T: embedded_storage_async::nor_flash::NorFlash> embedded_storage_async::nor_flash::NorFlash for YieldingAsync<T> {
145 const WRITE_SIZE: usize = T::WRITE_SIZE;
146 const ERASE_SIZE: usize = T::ERASE_SIZE;
147
148 async fn write(&mut self, offset: u32, bytes: &[u8]) -> Result<(), Self::Error> {
149 self.wrapped.write(offset, bytes).await?;
150 yield_now().await;
151 Ok(())
152 }
153
154 async fn erase(&mut self, from: u32, to: u32) -> Result<(), Self::Error> {
155 // Yield between each actual erase
156 for from in (from..to).step_by(T::ERASE_SIZE) {
157 let to = core::cmp::min(from + T::ERASE_SIZE as u32, to);
158 self.wrapped.erase(from, to).await?;
159 yield_now().await;
160 }
161 Ok(())
162 }
163}
164
165#[cfg(test)]
166mod tests {
167 use embedded_storage_async::nor_flash::NorFlash;
168
169 use super::*;
170
171 extern crate std;
172
173 #[derive(Default)]
174 struct FakeFlash(Vec<(u32, u32)>);
175
176 impl embedded_storage::nor_flash::ErrorType for FakeFlash {
177 type Error = std::convert::Infallible;
178 }
179
180 impl embedded_storage_async::nor_flash::ReadNorFlash for FakeFlash {
181 const READ_SIZE: usize = 1;
182
183 async fn read(&mut self, _offset: u32, _bytes: &mut [u8]) -> Result<(), Self::Error> {
184 unimplemented!()
185 }
186
187 fn capacity(&self) -> usize {
188 unimplemented!()
189 }
190 }
191
192 impl embedded_storage_async::nor_flash::NorFlash for FakeFlash {
193 const WRITE_SIZE: usize = 4;
194 const ERASE_SIZE: usize = 128;
195
196 async fn write(&mut self, _offset: u32, _bytes: &[u8]) -> Result<(), Self::Error> {
197 unimplemented!()
198 }
199
200 async fn erase(&mut self, from: u32, to: u32) -> Result<(), Self::Error> {
201 self.0.push((from, to));
202 Ok(())
203 }
204 }
205
206 #[futures_test::test]
207 async fn can_erase() {
208 let fake = FakeFlash::default();
209 let mut yielding = YieldingAsync::new(fake);
210
211 yielding.erase(0, 256).await.unwrap();
212
213 let fake = yielding.wrapped;
214 assert_eq!(2, fake.0.len());
215 assert_eq!((0, 128), fake.0[0]);
216 assert_eq!((128, 256), fake.0[1]);
217 }
218
219 #[futures_test::test]
220 async fn can_erase_wrong_erase_size() {
221 let fake = FakeFlash::default();
222 let mut yielding = YieldingAsync::new(fake);
223
224 yielding.erase(0, 257).await.unwrap();
225
226 let fake = yielding.wrapped;
227 assert_eq!(3, fake.0.len());
228 assert_eq!((0, 128), fake.0[0]);
229 assert_eq!((128, 256), fake.0[1]);
230 assert_eq!((256, 257), fake.0[2]);
231 }
232}
diff --git a/embassy-embedded-hal/src/flash.rs b/embassy-embedded-hal/src/flash.rs
new file mode 100644
index 000000000..9a6e4bd92
--- /dev/null
+++ b/embassy-embedded-hal/src/flash.rs
@@ -0,0 +1,286 @@
1//! Utilities related to flash.
2
3use embedded_storage::nor_flash::{ErrorType, NorFlash, NorFlashError, ReadNorFlash};
4#[cfg(feature = "nightly")]
5use embedded_storage_async::nor_flash::{NorFlash as AsyncNorFlash, ReadNorFlash as AsyncReadNorFlash};
6
7/// Convenience helper for concatenating two consecutive flashes into one.
8/// This is especially useful if used with "flash regions", where one may
9/// want to concatenate multiple regions into one larger region.
10pub struct ConcatFlash<First, Second>(First, Second);
11
12impl<First, Second> ConcatFlash<First, Second> {
13 /// Create a new flash that concatenates two consecutive flashes.
14 pub fn new(first: First, second: Second) -> Self {
15 Self(first, second)
16 }
17}
18
19const fn get_read_size(first_read_size: usize, second_read_size: usize) -> usize {
20 if first_read_size != second_read_size {
21 panic!("The read size for the concatenated flashes must be the same");
22 }
23 first_read_size
24}
25
26const fn get_write_size(first_write_size: usize, second_write_size: usize) -> usize {
27 if first_write_size != second_write_size {
28 panic!("The write size for the concatenated flashes must be the same");
29 }
30 first_write_size
31}
32
33const fn get_max_erase_size(first_erase_size: usize, second_erase_size: usize) -> usize {
34 let max_erase_size = if first_erase_size > second_erase_size {
35 first_erase_size
36 } else {
37 second_erase_size
38 };
39 if max_erase_size % first_erase_size != 0 || max_erase_size % second_erase_size != 0 {
40 panic!("The erase sizes for the concatenated flashes must have have a gcd equal to the max erase size");
41 }
42 max_erase_size
43}
44
45impl<First, Second, E> ErrorType for ConcatFlash<First, Second>
46where
47 First: ErrorType<Error = E>,
48 Second: ErrorType<Error = E>,
49 E: NorFlashError,
50{
51 type Error = E;
52}
53
54impl<First, Second, E> ReadNorFlash for ConcatFlash<First, Second>
55where
56 First: ReadNorFlash<Error = E>,
57 Second: ReadNorFlash<Error = E>,
58 E: NorFlashError,
59{
60 const READ_SIZE: usize = get_read_size(First::READ_SIZE, Second::READ_SIZE);
61
62 fn read(&mut self, mut offset: u32, mut bytes: &mut [u8]) -> Result<(), E> {
63 if offset < self.0.capacity() as u32 {
64 let len = core::cmp::min(self.0.capacity() - offset as usize, bytes.len());
65 self.0.read(offset, &mut bytes[..len])?;
66 offset += len as u32;
67 bytes = &mut bytes[len..];
68 }
69
70 if !bytes.is_empty() {
71 self.1.read(offset - self.0.capacity() as u32, bytes)?;
72 }
73
74 Ok(())
75 }
76
77 fn capacity(&self) -> usize {
78 self.0.capacity() + self.1.capacity()
79 }
80}
81
82impl<First, Second, E> NorFlash for ConcatFlash<First, Second>
83where
84 First: NorFlash<Error = E>,
85 Second: NorFlash<Error = E>,
86 E: NorFlashError,
87{
88 const WRITE_SIZE: usize = get_write_size(First::WRITE_SIZE, Second::WRITE_SIZE);
89 const ERASE_SIZE: usize = get_max_erase_size(First::ERASE_SIZE, Second::ERASE_SIZE);
90
91 fn write(&mut self, mut offset: u32, mut bytes: &[u8]) -> Result<(), E> {
92 if offset < self.0.capacity() as u32 {
93 let len = core::cmp::min(self.0.capacity() - offset as usize, bytes.len());
94 self.0.write(offset, &bytes[..len])?;
95 offset += len as u32;
96 bytes = &bytes[len..];
97 }
98
99 if !bytes.is_empty() {
100 self.1.write(offset - self.0.capacity() as u32, bytes)?;
101 }
102
103 Ok(())
104 }
105
106 fn erase(&mut self, mut from: u32, to: u32) -> Result<(), E> {
107 if from < self.0.capacity() as u32 {
108 let to = core::cmp::min(self.0.capacity() as u32, to);
109 self.0.erase(from, to)?;
110 from = self.0.capacity() as u32;
111 }
112
113 if from < to {
114 self.1
115 .erase(from - self.0.capacity() as u32, to - self.0.capacity() as u32)?;
116 }
117
118 Ok(())
119 }
120}
121
122#[cfg(feature = "nightly")]
123impl<First, Second, E> AsyncReadNorFlash for ConcatFlash<First, Second>
124where
125 First: AsyncReadNorFlash<Error = E>,
126 Second: AsyncReadNorFlash<Error = E>,
127 E: NorFlashError,
128{
129 const READ_SIZE: usize = get_read_size(First::READ_SIZE, Second::READ_SIZE);
130
131 async fn read(&mut self, mut offset: u32, mut bytes: &mut [u8]) -> Result<(), E> {
132 if offset < self.0.capacity() as u32 {
133 let len = core::cmp::min(self.0.capacity() - offset as usize, bytes.len());
134 self.0.read(offset, &mut bytes[..len]).await?;
135 offset += len as u32;
136 bytes = &mut bytes[len..];
137 }
138
139 if !bytes.is_empty() {
140 self.1.read(offset - self.0.capacity() as u32, bytes).await?;
141 }
142
143 Ok(())
144 }
145
146 fn capacity(&self) -> usize {
147 self.0.capacity() + self.1.capacity()
148 }
149}
150
151#[cfg(feature = "nightly")]
152impl<First, Second, E> AsyncNorFlash for ConcatFlash<First, Second>
153where
154 First: AsyncNorFlash<Error = E>,
155 Second: AsyncNorFlash<Error = E>,
156 E: NorFlashError,
157{
158 const WRITE_SIZE: usize = get_write_size(First::WRITE_SIZE, Second::WRITE_SIZE);
159 const ERASE_SIZE: usize = get_max_erase_size(First::ERASE_SIZE, Second::ERASE_SIZE);
160
161 async fn write(&mut self, mut offset: u32, mut bytes: &[u8]) -> Result<(), E> {
162 if offset < self.0.capacity() as u32 {
163 let len = core::cmp::min(self.0.capacity() - offset as usize, bytes.len());
164 self.0.write(offset, &bytes[..len]).await?;
165 offset += len as u32;
166 bytes = &bytes[len..];
167 }
168
169 if !bytes.is_empty() {
170 self.1.write(offset - self.0.capacity() as u32, bytes).await?;
171 }
172
173 Ok(())
174 }
175
176 async fn erase(&mut self, mut from: u32, to: u32) -> Result<(), E> {
177 if from < self.0.capacity() as u32 {
178 let to = core::cmp::min(self.0.capacity() as u32, to);
179 self.0.erase(from, to).await?;
180 from = self.0.capacity() as u32;
181 }
182
183 if from < to {
184 self.1
185 .erase(from - self.0.capacity() as u32, to - self.0.capacity() as u32)
186 .await?;
187 }
188
189 Ok(())
190 }
191}
192
193#[cfg(test)]
194mod tests {
195 use super::*;
196
197 #[test]
198 fn can_write_and_read_across_flashes() {
199 let first = MemFlash::<64, 16, 4>::new();
200 let second = MemFlash::<64, 64, 4>::new();
201 let mut f = ConcatFlash::new(first, second);
202
203 f.write(60, &[0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0x88]).unwrap();
204
205 assert_eq!(&[0x11, 0x22, 0x33, 0x44], &f.0 .0[60..]);
206 assert_eq!(&[0x55, 0x66, 0x77, 0x88], &f.1 .0[0..4]);
207
208 let mut read_buf = [0; 8];
209 f.read(60, &mut read_buf).unwrap();
210
211 assert_eq!(&[0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0x88], &read_buf);
212 }
213
214 #[test]
215 fn can_erase_across_flashes() {
216 let mut first = MemFlash::<128, 16, 4>::new();
217 let mut second = MemFlash::<128, 64, 4>::new();
218 first.0.fill(0x00);
219 second.0.fill(0x00);
220
221 let mut f = ConcatFlash::new(first, second);
222
223 f.erase(64, 192).unwrap();
224
225 assert_eq!(&[0x00; 64], &f.0 .0[0..64]);
226 assert_eq!(&[0xff; 64], &f.0 .0[64..128]);
227 assert_eq!(&[0xff; 64], &f.1 .0[0..64]);
228 assert_eq!(&[0x00; 64], &f.1 .0[64..128]);
229 }
230
231 pub struct MemFlash<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize>([u8; SIZE]);
232
233 impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE> {
234 pub const fn new() -> Self {
235 Self([0xff; SIZE])
236 }
237 }
238
239 impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> ErrorType
240 for MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE>
241 {
242 type Error = core::convert::Infallible;
243 }
244
245 impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> ReadNorFlash
246 for MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE>
247 {
248 const READ_SIZE: usize = 1;
249
250 fn read(&mut self, offset: u32, bytes: &mut [u8]) -> Result<(), Self::Error> {
251 let len = bytes.len();
252 bytes.copy_from_slice(&self.0[offset as usize..offset as usize + len]);
253 Ok(())
254 }
255
256 fn capacity(&self) -> usize {
257 SIZE
258 }
259 }
260
261 impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> NorFlash
262 for MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE>
263 {
264 const WRITE_SIZE: usize = WRITE_SIZE;
265 const ERASE_SIZE: usize = ERASE_SIZE;
266
267 fn erase(&mut self, from: u32, to: u32) -> Result<(), Self::Error> {
268 let from = from as usize;
269 let to = to as usize;
270 assert_eq!(0, from % ERASE_SIZE);
271 assert_eq!(0, to % ERASE_SIZE);
272 self.0[from..to].fill(0xff);
273 Ok(())
274 }
275
276 fn write(&mut self, offset: u32, bytes: &[u8]) -> Result<(), Self::Error> {
277 let offset = offset as usize;
278 assert_eq!(0, bytes.len() % WRITE_SIZE);
279 assert_eq!(0, offset % WRITE_SIZE);
280 assert!(offset + bytes.len() <= SIZE);
281
282 self.0[offset..offset + bytes.len()].copy_from_slice(bytes);
283 Ok(())
284 }
285 }
286}
diff --git a/embassy-embedded-hal/src/lib.rs b/embassy-embedded-hal/src/lib.rs
index 73c81b465..3aad838bd 100644
--- a/embassy-embedded-hal/src/lib.rs
+++ b/embassy-embedded-hal/src/lib.rs
@@ -7,6 +7,8 @@
7#[cfg(feature = "nightly")] 7#[cfg(feature = "nightly")]
8pub mod adapter; 8pub mod adapter;
9 9
10pub mod flash;
11
10pub mod shared_bus; 12pub mod shared_bus;
11 13
12/// Set the configuration of a peripheral driver. 14/// Set the configuration of a peripheral driver.