kernel/hil/
flash.rs

1// Licensed under the Apache License, Version 2.0 or the MIT License.
2// SPDX-License-Identifier: Apache-2.0 OR MIT
3// Copyright Tock Contributors 2022.
4
5//! Interface for reading, writing, and erasing flash storage pages.
6//!
7//! Operates on single pages. The page size is set by the associated type
8//! `page`. Here is an example of a page type and implementation of this trait:
9//!
10//! ```rust
11//! use core::ops::{Index, IndexMut};
12//!
13//! use kernel::hil;
14//! use kernel::ErrorCode;
15//!
16//! // Size in bytes
17//! const PAGE_SIZE: u32 = 1024;
18//!
19//! struct NewChipPage(pub [u8; PAGE_SIZE as usize]);
20//!
21//! impl Default for NewChipPage {
22//!     fn default() -> Self {
23//!         Self {
24//!             0: [0; PAGE_SIZE as usize],
25//!         }
26//!     }
27//! }
28//!
29//! impl NewChipPage {
30//!     fn len(&self) -> usize {
31//!         self.0.len()
32//!     }
33//! }
34//!
35//! impl Index<usize> for NewChipPage {
36//!     type Output = u8;
37//!
38//!     fn index(&self, idx: usize) -> &u8 {
39//!         &self.0[idx]
40//!     }
41//! }
42//!
43//! impl IndexMut<usize> for NewChipPage {
44//!     fn index_mut(&mut self, idx: usize) -> &mut u8 {
45//!         &mut self.0[idx]
46//!     }
47//! }
48//!
49//! impl AsMut<[u8]> for NewChipPage {
50//!     fn as_mut(&mut self) -> &mut [u8] {
51//!         &mut self.0
52//!     }
53//! }
54//!
55//! struct NewChipStruct {};
56//!
57//! impl<'a, C> hil::flash::HasClient<'a, C> for NewChipStruct {
58//!     fn set_client(&'a self, client: &'a C) { }
59//! }
60//!
61//! impl hil::flash::Flash for NewChipStruct {
62//!     type Page = NewChipPage;
63//!
64//!     fn read_page(&self, page_number: usize, buf: &'static mut Self::Page) -> Result<(), (ErrorCode, &'static mut Self::Page)> { Err((ErrorCode::FAIL, buf)) }
65//!     fn write_page(&self, page_number: usize, buf: &'static mut Self::Page) -> Result<(), (ErrorCode, &'static mut Self::Page)> { Err((ErrorCode::FAIL, buf)) }
66//!     fn erase_page(&self, page_number: usize) -> Result<(), ErrorCode> { Err(ErrorCode::FAIL) }
67//! }
68//! ```
69//!
70//! A user of this flash interface might look like:
71//!
72//! ```rust
73//! use kernel::utilities::cells::TakeCell;
74//! use kernel::hil;
75//!
76//! pub struct FlashUser<'a, F: hil::flash::Flash + 'static> {
77//!     driver: &'a F,
78//!     buffer: TakeCell<'static, F::Page>,
79//! }
80//!
81//! impl<'a, F: hil::flash::Flash> FlashUser<'a, F> {
82//!     pub fn new(driver: &'a F, buffer: &'static mut F::Page) -> FlashUser<'a, F> {
83//!         FlashUser {
84//!             driver: driver,
85//!             buffer: TakeCell::new(buffer),
86//!         }
87//!     }
88//! }
89//!
90//! impl<'a, F: hil::flash::Flash> hil::flash::Client<F> for FlashUser<'a, F> {
91//!     fn read_complete(&self, buffer: &'static mut F::Page, result: Result<(), hil::flash::Error>) {}
92//!     fn write_complete(&self, buffer: &'static mut F::Page, result: Result<(), hil::flash::Error>) { }
93//!     fn erase_complete(&self, result: Result<(), hil::flash::Error>) {}
94//! }
95//! ```
96
97use crate::ErrorCode;
98
99/// Flash errors returned in the callbacks.
100#[derive(Copy, Clone, Debug, Eq, PartialEq)]
101pub enum Error {
102    /// An error occurred during the flash operation.
103    FlashError,
104
105    /// A flash memory protection violation was detected.
106    FlashMemoryProtectionError,
107}
108
109pub trait HasClient<'a, C> {
110    /// Set the client for this flash peripheral. The client will be called
111    /// when operations complete.
112    fn set_client(&'a self, client: &'a C);
113}
114
115/// A page of writable persistent flash memory.
116pub trait Flash {
117    /// Type of a single flash page for the given implementation.
118    type Page: AsMut<[u8]> + Default;
119
120    /// Read a page of flash into the buffer.
121    fn read_page(
122        &self,
123        page_number: usize,
124        buf: &'static mut Self::Page,
125    ) -> Result<(), (ErrorCode, &'static mut Self::Page)>;
126
127    /// Write a page of flash from the buffer.
128    fn write_page(
129        &self,
130        page_number: usize,
131        buf: &'static mut Self::Page,
132    ) -> Result<(), (ErrorCode, &'static mut Self::Page)>;
133
134    /// Erase a page of flash by setting every byte to 0xFF.
135    fn erase_page(&self, page_number: usize) -> Result<(), ErrorCode>;
136}
137
138/// Implement `Client` to receive callbacks from `Flash`.
139pub trait Client<F: Flash> {
140    /// Flash read complete.
141    fn read_complete(&self, read_buffer: &'static mut F::Page, result: Result<(), Error>);
142
143    /// Flash write complete.
144    fn write_complete(&self, write_buffer: &'static mut F::Page, result: Result<(), Error>);
145
146    /// Flash erase complete.
147    fn erase_complete(&self, result: Result<(), Error>);
148}