mirror of
https://github.com/servo/servo
synced 2026-04-26 01:25:32 +02:00
Follow up of #43526. This addresses Nico's comment: https://github.com/servo/servo/pull/43526#issuecomment-4104953308 - `bluetooth_traits` -> `servo_bluetooth_traits` - `base` -> `servo_base` - `bluetooth` -> `servo_bluetooth` - `background_hang_monitor` -> `servo_background_hang_monitor` Testing: This should not change any behaviour. --------- Signed-off-by: Euclid Ye <yezhizhenjiakang@gmail.com>
213 lines
9.1 KiB
Rust
213 lines
9.1 KiB
Rust
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
|
|
|
//! Implementation of [microtasks](https://html.spec.whatwg.org/multipage/#microtask) and
|
|
//! microtask queues. It is up to implementations of event loops to store a queue and
|
|
//! perform checkpoints at appropriate times, as well as enqueue microtasks as required.
|
|
|
|
use std::cell::Cell;
|
|
use std::mem;
|
|
use std::rc::Rc;
|
|
|
|
use js::jsapi::JobQueueMayNotBeEmpty;
|
|
use js::realm::AutoRealm;
|
|
use servo_base::id::PipelineId;
|
|
|
|
use crate::dom::bindings::callback::ExceptionHandling;
|
|
use crate::dom::bindings::cell::DomRefCell;
|
|
use crate::dom::bindings::codegen::Bindings::PromiseBinding::PromiseJobCallback;
|
|
use crate::dom::bindings::codegen::Bindings::VoidFunctionBinding::VoidFunction;
|
|
use crate::dom::bindings::root::DomRoot;
|
|
use crate::dom::globalscope::GlobalScope;
|
|
use crate::dom::html::htmlimageelement::ImageElementMicrotask;
|
|
use crate::dom::html::htmlmediaelement::MediaElementMicrotask;
|
|
use crate::dom::promise::WaitForAllSuccessStepsMicrotask;
|
|
use crate::dom::stream::byteteereadintorequest::ByteTeeReadIntoRequestMicrotask;
|
|
use crate::dom::stream::byteteereadrequest::ByteTeeReadRequestMicrotask;
|
|
use crate::dom::stream::defaultteereadrequest::DefaultTeeReadRequestMicrotask;
|
|
use crate::realms::enter_auto_realm;
|
|
use crate::script_runtime::{CanGc, JSContext, notify_about_rejected_promises};
|
|
use crate::script_thread::ScriptThread;
|
|
|
|
/// A collection of microtasks in FIFO order.
|
|
#[derive(Default, JSTraceable, MallocSizeOf)]
|
|
pub(crate) struct MicrotaskQueue {
|
|
/// The list of enqueued microtasks that will be invoked at the next microtask checkpoint.
|
|
microtask_queue: DomRefCell<Vec<Microtask>>,
|
|
/// <https://html.spec.whatwg.org/multipage/#performing-a-microtask-checkpoint>
|
|
performing_a_microtask_checkpoint: Cell<bool>,
|
|
}
|
|
|
|
#[derive(JSTraceable, MallocSizeOf)]
|
|
pub(crate) enum Microtask {
|
|
Promise(EnqueuedPromiseCallback),
|
|
User(UserMicrotask),
|
|
MediaElement(MediaElementMicrotask),
|
|
ImageElement(ImageElementMicrotask),
|
|
ReadableStreamTeeReadRequest(DefaultTeeReadRequestMicrotask),
|
|
WaitForAllSuccessSteps(WaitForAllSuccessStepsMicrotask),
|
|
ReadableStreamByteTeeReadRequest(ByteTeeReadRequestMicrotask),
|
|
ReadableStreamByteTeeReadIntoRequest(ByteTeeReadIntoRequestMicrotask),
|
|
CustomElementReaction,
|
|
NotifyMutationObservers,
|
|
}
|
|
|
|
pub(crate) trait MicrotaskRunnable {
|
|
fn handler(&self, _cx: &mut js::context::JSContext) {}
|
|
fn enter_realm<'cx>(&self, cx: &'cx mut js::context::JSContext) -> AutoRealm<'cx>;
|
|
}
|
|
|
|
/// A promise callback scheduled to run during the next microtask checkpoint (#4283).
|
|
#[derive(JSTraceable, MallocSizeOf)]
|
|
pub(crate) struct EnqueuedPromiseCallback {
|
|
#[conditional_malloc_size_of]
|
|
pub(crate) callback: Rc<PromiseJobCallback>,
|
|
#[no_trace]
|
|
pub(crate) pipeline: PipelineId,
|
|
pub(crate) is_user_interacting: bool,
|
|
}
|
|
|
|
/// A microtask that comes from a queueMicrotask() Javascript call,
|
|
/// identical to EnqueuedPromiseCallback once it's on the queue
|
|
#[derive(JSTraceable, MallocSizeOf)]
|
|
pub(crate) struct UserMicrotask {
|
|
#[conditional_malloc_size_of]
|
|
pub(crate) callback: Rc<VoidFunction>,
|
|
#[no_trace]
|
|
pub(crate) pipeline: PipelineId,
|
|
}
|
|
|
|
impl MicrotaskQueue {
|
|
/// Add a new microtask to this queue. It will be invoked as part of the next
|
|
/// microtask checkpoint.
|
|
#[expect(unsafe_code)]
|
|
pub(crate) fn enqueue(&self, job: Microtask, cx: JSContext) {
|
|
self.microtask_queue.borrow_mut().push(job);
|
|
unsafe { JobQueueMayNotBeEmpty(*cx) };
|
|
}
|
|
|
|
/// <https://html.spec.whatwg.org/multipage/#perform-a-microtask-checkpoint>
|
|
/// Perform a microtask checkpoint, executing all queued microtasks until the queue is empty.
|
|
#[expect(unsafe_code)]
|
|
pub(crate) fn checkpoint<F>(
|
|
&self,
|
|
cx: &mut js::context::JSContext,
|
|
target_provider: F,
|
|
globalscopes: Vec<DomRoot<GlobalScope>>,
|
|
) where
|
|
F: Fn(PipelineId) -> Option<DomRoot<GlobalScope>>,
|
|
{
|
|
// Step 1. If the event loop's performing a microtask checkpoint is true, then return.
|
|
if self.performing_a_microtask_checkpoint.get() {
|
|
return;
|
|
}
|
|
|
|
// Step 2. Set the event loop's performing a microtask checkpoint to true.
|
|
self.performing_a_microtask_checkpoint.set(true);
|
|
|
|
debug!("Now performing a microtask checkpoint");
|
|
|
|
// Step 3. While the event loop's microtask queue is not empty:
|
|
while !self.microtask_queue.borrow().is_empty() {
|
|
rooted_vec!(let mut pending_queue);
|
|
mem::swap(&mut *pending_queue, &mut *self.microtask_queue.borrow_mut());
|
|
|
|
for (idx, job) in pending_queue.iter().enumerate() {
|
|
if idx == pending_queue.len() - 1 && self.microtask_queue.borrow().is_empty() {
|
|
unsafe { js::rust::wrappers2::JobQueueIsEmpty(cx) };
|
|
}
|
|
|
|
match *job {
|
|
Microtask::Promise(ref job) => {
|
|
if let Some(target) = target_provider(job.pipeline) {
|
|
let _guard = ScriptThread::user_interacting_guard();
|
|
let mut realm = enter_auto_realm(cx, &*target);
|
|
let cx = &mut realm;
|
|
let _ = job.callback.Call_(
|
|
&*target,
|
|
ExceptionHandling::Report,
|
|
CanGc::from_cx(cx),
|
|
);
|
|
}
|
|
},
|
|
Microtask::User(ref job) => {
|
|
if let Some(target) = target_provider(job.pipeline) {
|
|
let mut realm = enter_auto_realm(cx, &*target);
|
|
let cx = &mut realm;
|
|
let _ = job.callback.Call_(
|
|
&*target,
|
|
ExceptionHandling::Report,
|
|
CanGc::from_cx(cx),
|
|
);
|
|
}
|
|
},
|
|
Microtask::MediaElement(ref task) => {
|
|
let mut realm = task.enter_realm(cx);
|
|
let cx = &mut realm;
|
|
task.handler(cx);
|
|
},
|
|
Microtask::ImageElement(ref task) => {
|
|
let mut realm = task.enter_realm(cx);
|
|
let cx = &mut realm;
|
|
task.handler(cx);
|
|
},
|
|
Microtask::ReadableStreamTeeReadRequest(ref task) => {
|
|
let mut realm = task.enter_realm(cx);
|
|
let cx = &mut realm;
|
|
task.handler(cx);
|
|
},
|
|
Microtask::WaitForAllSuccessSteps(ref task) => {
|
|
let mut realm = task.enter_realm(cx);
|
|
let cx = &mut realm;
|
|
task.handler(cx);
|
|
},
|
|
Microtask::CustomElementReaction => {
|
|
ScriptThread::invoke_backup_element_queue(cx);
|
|
},
|
|
Microtask::NotifyMutationObservers => {
|
|
ScriptThread::mutation_observers()
|
|
.notify_mutation_observers(CanGc::from_cx(cx));
|
|
},
|
|
Microtask::ReadableStreamByteTeeReadRequest(ref task) => {
|
|
task.microtask_chunk_steps(cx)
|
|
},
|
|
Microtask::ReadableStreamByteTeeReadIntoRequest(ref task) => {
|
|
task.microtask_chunk_steps(cx)
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
// Step 4. For each environment settings object settingsObject whose responsible
|
|
// event loop is this event loop, notify about rejected promises given
|
|
// settingsObject's global object.
|
|
for global in globalscopes.clone().into_iter() {
|
|
notify_about_rejected_promises(&global);
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#perform-a-microtask-checkpoint
|
|
// Step 5. Cleanup Indexed Database transactions.
|
|
// https://w3c.github.io/IndexedDB/#cleanup-indexed-database-transactions
|
|
// “These steps are invoked by [HTML]. They ensure that transactions created by a script call
|
|
// to transaction() are deactivated once the task that invoked the script has completed.”
|
|
for global in globalscopes.iter() {
|
|
let _ = global.get_indexeddb().cleanup_indexeddb_transactions();
|
|
}
|
|
|
|
// TODO: Step 6. Perform ClearKeptObjects().
|
|
|
|
// Step 7. Set the event loop's performing a microtask checkpoint to false.
|
|
self.performing_a_microtask_checkpoint.set(false);
|
|
// TODO: Step 8. Record timing info for microtask checkpoint.
|
|
}
|
|
|
|
pub(crate) fn empty(&self) -> bool {
|
|
self.microtask_queue.borrow().is_empty()
|
|
}
|
|
|
|
pub(crate) fn clear(&self) {
|
|
self.microtask_queue.borrow_mut().clear();
|
|
}
|
|
}
|