Remove the usage of the local Partition type in BootLoader
This commit is contained in:
parent
3b38079490
commit
94046f30ff
@ -39,6 +39,7 @@ env_logger = "0.9"
|
|||||||
rand = "0.7" # ed25519-dalek v1.0.1 depends on this exact version
|
rand = "0.7" # ed25519-dalek v1.0.1 depends on this exact version
|
||||||
futures = { version = "0.3", features = ["executor"] }
|
futures = { version = "0.3", features = ["executor"] }
|
||||||
sha1 = "0.10.5"
|
sha1 = "0.10.5"
|
||||||
|
embassy-embedded-hal = { version = "0.1.0", path = "../../embassy-embedded-hal" }
|
||||||
|
|
||||||
[dev-dependencies.ed25519-dalek]
|
[dev-dependencies.ed25519-dalek]
|
||||||
default_features = false
|
default_features = false
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
use embedded_storage::nor_flash::{ErrorType, NorFlash, NorFlashError, NorFlashErrorKind, ReadNorFlash};
|
use embedded_storage::nor_flash::{ErrorType, NorFlash, NorFlashError, NorFlashErrorKind, ReadNorFlash};
|
||||||
|
|
||||||
use crate::{Partition, State, BOOT_MAGIC, SWAP_MAGIC};
|
use crate::{State, BOOT_MAGIC, SWAP_MAGIC};
|
||||||
|
|
||||||
|
const STATE_ERASE_VALUE: u8 = 0xFF;
|
||||||
|
|
||||||
/// Errors returned by bootloader
|
/// Errors returned by bootloader
|
||||||
#[derive(PartialEq, Eq, Debug)]
|
#[derive(PartialEq, Eq, Debug)]
|
||||||
@ -30,65 +32,39 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Trait defining the flash handles used for active and DFU partition.
|
|
||||||
pub trait FlashConfig {
|
|
||||||
/// The erase value of the state flash. Typically the default of 0xFF is used, but some flashes use a different value.
|
|
||||||
const STATE_ERASE_VALUE: u8 = 0xFF;
|
|
||||||
/// Flash type used for the state partition.
|
|
||||||
type STATE: NorFlash;
|
|
||||||
/// Flash type used for the active partition.
|
|
||||||
type ACTIVE: NorFlash;
|
|
||||||
/// Flash type used for the dfu partition.
|
|
||||||
type DFU: NorFlash;
|
|
||||||
|
|
||||||
/// Return flash instance used to write/read to/from active partition.
|
|
||||||
fn active(&mut self) -> &mut Self::ACTIVE;
|
|
||||||
/// Return flash instance used to write/read to/from dfu partition.
|
|
||||||
fn dfu(&mut self) -> &mut Self::DFU;
|
|
||||||
/// Return flash instance used to write/read to/from bootloader state.
|
|
||||||
fn state(&mut self) -> &mut Self::STATE;
|
|
||||||
}
|
|
||||||
|
|
||||||
trait FlashConfigEx {
|
|
||||||
fn page_size() -> u32;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: FlashConfig> FlashConfigEx for T {
|
|
||||||
/// Get the page size which is the "unit of operation" within the bootloader.
|
|
||||||
fn page_size() -> u32 {
|
|
||||||
core::cmp::max(T::ACTIVE::ERASE_SIZE, T::DFU::ERASE_SIZE) as u32
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// BootLoader works with any flash implementing embedded_storage.
|
/// BootLoader works with any flash implementing embedded_storage.
|
||||||
pub struct BootLoader {
|
pub struct BootLoader<ACTIVE: NorFlash, DFU: NorFlash, STATE: NorFlash> {
|
||||||
// Page with current state of bootloader. The state partition has the following format:
|
/// Flash type used for the active partition - the partition which will be booted from.
|
||||||
// All ranges are in multiples of WRITE_SIZE bytes.
|
active: ACTIVE,
|
||||||
// | Range | Description |
|
/// Flash type used for the dfu partition - he partition which will be swapped in when requested.
|
||||||
// | 0..1 | Magic indicating bootloader state. BOOT_MAGIC means boot, SWAP_MAGIC means swap. |
|
dfu: DFU,
|
||||||
// | 1..2 | Progress validity. ERASE_VALUE means valid, !ERASE_VALUE means invalid. |
|
/// Flash type used for the state partition.
|
||||||
// | 2..2 + N | Progress index used while swapping or reverting |
|
///
|
||||||
state: Partition,
|
/// The state partition has the following format:
|
||||||
// Location of the partition which will be booted from
|
/// All ranges are in multiples of WRITE_SIZE bytes.
|
||||||
active: Partition,
|
/// | Range | Description |
|
||||||
// Location of the partition which will be swapped in when requested
|
/// | 0..1 | Magic indicating bootloader state. BOOT_MAGIC means boot, SWAP_MAGIC means swap. |
|
||||||
dfu: Partition,
|
/// | 1..2 | Progress validity. ERASE_VALUE means valid, !ERASE_VALUE means invalid. |
|
||||||
|
/// | 2..2 + N | Progress index used while swapping or reverting
|
||||||
|
state: STATE,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BootLoader {
|
impl<ACTIVE: NorFlash, DFU: NorFlash, STATE: NorFlash> BootLoader<ACTIVE, DFU, STATE> {
|
||||||
/// Create a new instance of a bootloader with the given partitions.
|
/// Get the page size which is the "unit of operation" within the bootloader.
|
||||||
|
const PAGE_SIZE: u32 = if ACTIVE::ERASE_SIZE > DFU::ERASE_SIZE {
|
||||||
|
ACTIVE::ERASE_SIZE as u32
|
||||||
|
} else {
|
||||||
|
DFU::ERASE_SIZE as u32
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Create a new instance of a bootloader with the flash partitions.
|
||||||
///
|
///
|
||||||
/// - All partitions must be aligned with the PAGE_SIZE const generic parameter.
|
/// - All partitions must be aligned with the PAGE_SIZE const generic parameter.
|
||||||
/// - The dfu partition must be at least PAGE_SIZE bigger than the active partition.
|
/// - The dfu partition must be at least PAGE_SIZE bigger than the active partition.
|
||||||
pub fn new(active: Partition, dfu: Partition, state: Partition) -> Self {
|
pub fn new(active: ACTIVE, dfu: DFU, state: STATE) -> Self {
|
||||||
Self { active, dfu, state }
|
Self { active, dfu, state }
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Return the offset of the active partition into the active flash.
|
|
||||||
pub fn boot_address(&self) -> usize {
|
|
||||||
self.active.from as usize
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Perform necessary boot preparations like swapping images.
|
/// Perform necessary boot preparations like swapping images.
|
||||||
///
|
///
|
||||||
/// The DFU partition is assumed to be 1 page bigger than the active partition for the swap
|
/// The DFU partition is assumed to be 1 page bigger than the active partition for the swap
|
||||||
@ -175,195 +151,174 @@ impl BootLoader {
|
|||||||
/// | DFU | 3 | 3 | 2 | 1 | 3 |
|
/// | DFU | 3 | 3 | 2 | 1 | 3 |
|
||||||
/// +-----------+--------------+--------+--------+--------+--------+
|
/// +-----------+--------------+--------+--------+--------+--------+
|
||||||
///
|
///
|
||||||
pub fn prepare_boot<P: FlashConfig>(&mut self, p: &mut P, aligned_buf: &mut [u8]) -> Result<State, BootError> {
|
pub fn prepare_boot(&mut self, aligned_buf: &mut [u8]) -> Result<State, BootError> {
|
||||||
// Ensure we have enough progress pages to store copy progress
|
// Ensure we have enough progress pages to store copy progress
|
||||||
assert_eq!(0, P::page_size() % aligned_buf.len() as u32);
|
assert_eq!(0, Self::PAGE_SIZE % aligned_buf.len() as u32);
|
||||||
assert_eq!(0, P::page_size() % P::ACTIVE::WRITE_SIZE as u32);
|
assert_eq!(0, Self::PAGE_SIZE % ACTIVE::WRITE_SIZE as u32);
|
||||||
assert_eq!(0, P::page_size() % P::ACTIVE::ERASE_SIZE as u32);
|
assert_eq!(0, Self::PAGE_SIZE % ACTIVE::ERASE_SIZE as u32);
|
||||||
assert_eq!(0, P::page_size() % P::DFU::WRITE_SIZE as u32);
|
assert_eq!(0, Self::PAGE_SIZE % DFU::WRITE_SIZE as u32);
|
||||||
assert_eq!(0, P::page_size() % P::DFU::ERASE_SIZE as u32);
|
assert_eq!(0, Self::PAGE_SIZE % DFU::ERASE_SIZE as u32);
|
||||||
assert!(aligned_buf.len() >= P::STATE::WRITE_SIZE);
|
assert!(aligned_buf.len() >= STATE::WRITE_SIZE);
|
||||||
assert_eq!(0, aligned_buf.len() % P::ACTIVE::WRITE_SIZE);
|
assert_eq!(0, aligned_buf.len() % ACTIVE::WRITE_SIZE);
|
||||||
assert_eq!(0, aligned_buf.len() % P::DFU::WRITE_SIZE);
|
assert_eq!(0, aligned_buf.len() % DFU::WRITE_SIZE);
|
||||||
assert_partitions(self.active, self.dfu, self.state, P::page_size(), P::STATE::WRITE_SIZE);
|
|
||||||
|
assert_partitions(&self.active, &self.dfu, &self.state, Self::PAGE_SIZE);
|
||||||
|
|
||||||
// Copy contents from partition N to active
|
// Copy contents from partition N to active
|
||||||
let state = self.read_state(p, aligned_buf)?;
|
let state = self.read_state(aligned_buf)?;
|
||||||
if state == State::Swap {
|
if state == State::Swap {
|
||||||
//
|
//
|
||||||
// Check if we already swapped. If we're in the swap state, this means we should revert
|
// Check if we already swapped. If we're in the swap state, this means we should revert
|
||||||
// since the app has failed to mark boot as successful
|
// since the app has failed to mark boot as successful
|
||||||
//
|
//
|
||||||
if !self.is_swapped(p, aligned_buf)? {
|
if !self.is_swapped(aligned_buf)? {
|
||||||
trace!("Swapping");
|
trace!("Swapping");
|
||||||
self.swap(p, aligned_buf)?;
|
self.swap(aligned_buf)?;
|
||||||
trace!("Swapping done");
|
trace!("Swapping done");
|
||||||
} else {
|
} else {
|
||||||
trace!("Reverting");
|
trace!("Reverting");
|
||||||
self.revert(p, aligned_buf)?;
|
self.revert(aligned_buf)?;
|
||||||
|
|
||||||
let state_flash = p.state();
|
let state_word = &mut aligned_buf[..STATE::WRITE_SIZE];
|
||||||
let state_word = &mut aligned_buf[..P::STATE::WRITE_SIZE];
|
|
||||||
|
|
||||||
// Invalidate progress
|
// Invalidate progress
|
||||||
state_word.fill(!P::STATE_ERASE_VALUE);
|
state_word.fill(!STATE_ERASE_VALUE);
|
||||||
self.state
|
self.state.write(STATE::WRITE_SIZE as u32, state_word)?;
|
||||||
.write_blocking(state_flash, P::STATE::WRITE_SIZE as u32, state_word)?;
|
|
||||||
|
|
||||||
// Clear magic and progress
|
// Clear magic and progress
|
||||||
self.state.wipe_blocking(state_flash)?;
|
self.state.erase(0, self.state.capacity() as u32)?;
|
||||||
|
|
||||||
// Set magic
|
// Set magic
|
||||||
state_word.fill(BOOT_MAGIC);
|
state_word.fill(BOOT_MAGIC);
|
||||||
self.state.write_blocking(state_flash, 0, state_word)?;
|
self.state.write(0, state_word)?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(state)
|
Ok(state)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn is_swapped<P: FlashConfig>(&mut self, p: &mut P, aligned_buf: &mut [u8]) -> Result<bool, BootError> {
|
fn is_swapped(&mut self, aligned_buf: &mut [u8]) -> Result<bool, BootError> {
|
||||||
let page_count = (self.active.size() / P::page_size()) as usize;
|
let page_count = self.active.capacity() / Self::PAGE_SIZE as usize;
|
||||||
let progress = self.current_progress(p, aligned_buf)?;
|
let progress = self.current_progress(aligned_buf)?;
|
||||||
|
|
||||||
Ok(progress >= page_count * 2)
|
Ok(progress >= page_count * 2)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn current_progress<P: FlashConfig>(&mut self, config: &mut P, aligned_buf: &mut [u8]) -> Result<usize, BootError> {
|
fn current_progress(&mut self, aligned_buf: &mut [u8]) -> Result<usize, BootError> {
|
||||||
let write_size = P::STATE::WRITE_SIZE as u32;
|
let write_size = STATE::WRITE_SIZE as u32;
|
||||||
let max_index = (((self.state.size() - write_size) / write_size) - 2) as usize;
|
let max_index = ((self.state.capacity() - STATE::WRITE_SIZE) / STATE::WRITE_SIZE) - 2;
|
||||||
let state_flash = config.state();
|
|
||||||
let state_word = &mut aligned_buf[..write_size as usize];
|
let state_word = &mut aligned_buf[..write_size as usize];
|
||||||
|
|
||||||
self.state.read_blocking(state_flash, write_size, state_word)?;
|
self.state.read(write_size, state_word)?;
|
||||||
if state_word.iter().any(|&b| b != P::STATE_ERASE_VALUE) {
|
if state_word.iter().any(|&b| b != STATE_ERASE_VALUE) {
|
||||||
// Progress is invalid
|
// Progress is invalid
|
||||||
return Ok(max_index);
|
return Ok(max_index);
|
||||||
}
|
}
|
||||||
|
|
||||||
for index in 0..max_index {
|
for index in 0..max_index {
|
||||||
self.state
|
self.state.read((2 + index) as u32 * write_size, state_word)?;
|
||||||
.read_blocking(state_flash, (2 + index) as u32 * write_size, state_word)?;
|
|
||||||
|
|
||||||
if state_word.iter().any(|&b| b == P::STATE_ERASE_VALUE) {
|
if state_word.iter().any(|&b| b == STATE_ERASE_VALUE) {
|
||||||
return Ok(index);
|
return Ok(index);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(max_index)
|
Ok(max_index)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn update_progress<P: FlashConfig>(
|
fn update_progress(&mut self, progress_index: usize, aligned_buf: &mut [u8]) -> Result<(), BootError> {
|
||||||
&mut self,
|
let state_word = &mut aligned_buf[..STATE::WRITE_SIZE];
|
||||||
progress_index: usize,
|
state_word.fill(!STATE_ERASE_VALUE);
|
||||||
p: &mut P,
|
self.state
|
||||||
aligned_buf: &mut [u8],
|
.write((2 + progress_index) as u32 * STATE::WRITE_SIZE as u32, state_word)?;
|
||||||
) -> Result<(), BootError> {
|
|
||||||
let state_word = &mut aligned_buf[..P::STATE::WRITE_SIZE];
|
|
||||||
state_word.fill(!P::STATE_ERASE_VALUE);
|
|
||||||
self.state.write_blocking(
|
|
||||||
p.state(),
|
|
||||||
(2 + progress_index) as u32 * P::STATE::WRITE_SIZE as u32,
|
|
||||||
state_word,
|
|
||||||
)?;
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn copy_page_once_to_active<P: FlashConfig>(
|
fn copy_page_once_to_active(
|
||||||
&mut self,
|
&mut self,
|
||||||
progress_index: usize,
|
progress_index: usize,
|
||||||
from_offset: u32,
|
from_offset: u32,
|
||||||
to_offset: u32,
|
to_offset: u32,
|
||||||
p: &mut P,
|
|
||||||
aligned_buf: &mut [u8],
|
aligned_buf: &mut [u8],
|
||||||
) -> Result<(), BootError> {
|
) -> Result<(), BootError> {
|
||||||
if self.current_progress(p, aligned_buf)? <= progress_index {
|
if self.current_progress(aligned_buf)? <= progress_index {
|
||||||
let page_size = P::page_size() as u32;
|
let page_size = Self::PAGE_SIZE as u32;
|
||||||
|
|
||||||
self.active
|
self.active.erase(to_offset, to_offset + page_size)?;
|
||||||
.erase_blocking(p.active(), to_offset, to_offset + page_size)?;
|
|
||||||
|
|
||||||
for offset_in_page in (0..page_size).step_by(aligned_buf.len()) {
|
for offset_in_page in (0..page_size).step_by(aligned_buf.len()) {
|
||||||
self.dfu
|
self.dfu.read(from_offset + offset_in_page as u32, aligned_buf)?;
|
||||||
.read_blocking(p.dfu(), from_offset + offset_in_page as u32, aligned_buf)?;
|
self.active.write(to_offset + offset_in_page as u32, aligned_buf)?;
|
||||||
self.active
|
|
||||||
.write_blocking(p.active(), to_offset + offset_in_page as u32, aligned_buf)?;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
self.update_progress(progress_index, p, aligned_buf)?;
|
self.update_progress(progress_index, aligned_buf)?;
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn copy_page_once_to_dfu<P: FlashConfig>(
|
fn copy_page_once_to_dfu(
|
||||||
&mut self,
|
&mut self,
|
||||||
progress_index: usize,
|
progress_index: usize,
|
||||||
from_offset: u32,
|
from_offset: u32,
|
||||||
to_offset: u32,
|
to_offset: u32,
|
||||||
p: &mut P,
|
|
||||||
aligned_buf: &mut [u8],
|
aligned_buf: &mut [u8],
|
||||||
) -> Result<(), BootError> {
|
) -> Result<(), BootError> {
|
||||||
if self.current_progress(p, aligned_buf)? <= progress_index {
|
if self.current_progress(aligned_buf)? <= progress_index {
|
||||||
let page_size = P::page_size() as u32;
|
let page_size = Self::PAGE_SIZE as u32;
|
||||||
|
|
||||||
self.dfu
|
self.dfu.erase(to_offset as u32, to_offset + page_size)?;
|
||||||
.erase_blocking(p.dfu(), to_offset as u32, to_offset + page_size)?;
|
|
||||||
|
|
||||||
for offset_in_page in (0..page_size).step_by(aligned_buf.len()) {
|
for offset_in_page in (0..page_size).step_by(aligned_buf.len()) {
|
||||||
self.active
|
self.active.read(from_offset + offset_in_page as u32, aligned_buf)?;
|
||||||
.read_blocking(p.active(), from_offset + offset_in_page as u32, aligned_buf)?;
|
self.dfu.write(to_offset + offset_in_page as u32, aligned_buf)?;
|
||||||
self.dfu
|
|
||||||
.write_blocking(p.dfu(), to_offset + offset_in_page as u32, aligned_buf)?;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
self.update_progress(progress_index, p, aligned_buf)?;
|
self.update_progress(progress_index, aligned_buf)?;
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn swap<P: FlashConfig>(&mut self, p: &mut P, aligned_buf: &mut [u8]) -> Result<(), BootError> {
|
fn swap(&mut self, aligned_buf: &mut [u8]) -> Result<(), BootError> {
|
||||||
let page_size = P::page_size();
|
let page_count = self.active.capacity() as u32 / Self::PAGE_SIZE;
|
||||||
let page_count = self.active.size() / page_size;
|
|
||||||
for page_num in 0..page_count {
|
for page_num in 0..page_count {
|
||||||
let progress_index = (page_num * 2) as usize;
|
let progress_index = (page_num * 2) as usize;
|
||||||
|
|
||||||
// Copy active page to the 'next' DFU page.
|
// Copy active page to the 'next' DFU page.
|
||||||
let active_from_offset = (page_count - 1 - page_num) * page_size;
|
let active_from_offset = (page_count - 1 - page_num) * Self::PAGE_SIZE;
|
||||||
let dfu_to_offset = (page_count - page_num) * page_size;
|
let dfu_to_offset = (page_count - page_num) * Self::PAGE_SIZE;
|
||||||
//trace!("Copy active {} to dfu {}", active_from_offset, dfu_to_offset);
|
//trace!("Copy active {} to dfu {}", active_from_offset, dfu_to_offset);
|
||||||
self.copy_page_once_to_dfu(progress_index, active_from_offset, dfu_to_offset, p, aligned_buf)?;
|
self.copy_page_once_to_dfu(progress_index, active_from_offset, dfu_to_offset, aligned_buf)?;
|
||||||
|
|
||||||
// Copy DFU page to the active page
|
// Copy DFU page to the active page
|
||||||
let active_to_offset = (page_count - 1 - page_num) * page_size;
|
let active_to_offset = (page_count - 1 - page_num) * Self::PAGE_SIZE;
|
||||||
let dfu_from_offset = (page_count - 1 - page_num) * page_size;
|
let dfu_from_offset = (page_count - 1 - page_num) * Self::PAGE_SIZE;
|
||||||
//trace!("Copy dfy {} to active {}", dfu_from_offset, active_to_offset);
|
//trace!("Copy dfy {} to active {}", dfu_from_offset, active_to_offset);
|
||||||
self.copy_page_once_to_active(progress_index + 1, dfu_from_offset, active_to_offset, p, aligned_buf)?;
|
self.copy_page_once_to_active(progress_index + 1, dfu_from_offset, active_to_offset, aligned_buf)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn revert<P: FlashConfig>(&mut self, p: &mut P, aligned_buf: &mut [u8]) -> Result<(), BootError> {
|
fn revert(&mut self, aligned_buf: &mut [u8]) -> Result<(), BootError> {
|
||||||
let page_size = P::page_size();
|
let page_count = self.active.capacity() as u32 / Self::PAGE_SIZE;
|
||||||
let page_count = self.active.size() / page_size;
|
|
||||||
for page_num in 0..page_count {
|
for page_num in 0..page_count {
|
||||||
let progress_index = (page_count * 2 + page_num * 2) as usize;
|
let progress_index = (page_count * 2 + page_num * 2) as usize;
|
||||||
|
|
||||||
// Copy the bad active page to the DFU page
|
// Copy the bad active page to the DFU page
|
||||||
let active_from_offset = page_num * page_size;
|
let active_from_offset = page_num * Self::PAGE_SIZE;
|
||||||
let dfu_to_offset = page_num * page_size;
|
let dfu_to_offset = page_num * Self::PAGE_SIZE;
|
||||||
self.copy_page_once_to_dfu(progress_index, active_from_offset, dfu_to_offset, p, aligned_buf)?;
|
self.copy_page_once_to_dfu(progress_index, active_from_offset, dfu_to_offset, aligned_buf)?;
|
||||||
|
|
||||||
// Copy the DFU page back to the active page
|
// Copy the DFU page back to the active page
|
||||||
let active_to_offset = page_num * page_size;
|
let active_to_offset = page_num * Self::PAGE_SIZE;
|
||||||
let dfu_from_offset = (page_num + 1) * page_size;
|
let dfu_from_offset = (page_num + 1) * Self::PAGE_SIZE;
|
||||||
self.copy_page_once_to_active(progress_index + 1, dfu_from_offset, active_to_offset, p, aligned_buf)?;
|
self.copy_page_once_to_active(progress_index + 1, dfu_from_offset, active_to_offset, aligned_buf)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_state<P: FlashConfig>(&mut self, config: &mut P, aligned_buf: &mut [u8]) -> Result<State, BootError> {
|
fn read_state(&mut self, aligned_buf: &mut [u8]) -> Result<State, BootError> {
|
||||||
let state_word = &mut aligned_buf[..P::STATE::WRITE_SIZE];
|
let state_word = &mut aligned_buf[..STATE::WRITE_SIZE];
|
||||||
self.state.read_blocking(config.state(), 0, state_word)?;
|
self.state.read(0, state_word)?;
|
||||||
|
|
||||||
if !state_word.iter().any(|&b| b != SWAP_MAGIC) {
|
if !state_word.iter().any(|&b| b != SWAP_MAGIC) {
|
||||||
Ok(State::Swap)
|
Ok(State::Swap)
|
||||||
@ -373,11 +328,16 @@ impl BootLoader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn assert_partitions(active: Partition, dfu: Partition, state: Partition, page_size: u32, state_write_size: usize) {
|
fn assert_partitions<ACTIVE: NorFlash, DFU: NorFlash, STATE: NorFlash>(
|
||||||
assert_eq!(active.size() % page_size, 0);
|
active: &ACTIVE,
|
||||||
assert_eq!(dfu.size() % page_size, 0);
|
dfu: &DFU,
|
||||||
assert!(dfu.size() - active.size() >= page_size);
|
state: &STATE,
|
||||||
assert!(2 + 2 * (active.size() / page_size) <= state.size() / state_write_size as u32);
|
page_size: u32,
|
||||||
|
) {
|
||||||
|
assert_eq!(active.capacity() as u32 % page_size, 0);
|
||||||
|
assert_eq!(dfu.capacity() as u32 % page_size, 0);
|
||||||
|
assert!(dfu.capacity() as u32 - active.capacity() as u32 >= page_size);
|
||||||
|
assert!(2 + 2 * (active.capacity() as u32 / page_size) <= state.capacity() as u32 / STATE::WRITE_SIZE as u32);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// A flash wrapper implementing the Flash and embedded_storage traits.
|
/// A flash wrapper implementing the Flash and embedded_storage traits.
|
||||||
@ -436,98 +396,20 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Convenience provider that uses a single flash for all partitions.
|
|
||||||
pub struct SingleFlashConfig<'a, F>
|
|
||||||
where
|
|
||||||
F: NorFlash,
|
|
||||||
{
|
|
||||||
flash: &'a mut F,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, F> SingleFlashConfig<'a, F>
|
|
||||||
where
|
|
||||||
F: NorFlash,
|
|
||||||
{
|
|
||||||
/// Create a provider for a single flash.
|
|
||||||
pub fn new(flash: &'a mut F) -> Self {
|
|
||||||
Self { flash }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, F> FlashConfig for SingleFlashConfig<'a, F>
|
|
||||||
where
|
|
||||||
F: NorFlash,
|
|
||||||
{
|
|
||||||
type STATE = F;
|
|
||||||
type ACTIVE = F;
|
|
||||||
type DFU = F;
|
|
||||||
|
|
||||||
fn active(&mut self) -> &mut Self::STATE {
|
|
||||||
self.flash
|
|
||||||
}
|
|
||||||
fn dfu(&mut self) -> &mut Self::ACTIVE {
|
|
||||||
self.flash
|
|
||||||
}
|
|
||||||
fn state(&mut self) -> &mut Self::DFU {
|
|
||||||
self.flash
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Convenience flash provider that uses separate flash instances for each partition.
|
|
||||||
pub struct MultiFlashConfig<'a, ACTIVE, STATE, DFU>
|
|
||||||
where
|
|
||||||
ACTIVE: NorFlash,
|
|
||||||
STATE: NorFlash,
|
|
||||||
DFU: NorFlash,
|
|
||||||
{
|
|
||||||
active: &'a mut ACTIVE,
|
|
||||||
state: &'a mut STATE,
|
|
||||||
dfu: &'a mut DFU,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, ACTIVE, STATE, DFU> MultiFlashConfig<'a, ACTIVE, STATE, DFU>
|
|
||||||
where
|
|
||||||
ACTIVE: NorFlash,
|
|
||||||
STATE: NorFlash,
|
|
||||||
DFU: NorFlash,
|
|
||||||
{
|
|
||||||
/// Create a new flash provider with separate configuration for all three partitions.
|
|
||||||
pub fn new(active: &'a mut ACTIVE, state: &'a mut STATE, dfu: &'a mut DFU) -> Self {
|
|
||||||
Self { active, state, dfu }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, ACTIVE, STATE, DFU> FlashConfig for MultiFlashConfig<'a, ACTIVE, STATE, DFU>
|
|
||||||
where
|
|
||||||
ACTIVE: NorFlash,
|
|
||||||
STATE: NorFlash,
|
|
||||||
DFU: NorFlash,
|
|
||||||
{
|
|
||||||
type STATE = STATE;
|
|
||||||
type ACTIVE = ACTIVE;
|
|
||||||
type DFU = DFU;
|
|
||||||
|
|
||||||
fn active(&mut self) -> &mut Self::ACTIVE {
|
|
||||||
self.active
|
|
||||||
}
|
|
||||||
fn dfu(&mut self) -> &mut Self::DFU {
|
|
||||||
self.dfu
|
|
||||||
}
|
|
||||||
fn state(&mut self) -> &mut Self::STATE {
|
|
||||||
self.state
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
|
use crate::mem_flash::MemFlash;
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[should_panic]
|
#[should_panic]
|
||||||
fn test_range_asserts() {
|
fn test_range_asserts() {
|
||||||
const ACTIVE: Partition = Partition::new(4096, 4194304);
|
const ACTIVE_SIZE: usize = 4194304 - 4096;
|
||||||
const DFU: Partition = Partition::new(4194304, 2 * 4194304);
|
const DFU_SIZE: usize = 4194304;
|
||||||
const STATE: Partition = Partition::new(0, 4096);
|
const STATE_SIZE: usize = 4096;
|
||||||
assert_partitions(ACTIVE, DFU, STATE, 4096, 4);
|
static ACTIVE: MemFlash<ACTIVE_SIZE, 4, 4> = MemFlash::new(0xFF);
|
||||||
|
static DFU: MemFlash<DFU_SIZE, 4, 4> = MemFlash::new(0xFF);
|
||||||
|
static STATE: MemFlash<STATE_SIZE, 4, 4> = MemFlash::new(0xFF);
|
||||||
|
assert_partitions(&ACTIVE, &DFU, &STATE, 4096);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user