Add a facility for delaying quit until critical tasks finish

Co-Authored-By: Antonio Scandurra <me@as-cii.com>
This commit is contained in:
Max Brunsfeld 2021-11-01 11:56:49 -07:00
parent 6e5ec2a00d
commit b8994c2a89
7 changed files with 70 additions and 4 deletions

1
Cargo.lock generated
View file

@ -2309,6 +2309,7 @@ dependencies = [
"etagere", "etagere",
"font-kit", "font-kit",
"foreign-types", "foreign-types",
"futures",
"gpui_macros", "gpui_macros",
"image 0.23.14", "image 0.23.14",
"lazy_static", "lazy_static",

View file

@ -15,6 +15,7 @@ backtrace = "0.3"
ctor = "0.1" ctor = "0.1"
env_logger = { version = "0.8", optional = true } env_logger = { version = "0.8", optional = true }
etagere = "0.2" etagere = "0.2"
futures = "0.3"
image = "0.23" image = "0.23"
lazy_static = "1.4.0" lazy_static = "1.4.0"
log = "0.4" log = "0.4"

View file

@ -265,6 +265,18 @@ impl App {
self self
} }
pub fn on_quit<F>(self, mut callback: F) -> Self
where
F: 'static + FnMut(&mut MutableAppContext),
{
let cx = self.0.clone();
self.0
.borrow_mut()
.foreground_platform
.on_quit(Box::new(move || callback(&mut *cx.borrow_mut())));
self
}
pub fn on_event<F>(self, mut callback: F) -> Self pub fn on_event<F>(self, mut callback: F) -> Self
where where
F: 'static + FnMut(Event, &mut MutableAppContext) -> bool, F: 'static + FnMut(Event, &mut MutableAppContext) -> bool,

View file

@ -38,9 +38,13 @@ pub enum Foreground {
} }
pub enum Background { pub enum Background {
Deterministic(Arc<Deterministic>), Deterministic {
executor: Arc<Deterministic>,
critical_tasks: Mutex<Vec<Task<()>>>,
},
Production { Production {
executor: Arc<smol::Executor<'static>>, executor: Arc<smol::Executor<'static>>,
critical_tasks: Mutex<Vec<Task<()>>>,
_stop: channel::Sender<()>, _stop: channel::Sender<()>,
}, },
} }
@ -500,6 +504,7 @@ impl Background {
Self::Production { Self::Production {
executor, executor,
critical_tasks: Default::default(),
_stop: stop.0, _stop: stop.0,
} }
} }
@ -516,11 +521,36 @@ impl Background {
let future = any_future(future); let future = any_future(future);
let any_task = match self { let any_task = match self {
Self::Production { executor, .. } => executor.spawn(future), Self::Production { executor, .. } => executor.spawn(future),
Self::Deterministic(executor) => executor.spawn(future), Self::Deterministic { executor, .. } => executor.spawn(future),
}; };
Task::send(any_task) Task::send(any_task)
} }
pub fn spawn_critical<T, F>(&self, future: F)
where
T: 'static + Send,
F: Send + Future<Output = T> + 'static,
{
let task = self.spawn(async move {
future.await;
});
match self {
Self::Production { critical_tasks, .. }
| Self::Deterministic { critical_tasks, .. } => critical_tasks.lock().push(task),
}
}
pub fn block_on_critical_tasks(&self, timeout: Duration) -> bool {
match self {
Background::Production { critical_tasks, .. }
| Self::Deterministic { critical_tasks, .. } => {
let tasks = mem::take(&mut *critical_tasks.lock());
self.block_with_timeout(timeout, futures::future::join_all(tasks))
.is_ok()
}
}
}
pub fn block_with_timeout<F, T>( pub fn block_with_timeout<F, T>(
&self, &self,
timeout: Duration, timeout: Duration,
@ -534,7 +564,7 @@ impl Background {
if !timeout.is_zero() { if !timeout.is_zero() {
let output = match self { let output = match self {
Self::Production { .. } => smol::block_on(util::timeout(timeout, &mut future)).ok(), Self::Production { .. } => smol::block_on(util::timeout(timeout, &mut future)).ok(),
Self::Deterministic(executor) => executor.block_on(&mut future), Self::Deterministic { executor, .. } => executor.block_on(&mut future),
}; };
if let Some(output) = output { if let Some(output) = output {
return Ok(*output.downcast().unwrap()); return Ok(*output.downcast().unwrap());
@ -587,7 +617,10 @@ pub fn deterministic(seed: u64) -> (Rc<Foreground>, Arc<Background>) {
let executor = Arc::new(Deterministic::new(seed)); let executor = Arc::new(Deterministic::new(seed));
( (
Rc::new(Foreground::Deterministic(executor.clone())), Rc::new(Foreground::Deterministic(executor.clone())),
Arc::new(Background::Deterministic(executor)), Arc::new(Background::Deterministic {
executor,
critical_tasks: Default::default(),
}),
) )
} }

View file

@ -60,6 +60,7 @@ pub trait Platform: Send + Sync {
pub(crate) trait ForegroundPlatform { pub(crate) trait ForegroundPlatform {
fn on_become_active(&self, callback: Box<dyn FnMut()>); fn on_become_active(&self, callback: Box<dyn FnMut()>);
fn on_resign_active(&self, callback: Box<dyn FnMut()>); fn on_resign_active(&self, callback: Box<dyn FnMut()>);
fn on_quit(&self, callback: Box<dyn FnMut()>);
fn on_event(&self, callback: Box<dyn FnMut(Event) -> bool>); fn on_event(&self, callback: Box<dyn FnMut(Event) -> bool>);
fn on_open_files(&self, callback: Box<dyn FnMut(Vec<PathBuf>)>); fn on_open_files(&self, callback: Box<dyn FnMut(Vec<PathBuf>)>);
fn run(&self, on_finish_launching: Box<dyn FnOnce() -> ()>); fn run(&self, on_finish_launching: Box<dyn FnOnce() -> ()>);

View file

@ -81,6 +81,10 @@ unsafe fn build_classes() {
sel!(applicationDidResignActive:), sel!(applicationDidResignActive:),
did_resign_active as extern "C" fn(&mut Object, Sel, id), did_resign_active as extern "C" fn(&mut Object, Sel, id),
); );
decl.add_method(
sel!(applicationWillTerminate:),
will_terminate as extern "C" fn(&mut Object, Sel, id),
);
decl.add_method( decl.add_method(
sel!(handleGPUIMenuItem:), sel!(handleGPUIMenuItem:),
handle_menu_item as extern "C" fn(&mut Object, Sel, id), handle_menu_item as extern "C" fn(&mut Object, Sel, id),
@ -100,6 +104,7 @@ pub struct MacForegroundPlatform(RefCell<MacForegroundPlatformState>);
pub struct MacForegroundPlatformState { pub struct MacForegroundPlatformState {
become_active: Option<Box<dyn FnMut()>>, become_active: Option<Box<dyn FnMut()>>,
resign_active: Option<Box<dyn FnMut()>>, resign_active: Option<Box<dyn FnMut()>>,
quit: Option<Box<dyn FnMut()>>,
event: Option<Box<dyn FnMut(crate::Event) -> bool>>, event: Option<Box<dyn FnMut(crate::Event) -> bool>>,
menu_command: Option<Box<dyn FnMut(&dyn AnyAction)>>, menu_command: Option<Box<dyn FnMut(&dyn AnyAction)>>,
open_files: Option<Box<dyn FnMut(Vec<PathBuf>)>>, open_files: Option<Box<dyn FnMut(Vec<PathBuf>)>>,
@ -196,6 +201,10 @@ impl platform::ForegroundPlatform for MacForegroundPlatform {
self.0.borrow_mut().resign_active = Some(callback); self.0.borrow_mut().resign_active = Some(callback);
} }
fn on_quit(&self, callback: Box<dyn FnMut()>) {
self.0.borrow_mut().quit = Some(callback);
}
fn on_event(&self, callback: Box<dyn FnMut(crate::Event) -> bool>) { fn on_event(&self, callback: Box<dyn FnMut(crate::Event) -> bool>) {
self.0.borrow_mut().event = Some(callback); self.0.borrow_mut().event = Some(callback);
} }
@ -664,6 +673,13 @@ extern "C" fn did_resign_active(this: &mut Object, _: Sel, _: id) {
} }
} }
extern "C" fn will_terminate(this: &mut Object, _: Sel, _: id) {
let platform = unsafe { get_foreground_platform(this) };
if let Some(callback) = platform.0.borrow_mut().quit.as_mut() {
callback();
}
}
extern "C" fn open_files(this: &mut Object, _: Sel, _: id, paths: id) { extern "C" fn open_files(this: &mut Object, _: Sel, _: id, paths: id) {
let paths = unsafe { let paths = unsafe {
(0..paths.count()) (0..paths.count())

View file

@ -58,6 +58,8 @@ impl super::ForegroundPlatform for ForegroundPlatform {
fn on_resign_active(&self, _: Box<dyn FnMut()>) {} fn on_resign_active(&self, _: Box<dyn FnMut()>) {}
fn on_quit(&self, _: Box<dyn FnMut()>) {}
fn on_event(&self, _: Box<dyn FnMut(crate::Event) -> bool>) {} fn on_event(&self, _: Box<dyn FnMut(crate::Event) -> bool>) {}
fn on_open_files(&self, _: Box<dyn FnMut(Vec<std::path::PathBuf>)>) {} fn on_open_files(&self, _: Box<dyn FnMut(Vec<std::path::PathBuf>)>) {}