84 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			84 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| #[cfg(feature = "executor-interrupt")]
 | |
| compile_error!("`executor-interrupt` is not supported with `arch-wasm`.");
 | |
| 
 | |
| #[cfg(feature = "executor-thread")]
 | |
| pub use thread::*;
 | |
| #[cfg(feature = "executor-thread")]
 | |
| mod thread {
 | |
| 
 | |
|     use core::marker::PhantomData;
 | |
| 
 | |
|     pub use embassy_executor_macros::main_wasm as main;
 | |
|     use js_sys::Promise;
 | |
|     use wasm_bindgen::prelude::*;
 | |
| 
 | |
|     use crate::raw::util::UninitCell;
 | |
|     use crate::{raw, Spawner};
 | |
| 
 | |
|     #[export_name = "__pender"]
 | |
|     fn __pender(context: *mut ()) {
 | |
|         let signaler: &'static WasmContext = unsafe { std::mem::transmute(context) };
 | |
|         let _ = signaler.promise.then(unsafe { signaler.closure.as_mut() });
 | |
|     }
 | |
| 
 | |
|     pub(crate) struct WasmContext {
 | |
|         promise: Promise,
 | |
|         closure: UninitCell<Closure<dyn FnMut(JsValue)>>,
 | |
|     }
 | |
| 
 | |
|     impl WasmContext {
 | |
|         pub fn new() -> Self {
 | |
|             Self {
 | |
|                 promise: Promise::resolve(&JsValue::undefined()),
 | |
|                 closure: UninitCell::uninit(),
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /// WASM executor, wasm_bindgen to schedule tasks on the JS event loop.
 | |
|     pub struct Executor {
 | |
|         inner: raw::Executor,
 | |
|         ctx: &'static WasmContext,
 | |
|         not_send: PhantomData<*mut ()>,
 | |
|     }
 | |
| 
 | |
|     impl Executor {
 | |
|         /// Create a new Executor.
 | |
|         pub fn new() -> Self {
 | |
|             let ctx = Box::leak(Box::new(WasmContext::new()));
 | |
|             Self {
 | |
|                 inner: raw::Executor::new(ctx as *mut WasmContext as *mut ()),
 | |
|                 ctx,
 | |
|                 not_send: PhantomData,
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// Run the executor.
 | |
|         ///
 | |
|         /// The `init` closure is called with a [`Spawner`] that spawns tasks on
 | |
|         /// this executor. Use it to spawn the initial task(s). After `init` returns,
 | |
|         /// the executor starts running the tasks.
 | |
|         ///
 | |
|         /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`),
 | |
|         /// for example by passing it as an argument to the initial tasks.
 | |
|         ///
 | |
|         /// This function requires `&'static mut self`. This means you have to store the
 | |
|         /// Executor instance in a place where it'll live forever and grants you mutable
 | |
|         /// access. There's a few ways to do this:
 | |
|         ///
 | |
|         /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe)
 | |
|         /// - a `static mut` (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)) {
 | |
|             unsafe {
 | |
|                 let executor = &self.inner;
 | |
|                 let future = Closure::new(move |_| {
 | |
|                     executor.poll();
 | |
|                 });
 | |
|                 self.ctx.closure.write_in_place(|| future);
 | |
|                 init(self.inner.spawner());
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |