From b0106898f73b70a5487034abbcd4be8296fd2894 Mon Sep 17 00:00:00 2001 From: dAxpeDDa Date: Sun, 25 Jun 2023 12:03:22 +0200 Subject: [PATCH] Don't allow event loops to run in parallel --- src/platform_impl/web/event_loop/mod.rs | 18 ++++++++---------- src/platform_impl/web/event_loop/runner.rs | 9 +++++++++ .../web/event_loop/window_target.rs | 3 ++- 3 files changed, 19 insertions(+), 11 deletions(-) diff --git a/src/platform_impl/web/event_loop/mod.rs b/src/platform_impl/web/event_loop/mod.rs index 28598dde..1dcee878 100644 --- a/src/platform_impl/web/event_loop/mod.rs +++ b/src/platform_impl/web/event_loop/mod.rs @@ -8,9 +8,7 @@ pub use self::window_target::EventLoopWindowTarget; use super::{backend, device, window}; use crate::event::Event; -use crate::event_loop::{ - ControlFlow, EventLoopBuilder, EventLoopWindowTarget as RootEventLoopWindowTarget, -}; +use crate::event_loop::{ControlFlow, EventLoopWindowTarget as RootEventLoopWindowTarget}; use std::marker::PhantomData; @@ -35,7 +33,7 @@ impl EventLoop { where F: 'static + FnMut(Event<'_, T>, &RootEventLoopWindowTarget, &mut ControlFlow), { - self.spawn_inner(event_handler); + self.spawn_inner(event_handler, false); // Throw an exception to break out of Rust execution and use unreachable to tell the // compiler this function won't return, giving it a return type of '!' @@ -50,11 +48,10 @@ impl EventLoop { where F: 'static + FnMut(Event<'_, T>, &RootEventLoopWindowTarget, &mut ControlFlow), { - EventLoopBuilder::::allow_event_loop_recreation(); - self.spawn_inner(event_handler); + self.spawn_inner(event_handler, true); } - fn spawn_inner(self, mut event_handler: F) + fn spawn_inner(self, mut event_handler: F, event_loop_recreation: bool) where F: 'static + FnMut(Event<'_, T>, &RootEventLoopWindowTarget, &mut ControlFlow), { @@ -63,9 +60,10 @@ impl EventLoop { _marker: PhantomData, }; - self.elw.p.run(Box::new(move |event, flow| { - event_handler(event, &target, flow) - })); + self.elw.p.run( + Box::new(move |event, flow| event_handler(event, &target, flow)), + event_loop_recreation, + ); } pub fn create_proxy(&self) -> EventLoopProxy { diff --git a/src/platform_impl/web/event_loop/runner.rs b/src/platform_impl/web/event_loop/runner.rs index 90a24f9a..a47dc8c5 100644 --- a/src/platform_impl/web/event_loop/runner.rs +++ b/src/platform_impl/web/event_loop/runner.rs @@ -35,6 +35,7 @@ type OnEventHandle = RefCell>>; pub struct Execution { runner: RefCell>, + event_loop_recreation: Cell, events: RefCell>>, id: RefCell, window: web_sys::Window, @@ -139,6 +140,7 @@ impl Shared { pub fn new() -> Self { Shared(Rc::new(Execution { runner: RefCell::new(RunnerEnum::Pending), + event_loop_recreation: Cell::new(false), events: RefCell::new(VecDeque::new()), #[allow(clippy::disallowed_methods)] window: web_sys::window().expect("only callable from inside the `Window`"), @@ -649,6 +651,9 @@ impl Shared { // * For each undropped `Window`: // * The `register_redraw_request` closure. // * The `destroy_fn` closure. + if self.0.event_loop_recreation.get() { + crate::event_loop::EventLoopBuilder::::allow_event_loop_recreation(); + } } // Check if the event loop is currently closed @@ -691,6 +696,10 @@ impl Shared { DeviceEvents::Never => false, } } + + pub fn event_loop_recreation(&self, allow: bool) { + self.0.event_loop_recreation.set(allow) + } } pub(crate) enum EventWrapper { diff --git a/src/platform_impl/web/event_loop/window_target.rs b/src/platform_impl/web/event_loop/window_target.rs index 8cfbb427..7534aa7b 100644 --- a/src/platform_impl/web/event_loop/window_target.rs +++ b/src/platform_impl/web/event_loop/window_target.rs @@ -69,7 +69,8 @@ impl EventLoopWindowTarget { EventLoopProxy::new(self.runner.clone()) } - pub fn run(&self, event_handler: Box>) { + pub fn run(&self, event_handler: Box>, event_loop_recreation: bool) { + self.runner.event_loop_recreation(event_loop_recreation); self.runner.set_listener(event_handler); }