add failure and success apps

This commit is contained in:
Ulf Lilleengen 2023-09-15 22:40:25 +02:00
parent 2d78aa763a
commit 2a437d31c4
6 changed files with 194 additions and 40 deletions

View File

@ -12,9 +12,13 @@ embassy-sync = { version = "0.2.0", path = "../../../embassy-sync", features = [
embassy-executor = { version = "0.3.0", path = "../../../embassy-executor", features = ["arch-cortex-m", "executor-thread", "defmt", "nightly", "integrated-timers"] } embassy-executor = { version = "0.3.0", path = "../../../embassy-executor", features = ["arch-cortex-m", "executor-thread", "defmt", "nightly", "integrated-timers"] }
embassy-time = { version = "0.1.3", path = "../../../embassy-time", features = ["defmt", "nightly", "unstable-traits", "defmt-timestamp-uptime"] } embassy-time = { version = "0.1.3", path = "../../../embassy-time", features = ["defmt", "nightly", "unstable-traits", "defmt-timestamp-uptime"] }
embassy-nrf = { version = "0.1.0", path = "../../../embassy-nrf", features = ["defmt", "nightly", "unstable-traits", "nrf52840", "time-driver-rtc1", "gpiote", "unstable-pac"] } embassy-nrf = { version = "0.1.0", path = "../../../embassy-nrf", features = ["defmt", "nightly", "unstable-traits", "nrf52840", "time-driver-rtc1", "gpiote", "unstable-pac"] }
embassy-embedded-hal = { version = "0.1.0", path = "../../../embassy-embedded-hal", features = ["nightly"] }
embassy-boot-nrf = { version = "0.1.0", path = "../../../embassy-boot/nrf" }
embedded-io-async = { version = "0.5.0" } embedded-io-async = { version = "0.5.0" }
embedded-hal-async = { version = "1.0.0-rc.1" } embedded-hal-async = { version = "1.0.0-rc.1" }
embedded-hal-bus = { version = "0.1.0-rc.1", features = ["async"] } embedded-hal-bus = { version = "0.1.0-rc.1", features = ["async"] }
embedded-storage = "0.3.0"
embedded-storage-async = { version = "0.4.0" }
static_cell = { version = "1.1", features = [ "nightly" ] } static_cell = { version = "1.1", features = [ "nightly" ] }
defmt = "0.3" defmt = "0.3"

View File

@ -5,7 +5,7 @@ MEMORY
BOOTLOADER_STATE : ORIGIN = 0x00006000, LENGTH = 4K BOOTLOADER_STATE : ORIGIN = 0x00006000, LENGTH = 4K
FLASH : ORIGIN = 0x00007000, LENGTH = 64K FLASH : ORIGIN = 0x00007000, LENGTH = 64K
DFU : ORIGIN = 0x00017000, LENGTH = 68K DFU : ORIGIN = 0x00017000, LENGTH = 68K
TESTSTATE : ORIGIN = 0x00028000, LENGTH = 4K TEST_STATE : ORIGIN = 0x00028000, LENGTH = 4K
RAM (rwx) : ORIGIN = 0x20000000, LENGTH = 32K RAM (rwx) : ORIGIN = 0x20000000, LENGTH = 32K
} }
@ -15,6 +15,5 @@ __bootloader_state_end = ORIGIN(BOOTLOADER_STATE) + LENGTH(BOOTLOADER_STATE);
__bootloader_dfu_start = ORIGIN(DFU); __bootloader_dfu_start = ORIGIN(DFU);
__bootloader_dfu_end = ORIGIN(DFU) + LENGTH(DFU); __bootloader_dfu_end = ORIGIN(DFU) + LENGTH(DFU);
__test_state_start = ORIGIN(TEST_STATE);
__teststate_start = ORIGIN(TESTSTATE); __test-state_end = ORIGIN(TEST_STATE) + LENGTH(TEST_STATE);
__teststate_end = ORIGIN(TESTSTATE) + LENGTH(TESTSTATE);

View File

@ -0,0 +1,68 @@
#![no_std]
#![no_main]
#![feature(type_alias_impl_trait)]
teleprobe_meta::target!(b"nrf52840-dk");
#[path = "../common.rs"]
mod common;
use common::*;
use embassy_boot_nrf::{BlockingFirmwareUpdater, FirmwareUpdaterConfig};
use embassy_executor::Spawner;
use embassy_nrf::nvmc::Nvmc;
use embassy_time::{Timer, Duration};
use embassy_sync::blocking_mutex::Mutex;
use core::cell::RefCell;
use embedded_storage::nor_flash::NorFlash;
static APP_SUCCESS: &[u8] = include_bytes!("../../update_success.bin");
static APP_FAILURE: &[u8] = include_bytes!("../../update_failure.bin");
#[embassy_executor::main]
async fn main(s: Spawner) {
let p = embassy_nrf::init(Default::default());
s.spawn(watchdog_task()).unwrap();
let nvmc = Nvmc::new(p.NVMC);
let nvmc = Mutex::new(RefCell::new(nvmc));
let state = TestState::init(&nvmc);
// First time we boot the bad firmare
let app = if state.num_boots == 0 {
APP_FAILURE
// Second time we boot the good firmware
} else if state.num_boots == 1 {
APP_SUCCESS
} else {
defmt::panic!("Unexpected number of boots");
};
let config = FirmwareUpdaterConfig::from_linkerfile_blocking(&nvmc);
let mut magic = [0; 4];
let mut updater = BlockingFirmwareUpdater::new(config, &mut magic);
// Check state if we've attempted update yet
let dfu = updater.prepare_update().unwrap();
let mut offset = 0;
for chunk in app.chunks(4096) {
let mut buf: [u8; 4096] = [0; 4096];
buf[..chunk.len()].copy_from_slice(chunk);
dfu.write(offset, &buf).unwrap();
offset += chunk.len() as u32;
}
updater.mark_updated().unwrap();
cortex_m::peripheral::SCB::sys_reset();
}
// Keeps our system alive
#[embassy_executor::task]
async fn watchdog_task() {
let mut handle = unsafe { embassy_nrf::wdt::WatchdogHandle::steal(0) };
loop {
handle.pet();
Timer::after(Duration::from_secs(2)).await;
}
}

View File

@ -0,0 +1,40 @@
#![no_std]
#![no_main]
#![feature(type_alias_impl_trait)]
teleprobe_meta::target!(b"nrf52840-dk");
#[path = "../common.rs"]
mod common;
use common::*;
use embassy_executor::Spawner;
use embassy_time::{Timer, Duration};
use embassy_nrf::nvmc::Nvmc;
use embassy_sync::blocking_mutex::Mutex;
use core::cell::RefCell;
#[embassy_executor::main]
async fn main(s: Spawner) {
let p = embassy_nrf::init(Default::default());
s.spawn(watchdog_task()).unwrap();
let nvmc = Nvmc::new(p.NVMC);
let nvmc = Mutex::new(RefCell::new(nvmc));
let mut state = TestState::init(&nvmc);
state.inc(&nvmc);
// Simulate hang
loop {}
}
// Keeps our system alive
#[embassy_executor::task]
async fn watchdog_task() {
let mut handle = unsafe { embassy_nrf::wdt::WatchdogHandle::steal(0) };
loop {
handle.pet();
Timer::after(Duration::from_secs(2)).await;
}
}

View File

@ -7,49 +7,46 @@ teleprobe_meta::target!(b"nrf52840-dk");
mod common; mod common;
use common::*; use common::*;
use embassy_boot_nrf::{FirmwareUpdater, FirmwareUpdaterConfig}; use embassy_boot_nrf::{BlockingFirmwareUpdater, FirmwareUpdaterConfig};
use embassy_embedded_hal::adapter::BlockingAsync;
use embassy_executor::Spawner; use embassy_executor::Spawner;
use embassy_nrf::gpio::{Input, Level, Output, OutputDrive, Pull}; use embassy_time::{Timer, Duration};
use embassy_nrf::nvmc::Nvmc; use embassy_nrf::nvmc::Nvmc;
use embassy_nrf::wdt::{self, Watchdog}; use embassy_sync::blocking_mutex::Mutex;
use embassy_sync::mutex::Mutex; use core::cell::RefCell;
use panic_reset as _;
#[embassy_executor::main] #[embassy_executor::main]
async fn main(_spawner: Spawner) { async fn main(s: Spawner) {
let p = embassy_nrf::init(Default::default()); let p = embassy_nrf::init(Default::default());
let wdt_config = wdt::Config::try_new(&p.WDT).unwrap(); s.spawn(watchdog_task()).unwrap();
let (_wdt, [_wdt_handle]) = match Watchdog::try_new(p.WDT, wdt_config) {
Ok(x) => x,
Err(_) => {
// Watchdog already active with the wrong number of handles, waiting for it to timeout...
loop {
cortex_m::asm::wfe();
}
}
};
let nvmc = Nvmc::new(p.NVMC); let nvmc = Nvmc::new(p.NVMC);
let nvmc = Mutex::new(BlockingAsync::new(nvmc)); let nvmc = Mutex::new(RefCell::new(nvmc));
let test_state = Partition::new(&mut ) let mut state = TestState::init(&nvmc);
let config = FirmwareUpdaterConfig::from_linkerfile(&nvmc); if state.num_boots == 1 {
let mut magic = [0; 4]; let config = FirmwareUpdaterConfig::from_linkerfile_blocking(&nvmc);
let mut updater = FirmwareUpdater::new(config, &mut magic);
// Check state if we've attempted update yet let mut magic = [0; 4];
let dfu = updater.prepare().await.unwrap(); let mut updater = BlockingFirmwareUpdater::new(config, &mut magic);
let mut offset = 0; updater.mark_booted().unwrap();
for chunk in APP_B.chunks(4096) { state.inc(&nvmc);
let mut buf: [u8; 4096] = [0; 4096]; cortex_m::peripheral::SCB::sys_reset();
buf[..chunk.len()].copy_from_slice(chunk); } else if state.num_boots == 2 {
updater.write_firmware(offset, &buf).await.unwrap(); // Test success!
offset += chunk.len(); cortex_m::asm::bkpt();
} else {
defmt::panic!("Unexpected number of boots");
}
}
// Keeps our system alive
#[embassy_executor::task]
async fn watchdog_task() {
let mut handle = unsafe { embassy_nrf::wdt::WatchdogHandle::steal(0) };
loop {
handle.pet();
Timer::after(Duration::from_secs(2)).await;
} }
updater.mark_updated().await.unwrap();
led.set_high();
cortex_m::peripheral::SCB::sys_reset();
} }

View File

@ -1,12 +1,58 @@
#![macro_use] #![macro_use]
use core::cell::RefCell;
#[allow(unused)] #[allow(unused)]
pub use defmt::*; pub use defmt::*;
use embassy_embedded_hal::flash::partition::asynch::Partition; use embassy_embedded_hal::flash::partition::BlockingPartition;
use embassy_nrf::nvmc::Nvmc; use embassy_nrf::nvmc::Nvmc;
use embassy_sync::blocking_mutex::raw::NoopRawMutex; use embassy_sync::blocking_mutex::raw::NoopRawMutex;
use embassy_sync::blocking_mutex::Mutex;
use embedded_storage::nor_flash::{NorFlash, ReadNorFlash};
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
pub struct TestState<'a> { pub struct TestState {
flash: Partition<'a, NoopRawMutex, Nvmc<'a>>, pub num_boots: u32,
}
impl TestState {
pub fn init(flash: &Mutex<NoopRawMutex, RefCell<Nvmc<'_>>>) -> Self {
extern "C" {
static __test_state_start: u32;
static __test_state_end: u32;
}
let mut flash = unsafe {
let start = &__test_state_start as *const u32 as u32;
let end = &__test_state_end as *const u32 as u32;
BlockingPartition::new(flash, start, end - start)
};
let mut buf = [0; 4];
let _ = flash.read(0, &mut buf).unwrap();
if &buf[..] == &[0xff, 0xff, 0xff, 0xff] {
TestState { num_boots: 0 }
} else {
TestState {
num_boots: u32::from_be_bytes([buf[0], buf[1], buf[2], buf[3]]),
}
}
}
pub fn inc(&mut self, flash: &Mutex<NoopRawMutex, RefCell<Nvmc<'_>>>) {
extern "C" {
static __test_state_start: u32;
static __test_state_end: u32;
}
let mut flash = unsafe {
let start = &__test_state_start as *const u32 as u32;
let end = &__test_state_end as *const u32 as u32;
BlockingPartition::new(flash, start, end - start)
};
self.num_boots += 1;
flash.erase(0, 4096).unwrap();
flash.write(0, &self.num_boots.to_be_bytes()).unwrap();
}
} }