diff options
| -rw-r--r-- | embassy-cortex-m/src/executor.rs | 99 | ||||
| -rw-r--r-- | examples/nrf52840/src/bin/multiprio.rs | 32 | ||||
| -rw-r--r-- | examples/stm32f0/Cargo.toml | 2 | ||||
| -rw-r--r-- | examples/stm32f0/src/bin/multiprio.rs (renamed from examples/stm32f0/src/bin/priority.rs) | 32 | ||||
| -rw-r--r-- | examples/stm32f3/src/bin/multiprio.rs | 36 | ||||
| -rw-r--r-- | examples/stm32f4/src/bin/multiprio.rs | 36 |
6 files changed, 152 insertions, 85 deletions
diff --git a/embassy-cortex-m/src/executor.rs b/embassy-cortex-m/src/executor.rs index 0d1745d8a..558539e73 100644 --- a/embassy-cortex-m/src/executor.rs +++ b/embassy-cortex-m/src/executor.rs | |||
| @@ -1,18 +1,22 @@ | |||
| 1 | //! Executor specific to cortex-m devices. | 1 | //! Executor specific to cortex-m devices. |
| 2 | use core::marker::PhantomData; | ||
| 3 | 2 | ||
| 3 | use core::cell::UnsafeCell; | ||
| 4 | use core::mem::MaybeUninit; | ||
| 5 | |||
| 6 | use atomic_polyfill::{AtomicBool, Ordering}; | ||
| 7 | use cortex_m::interrupt::InterruptNumber; | ||
| 8 | use cortex_m::peripheral::NVIC; | ||
| 4 | pub use embassy_executor::*; | 9 | pub use embassy_executor::*; |
| 5 | 10 | ||
| 6 | use crate::interrupt::{Interrupt, InterruptExt}; | 11 | #[derive(Clone, Copy)] |
| 12 | struct N(u16); | ||
| 13 | unsafe impl cortex_m::interrupt::InterruptNumber for N { | ||
| 14 | fn number(self) -> u16 { | ||
| 15 | self.0 | ||
| 16 | } | ||
| 17 | } | ||
| 7 | 18 | ||
| 8 | fn pend_by_number(n: u16) { | 19 | fn pend_by_number(n: u16) { |
| 9 | #[derive(Clone, Copy)] | ||
| 10 | struct N(u16); | ||
| 11 | unsafe impl cortex_m::interrupt::InterruptNumber for N { | ||
| 12 | fn number(self) -> u16 { | ||
| 13 | self.0 | ||
| 14 | } | ||
| 15 | } | ||
| 16 | cortex_m::peripheral::NVIC::pend(N(n)) | 20 | cortex_m::peripheral::NVIC::pend(N(n)) |
| 17 | } | 21 | } |
| 18 | 22 | ||
| @@ -37,26 +41,37 @@ fn pend_by_number(n: u16) { | |||
| 37 | /// | 41 | /// |
| 38 | /// It is somewhat more complex to use, it's recommended to use the thread-mode | 42 | /// It is somewhat more complex to use, it's recommended to use the thread-mode |
| 39 | /// [`Executor`] instead, if it works for your use case. | 43 | /// [`Executor`] instead, if it works for your use case. |
| 40 | pub struct InterruptExecutor<I: Interrupt> { | 44 | pub struct InterruptExecutor { |
| 41 | irq: I, | 45 | started: AtomicBool, |
| 42 | inner: raw::Executor, | 46 | executor: UnsafeCell<MaybeUninit<raw::Executor>>, |
| 43 | not_send: PhantomData<*mut ()>, | ||
| 44 | } | 47 | } |
| 45 | 48 | ||
| 46 | impl<I: Interrupt> InterruptExecutor<I> { | 49 | unsafe impl Send for InterruptExecutor {} |
| 47 | /// Create a new Executor. | 50 | unsafe impl Sync for InterruptExecutor {} |
| 48 | pub fn new(irq: I) -> Self { | 51 | |
| 49 | let ctx = irq.number() as *mut (); | 52 | impl InterruptExecutor { |
| 53 | /// Create a new, not started `InterruptExecutor`. | ||
| 54 | #[inline] | ||
| 55 | pub const fn new() -> Self { | ||
| 50 | Self { | 56 | Self { |
| 51 | irq, | 57 | started: AtomicBool::new(false), |
| 52 | inner: raw::Executor::new(|ctx| pend_by_number(ctx as u16), ctx), | 58 | executor: UnsafeCell::new(MaybeUninit::uninit()), |
| 53 | not_send: PhantomData, | ||
| 54 | } | 59 | } |
| 55 | } | 60 | } |
| 56 | 61 | ||
| 62 | /// Executor interrupt callback. | ||
| 63 | /// | ||
| 64 | /// # Safety | ||
| 65 | /// | ||
| 66 | /// You MUST call this from the interrupt handler, and from nowhere else. | ||
| 67 | pub unsafe fn on_interrupt(&'static self) { | ||
| 68 | let executor = unsafe { (&*self.executor.get()).assume_init_ref() }; | ||
| 69 | executor.poll(); | ||
| 70 | } | ||
| 71 | |||
| 57 | /// Start the executor. | 72 | /// Start the executor. |
| 58 | /// | 73 | /// |
| 59 | /// This initializes the executor, configures and enables the interrupt, and returns. | 74 | /// This initializes the executor, enables the interrupt, and returns. |
| 60 | /// The executor keeps running in the background through the interrupt. | 75 | /// The executor keeps running in the background through the interrupt. |
| 61 | /// | 76 | /// |
| 62 | /// This returns a [`SendSpawner`] you can use to spawn tasks on it. A [`SendSpawner`] | 77 | /// This returns a [`SendSpawner`] you can use to spawn tasks on it. A [`SendSpawner`] |
| @@ -67,23 +82,35 @@ impl<I: Interrupt> InterruptExecutor<I> { | |||
| 67 | /// To obtain a [`Spawner`](embassy_executor::Spawner) for this executor, use [`Spawner::for_current_executor()`](embassy_executor::Spawner::for_current_executor()) from | 82 | /// To obtain a [`Spawner`](embassy_executor::Spawner) for this executor, use [`Spawner::for_current_executor()`](embassy_executor::Spawner::for_current_executor()) from |
| 68 | /// a task running in it. | 83 | /// a task running in it. |
| 69 | /// | 84 | /// |
| 70 | /// This function requires `&'static mut self`. This means you have to store the | 85 | /// # Interrupt requirements |
| 71 | /// Executor instance in a place where it'll live forever and grants you mutable | 86 | /// |
| 72 | /// access. There's a few ways to do this: | 87 | /// You must write the interrupt handler yourself, and make it call [`on_interrupt()`](Self::on_interrupt). |
| 88 | /// | ||
| 89 | /// This method already enables (unmasks) the interrupt, you must NOT do it yourself. | ||
| 90 | /// | ||
| 91 | /// You must set the interrupt priority before calling this method. You MUST NOT | ||
| 92 | /// do it after. | ||
| 73 | /// | 93 | /// |
| 74 | /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | 94 | pub fn start(&'static self, irq: impl InterruptNumber) -> SendSpawner { |
| 75 | /// - a `static mut` (unsafe) | 95 | if self |
| 76 | /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | 96 | .started |
| 77 | pub fn start(&'static mut self) -> SendSpawner { | 97 | .compare_exchange(false, true, Ordering::Acquire, Ordering::Relaxed) |
| 78 | self.irq.disable(); | 98 | .is_err() |
| 99 | { | ||
| 100 | panic!("InterruptExecutor::start() called multiple times on the same executor."); | ||
| 101 | } | ||
| 102 | |||
| 103 | unsafe { | ||
| 104 | (&mut *self.executor.get()).as_mut_ptr().write(raw::Executor::new( | ||
| 105 | |ctx| pend_by_number(ctx as u16), | ||
| 106 | irq.number() as *mut (), | ||
| 107 | )) | ||
| 108 | } | ||
| 109 | |||
| 110 | let executor = unsafe { (&*self.executor.get()).assume_init_ref() }; | ||
| 79 | 111 | ||
| 80 | self.irq.set_handler(|ctx| unsafe { | 112 | unsafe { NVIC::unmask(irq) } |
| 81 | let executor = &*(ctx as *const raw::Executor); | ||
| 82 | executor.poll(); | ||
| 83 | }); | ||
| 84 | self.irq.set_handler_context(&self.inner as *const _ as _); | ||
| 85 | self.irq.enable(); | ||
| 86 | 113 | ||
| 87 | self.inner.spawner().make_send() | 114 | executor.spawner().make_send() |
| 88 | } | 115 | } |
| 89 | } | 116 | } |
diff --git a/examples/nrf52840/src/bin/multiprio.rs b/examples/nrf52840/src/bin/multiprio.rs index 25806ae48..851e189ea 100644 --- a/examples/nrf52840/src/bin/multiprio.rs +++ b/examples/nrf52840/src/bin/multiprio.rs | |||
| @@ -57,11 +57,14 @@ | |||
| 57 | #![no_main] | 57 | #![no_main] |
| 58 | #![feature(type_alias_impl_trait)] | 58 | #![feature(type_alias_impl_trait)] |
| 59 | 59 | ||
| 60 | use core::mem; | ||
| 61 | |||
| 62 | use cortex_m::peripheral::NVIC; | ||
| 60 | use cortex_m_rt::entry; | 63 | use cortex_m_rt::entry; |
| 61 | use defmt::{info, unwrap}; | 64 | use defmt::{info, unwrap}; |
| 62 | use embassy_nrf::executor::{Executor, InterruptExecutor}; | 65 | use embassy_nrf::executor::{Executor, InterruptExecutor}; |
| 63 | use embassy_nrf::interrupt; | 66 | use embassy_nrf::interrupt; |
| 64 | use embassy_nrf::interrupt::InterruptExt; | 67 | use embassy_nrf::pac::Interrupt; |
| 65 | use embassy_time::{Duration, Instant, Timer}; | 68 | use embassy_time::{Duration, Instant, Timer}; |
| 66 | use static_cell::StaticCell; | 69 | use static_cell::StaticCell; |
| 67 | use {defmt_rtt as _, panic_probe as _}; | 70 | use {defmt_rtt as _, panic_probe as _}; |
| @@ -108,28 +111,35 @@ async fn run_low() { | |||
| 108 | } | 111 | } |
| 109 | } | 112 | } |
| 110 | 113 | ||
| 111 | static EXECUTOR_HIGH: StaticCell<InterruptExecutor<interrupt::SWI1_EGU1>> = StaticCell::new(); | 114 | static EXECUTOR_HIGH: InterruptExecutor = InterruptExecutor::new(); |
| 112 | static EXECUTOR_MED: StaticCell<InterruptExecutor<interrupt::SWI0_EGU0>> = StaticCell::new(); | 115 | static EXECUTOR_MED: InterruptExecutor = InterruptExecutor::new(); |
| 113 | static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new(); | 116 | static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new(); |
| 114 | 117 | ||
| 118 | #[interrupt] | ||
| 119 | unsafe fn SWI1_EGU1() { | ||
| 120 | EXECUTOR_HIGH.on_interrupt() | ||
| 121 | } | ||
| 122 | |||
| 123 | #[interrupt] | ||
| 124 | unsafe fn SWI0_EGU0() { | ||
| 125 | EXECUTOR_MED.on_interrupt() | ||
| 126 | } | ||
| 127 | |||
| 115 | #[entry] | 128 | #[entry] |
| 116 | fn main() -> ! { | 129 | fn main() -> ! { |
| 117 | info!("Hello World!"); | 130 | info!("Hello World!"); |
| 118 | 131 | ||
| 119 | let _p = embassy_nrf::init(Default::default()); | 132 | let _p = embassy_nrf::init(Default::default()); |
| 133 | let mut nvic: NVIC = unsafe { mem::transmute(()) }; | ||
| 120 | 134 | ||
| 121 | // High-priority executor: SWI1_EGU1, priority level 6 | 135 | // High-priority executor: SWI1_EGU1, priority level 6 |
| 122 | let irq = interrupt::take!(SWI1_EGU1); | 136 | unsafe { nvic.set_priority(Interrupt::SWI1_EGU1, 6 << 5) }; |
| 123 | irq.set_priority(interrupt::Priority::P6); | 137 | let spawner = EXECUTOR_HIGH.start(Interrupt::SWI1_EGU1); |
| 124 | let executor = EXECUTOR_HIGH.init(InterruptExecutor::new(irq)); | ||
| 125 | let spawner = executor.start(); | ||
| 126 | unwrap!(spawner.spawn(run_high())); | 138 | unwrap!(spawner.spawn(run_high())); |
| 127 | 139 | ||
| 128 | // Medium-priority executor: SWI0_EGU0, priority level 7 | 140 | // Medium-priority executor: SWI0_EGU0, priority level 7 |
| 129 | let irq = interrupt::take!(SWI0_EGU0); | 141 | unsafe { nvic.set_priority(Interrupt::SWI0_EGU0, 7 << 5) }; |
| 130 | irq.set_priority(interrupt::Priority::P7); | 142 | let spawner = EXECUTOR_MED.start(Interrupt::SWI0_EGU0); |
| 131 | let executor = EXECUTOR_MED.init(InterruptExecutor::new(irq)); | ||
| 132 | let spawner = executor.start(); | ||
| 133 | unwrap!(spawner.spawn(run_med())); | 143 | unwrap!(spawner.spawn(run_med())); |
| 134 | 144 | ||
| 135 | // Low priority executor: runs in thread mode, using WFE/SEV | 145 | // Low priority executor: runs in thread mode, using WFE/SEV |
diff --git a/examples/stm32f0/Cargo.toml b/examples/stm32f0/Cargo.toml index d4afbb8f8..89d99b6d3 100644 --- a/examples/stm32f0/Cargo.toml +++ b/examples/stm32f0/Cargo.toml | |||
| @@ -15,5 +15,5 @@ panic-probe = "0.3" | |||
| 15 | embassy-sync = { version = "0.1.0", path = "../../embassy-sync", features = ["defmt"] } | 15 | embassy-sync = { version = "0.1.0", path = "../../embassy-sync", features = ["defmt"] } |
| 16 | embassy-executor = { version = "0.1.0", path = "../../embassy-executor", features = ["defmt", "integrated-timers"] } | 16 | embassy-executor = { version = "0.1.0", path = "../../embassy-executor", features = ["defmt", "integrated-timers"] } |
| 17 | embassy-time = { version = "0.1.0", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime", "tick-hz-32_768"] } | 17 | embassy-time = { version = "0.1.0", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime", "tick-hz-32_768"] } |
| 18 | embassy-stm32 = { version = "0.1.0", path = "../../embassy-stm32", features = ["nightly", "defmt", "memory-x", "stm32f091rc", "time-driver-any", "exti"] } | 18 | embassy-stm32 = { version = "0.1.0", path = "../../embassy-stm32", features = ["nightly", "defmt", "memory-x", "stm32f091rc", "time-driver-any", "exti", "unstable-pac"] } |
| 19 | static_cell = "1.0" | 19 | static_cell = "1.0" |
diff --git a/examples/stm32f0/src/bin/priority.rs b/examples/stm32f0/src/bin/multiprio.rs index 7fed6a773..e0dc8c989 100644 --- a/examples/stm32f0/src/bin/priority.rs +++ b/examples/stm32f0/src/bin/multiprio.rs | |||
| @@ -57,11 +57,14 @@ | |||
| 57 | #![no_main] | 57 | #![no_main] |
| 58 | #![feature(type_alias_impl_trait)] | 58 | #![feature(type_alias_impl_trait)] |
| 59 | 59 | ||
| 60 | use core::mem; | ||
| 61 | |||
| 62 | use cortex_m::peripheral::NVIC; | ||
| 60 | use cortex_m_rt::entry; | 63 | use cortex_m_rt::entry; |
| 61 | use defmt::*; | 64 | use defmt::*; |
| 62 | use embassy_stm32::executor::{Executor, InterruptExecutor}; | 65 | use embassy_stm32::executor::{Executor, InterruptExecutor}; |
| 63 | use embassy_stm32::interrupt; | 66 | use embassy_stm32::interrupt; |
| 64 | use embassy_stm32::interrupt::InterruptExt; | 67 | use embassy_stm32::pac::Interrupt; |
| 65 | use embassy_time::{Duration, Instant, Timer}; | 68 | use embassy_time::{Duration, Instant, Timer}; |
| 66 | use static_cell::StaticCell; | 69 | use static_cell::StaticCell; |
| 67 | use {defmt_rtt as _, panic_probe as _}; | 70 | use {defmt_rtt as _, panic_probe as _}; |
| @@ -108,27 +111,34 @@ async fn run_low() { | |||
| 108 | } | 111 | } |
| 109 | } | 112 | } |
| 110 | 113 | ||
| 111 | static EXECUTOR_HIGH: StaticCell<InterruptExecutor<interrupt::USART1>> = StaticCell::new(); | 114 | static EXECUTOR_HIGH: InterruptExecutor = InterruptExecutor::new(); |
| 112 | static EXECUTOR_MED: StaticCell<InterruptExecutor<interrupt::USART2>> = StaticCell::new(); | 115 | static EXECUTOR_MED: InterruptExecutor = InterruptExecutor::new(); |
| 113 | static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new(); | 116 | static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new(); |
| 114 | 117 | ||
| 118 | #[interrupt] | ||
| 119 | unsafe fn USART1() { | ||
| 120 | EXECUTOR_HIGH.on_interrupt() | ||
| 121 | } | ||
| 122 | |||
| 123 | #[interrupt] | ||
| 124 | unsafe fn USART2() { | ||
| 125 | EXECUTOR_MED.on_interrupt() | ||
| 126 | } | ||
| 127 | |||
| 115 | #[entry] | 128 | #[entry] |
| 116 | fn main() -> ! { | 129 | fn main() -> ! { |
| 117 | // Initialize and create handle for devicer peripherals | 130 | // Initialize and create handle for devicer peripherals |
| 118 | let _p = embassy_stm32::init(Default::default()); | 131 | let _p = embassy_stm32::init(Default::default()); |
| 132 | let mut nvic: NVIC = unsafe { mem::transmute(()) }; | ||
| 119 | 133 | ||
| 120 | // High-priority executor: USART1, priority level 6 | 134 | // High-priority executor: USART1, priority level 6 |
| 121 | let irq = interrupt::take!(USART1); | 135 | unsafe { nvic.set_priority(Interrupt::USART1, 6 << 4) }; |
| 122 | irq.set_priority(interrupt::Priority::P6); | 136 | let spawner = EXECUTOR_HIGH.start(Interrupt::USART1); |
| 123 | let executor = EXECUTOR_HIGH.init(InterruptExecutor::new(irq)); | ||
| 124 | let spawner = executor.start(); | ||
| 125 | unwrap!(spawner.spawn(run_high())); | 137 | unwrap!(spawner.spawn(run_high())); |
| 126 | 138 | ||
| 127 | // Medium-priority executor: USART2, priority level 7 | 139 | // Medium-priority executor: USART2, priority level 7 |
| 128 | let irq = interrupt::take!(USART2); | 140 | unsafe { nvic.set_priority(Interrupt::USART2, 7 << 4) }; |
| 129 | irq.set_priority(interrupt::Priority::P7); | 141 | let spawner = EXECUTOR_MED.start(Interrupt::USART2); |
| 130 | let executor = EXECUTOR_MED.init(InterruptExecutor::new(irq)); | ||
| 131 | let spawner = executor.start(); | ||
| 132 | unwrap!(spawner.spawn(run_med())); | 142 | unwrap!(spawner.spawn(run_med())); |
| 133 | 143 | ||
| 134 | // Low priority executor: runs in thread mode, using WFE/SEV | 144 | // Low priority executor: runs in thread mode, using WFE/SEV |
diff --git a/examples/stm32f3/src/bin/multiprio.rs b/examples/stm32f3/src/bin/multiprio.rs index 9e8228a4b..77df51ac7 100644 --- a/examples/stm32f3/src/bin/multiprio.rs +++ b/examples/stm32f3/src/bin/multiprio.rs | |||
| @@ -57,11 +57,14 @@ | |||
| 57 | #![no_main] | 57 | #![no_main] |
| 58 | #![feature(type_alias_impl_trait)] | 58 | #![feature(type_alias_impl_trait)] |
| 59 | 59 | ||
| 60 | use core::mem; | ||
| 61 | |||
| 62 | use cortex_m::peripheral::NVIC; | ||
| 60 | use cortex_m_rt::entry; | 63 | use cortex_m_rt::entry; |
| 61 | use defmt::*; | 64 | use defmt::*; |
| 62 | use embassy_stm32::executor::{Executor, InterruptExecutor}; | 65 | use embassy_stm32::executor::{Executor, InterruptExecutor}; |
| 63 | use embassy_stm32::interrupt; | 66 | use embassy_stm32::interrupt; |
| 64 | use embassy_stm32::interrupt::InterruptExt; | 67 | use embassy_stm32::pac::Interrupt; |
| 65 | use embassy_time::{Duration, Instant, Timer}; | 68 | use embassy_time::{Duration, Instant, Timer}; |
| 66 | use static_cell::StaticCell; | 69 | use static_cell::StaticCell; |
| 67 | use {defmt_rtt as _, panic_probe as _}; | 70 | use {defmt_rtt as _, panic_probe as _}; |
| @@ -108,28 +111,35 @@ async fn run_low() { | |||
| 108 | } | 111 | } |
| 109 | } | 112 | } |
| 110 | 113 | ||
| 111 | static EXECUTOR_HIGH: StaticCell<InterruptExecutor<interrupt::UART4>> = StaticCell::new(); | 114 | static EXECUTOR_HIGH: InterruptExecutor = InterruptExecutor::new(); |
| 112 | static EXECUTOR_MED: StaticCell<InterruptExecutor<interrupt::UART5>> = StaticCell::new(); | 115 | static EXECUTOR_MED: InterruptExecutor = InterruptExecutor::new(); |
| 113 | static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new(); | 116 | static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new(); |
| 114 | 117 | ||
| 118 | #[interrupt] | ||
| 119 | unsafe fn UART4() { | ||
| 120 | EXECUTOR_HIGH.on_interrupt() | ||
| 121 | } | ||
| 122 | |||
| 123 | #[interrupt] | ||
| 124 | unsafe fn UART5() { | ||
| 125 | EXECUTOR_MED.on_interrupt() | ||
| 126 | } | ||
| 127 | |||
| 115 | #[entry] | 128 | #[entry] |
| 116 | fn main() -> ! { | 129 | fn main() -> ! { |
| 117 | info!("Hello World!"); | 130 | info!("Hello World!"); |
| 118 | 131 | ||
| 119 | let _p = embassy_stm32::init(Default::default()); | 132 | let _p = embassy_stm32::init(Default::default()); |
| 133 | let mut nvic: NVIC = unsafe { mem::transmute(()) }; | ||
| 120 | 134 | ||
| 121 | // High-priority executor: SWI1_EGU1, priority level 6 | 135 | // High-priority executor: UART4, priority level 6 |
| 122 | let irq = interrupt::take!(UART4); | 136 | unsafe { nvic.set_priority(Interrupt::UART4, 6 << 4) }; |
| 123 | irq.set_priority(interrupt::Priority::P6); | 137 | let spawner = EXECUTOR_HIGH.start(Interrupt::UART4); |
| 124 | let executor = EXECUTOR_HIGH.init(InterruptExecutor::new(irq)); | ||
| 125 | let spawner = executor.start(); | ||
| 126 | unwrap!(spawner.spawn(run_high())); | 138 | unwrap!(spawner.spawn(run_high())); |
| 127 | 139 | ||
| 128 | // Medium-priority executor: SWI0_EGU0, priority level 7 | 140 | // Medium-priority executor: UART5, priority level 7 |
| 129 | let irq = interrupt::take!(UART5); | 141 | unsafe { nvic.set_priority(Interrupt::UART5, 7 << 4) }; |
| 130 | irq.set_priority(interrupt::Priority::P7); | 142 | let spawner = EXECUTOR_MED.start(Interrupt::UART5); |
| 131 | let executor = EXECUTOR_MED.init(InterruptExecutor::new(irq)); | ||
| 132 | let spawner = executor.start(); | ||
| 133 | unwrap!(spawner.spawn(run_med())); | 143 | unwrap!(spawner.spawn(run_med())); |
| 134 | 144 | ||
| 135 | // Low priority executor: runs in thread mode, using WFE/SEV | 145 | // Low priority executor: runs in thread mode, using WFE/SEV |
diff --git a/examples/stm32f4/src/bin/multiprio.rs b/examples/stm32f4/src/bin/multiprio.rs index 9e8228a4b..77df51ac7 100644 --- a/examples/stm32f4/src/bin/multiprio.rs +++ b/examples/stm32f4/src/bin/multiprio.rs | |||
| @@ -57,11 +57,14 @@ | |||
| 57 | #![no_main] | 57 | #![no_main] |
| 58 | #![feature(type_alias_impl_trait)] | 58 | #![feature(type_alias_impl_trait)] |
| 59 | 59 | ||
| 60 | use core::mem; | ||
| 61 | |||
| 62 | use cortex_m::peripheral::NVIC; | ||
| 60 | use cortex_m_rt::entry; | 63 | use cortex_m_rt::entry; |
| 61 | use defmt::*; | 64 | use defmt::*; |
| 62 | use embassy_stm32::executor::{Executor, InterruptExecutor}; | 65 | use embassy_stm32::executor::{Executor, InterruptExecutor}; |
| 63 | use embassy_stm32::interrupt; | 66 | use embassy_stm32::interrupt; |
| 64 | use embassy_stm32::interrupt::InterruptExt; | 67 | use embassy_stm32::pac::Interrupt; |
| 65 | use embassy_time::{Duration, Instant, Timer}; | 68 | use embassy_time::{Duration, Instant, Timer}; |
| 66 | use static_cell::StaticCell; | 69 | use static_cell::StaticCell; |
| 67 | use {defmt_rtt as _, panic_probe as _}; | 70 | use {defmt_rtt as _, panic_probe as _}; |
| @@ -108,28 +111,35 @@ async fn run_low() { | |||
| 108 | } | 111 | } |
| 109 | } | 112 | } |
| 110 | 113 | ||
| 111 | static EXECUTOR_HIGH: StaticCell<InterruptExecutor<interrupt::UART4>> = StaticCell::new(); | 114 | static EXECUTOR_HIGH: InterruptExecutor = InterruptExecutor::new(); |
| 112 | static EXECUTOR_MED: StaticCell<InterruptExecutor<interrupt::UART5>> = StaticCell::new(); | 115 | static EXECUTOR_MED: InterruptExecutor = InterruptExecutor::new(); |
| 113 | static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new(); | 116 | static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new(); |
| 114 | 117 | ||
| 118 | #[interrupt] | ||
| 119 | unsafe fn UART4() { | ||
| 120 | EXECUTOR_HIGH.on_interrupt() | ||
| 121 | } | ||
| 122 | |||
| 123 | #[interrupt] | ||
| 124 | unsafe fn UART5() { | ||
| 125 | EXECUTOR_MED.on_interrupt() | ||
| 126 | } | ||
| 127 | |||
| 115 | #[entry] | 128 | #[entry] |
| 116 | fn main() -> ! { | 129 | fn main() -> ! { |
| 117 | info!("Hello World!"); | 130 | info!("Hello World!"); |
| 118 | 131 | ||
| 119 | let _p = embassy_stm32::init(Default::default()); | 132 | let _p = embassy_stm32::init(Default::default()); |
| 133 | let mut nvic: NVIC = unsafe { mem::transmute(()) }; | ||
| 120 | 134 | ||
| 121 | // High-priority executor: SWI1_EGU1, priority level 6 | 135 | // High-priority executor: UART4, priority level 6 |
| 122 | let irq = interrupt::take!(UART4); | 136 | unsafe { nvic.set_priority(Interrupt::UART4, 6 << 4) }; |
| 123 | irq.set_priority(interrupt::Priority::P6); | 137 | let spawner = EXECUTOR_HIGH.start(Interrupt::UART4); |
| 124 | let executor = EXECUTOR_HIGH.init(InterruptExecutor::new(irq)); | ||
| 125 | let spawner = executor.start(); | ||
| 126 | unwrap!(spawner.spawn(run_high())); | 138 | unwrap!(spawner.spawn(run_high())); |
| 127 | 139 | ||
| 128 | // Medium-priority executor: SWI0_EGU0, priority level 7 | 140 | // Medium-priority executor: UART5, priority level 7 |
| 129 | let irq = interrupt::take!(UART5); | 141 | unsafe { nvic.set_priority(Interrupt::UART5, 7 << 4) }; |
| 130 | irq.set_priority(interrupt::Priority::P7); | 142 | let spawner = EXECUTOR_MED.start(Interrupt::UART5); |
| 131 | let executor = EXECUTOR_MED.init(InterruptExecutor::new(irq)); | ||
| 132 | let spawner = executor.start(); | ||
| 133 | unwrap!(spawner.spawn(run_med())); | 143 | unwrap!(spawner.spawn(run_med())); |
| 134 | 144 | ||
| 135 | // Low priority executor: runs in thread mode, using WFE/SEV | 145 | // Low priority executor: runs in thread mode, using WFE/SEV |
