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::{JSContext, notify_about_rejected_promises};
use crate::script_thread::ScriptThread;
#[derive(Default, JSTraceable, MallocSizeOf)]
pub(crate) struct MicrotaskQueue {
microtask_queue: DomRefCell<Vec<Microtask>>,
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>;
}
#[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,
}
#[derive(JSTraceable, MallocSizeOf)]
pub(crate) struct UserMicrotask {
#[conditional_malloc_size_of]
pub(crate) callback: Rc<VoidFunction>,
#[no_trace]
pub(crate) pipeline: PipelineId,
}
impl MicrotaskQueue {
#[expect(unsafe_code)]
pub(crate) fn enqueue(&self, job: Microtask, cx: JSContext) {
self.microtask_queue.borrow_mut().push(job);
unsafe { JobQueueMayNotBeEmpty(*cx) };
}
#[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>>,
{
if self.performing_a_microtask_checkpoint.get() {
return;
}
self.performing_a_microtask_checkpoint.set(true);
debug!("Now performing a microtask checkpoint");
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_(cx, &*target, ExceptionHandling::Report);
}
},
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_(cx, &*target, ExceptionHandling::Report);
}
},
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(cx);
},
Microtask::ReadableStreamByteTeeReadRequest(ref task) => {
task.microtask_chunk_steps(cx)
},
Microtask::ReadableStreamByteTeeReadIntoRequest(ref task) => {
task.microtask_chunk_steps(cx)
},
}
}
}
for global in globalscopes.clone().into_iter() {
notify_about_rejected_promises(&global);
}
for global in globalscopes.iter() {
let _ = global.get_indexeddb().cleanup_indexeddb_transactions();
}
self.performing_a_microtask_checkpoint.set(false);
}
pub(crate) fn empty(&self) -> bool {
self.microtask_queue.borrow().is_empty()
}
pub(crate) fn clear(&self) {
self.microtask_queue.borrow_mut().clear();
}
}