1 | use core::marker::PhantomData; |
2 | use core::sync::atomic::{fence, Ordering}; |
3 | |
4 | use embassy_hal_internal::drop::OnDrop; |
5 | use embassy_hal_internal::{into_ref, PeripheralRef}; |
6 | |
7 | use super::{ |
8 | family, Async, Blocking, Error, FlashBank, FlashLayout, FlashRegion, FlashSector, FLASH_SIZE, MAX_ERASE_SIZE, |
9 | READ_SIZE, WRITE_SIZE, |
10 | }; |
11 | use crate::_generated::FLASH_BASE; |
12 | use crate::peripherals::FLASH; |
13 | use crate::Peripheral; |
14 | |
15 | /// Internal flash memory driver. |
16 | pub struct Flash<'d, MODE = Async> { |
17 | pub(crate) inner: PeripheralRef<'d, FLASH>, |
18 | pub(crate) _mode: PhantomData<MODE>, |
19 | } |
20 | |
21 | impl<'d> Flash<'d, Blocking> { |
22 | /// Create a new flash driver, usable in blocking mode. |
23 | pub fn new_blocking(p: impl Peripheral<P = FLASH> + 'd) -> Self { |
24 | into_ref!(p); |
25 | |
26 | Self { |
27 | inner: p, |
28 | _mode: PhantomData, |
29 | } |
30 | } |
31 | } |
32 | |
33 | impl<'d, MODE> Flash<'d, MODE> { |
34 | /// Split this flash driver into one instance per flash memory region. |
35 | /// |
36 | /// See module-level documentation for details on how memory regions work. |
37 | pub fn into_blocking_regions(self) -> FlashLayout<'d, Blocking> { |
38 | assert!(family::is_default_layout()); |
39 | FlashLayout::new(self.inner) |
40 | } |
41 | |
42 | /// Blocking read. |
43 | /// |
44 | /// NOTE: `offset` is an offset from the flash start, NOT an absolute address. |
45 | /// For example, to read address `0x0800_1234` you have to use offset `0x1234`. |
46 | pub fn blocking_read(&mut self, offset: u32, bytes: &mut [u8]) -> Result<(), Error> { |
47 | blocking_read(FLASH_BASE as u32, FLASH_SIZE as u32, offset, bytes) |
48 | } |
49 | |
50 | /// Blocking write. |
51 | /// |
52 | /// NOTE: `offset` is an offset from the flash start, NOT an absolute address. |
53 | /// For example, to write address `0x0800_1234` you have to use offset `0x1234`. |
54 | pub fn blocking_write(&mut self, offset: u32, bytes: &[u8]) -> Result<(), Error> { |
55 | unsafe { |
56 | blocking_write( |
57 | FLASH_BASE as u32, |
58 | FLASH_SIZE as u32, |
59 | offset, |
60 | bytes, |
61 | write_chunk_unlocked, |
62 | ) |
63 | } |
64 | } |
65 | |
66 | /// Blocking erase. |
67 | /// |
68 | /// NOTE: `from` and `to` are offsets from the flash start, NOT an absolute address. |
69 | /// For example, to erase address `0x0801_0000` you have to use offset `0x1_0000`. |
70 | pub fn blocking_erase(&mut self, from: u32, to: u32) -> Result<(), Error> { |
71 | unsafe { blocking_erase(FLASH_BASE as u32, from, to, erase_sector_unlocked) } |
72 | } |
73 | } |
74 | |
75 | pub(super) fn blocking_read(base: u32, size: u32, offset: u32, bytes: &mut [u8]) -> Result<(), Error> { |
76 | if offset + bytes.len() as u32 > size { |
77 | return Err(Error::Size); |
78 | } |
79 | |
80 | let start_address: u32 = base + offset; |
81 | |
82 | #[cfg (flash_f4)] |
83 | family::assert_not_corrupted_read(start_address + bytes.len() as u32); |
84 | |
85 | let flash_data: &[u8] = unsafe { core::slice::from_raw_parts(data:start_address as *const u8, bytes.len()) }; |
86 | bytes.copy_from_slice(src:flash_data); |
87 | Ok(()) |
88 | } |
89 | |
90 | pub(super) unsafe fn blocking_write( |
91 | base: u32, |
92 | size: u32, |
93 | offset: u32, |
94 | bytes: &[u8], |
95 | write_chunk: unsafe fn(u32, &[u8]) -> Result<(), Error>, |
96 | ) -> Result<(), Error> { |
97 | if offset + bytes.len() as u32 > size { |
98 | return Err(Error::Size); |
99 | } |
100 | if offset % WRITE_SIZE as u32 != 0 || bytes.len() % WRITE_SIZE != 0 { |
101 | return Err(Error::Unaligned); |
102 | } |
103 | |
104 | let mut address: u32 = base + offset; |
105 | trace!("Writing {} bytes at 0x{:x}" , bytes.len(), address); |
106 | |
107 | for chunk: &[u8] in bytes.chunks(WRITE_SIZE) { |
108 | write_chunk(address, chunk)?; |
109 | address += WRITE_SIZE as u32; |
110 | } |
111 | Ok(()) |
112 | } |
113 | |
114 | pub(super) unsafe fn write_chunk_unlocked(address: u32, chunk: &[u8]) -> Result<(), Error> { |
115 | family::clear_all_err(); |
116 | fence(order:Ordering::SeqCst); |
117 | family::unlock(); |
118 | fence(order:Ordering::SeqCst); |
119 | family::enable_blocking_write(); |
120 | fence(order:Ordering::SeqCst); |
121 | |
122 | let _on_drop: OnDrop = OnDrop::new(|| { |
123 | family::disable_blocking_write(); |
124 | fence(order:Ordering::SeqCst); |
125 | family::lock(); |
126 | }); |
127 | |
128 | family::blocking_write(address, buf:unwrap!(chunk.try_into())) |
129 | } |
130 | |
131 | pub(super) unsafe fn write_chunk_with_critical_section(address: u32, chunk: &[u8]) -> Result<(), Error> { |
132 | critical_section::with(|_| write_chunk_unlocked(address, chunk)) |
133 | } |
134 | |
135 | pub(super) unsafe fn blocking_erase( |
136 | base: u32, |
137 | from: u32, |
138 | to: u32, |
139 | erase_sector: unsafe fn(&FlashSector) -> Result<(), Error>, |
140 | ) -> Result<(), Error> { |
141 | let start_address: u32 = base + from; |
142 | let end_address: u32 = base + to; |
143 | let regions: &'static [&'static FlashRegion] = family::get_flash_regions(); |
144 | |
145 | ensure_sector_aligned(start_address, end_address, regions)?; |
146 | |
147 | trace!("Erasing from 0x{:x} to 0x{:x}" , start_address, end_address); |
148 | |
149 | let mut address: u32 = start_address; |
150 | while address < end_address { |
151 | let sector: FlashSector = get_sector(address, regions); |
152 | trace!("Erasing sector: {:?}" , sector); |
153 | erase_sector(§or)?; |
154 | address += sector.size; |
155 | } |
156 | Ok(()) |
157 | } |
158 | |
159 | pub(super) unsafe fn erase_sector_unlocked(sector: &FlashSector) -> Result<(), Error> { |
160 | family::clear_all_err(); |
161 | fence(order:Ordering::SeqCst); |
162 | family::unlock(); |
163 | fence(order:Ordering::SeqCst); |
164 | |
165 | let _on_drop: OnDrop = OnDrop::new(|| family::lock()); |
166 | |
167 | family::blocking_erase_sector(sector) |
168 | } |
169 | |
170 | pub(super) unsafe fn erase_sector_with_critical_section(sector: &FlashSector) -> Result<(), Error> { |
171 | critical_section::with(|_| erase_sector_unlocked(sector)) |
172 | } |
173 | |
174 | pub(super) fn get_sector(address: u32, regions: &[&FlashRegion]) -> FlashSector { |
175 | let mut current_bank: FlashBank = FlashBank::Bank1; |
176 | let mut bank_offset: u8 = 0; |
177 | for region: &&FlashRegion in regions { |
178 | if region.bank != current_bank { |
179 | current_bank = region.bank; |
180 | bank_offset = 0; |
181 | } |
182 | |
183 | if address >= region.base && address < region.end() { |
184 | let index_in_region: u32 = (address - region.base) / region.erase_size; |
185 | return FlashSector { |
186 | bank: region.bank, |
187 | index_in_bank: bank_offset + index_in_region as u8, |
188 | start: region.base + index_in_region * region.erase_size, |
189 | size: region.erase_size, |
190 | }; |
191 | } |
192 | |
193 | bank_offset += region.sectors(); |
194 | } |
195 | |
196 | panic!("Flash sector not found" ); |
197 | } |
198 | |
199 | pub(super) fn ensure_sector_aligned( |
200 | start_address: u32, |
201 | end_address: u32, |
202 | regions: &[&FlashRegion], |
203 | ) -> Result<(), Error> { |
204 | let mut address: u32 = start_address; |
205 | while address < end_address { |
206 | let sector: FlashSector = get_sector(address, regions); |
207 | if sector.start != address { |
208 | return Err(Error::Unaligned); |
209 | } |
210 | address += sector.size; |
211 | } |
212 | if address != end_address { |
213 | return Err(Error::Unaligned); |
214 | } |
215 | Ok(()) |
216 | } |
217 | |
218 | impl<MODE> embedded_storage::nor_flash::ErrorType for Flash<'_, MODE> { |
219 | type Error = Error; |
220 | } |
221 | |
222 | impl<MODE> embedded_storage::nor_flash::ReadNorFlash for Flash<'_, MODE> { |
223 | const READ_SIZE: usize = READ_SIZE; |
224 | |
225 | fn read(&mut self, offset: u32, bytes: &mut [u8]) -> Result<(), Self::Error> { |
226 | self.blocking_read(offset, bytes) |
227 | } |
228 | |
229 | fn capacity(&self) -> usize { |
230 | FLASH_SIZE |
231 | } |
232 | } |
233 | |
234 | impl<MODE> embedded_storage::nor_flash::NorFlash for Flash<'_, MODE> { |
235 | const WRITE_SIZE: usize = WRITE_SIZE; |
236 | const ERASE_SIZE: usize = MAX_ERASE_SIZE; |
237 | |
238 | fn write(&mut self, offset: u32, bytes: &[u8]) -> Result<(), Self::Error> { |
239 | self.blocking_write(offset, bytes) |
240 | } |
241 | |
242 | fn erase(&mut self, from: u32, to: u32) -> Result<(), Self::Error> { |
243 | self.blocking_erase(from, to) |
244 | } |
245 | } |
246 | |
247 | foreach_flash_region! { |
248 | ($type_name:ident, $write_size:literal, $erase_size:literal) => { |
249 | impl<MODE> crate::_generated::flash_regions::$type_name<'_, MODE> { |
250 | /// Blocking read. |
251 | /// |
252 | /// NOTE: `offset` is an offset from the flash start, NOT an absolute address. |
253 | /// For example, to read address `0x0800_1234` you have to use offset `0x1234`. |
254 | pub fn blocking_read(&mut self, offset: u32, bytes: &mut [u8]) -> Result<(), Error> { |
255 | blocking_read(self.0.base, self.0.size, offset, bytes) |
256 | } |
257 | } |
258 | |
259 | impl crate::_generated::flash_regions::$type_name<'_, Blocking> { |
260 | /// Blocking write. |
261 | /// |
262 | /// NOTE: `offset` is an offset from the flash start, NOT an absolute address. |
263 | /// For example, to write address `0x0800_1234` you have to use offset `0x1234`. |
264 | pub fn blocking_write(&mut self, offset: u32, bytes: &[u8]) -> Result<(), Error> { |
265 | unsafe { blocking_write(self.0.base, self.0.size, offset, bytes, write_chunk_with_critical_section) } |
266 | } |
267 | |
268 | /// Blocking erase. |
269 | /// |
270 | /// NOTE: `from` and `to` are offsets from the flash start, NOT an absolute address. |
271 | /// For example, to erase address `0x0801_0000` you have to use offset `0x1_0000`. |
272 | pub fn blocking_erase(&mut self, from: u32, to: u32) -> Result<(), Error> { |
273 | unsafe { blocking_erase(self.0.base, from, to, erase_sector_with_critical_section) } |
274 | } |
275 | } |
276 | |
277 | impl<MODE> embedded_storage::nor_flash::ErrorType for crate::_generated::flash_regions::$type_name<'_, MODE> { |
278 | type Error = Error; |
279 | } |
280 | |
281 | impl<MODE> embedded_storage::nor_flash::ReadNorFlash for crate::_generated::flash_regions::$type_name<'_, MODE> { |
282 | const READ_SIZE: usize = READ_SIZE; |
283 | |
284 | fn read(&mut self, offset: u32, bytes: &mut [u8]) -> Result<(), Self::Error> { |
285 | self.blocking_read(offset, bytes) |
286 | } |
287 | |
288 | fn capacity(&self) -> usize { |
289 | self.0.size as usize |
290 | } |
291 | } |
292 | |
293 | impl embedded_storage::nor_flash::NorFlash for crate::_generated::flash_regions::$type_name<'_, Blocking> { |
294 | const WRITE_SIZE: usize = $write_size; |
295 | const ERASE_SIZE: usize = $erase_size; |
296 | |
297 | fn write(&mut self, offset: u32, bytes: &[u8]) -> Result<(), Self::Error> { |
298 | self.blocking_write(offset, bytes) |
299 | } |
300 | |
301 | fn erase(&mut self, from: u32, to: u32) -> Result<(), Self::Error> { |
302 | self.blocking_erase(from, to) |
303 | } |
304 | } |
305 | }; |
306 | } |
307 | |