add failure and success apps
This commit is contained in:
parent
2d78aa763a
commit
2a437d31c4
@ -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"
|
||||||
|
@ -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);
|
|
||||||
|
68
tests/boot/nrf/src/bin/update_base.rs
Normal file
68
tests/boot/nrf/src/bin/update_base.rs
Normal 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;
|
||||||
|
}
|
||||||
|
}
|
40
tests/boot/nrf/src/bin/update_failure.rs
Normal file
40
tests/boot/nrf/src/bin/update_failure.rs
Normal 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;
|
||||||
|
}
|
||||||
|
}
|
@ -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();
|
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user