Remove Forever, switch to static_cell.

This commit is contained in:
Dario Nieuwenhuis 2022-08-22 15:51:44 +02:00
parent 1b95990258
commit 478f472784
45 changed files with 139 additions and 220 deletions

View file

@ -71,7 +71,7 @@ impl<I: Interrupt> InterruptExecutor<I> {
/// Executor instance in a place where it'll live forever and grants you mutable /// Executor instance in a place where it'll live forever and grants you mutable
/// access. There's a few ways to do this: /// access. There's a few ways to do this:
/// ///
/// - a [Forever](embassy_util::Forever) (safe) /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe)
/// - a `static mut` (unsafe) /// - a `static mut` (unsafe)
/// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe)
pub fn start(&'static mut self) -> SendSpawner { pub fn start(&'static mut self) -> SendSpawner {

View file

@ -7,12 +7,12 @@
//! use embassy_util::mutex::Mutex; //! use embassy_util::mutex::Mutex;
//! use embassy_util::blocking_mutex::raw::ThreadModeRawMutex; //! use embassy_util::blocking_mutex::raw::ThreadModeRawMutex;
//! //!
//! static I2C_BUS: Forever<Mutex::<ThreadModeRawMutex, Twim<TWISPI0>>> = Forever::new(); //! static I2C_BUS: StaticCell<Mutex::<ThreadModeRawMutex, Twim<TWISPI0>>> = StaticCell::new();
//! let config = twim::Config::default(); //! let config = twim::Config::default();
//! let irq = interrupt::take!(SPIM0_SPIS0_TWIM0_TWIS0_SPI0_TWI0); //! let irq = interrupt::take!(SPIM0_SPIS0_TWIM0_TWIS0_SPI0_TWI0);
//! let i2c = Twim::new(p.TWISPI0, irq, p.P0_03, p.P0_04, config); //! let i2c = Twim::new(p.TWISPI0, irq, p.P0_03, p.P0_04, config);
//! let i2c_bus = Mutex::<ThreadModeRawMutex, _>::new(i2c); //! let i2c_bus = Mutex::<ThreadModeRawMutex, _>::new(i2c);
//! let i2c_bus = I2C_BUS.put(i2c_bus); //! let i2c_bus = I2C_BUS.init(i2c_bus);
//! //!
//! // Device 1, using embedded-hal-async compatible driver for QMC5883L compass //! // Device 1, using embedded-hal-async compatible driver for QMC5883L compass
//! let i2c_dev1 = I2cDevice::new(i2c_bus); //! let i2c_dev1 = I2cDevice::new(i2c_bus);

View file

@ -7,13 +7,13 @@
//! use embassy_util::mutex::Mutex; //! use embassy_util::mutex::Mutex;
//! use embassy_util::blocking_mutex::raw::ThreadModeRawMutex; //! use embassy_util::blocking_mutex::raw::ThreadModeRawMutex;
//! //!
//! static SPI_BUS: Forever<Mutex<ThreadModeRawMutex, spim::Spim<SPI3>>> = Forever::new(); //! static SPI_BUS: StaticCell<Mutex<ThreadModeRawMutex, spim::Spim<SPI3>>> = StaticCell::new();
//! let mut config = spim::Config::default(); //! let mut config = spim::Config::default();
//! config.frequency = spim::Frequency::M32; //! config.frequency = spim::Frequency::M32;
//! let irq = interrupt::take!(SPIM3); //! let irq = interrupt::take!(SPIM3);
//! let spi = spim::Spim::new_txonly(p.SPI3, irq, p.P0_15, p.P0_18, config); //! let spi = spim::Spim::new_txonly(p.SPI3, irq, p.P0_15, p.P0_18, config);
//! let spi_bus = Mutex::<ThreadModeRawMutex, _>::new(spi); //! let spi_bus = Mutex::<ThreadModeRawMutex, _>::new(spi);
//! let spi_bus = SPI_BUS.put(spi_bus); //! let spi_bus = SPI_BUS.init(spi_bus);
//! //!
//! // Device 1, using embedded-hal-async compatible driver for ST7735 LCD display //! // Device 1, using embedded-hal-async compatible driver for ST7735 LCD display
//! let cs_pin1 = Output::new(p.P0_24, Level::Low, OutputDrive::Standard); //! let cs_pin1 = Output::new(p.P0_24, Level::Low, OutputDrive::Standard);

View file

@ -6,11 +6,11 @@
//! use embassy_embedded_hal::shared_bus::blocking::i2c::I2cDevice; //! use embassy_embedded_hal::shared_bus::blocking::i2c::I2cDevice;
//! use embassy_util::blocking_mutex::{NoopMutex, raw::NoopRawMutex}; //! use embassy_util::blocking_mutex::{NoopMutex, raw::NoopRawMutex};
//! //!
//! static I2C_BUS: Forever<NoopMutex<RefCell<Twim<TWISPI0>>>> = Forever::new(); //! static I2C_BUS: StaticCell<NoopMutex<RefCell<Twim<TWISPI0>>>> = StaticCell::new();
//! let irq = interrupt::take!(SPIM0_SPIS0_TWIM0_TWIS0_SPI0_TWI0); //! let irq = interrupt::take!(SPIM0_SPIS0_TWIM0_TWIS0_SPI0_TWI0);
//! let i2c = Twim::new(p.TWISPI0, irq, p.P0_03, p.P0_04, Config::default()); //! let i2c = Twim::new(p.TWISPI0, irq, p.P0_03, p.P0_04, Config::default());
//! let i2c_bus = NoopMutex::new(RefCell::new(i2c)); //! let i2c_bus = NoopMutex::new(RefCell::new(i2c));
//! let i2c_bus = I2C_BUS.put(i2c_bus); //! let i2c_bus = I2C_BUS.init(i2c_bus);
//! //!
//! let i2c_dev1 = I2cDevice::new(i2c_bus); //! let i2c_dev1 = I2cDevice::new(i2c_bus);
//! let mpu = Mpu6050::new(i2c_dev1); //! let mpu = Mpu6050::new(i2c_dev1);

View file

@ -6,11 +6,11 @@
//! use embassy_embedded_hal::shared_bus::blocking::spi::SpiDevice; //! use embassy_embedded_hal::shared_bus::blocking::spi::SpiDevice;
//! use embassy_util::blocking_mutex::{NoopMutex, raw::NoopRawMutex}; //! use embassy_util::blocking_mutex::{NoopMutex, raw::NoopRawMutex};
//! //!
//! static SPI_BUS: Forever<NoopMutex<RefCell<Spim<SPI3>>>> = Forever::new(); //! static SPI_BUS: StaticCell<NoopMutex<RefCell<Spim<SPI3>>>> = StaticCell::new();
//! let irq = interrupt::take!(SPIM3); //! let irq = interrupt::take!(SPIM3);
//! let spi = Spim::new_txonly(p.SPI3, irq, p.P0_15, p.P0_18, Config::default()); //! let spi = Spim::new_txonly(p.SPI3, irq, p.P0_15, p.P0_18, Config::default());
//! let spi_bus = NoopMutex::new(RefCell::new(spi)); //! let spi_bus = NoopMutex::new(RefCell::new(spi));
//! let spi_bus = SPI_BUS.put(spi_bus); //! let spi_bus = SPI_BUS.init(spi_bus);
//! //!
//! // Device 1, using embedded-hal compatible driver for ST7735 LCD display //! // Device 1, using embedded-hal compatible driver for ST7735 LCD display
//! let cs_pin1 = Output::new(p.P0_24, Level::Low, OutputDrive::Standard); //! let cs_pin1 = Output::new(p.P0_24, Level::Low, OutputDrive::Standard);

View file

@ -44,6 +44,7 @@ embassy-time = { version = "0.1.0", path = "../embassy-time", optional = true}
atomic-polyfill = "1.0.1" atomic-polyfill = "1.0.1"
critical-section = "1.1" critical-section = "1.1"
cfg-if = "1.0.0" cfg-if = "1.0.0"
static_cell = "1.0"
# WASM dependencies # WASM dependencies
wasm-bindgen = { version = "0.2.76", features = ["nightly"], optional = true } wasm-bindgen = { version = "0.2.76", features = ["nightly"], optional = true }

View file

@ -41,7 +41,7 @@ impl Executor {
/// Executor instance in a place where it'll live forever and grants you mutable /// Executor instance in a place where it'll live forever and grants you mutable
/// access. There's a few ways to do this: /// access. There's a few ways to do this:
/// ///
/// - a [Forever](crate::util::Forever) (safe) /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe)
/// - a `static mut` (unsafe) /// - a `static mut` (unsafe)
/// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe)
/// ///

View file

@ -43,7 +43,7 @@ impl Executor {
/// Executor instance in a place where it'll live forever and grants you mutable /// Executor instance in a place where it'll live forever and grants you mutable
/// access. There's a few ways to do this: /// access. There's a few ways to do this:
/// ///
/// - a [Forever](crate::util::Forever) (safe) /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe)
/// - a `static mut` (unsafe) /// - a `static mut` (unsafe)
/// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe)
/// ///

View file

@ -40,7 +40,7 @@ impl Executor {
/// Executor instance in a place where it'll live forever and grants you mutable /// Executor instance in a place where it'll live forever and grants you mutable
/// access. There's a few ways to do this: /// access. There's a few ways to do this:
/// ///
/// - a [Forever](crate::util::Forever) (safe) /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe)
/// - a `static mut` (unsafe) /// - a `static mut` (unsafe)
/// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe)
/// ///

View file

@ -59,7 +59,7 @@ impl Executor {
/// Executor instance in a place where it'll live forever and grants you mutable /// Executor instance in a place where it'll live forever and grants you mutable
/// access. There's a few ways to do this: /// access. There's a few ways to do this:
/// ///
/// - a [Forever](crate::util::Forever) (safe) /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe)
/// - a `static mut` (unsafe) /// - a `static mut` (unsafe)
/// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe)
pub fn start(&'static mut self, init: impl FnOnce(Spawner)) { pub fn start(&'static mut self, init: impl FnOnce(Spawner)) {

View file

@ -43,7 +43,7 @@ impl Executor {
/// Executor instance in a place where it'll live forever and grants you mutable /// Executor instance in a place where it'll live forever and grants you mutable
/// access. There's a few ways to do this: /// access. There's a few ways to do this:
/// ///
/// - a [Forever](crate::util::Forever) (safe) /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe)
/// - a `static mut` (unsafe) /// - a `static mut` (unsafe)
/// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe)
/// ///

View file

@ -67,3 +67,9 @@ pub mod raw;
mod spawner; mod spawner;
pub use spawner::*; pub use spawner::*;
/// Do not use. Used for macros and HALs only. Not covered by semver guarantees.
#[doc(hidden)]
pub mod _export {
pub use static_cell::StaticCell;
}

View file

@ -34,8 +34,8 @@ pub fn run(args: syn::AttributeArgs, f: syn::ItemFn) -> Result<TokenStream, Toke
let main = quote! { let main = quote! {
#[wasm_bindgen::prelude::wasm_bindgen(start)] #[wasm_bindgen::prelude::wasm_bindgen(start)]
pub fn main() -> Result<(), wasm_bindgen::JsValue> { pub fn main() -> Result<(), wasm_bindgen::JsValue> {
static EXECUTOR: ::embassy_util::Forever<::embassy_executor::Executor> = ::embassy_util::Forever::new(); static EXECUTOR: ::embassy_executor::_export::StaticCell<::embassy_executor::Executor> = ::embassy_executor::_export::StaticCell::new();
let executor = EXECUTOR.put(::embassy_executor::Executor::new()); let executor = EXECUTOR.init(::embassy_executor::Executor::new());
executor.start(|spawner| { executor.start(|spawner| {
spawner.spawn(__embassy_main(spawner)).unwrap(); spawner.spawn(__embassy_main(spawner)).unwrap();

View file

@ -31,3 +31,4 @@ futures-util = { version = "0.3.17", features = [ "channel" ] }
# Enable critical-section implementation for std, for tests # Enable critical-section implementation for std, for tests
critical-section = { version = "1.1", features = ["std"] } critical-section = { version = "1.1", features = ["std"] }
static_cell = "1.0"

View file

@ -471,10 +471,10 @@ mod tests {
use futures_executor::ThreadPool; use futures_executor::ThreadPool;
use futures_timer::Delay; use futures_timer::Delay;
use futures_util::task::SpawnExt; use futures_util::task::SpawnExt;
use static_cell::StaticCell;
use super::*; use super::*;
use crate::blocking_mutex::raw::{CriticalSectionRawMutex, NoopRawMutex}; use crate::blocking_mutex::raw::{CriticalSectionRawMutex, NoopRawMutex};
use crate::Forever;
fn capacity<T, const N: usize>(c: &ChannelState<T, N>) -> usize { fn capacity<T, const N: usize>(c: &ChannelState<T, N>) -> usize {
c.queue.capacity() - c.queue.len() c.queue.capacity() - c.queue.len()
@ -549,8 +549,8 @@ mod tests {
async fn receiver_receives_given_try_send_async() { async fn receiver_receives_given_try_send_async() {
let executor = ThreadPool::new().unwrap(); let executor = ThreadPool::new().unwrap();
static CHANNEL: Forever<Channel<CriticalSectionRawMutex, u32, 3>> = Forever::new(); static CHANNEL: StaticCell<Channel<CriticalSectionRawMutex, u32, 3>> = StaticCell::new();
let c = &*CHANNEL.put(Channel::new()); let c = &*CHANNEL.init(Channel::new());
let c2 = c; let c2 = c;
assert!(executor assert!(executor
.spawn(async move { .spawn(async move {
@ -571,8 +571,8 @@ mod tests {
async fn senders_sends_wait_until_capacity() { async fn senders_sends_wait_until_capacity() {
let executor = ThreadPool::new().unwrap(); let executor = ThreadPool::new().unwrap();
static CHANNEL: Forever<Channel<CriticalSectionRawMutex, u32, 1>> = Forever::new(); static CHANNEL: StaticCell<Channel<CriticalSectionRawMutex, u32, 1>> = StaticCell::new();
let c = &*CHANNEL.put(Channel::new()); let c = &*CHANNEL.init(Channel::new());
assert!(c.try_send(1).is_ok()); assert!(c.try_send(1).is_ok());
let c2 = c; let c2 = c;

View file

@ -1,95 +0,0 @@
use core::cell::UnsafeCell;
use core::mem::MaybeUninit;
use atomic_polyfill::{AtomicBool, Ordering};
/// Type with static lifetime that may be written to once at runtime.
///
/// This may be used to initialize static objects at runtime, typically in the init routine.
/// This is useful for objects such as Embassy's RTC, which cannot be initialized in a const
/// context.
///
/// Note: IF a global mutable variable is desired, use a CriticalSectionMutex or ThreadModeMutex instead.
///
/// ```
/// use embassy_util::Forever;
/// // Using an integer for the sake of keeping this example self-contained,
/// // see https://github.com/embassy-rs/embassy/wiki/Getting-Started for a more "proper" example.
/// static SOME_INT: Forever<u32> =Forever::new();
///
/// // put returns a mutable pointer to the object stored in the forever, which may then be passed
/// // around.
/// let mut x = SOME_INT.put(42);
/// assert_eq!(*x, 42);
/// ```
pub struct Forever<T> {
used: AtomicBool,
t: UnsafeCell<MaybeUninit<T>>,
}
unsafe impl<T> Send for Forever<T> {}
unsafe impl<T> Sync for Forever<T> {}
impl<T> Forever<T> {
/// Create a new `Forever`.
#[inline(always)]
pub const fn new() -> Self {
Self {
used: AtomicBool::new(false),
t: UnsafeCell::new(MaybeUninit::uninit()),
}
}
/// Store a value in this `Forever`, returning a mutable reference to it.
///
/// Using this method, the compiler usually constructs `val` in the stack and then moves
/// it into the `Forever`. If `T` is big, this is likely to cause stack overflows.
/// Considering using [`Signal::put_with`] instead, which will construct it in-place inside the `Forever`.
///
/// # Panics
///
/// Panics if this `Forever` already has a value stored in it.
#[inline(always)]
#[allow(clippy::mut_from_ref)]
pub fn put(&'static self, val: T) -> &'static mut T {
self.put_with(|| val)
}
/// Store the closure return value in this `Forever`, returning a mutable reference to it.
///
/// The advantage over [`Forever::put`] is that this method allows the closure to construct
/// the `T` value in-place directly inside the `Forever`, saving stack space.
///
/// # Panics
///
/// Panics if this `Forever` already has a value stored in it.
#[inline(always)]
#[allow(clippy::mut_from_ref)]
pub fn put_with(&'static self, val: impl FnOnce() -> T) -> &'static mut T {
if self
.used
.compare_exchange(false, true, Ordering::Relaxed, Ordering::Relaxed)
.is_err()
{
panic!("Forever.put() called multiple times");
}
let p: &'static mut MaybeUninit<T> = unsafe { &mut *self.t.get() };
p.write(val())
}
/// Unsafely get a mutable reference to the contents of this Forever.
///
/// # Safety
///
/// This is undefined behavior if:
///
/// - The `Forever` has not been initialized yet (with `put' or `put_with`), or
/// - A reference to the contents (mutable or not) already exists.
#[inline(always)]
#[allow(clippy::mut_from_ref)]
pub unsafe fn steal(&self) -> &mut T {
let p: &mut MaybeUninit<T> = &mut *self.t.get();
p.assume_init_mut()
}
}

View file

@ -16,10 +16,8 @@ pub mod mutex;
pub mod pipe; pub mod pipe;
pub mod waitqueue; pub mod waitqueue;
mod forever;
mod select; mod select;
mod yield_now; mod yield_now;
pub use forever::*;
pub use select::*; pub use select::*;
pub use yield_now::*; pub use yield_now::*;

View file

@ -461,10 +461,10 @@ mod io_impls {
mod tests { mod tests {
use futures_executor::ThreadPool; use futures_executor::ThreadPool;
use futures_util::task::SpawnExt; use futures_util::task::SpawnExt;
use static_cell::StaticCell;
use super::*; use super::*;
use crate::blocking_mutex::raw::{CriticalSectionRawMutex, NoopRawMutex}; use crate::blocking_mutex::raw::{CriticalSectionRawMutex, NoopRawMutex};
use crate::Forever;
fn capacity<const N: usize>(c: &PipeState<N>) -> usize { fn capacity<const N: usize>(c: &PipeState<N>) -> usize {
N - c.buffer.len() N - c.buffer.len()
@ -528,8 +528,8 @@ mod tests {
async fn receiver_receives_given_try_write_async() { async fn receiver_receives_given_try_write_async() {
let executor = ThreadPool::new().unwrap(); let executor = ThreadPool::new().unwrap();
static CHANNEL: Forever<Pipe<CriticalSectionRawMutex, 3>> = Forever::new(); static CHANNEL: StaticCell<Pipe<CriticalSectionRawMutex, 3>> = StaticCell::new();
let c = &*CHANNEL.put(Pipe::new()); let c = &*CHANNEL.init(Pipe::new());
let c2 = c; let c2 = c;
let f = async move { let f = async move {
assert_eq!(c2.try_write(&[42]), Ok(1)); assert_eq!(c2.try_write(&[42]), Ok(1));

View file

@ -22,6 +22,7 @@ embedded-io = "0.3.0"
defmt = "0.3" defmt = "0.3"
defmt-rtt = "0.3" defmt-rtt = "0.3"
static_cell = "1.0"
cortex-m = { version = "0.7.6", features = ["critical-section-single-core"] } cortex-m = { version = "0.7.6", features = ["critical-section-single-core"] }
cortex-m-rt = "0.7.0" cortex-m-rt = "0.7.0"
panic-probe = { version = "0.3", features = ["print-defmt"] } panic-probe = { version = "0.3", features = ["print-defmt"] }

View file

@ -8,7 +8,7 @@ use embassy_nrf::gpio::{AnyPin, Level, Output, OutputDrive, Pin};
use embassy_time::{Duration, Timer}; use embassy_time::{Duration, Timer};
use embassy_util::blocking_mutex::raw::NoopRawMutex; use embassy_util::blocking_mutex::raw::NoopRawMutex;
use embassy_util::channel::mpmc::{Channel, Receiver, Sender}; use embassy_util::channel::mpmc::{Channel, Receiver, Sender};
use embassy_util::Forever; use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
enum LedState { enum LedState {
@ -16,7 +16,7 @@ enum LedState {
Off, Off,
} }
static CHANNEL: Forever<Channel<NoopRawMutex, LedState, 1>> = Forever::new(); static CHANNEL: StaticCell<Channel<NoopRawMutex, LedState, 1>> = StaticCell::new();
#[embassy_executor::task] #[embassy_executor::task]
async fn send_task(sender: Sender<'static, NoopRawMutex, LedState, 1>) { async fn send_task(sender: Sender<'static, NoopRawMutex, LedState, 1>) {
@ -43,7 +43,7 @@ async fn recv_task(led: AnyPin, receiver: Receiver<'static, NoopRawMutex, LedSta
#[embassy_executor::main] #[embassy_executor::main]
async fn main(spawner: Spawner) { async fn main(spawner: Spawner) {
let p = embassy_nrf::init(Default::default()); let p = embassy_nrf::init(Default::default());
let channel = CHANNEL.put(Channel::new()); let channel = CHANNEL.init(Channel::new());
unwrap!(spawner.spawn(send_task(channel.sender()))); unwrap!(spawner.spawn(send_task(channel.sender())));
unwrap!(spawner.spawn(recv_task(p.P0_13.degrade(), channel.receiver()))); unwrap!(spawner.spawn(recv_task(p.P0_13.degrade(), channel.receiver())));

View file

@ -63,7 +63,7 @@ use embassy_nrf::executor::{Executor, InterruptExecutor};
use embassy_nrf::interrupt; use embassy_nrf::interrupt;
use embassy_nrf::interrupt::InterruptExt; use embassy_nrf::interrupt::InterruptExt;
use embassy_time::{Duration, Instant, Timer}; use embassy_time::{Duration, Instant, Timer};
use embassy_util::Forever; use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::task] #[embassy_executor::task]
@ -108,9 +108,9 @@ async fn run_low() {
} }
} }
static EXECUTOR_HIGH: Forever<InterruptExecutor<interrupt::SWI1_EGU1>> = Forever::new(); static EXECUTOR_HIGH: StaticCell<InterruptExecutor<interrupt::SWI1_EGU1>> = StaticCell::new();
static EXECUTOR_MED: Forever<InterruptExecutor<interrupt::SWI0_EGU0>> = Forever::new(); static EXECUTOR_MED: StaticCell<InterruptExecutor<interrupt::SWI0_EGU0>> = StaticCell::new();
static EXECUTOR_LOW: Forever<Executor> = Forever::new(); static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new();
#[entry] #[entry]
fn main() -> ! { fn main() -> ! {
@ -121,19 +121,19 @@ fn main() -> ! {
// High-priority executor: SWI1_EGU1, priority level 6 // High-priority executor: SWI1_EGU1, priority level 6
let irq = interrupt::take!(SWI1_EGU1); let irq = interrupt::take!(SWI1_EGU1);
irq.set_priority(interrupt::Priority::P6); irq.set_priority(interrupt::Priority::P6);
let executor = EXECUTOR_HIGH.put(InterruptExecutor::new(irq)); let executor = EXECUTOR_HIGH.init(InterruptExecutor::new(irq));
let spawner = executor.start(); let spawner = executor.start();
unwrap!(spawner.spawn(run_high())); unwrap!(spawner.spawn(run_high()));
// Medium-priority executor: SWI0_EGU0, priority level 7 // Medium-priority executor: SWI0_EGU0, priority level 7
let irq = interrupt::take!(SWI0_EGU0); let irq = interrupt::take!(SWI0_EGU0);
irq.set_priority(interrupt::Priority::P7); irq.set_priority(interrupt::Priority::P7);
let executor = EXECUTOR_MED.put(InterruptExecutor::new(irq)); let executor = EXECUTOR_MED.init(InterruptExecutor::new(irq));
let spawner = executor.start(); let spawner = executor.start();
unwrap!(spawner.spawn(run_med())); unwrap!(spawner.spawn(run_med()));
// Low priority executor: runs in thread mode, using WFE/SEV // Low priority executor: runs in thread mode, using WFE/SEV
let executor = EXECUTOR_LOW.put(Executor::new()); let executor = EXECUTOR_LOW.init(Executor::new());
executor.run(|spawner| { executor.run(|spawner| {
unwrap!(spawner.spawn(run_low())); unwrap!(spawner.spawn(run_low()));
}); });

View file

@ -8,7 +8,7 @@ use defmt::{info, unwrap};
use embassy_executor::raw::TaskStorage; use embassy_executor::raw::TaskStorage;
use embassy_executor::Executor; use embassy_executor::Executor;
use embassy_time::{Duration, Timer}; use embassy_time::{Duration, Timer};
use embassy_util::Forever; use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
async fn run1() { async fn run1() {
@ -25,14 +25,14 @@ async fn run2() {
} }
} }
static EXECUTOR: Forever<Executor> = Forever::new(); static EXECUTOR: StaticCell<Executor> = StaticCell::new();
#[entry] #[entry]
fn main() -> ! { fn main() -> ! {
info!("Hello World!"); info!("Hello World!");
let _p = embassy_nrf::init(Default::default()); let _p = embassy_nrf::init(Default::default());
let executor = EXECUTOR.put(Executor::new()); let executor = EXECUTOR.init(Executor::new());
let run1_task = TaskStorage::new(); let run1_task = TaskStorage::new();
let run2_task = TaskStorage::new(); let run2_task = TaskStorage::new();

View file

@ -18,17 +18,17 @@ use embassy_usb::{Builder, Config, UsbDevice};
use embassy_usb_ncm::{CdcNcmClass, Receiver, Sender, State}; use embassy_usb_ncm::{CdcNcmClass, Receiver, Sender, State};
use embassy_util::blocking_mutex::raw::ThreadModeRawMutex; use embassy_util::blocking_mutex::raw::ThreadModeRawMutex;
use embassy_util::channel::mpmc::Channel; use embassy_util::channel::mpmc::Channel;
use embassy_util::Forever;
use embedded_io::asynch::{Read, Write}; use embedded_io::asynch::{Read, Write};
use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
type MyDriver = Driver<'static, peripherals::USBD, PowerUsb>; type MyDriver = Driver<'static, peripherals::USBD, PowerUsb>;
macro_rules! forever { macro_rules! singleton {
($val:expr) => {{ ($val:expr) => {{
type T = impl Sized; type T = impl Sized;
static FOREVER: Forever<T> = Forever::new(); static STATIC_CELL: StaticCell<T> = StaticCell::new();
FOREVER.put_with(move || $val) STATIC_CELL.init_with(move || $val)
}}; }};
} }
@ -116,7 +116,7 @@ async fn main(spawner: Spawner) {
control_buf: [u8; 128], control_buf: [u8; 128],
serial_state: State<'static>, serial_state: State<'static>,
} }
let res: &mut Resources = forever!(Resources { let res: &mut Resources = singleton!(Resources {
device_descriptor: [0; 256], device_descriptor: [0; 256],
config_descriptor: [0; 256], config_descriptor: [0; 256],
bos_descriptor: [0; 256], bos_descriptor: [0; 256],
@ -174,10 +174,10 @@ async fn main(spawner: Spawner) {
// Init network stack // Init network stack
let device = Device { mac_addr: our_mac_addr }; let device = Device { mac_addr: our_mac_addr };
let stack = &*forever!(Stack::new( let stack = &*singleton!(Stack::new(
device, device,
config, config,
forever!(StackResources::<1, 2, 8>::new()), singleton!(StackResources::<1, 2, 8>::new()),
seed seed
)); ));

View file

@ -12,7 +12,7 @@ use embassy_nrf::{interrupt, pac, peripherals};
use embassy_usb::driver::EndpointError; use embassy_usb::driver::EndpointError;
use embassy_usb::{Builder, Config, UsbDevice}; use embassy_usb::{Builder, Config, UsbDevice};
use embassy_usb_serial::{CdcAcmClass, State}; use embassy_usb_serial::{CdcAcmClass, State};
use embassy_util::Forever; use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
type MyDriver = Driver<'static, peripherals::USBD, PowerUsb>; type MyDriver = Driver<'static, peripherals::USBD, PowerUsb>;
@ -67,8 +67,8 @@ async fn main(spawner: Spawner) {
control_buf: [u8; 64], control_buf: [u8; 64],
serial_state: State<'static>, serial_state: State<'static>,
} }
static RESOURCES: Forever<Resources> = Forever::new(); static RESOURCES: StaticCell<Resources> = StaticCell::new();
let res = RESOURCES.put(Resources { let res = RESOURCES.init(Resources {
device_descriptor: [0; 256], device_descriptor: [0; 256],
config_descriptor: [0; 256], config_descriptor: [0; 256],
bos_descriptor: [0; 256], bos_descriptor: [0; 256],

View file

@ -20,3 +20,4 @@ libc = "0.2.101"
clap = { version = "3.0.0-beta.5", features = ["derive"] } clap = { version = "3.0.0-beta.5", features = ["derive"] }
rand_core = { version = "0.6.3", features = ["std"] } rand_core = { version = "0.6.3", features = ["std"] }
heapless = { version = "0.7.5", default-features = false } heapless = { version = "0.7.5", default-features = false }
static_cell = "1.0"

View file

@ -4,22 +4,22 @@ use clap::Parser;
use embassy_executor::{Executor, Spawner}; use embassy_executor::{Executor, Spawner};
use embassy_net::tcp::TcpSocket; use embassy_net::tcp::TcpSocket;
use embassy_net::{ConfigStrategy, Ipv4Address, Ipv4Cidr, Stack, StackResources}; use embassy_net::{ConfigStrategy, Ipv4Address, Ipv4Cidr, Stack, StackResources};
use embassy_util::Forever;
use embedded_io::asynch::Write; use embedded_io::asynch::Write;
use heapless::Vec; use heapless::Vec;
use log::*; use log::*;
use rand_core::{OsRng, RngCore}; use rand_core::{OsRng, RngCore};
use static_cell::StaticCell;
#[path = "../tuntap.rs"] #[path = "../tuntap.rs"]
mod tuntap; mod tuntap;
use crate::tuntap::TunTapDevice; use crate::tuntap::TunTapDevice;
macro_rules! forever { macro_rules! singleton {
($val:expr) => {{ ($val:expr) => {{
type T = impl Sized; type T = impl Sized;
static FOREVER: Forever<T> = Forever::new(); static STATIC_CELL: StaticCell<T> = StaticCell::new();
FOREVER.put_with(move || $val) STATIC_CELL.init_with(move || $val)
}}; }};
} }
@ -63,10 +63,10 @@ async fn main_task(spawner: Spawner) {
let seed = u64::from_le_bytes(seed); let seed = u64::from_le_bytes(seed);
// Init network stack // Init network stack
let stack = &*forever!(Stack::new( let stack = &*singleton!(Stack::new(
device, device,
config, config,
forever!(StackResources::<1, 2, 8>::new()), singleton!(StackResources::<1, 2, 8>::new()),
seed seed
)); ));
@ -97,7 +97,7 @@ async fn main_task(spawner: Spawner) {
} }
} }
static EXECUTOR: Forever<Executor> = Forever::new(); static EXECUTOR: StaticCell<Executor> = StaticCell::new();
fn main() { fn main() {
env_logger::builder() env_logger::builder()
@ -106,7 +106,7 @@ fn main() {
.format_timestamp_nanos() .format_timestamp_nanos()
.init(); .init();
let executor = EXECUTOR.put(Executor::new()); let executor = EXECUTOR.init(Executor::new());
executor.run(|spawner| { executor.run(|spawner| {
spawner.spawn(main_task(spawner)).unwrap(); spawner.spawn(main_task(spawner)).unwrap();
}); });

View file

@ -4,21 +4,21 @@ use clap::Parser;
use embassy_executor::{Executor, Spawner}; use embassy_executor::{Executor, Spawner};
use embassy_net::udp::UdpSocket; use embassy_net::udp::UdpSocket;
use embassy_net::{ConfigStrategy, Ipv4Address, Ipv4Cidr, PacketMetadata, Stack, StackResources}; use embassy_net::{ConfigStrategy, Ipv4Address, Ipv4Cidr, PacketMetadata, Stack, StackResources};
use embassy_util::Forever;
use heapless::Vec; use heapless::Vec;
use log::*; use log::*;
use rand_core::{OsRng, RngCore}; use rand_core::{OsRng, RngCore};
use static_cell::StaticCell;
#[path = "../tuntap.rs"] #[path = "../tuntap.rs"]
mod tuntap; mod tuntap;
use crate::tuntap::TunTapDevice; use crate::tuntap::TunTapDevice;
macro_rules! forever { macro_rules! singleton {
($val:expr) => {{ ($val:expr) => {{
type T = impl Sized; type T = impl Sized;
static FOREVER: Forever<T> = Forever::new(); static STATIC_CELL: StaticCell<T> = StaticCell::new();
FOREVER.put_with(move || $val) STATIC_CELL.init_with(move || $val)
}}; }};
} }
@ -62,10 +62,10 @@ async fn main_task(spawner: Spawner) {
let seed = u64::from_le_bytes(seed); let seed = u64::from_le_bytes(seed);
// Init network stack // Init network stack
let stack = &*forever!(Stack::new( let stack = &*singleton!(Stack::new(
device, device,
config, config,
forever!(StackResources::<1, 2, 8>::new()), singleton!(StackResources::<1, 2, 8>::new()),
seed seed
)); ));
@ -93,7 +93,7 @@ async fn main_task(spawner: Spawner) {
} }
} }
static EXECUTOR: Forever<Executor> = Forever::new(); static EXECUTOR: StaticCell<Executor> = StaticCell::new();
fn main() { fn main() {
env_logger::builder() env_logger::builder()
@ -102,7 +102,7 @@ fn main() {
.format_timestamp_nanos() .format_timestamp_nanos()
.init(); .init();
let executor = EXECUTOR.put(Executor::new()); let executor = EXECUTOR.init(Executor::new());
executor.run(|spawner| { executor.run(|spawner| {
spawner.spawn(main_task(spawner)).unwrap(); spawner.spawn(main_task(spawner)).unwrap();
}); });

View file

@ -5,10 +5,10 @@ mod serial_port;
use async_io::Async; use async_io::Async;
use embassy_executor::Executor; use embassy_executor::Executor;
use embassy_util::Forever;
use embedded_io::asynch::Read; use embedded_io::asynch::Read;
use log::*; use log::*;
use nix::sys::termios; use nix::sys::termios;
use static_cell::StaticCell;
use self::serial_port::SerialPort; use self::serial_port::SerialPort;
@ -40,7 +40,7 @@ async fn run() {
} }
} }
static EXECUTOR: Forever<Executor> = Forever::new(); static EXECUTOR: StaticCell<Executor> = StaticCell::new();
fn main() { fn main() {
env_logger::builder() env_logger::builder()
@ -49,7 +49,7 @@ fn main() {
.format_timestamp_nanos() .format_timestamp_nanos()
.init(); .init();
let executor = EXECUTOR.put(Executor::new()); let executor = EXECUTOR.init(Executor::new());
executor.run(|spawner| { executor.run(|spawner| {
spawner.spawn(run()).unwrap(); spawner.spawn(run()).unwrap();
}); });

View file

@ -23,3 +23,4 @@ futures = { version = "0.3.17", default-features = false, features = ["async-awa
heapless = { version = "0.7.5", default-features = false } heapless = { version = "0.7.5", default-features = false }
nb = "1.0.0" nb = "1.0.0"
embedded-storage = "0.3.0" embedded-storage = "0.3.0"
static_cell = "1.0"

View file

@ -63,7 +63,7 @@ use embassy_stm32::executor::{Executor, InterruptExecutor};
use embassy_stm32::interrupt; use embassy_stm32::interrupt;
use embassy_stm32::interrupt::InterruptExt; use embassy_stm32::interrupt::InterruptExt;
use embassy_time::{Duration, Instant, Timer}; use embassy_time::{Duration, Instant, Timer};
use embassy_util::Forever; use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::task] #[embassy_executor::task]
@ -108,9 +108,9 @@ async fn run_low() {
} }
} }
static EXECUTOR_HIGH: Forever<InterruptExecutor<interrupt::UART4>> = Forever::new(); static EXECUTOR_HIGH: StaticCell<InterruptExecutor<interrupt::UART4>> = StaticCell::new();
static EXECUTOR_MED: Forever<InterruptExecutor<interrupt::UART5>> = Forever::new(); static EXECUTOR_MED: StaticCell<InterruptExecutor<interrupt::UART5>> = StaticCell::new();
static EXECUTOR_LOW: Forever<Executor> = Forever::new(); static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new();
#[entry] #[entry]
fn main() -> ! { fn main() -> ! {
@ -121,19 +121,19 @@ fn main() -> ! {
// High-priority executor: SWI1_EGU1, priority level 6 // High-priority executor: SWI1_EGU1, priority level 6
let irq = interrupt::take!(UART4); let irq = interrupt::take!(UART4);
irq.set_priority(interrupt::Priority::P6); irq.set_priority(interrupt::Priority::P6);
let executor = EXECUTOR_HIGH.put(InterruptExecutor::new(irq)); let executor = EXECUTOR_HIGH.init(InterruptExecutor::new(irq));
let spawner = executor.start(); let spawner = executor.start();
unwrap!(spawner.spawn(run_high())); unwrap!(spawner.spawn(run_high()));
// Medium-priority executor: SWI0_EGU0, priority level 7 // Medium-priority executor: SWI0_EGU0, priority level 7
let irq = interrupt::take!(UART5); let irq = interrupt::take!(UART5);
irq.set_priority(interrupt::Priority::P7); irq.set_priority(interrupt::Priority::P7);
let executor = EXECUTOR_MED.put(InterruptExecutor::new(irq)); let executor = EXECUTOR_MED.init(InterruptExecutor::new(irq));
let spawner = executor.start(); let spawner = executor.start();
unwrap!(spawner.spawn(run_med())); unwrap!(spawner.spawn(run_med()));
// Low priority executor: runs in thread mode, using WFE/SEV // Low priority executor: runs in thread mode, using WFE/SEV
let executor = EXECUTOR_LOW.put(Executor::new()); let executor = EXECUTOR_LOW.init(Executor::new());
executor.run(|spawner| { executor.run(|spawner| {
unwrap!(spawner.spawn(run_low())); unwrap!(spawner.spawn(run_low()));
}); });

View file

@ -23,6 +23,7 @@ heapless = { version = "0.7.5", default-features = false }
nb = "1.0.0" nb = "1.0.0"
embedded-storage = "0.3.0" embedded-storage = "0.3.0"
micromath = "2.0.0" micromath = "2.0.0"
static_cell = "1.0"
usb-device = "0.2" usb-device = "0.2"
usbd-serial = "0.1.1" usbd-serial = "0.1.1"

View file

@ -63,7 +63,7 @@ use embassy_stm32::executor::{Executor, InterruptExecutor};
use embassy_stm32::interrupt; use embassy_stm32::interrupt;
use embassy_stm32::interrupt::InterruptExt; use embassy_stm32::interrupt::InterruptExt;
use embassy_time::{Duration, Instant, Timer}; use embassy_time::{Duration, Instant, Timer};
use embassy_util::Forever; use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::task] #[embassy_executor::task]
@ -108,9 +108,9 @@ async fn run_low() {
} }
} }
static EXECUTOR_HIGH: Forever<InterruptExecutor<interrupt::UART4>> = Forever::new(); static EXECUTOR_HIGH: StaticCell<InterruptExecutor<interrupt::UART4>> = StaticCell::new();
static EXECUTOR_MED: Forever<InterruptExecutor<interrupt::UART5>> = Forever::new(); static EXECUTOR_MED: StaticCell<InterruptExecutor<interrupt::UART5>> = StaticCell::new();
static EXECUTOR_LOW: Forever<Executor> = Forever::new(); static EXECUTOR_LOW: StaticCell<Executor> = StaticCell::new();
#[entry] #[entry]
fn main() -> ! { fn main() -> ! {
@ -121,19 +121,19 @@ fn main() -> ! {
// High-priority executor: SWI1_EGU1, priority level 6 // High-priority executor: SWI1_EGU1, priority level 6
let irq = interrupt::take!(UART4); let irq = interrupt::take!(UART4);
irq.set_priority(interrupt::Priority::P6); irq.set_priority(interrupt::Priority::P6);
let executor = EXECUTOR_HIGH.put(InterruptExecutor::new(irq)); let executor = EXECUTOR_HIGH.init(InterruptExecutor::new(irq));
let spawner = executor.start(); let spawner = executor.start();
unwrap!(spawner.spawn(run_high())); unwrap!(spawner.spawn(run_high()));
// Medium-priority executor: SWI0_EGU0, priority level 7 // Medium-priority executor: SWI0_EGU0, priority level 7
let irq = interrupt::take!(UART5); let irq = interrupt::take!(UART5);
irq.set_priority(interrupt::Priority::P7); irq.set_priority(interrupt::Priority::P7);
let executor = EXECUTOR_MED.put(InterruptExecutor::new(irq)); let executor = EXECUTOR_MED.init(InterruptExecutor::new(irq));
let spawner = executor.start(); let spawner = executor.start();
unwrap!(spawner.spawn(run_med())); unwrap!(spawner.spawn(run_med()));
// Low priority executor: runs in thread mode, using WFE/SEV // Low priority executor: runs in thread mode, using WFE/SEV
let executor = EXECUTOR_LOW.put(Executor::new()); let executor = EXECUTOR_LOW.init(Executor::new());
executor.run(|spawner| { executor.run(|spawner| {
unwrap!(spawner.spawn(run_low())); unwrap!(spawner.spawn(run_low()));
}); });

View file

@ -24,3 +24,4 @@ nb = "1.0.0"
rand_core = "0.6.3" rand_core = "0.6.3"
critical-section = "1.1" critical-section = "1.1"
embedded-storage = "0.3.0" embedded-storage = "0.3.0"
static_cell = "1.0"

View file

@ -13,16 +13,16 @@ use embassy_stm32::rng::Rng;
use embassy_stm32::time::mhz; use embassy_stm32::time::mhz;
use embassy_stm32::{interrupt, Config}; use embassy_stm32::{interrupt, Config};
use embassy_time::{Duration, Timer}; use embassy_time::{Duration, Timer};
use embassy_util::Forever;
use embedded_io::asynch::Write; use embedded_io::asynch::Write;
use rand_core::RngCore; use rand_core::RngCore;
use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
macro_rules! forever { macro_rules! singleton {
($val:expr) => {{ ($val:expr) => {{
type T = impl Sized; type T = impl Sized;
static FOREVER: Forever<T> = Forever::new(); static STATIC_CELL: StaticCell<T> = StaticCell::new();
FOREVER.put_with(move || $val) STATIC_CELL.init_with(move || $val)
}}; }};
} }
@ -52,7 +52,7 @@ async fn main(spawner: Spawner) -> ! {
let device = unsafe { let device = unsafe {
Ethernet::new( Ethernet::new(
forever!(State::new()), singleton!(State::new()),
p.ETH, p.ETH,
eth_int, eth_int,
p.PA1, p.PA1,
@ -78,10 +78,10 @@ async fn main(spawner: Spawner) -> ! {
//}); //});
// Init network stack // Init network stack
let stack = &*forever!(Stack::new( let stack = &*singleton!(Stack::new(
device, device,
config, config,
forever!(StackResources::<1, 2, 8>::new()), singleton!(StackResources::<1, 2, 8>::new()),
seed seed
)); ));

View file

@ -28,6 +28,7 @@ critical-section = "1.1"
micromath = "2.0.0" micromath = "2.0.0"
stm32-fmc = "0.2.4" stm32-fmc = "0.2.4"
embedded-storage = "0.3.0" embedded-storage = "0.3.0"
static_cell = "1.0"
# cargo build/run # cargo build/run
[profile.dev] [profile.dev]

View file

@ -13,16 +13,16 @@ use embassy_stm32::rng::Rng;
use embassy_stm32::time::mhz; use embassy_stm32::time::mhz;
use embassy_stm32::{interrupt, Config}; use embassy_stm32::{interrupt, Config};
use embassy_time::{Duration, Timer}; use embassy_time::{Duration, Timer};
use embassy_util::Forever;
use embedded_io::asynch::Write; use embedded_io::asynch::Write;
use rand_core::RngCore; use rand_core::RngCore;
use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
macro_rules! forever { macro_rules! singleton {
($val:expr) => {{ ($val:expr) => {{
type T = impl Sized; type T = impl Sized;
static FOREVER: Forever<T> = Forever::new(); static STATIC_CELL: StaticCell<T> = StaticCell::new();
FOREVER.put_with(move || $val) STATIC_CELL.init_with(move || $val)
}}; }};
} }
@ -53,7 +53,7 @@ async fn main(spawner: Spawner) -> ! {
let device = unsafe { let device = unsafe {
Ethernet::new( Ethernet::new(
forever!(State::new()), singleton!(State::new()),
p.ETH, p.ETH,
eth_int, eth_int,
p.PA1, p.PA1,
@ -79,10 +79,10 @@ async fn main(spawner: Spawner) -> ! {
//}); //});
// Init network stack // Init network stack
let stack = &*forever!(Stack::new( let stack = &*singleton!(Stack::new(
device, device,
config, config,
forever!(StackResources::<1, 2, 8>::new()), singleton!(StackResources::<1, 2, 8>::new()),
seed seed
)); ));

View file

@ -13,17 +13,17 @@ use embassy_stm32::rng::Rng;
use embassy_stm32::time::mhz; use embassy_stm32::time::mhz;
use embassy_stm32::{interrupt, Config}; use embassy_stm32::{interrupt, Config};
use embassy_time::{Duration, Timer}; use embassy_time::{Duration, Timer};
use embassy_util::Forever;
use embedded_io::asynch::Write; use embedded_io::asynch::Write;
use embedded_nal_async::{Ipv4Addr, SocketAddr, SocketAddrV4, TcpConnect}; use embedded_nal_async::{Ipv4Addr, SocketAddr, SocketAddrV4, TcpConnect};
use rand_core::RngCore; use rand_core::RngCore;
use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
macro_rules! forever { macro_rules! singleton {
($val:expr) => {{ ($val:expr) => {{
type T = impl Sized; type T = impl Sized;
static FOREVER: Forever<T> = Forever::new(); static STATIC_CELL: StaticCell<T> = StaticCell::new();
FOREVER.put_with(move || $val) STATIC_CELL.init_with(move || $val)
}}; }};
} }
@ -54,7 +54,7 @@ async fn main(spawner: Spawner) -> ! {
let device = unsafe { let device = unsafe {
Ethernet::new( Ethernet::new(
forever!(State::new()), singleton!(State::new()),
p.ETH, p.ETH,
eth_int, eth_int,
p.PA1, p.PA1,
@ -80,10 +80,10 @@ async fn main(spawner: Spawner) -> ! {
//}); //});
// Init network stack // Init network stack
let stack = &*forever!(Stack::new( let stack = &*singleton!(Stack::new(
device, device,
config, config,
forever!(StackResources::<1, 2, 8>::new()), singleton!(StackResources::<1, 2, 8>::new()),
seed seed
)); ));

View file

@ -12,8 +12,8 @@ use embassy_stm32::dma::NoDma;
use embassy_stm32::peripherals::SPI3; use embassy_stm32::peripherals::SPI3;
use embassy_stm32::time::mhz; use embassy_stm32::time::mhz;
use embassy_stm32::{spi, Config}; use embassy_stm32::{spi, Config};
use embassy_util::Forever;
use heapless::String; use heapless::String;
use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::task] #[embassy_executor::task]
@ -31,7 +31,7 @@ async fn main_task(mut spi: spi::Spi<'static, SPI3, NoDma, NoDma>) {
} }
} }
static EXECUTOR: Forever<Executor> = Forever::new(); static EXECUTOR: StaticCell<Executor> = StaticCell::new();
#[entry] #[entry]
fn main() -> ! { fn main() -> ! {
@ -54,7 +54,7 @@ fn main() -> ! {
spi::Config::default(), spi::Config::default(),
); );
let executor = EXECUTOR.put(Executor::new()); let executor = EXECUTOR.init(Executor::new());
executor.run(|spawner| { executor.run(|spawner| {
unwrap!(spawner.spawn(main_task(spi))); unwrap!(spawner.spawn(main_task(spi)));

View file

@ -11,8 +11,8 @@ use embassy_executor::Executor;
use embassy_stm32::peripherals::{DMA1_CH3, DMA1_CH4, SPI3}; use embassy_stm32::peripherals::{DMA1_CH3, DMA1_CH4, SPI3};
use embassy_stm32::time::mhz; use embassy_stm32::time::mhz;
use embassy_stm32::{spi, Config}; use embassy_stm32::{spi, Config};
use embassy_util::Forever;
use heapless::String; use heapless::String;
use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::task] #[embassy_executor::task]
@ -27,7 +27,7 @@ async fn main_task(mut spi: spi::Spi<'static, SPI3, DMA1_CH3, DMA1_CH4>) {
} }
} }
static EXECUTOR: Forever<Executor> = Forever::new(); static EXECUTOR: StaticCell<Executor> = StaticCell::new();
#[entry] #[entry]
fn main() -> ! { fn main() -> ! {
@ -50,7 +50,7 @@ fn main() -> ! {
spi::Config::default(), spi::Config::default(),
); );
let executor = EXECUTOR.put(Executor::new()); let executor = EXECUTOR.init(Executor::new());
executor.run(|spawner| { executor.run(|spawner| {
unwrap!(spawner.spawn(main_task(spi))); unwrap!(spawner.spawn(main_task(spi)));

View file

@ -7,7 +7,7 @@ use defmt::*;
use embassy_executor::Executor; use embassy_executor::Executor;
use embassy_stm32::dma::NoDma; use embassy_stm32::dma::NoDma;
use embassy_stm32::usart::{Config, Uart}; use embassy_stm32::usart::{Config, Uart};
use embassy_util::Forever; use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::task] #[embassy_executor::task]
@ -27,13 +27,13 @@ async fn main_task() {
} }
} }
static EXECUTOR: Forever<Executor> = Forever::new(); static EXECUTOR: StaticCell<Executor> = StaticCell::new();
#[entry] #[entry]
fn main() -> ! { fn main() -> ! {
info!("Hello World!"); info!("Hello World!");
let executor = EXECUTOR.put(Executor::new()); let executor = EXECUTOR.init(Executor::new());
executor.run(|spawner| { executor.run(|spawner| {
unwrap!(spawner.spawn(main_task())); unwrap!(spawner.spawn(main_task()));

View file

@ -9,8 +9,8 @@ use defmt::*;
use embassy_executor::Executor; use embassy_executor::Executor;
use embassy_stm32::dma::NoDma; use embassy_stm32::dma::NoDma;
use embassy_stm32::usart::{Config, Uart}; use embassy_stm32::usart::{Config, Uart};
use embassy_util::Forever;
use heapless::String; use heapless::String;
use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::task] #[embassy_executor::task]
@ -30,13 +30,13 @@ async fn main_task() {
} }
} }
static EXECUTOR: Forever<Executor> = Forever::new(); static EXECUTOR: StaticCell<Executor> = StaticCell::new();
#[entry] #[entry]
fn main() -> ! { fn main() -> ! {
info!("Hello World!"); info!("Hello World!");
let executor = EXECUTOR.put(Executor::new()); let executor = EXECUTOR.init(Executor::new());
executor.run(|spawner| { executor.run(|spawner| {
unwrap!(spawner.spawn(main_task())); unwrap!(spawner.spawn(main_task()));

View file

@ -29,3 +29,4 @@ panic-probe = { version = "0.3", features = ["print-defmt"] }
futures = { version = "0.3.17", default-features = false, features = ["async-await"] } futures = { version = "0.3.17", default-features = false, features = ["async-await"] }
heapless = { version = "0.7.5", default-features = false } heapless = { version = "0.7.5", default-features = false }
embedded-hal = "0.2.6" embedded-hal = "0.2.6"
static_cell = "1.0"

View file

@ -8,7 +8,7 @@ use defmt::*;
use embassy_executor::raw::TaskStorage; use embassy_executor::raw::TaskStorage;
use embassy_executor::Executor; use embassy_executor::Executor;
use embassy_time::{Duration, Timer}; use embassy_time::{Duration, Timer};
use embassy_util::Forever; use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
async fn run1() { async fn run1() {
@ -25,14 +25,14 @@ async fn run2() {
} }
} }
static EXECUTOR: Forever<Executor> = Forever::new(); static EXECUTOR: StaticCell<Executor> = StaticCell::new();
#[entry] #[entry]
fn main() -> ! { fn main() -> ! {
info!("Hello World!"); info!("Hello World!");
let _p = embassy_stm32::init(Default::default()); let _p = embassy_stm32::init(Default::default());
let executor = EXECUTOR.put(Executor::new()); let executor = EXECUTOR.init(Executor::new());
let run1_task = TaskStorage::new(); let run1_task = TaskStorage::new();
let run2_task = TaskStorage::new(); let run2_task = TaskStorage::new();

View file

@ -28,3 +28,4 @@ futures = { version = "0.3.17", default-features = false, features = ["async-awa
heapless = { version = "0.7.5", default-features = false } heapless = { version = "0.7.5", default-features = false }
rand_core = { version = "0.6.3", default-features = false } rand_core = { version = "0.6.3", default-features = false }
embedded-io = { version = "0.3.0", features = ["async"] } embedded-io = { version = "0.3.0", features = ["async"] }
static_cell = "1.0"

View file

@ -19,18 +19,18 @@ use embassy_usb::{Builder, UsbDevice};
use embassy_usb_ncm::{CdcNcmClass, Receiver, Sender, State}; use embassy_usb_ncm::{CdcNcmClass, Receiver, Sender, State};
use embassy_util::blocking_mutex::raw::ThreadModeRawMutex; use embassy_util::blocking_mutex::raw::ThreadModeRawMutex;
use embassy_util::channel::mpmc::Channel; use embassy_util::channel::mpmc::Channel;
use embassy_util::Forever;
use embedded_io::asynch::{Read, Write}; use embedded_io::asynch::{Read, Write};
use rand_core::RngCore; use rand_core::RngCore;
use static_cell::StaticCell;
use {defmt_rtt as _, panic_probe as _}; use {defmt_rtt as _, panic_probe as _};
type MyDriver = Driver<'static, embassy_stm32::peripherals::USB>; type MyDriver = Driver<'static, embassy_stm32::peripherals::USB>;
macro_rules! forever { macro_rules! singleton {
($val:expr) => {{ ($val:expr) => {{
type T = impl Sized; type T = impl Sized;
static FOREVER: Forever<T> = Forever::new(); static STATIC_CELL: StaticCell<T> = StaticCell::new();
FOREVER.put_with(move || $val) STATIC_CELL.init_with(move || $val)
}}; }};
} }
@ -115,7 +115,7 @@ async fn main(spawner: Spawner) {
control_buf: [u8; 128], control_buf: [u8; 128],
serial_state: State<'static>, serial_state: State<'static>,
} }
let res: &mut Resources = forever!(Resources { let res: &mut Resources = singleton!(Resources {
device_descriptor: [0; 256], device_descriptor: [0; 256],
config_descriptor: [0; 256], config_descriptor: [0; 256],
bos_descriptor: [0; 256], bos_descriptor: [0; 256],
@ -171,10 +171,10 @@ async fn main(spawner: Spawner) {
// Init network stack // Init network stack
let device = Device { mac_addr: our_mac_addr }; let device = Device { mac_addr: our_mac_addr };
let stack = &*forever!(Stack::new( let stack = &*singleton!(Stack::new(
device, device,
config, config,
forever!(StackResources::<1, 2, 8>::new()), singleton!(StackResources::<1, 2, 8>::new()),
seed seed
)); ));