diff options
| author | bors[bot] <26634292+bors[bot]@users.noreply.github.com> | 2021-12-15 05:46:18 +0000 |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2021-12-15 05:46:18 +0000 |
| commit | 37facfb5bf9aca11c43868cb8880b12b9f6b336a (patch) | |
| tree | 7e501c06dba421f6874dc5879e70192cedb4321a /examples | |
| parent | 235484565d706d362f37c8bd388dbfff78401e8a (diff) | |
| parent | d81a4da6fb759c5561eacf4d99d953526e1ee895 (diff) | |
Merge #562
562: Replace default WFI with simpler NOP r=korken89 a=AfoHT
As noted by #561 there are multiple issues with various hardware implementations and debugging together with sleep modes.
As RTIC strives to be a generic framework (even though this is an implementation targeting cortex-m) the decision having `WFI` as the default `idle` task causes issues in some hardware, raising the barrier to entry.
This changes the default behaviour to do a simple `NOP` instead, adding documentation how to provide a custom `idle` task achieving the old default behaviour. Also removes the automatic setting of SLEEPONEXIT bit when no `idle` was provided, delegating this to the user.
This was discussed on the weekly meeting 2021-12-14 and this was the favoured solution.
Fixes #561
Co-authored-by: Henrik Tjäder <henrik@grepit.se>
Diffstat (limited to 'examples')
| -rw-r--r-- | examples/idle-wfi.rs | 47 |
1 files changed, 47 insertions, 0 deletions
diff --git a/examples/idle-wfi.rs b/examples/idle-wfi.rs new file mode 100644 index 0000000..4a8a8de --- /dev/null +++ b/examples/idle-wfi.rs @@ -0,0 +1,47 @@ +//! examples/idle-wfi.rs + +#![deny(unsafe_code)] +#![deny(warnings)] +#![no_main] +#![no_std] + +use panic_semihosting as _; + +#[rtic::app(device = lm3s6965)] +mod app { + use cortex_m_semihosting::{debug, hprintln}; + + #[shared] + struct Shared {} + + #[local] + struct Local {} + + #[init] + fn init(mut cx: init::Context) -> (Shared, Local, init::Monotonics) { + hprintln!("init").unwrap(); + + // Set the ARM SLEEPONEXIT bit to go to sleep after handling interrupts + // See https://developer.arm.com/docs/100737/0100/power-management/sleep-mode/sleep-on-exit-bit + cx.core.SCB.set_sleepdeep(); + + (Shared {}, Local {}, init::Monotonics()) + } + + #[idle(local = [x: u32 = 0])] + fn idle(cx: idle::Context) -> ! { + // Locals in idle have lifetime 'static + let _x: &'static mut u32 = cx.local.x; + + hprintln!("idle").unwrap(); + + debug::exit(debug::EXIT_SUCCESS); // Exit QEMU simulator + + loop { + // Now Wait For Interrupt is used instead of a busy-wait loop + // to allow MCU to sleep between interrupts + // https://developer.arm.com/documentation/ddi0406/c/Application-Level-Architecture/Instruction-Details/Alphabetical-list-of-instructions/WFI + rtic::export::wfi() + } + } +} |
