1use std::cell::{Cell, RefCell};
6use std::rc::Rc;
7use std::sync::{Arc, Mutex};
8
9use dom_struct::dom_struct;
10use indexmap::IndexSet;
11use js::context::JSContext;
12use js::jsapi::{ExceptionStackBehavior, Heap};
13use js::jsval::{JSVal, UndefinedValue};
14use js::realm::CurrentRealm;
15use js::rust::wrappers2::JS_SetPendingException;
16use js::rust::{HandleObject, HandleValue, MutableHandleValue};
17use script_bindings::cell::DomRefCell;
18use script_bindings::inheritance::Castable;
19use script_bindings::reflector::reflect_dom_object_with_proto_and_cx;
20use script_bindings::weakref::WeakRef;
21
22use crate::dom::bindings::codegen::Bindings::AbortSignalBinding::AbortSignalMethods;
23use crate::dom::bindings::codegen::Bindings::EventListenerBinding::EventListener;
24use crate::dom::bindings::codegen::Bindings::EventTargetBinding::EventListenerOptions;
25use crate::dom::bindings::error::{Error, ErrorToJsval, Fallible};
26use crate::dom::bindings::refcounted::Trusted;
27use crate::dom::bindings::reflector::DomGlobal;
28use crate::dom::bindings::root::{Dom, DomRoot};
29use crate::dom::bindings::str::DOMString;
30use crate::dom::eventtarget::EventTarget;
31use crate::dom::globalscope::GlobalScope;
32use crate::dom::readablestream::PipeTo;
33use crate::fetch::{DeferredFetchRecordId, FetchContext};
34use crate::realms::enter_auto_realm;
35use crate::script_runtime::JSContext as SafeJSContext;
36
37impl js::gc::Rootable for AbortAlgorithm {}
38
39#[derive(Clone, JSTraceable, MallocSizeOf)]
43#[cfg_attr(crown, crown::unrooted_must_root_lint::must_root)]
44pub(crate) enum AbortAlgorithm {
45 DomEventListener(RemovableDomEventListener),
47 StreamPiping(PipeTo),
49 Fetch(
51 #[no_trace]
52 #[conditional_malloc_size_of]
53 Arc<Mutex<Option<FetchContext>>>,
54 ),
55 FetchLater(#[no_trace] DeferredFetchRecordId),
57}
58
59#[derive(Clone, JSTraceable, MallocSizeOf)]
60#[cfg_attr(crown, crown::unrooted_must_root_lint::must_root)]
61pub(crate) struct RemovableDomEventListener {
62 pub(crate) event_target: Dom<EventTarget>,
63 pub(crate) ty: DOMString,
64 #[conditional_malloc_size_of]
65 pub(crate) listener: Option<Rc<EventListener>>,
66 pub(crate) options: EventListenerOptions,
67}
68
69#[dom_struct]
71pub(crate) struct AbortSignal {
72 eventtarget: EventTarget,
73
74 #[ignore_malloc_size_of = "mozjs"]
76 abort_reason: Heap<JSVal>,
77
78 abort_algorithms: RefCell<Vec<AbortAlgorithm>>,
80
81 dependent: Cell<bool>,
83
84 #[no_trace]
86 #[ignore_malloc_size_of = "WeakRef"]
87 source_signals: DomRefCell<IndexSet<WeakRef<AbortSignal>>>,
88
89 #[no_trace]
91 #[ignore_malloc_size_of = "WeakRef"]
92 dependent_signals: DomRefCell<IndexSet<WeakRef<AbortSignal>>>,
93}
94
95impl AbortSignal {
96 fn new_inherited() -> AbortSignal {
97 AbortSignal {
98 eventtarget: EventTarget::new_inherited(),
99 abort_reason: Default::default(),
100 abort_algorithms: Default::default(),
101 dependent: Default::default(),
102 source_signals: Default::default(),
103 dependent_signals: Default::default(),
104 }
105 }
106
107 pub(crate) fn new_with_proto(
108 cx: &mut JSContext,
109 global: &GlobalScope,
110 proto: Option<HandleObject>,
111 ) -> DomRoot<AbortSignal> {
112 reflect_dom_object_with_proto_and_cx(
113 Box::new(AbortSignal::new_inherited()),
114 global,
115 proto,
116 cx,
117 )
118 }
119
120 pub(crate) fn signal_abort(&self, cx: &mut CurrentRealm, reason: HandleValue) {
122 let global = self.global();
123
124 if self.Aborted() {
126 return;
127 }
128
129 let abort_reason = reason.get();
132 if !abort_reason.is_undefined() {
133 self.abort_reason.set(abort_reason);
134 } else {
135 rooted!(&in(cx) let mut rooted_error = UndefinedValue());
136 Error::Abort(None).to_jsval(cx, &global, rooted_error.handle_mut());
137 self.abort_reason.set(rooted_error.get())
138 }
139
140 let mut dependent_signals_to_abort = vec![];
142
143 for weak in self.dependent_signals.borrow().iter() {
145 if let Some(dependent_signal) = weak.root() {
146 if !dependent_signal.aborted() {
148 dependent_signal.abort_reason.set(self.abort_reason.get());
150 dependent_signals_to_abort.push(dependent_signal);
152 }
153 }
154 }
155
156 self.run_the_abort_steps(cx, &global);
158
159 for dependent_signal in dependent_signals_to_abort.iter() {
161 dependent_signal.run_the_abort_steps(cx, &global);
162 }
163 }
164
165 pub(crate) fn add(&self, algorithm: &AbortAlgorithm) {
167 if self.aborted() {
169 return;
170 }
171
172 self.abort_algorithms.borrow_mut().push(algorithm.clone());
174
175 if self.dependent.get() {
177 self.global().register_dependent_abort_signal(self);
178 }
179 }
180
181 pub(crate) fn run_abort_algorithm(
183 &self,
184 cx: &mut CurrentRealm,
185 global: &GlobalScope,
186 algorithm: &AbortAlgorithm,
187 ) {
188 match algorithm {
189 AbortAlgorithm::StreamPiping(pipe) => {
190 rooted!(&in(cx) let mut reason = UndefinedValue());
191 reason.set(self.abort_reason.get());
192 pipe.abort_with_reason(cx, global, reason.handle());
193 },
194 AbortAlgorithm::Fetch(fetch_context) => {
195 rooted!(&in(cx) let mut reason = UndefinedValue());
196 reason.set(self.abort_reason.get());
197 if let Some(fetch_context) = &mut *fetch_context.lock().unwrap() {
198 fetch_context.abort_fetch(reason.handle(), cx);
199 }
200 },
201 AbortAlgorithm::FetchLater(deferred_fetch_record_id) => {
202 global
203 .deferred_fetch_record_for_id(deferred_fetch_record_id)
204 .abort();
205 },
206 AbortAlgorithm::DomEventListener(removable_listener) => {
207 removable_listener.event_target.remove_event_listener(
208 removable_listener.ty.clone(),
209 &removable_listener.listener,
210 &removable_listener.options,
211 );
212 },
213 }
214 }
215
216 fn run_the_abort_steps(&self, cx: &mut CurrentRealm, global: &GlobalScope) {
218 for algo in self.abort_algorithms.borrow().iter() {
220 self.run_abort_algorithm(cx, global, algo);
221 }
222 self.abort_algorithms.borrow_mut().clear();
224
225 self.upcast::<EventTarget>().fire_event(cx, atom!("abort"));
227 }
228
229 pub(crate) fn aborted(&self) -> bool {
231 !self.abort_reason.get().is_undefined()
233 }
234
235 pub(crate) fn create_dependent_abort_signal(
237 cx: &mut JSContext,
238 signals: Vec<DomRoot<AbortSignal>>,
239 global: &GlobalScope,
240 ) -> DomRoot<AbortSignal> {
241 let result_signal = Self::new_with_proto(cx, global, None);
243 for signal in signals.iter() {
246 if signal.aborted() {
247 result_signal.abort_reason.set(signal.abort_reason.get());
248 return result_signal;
249 }
250 }
251 result_signal.dependent.set(true);
253 for signal in signals.iter() {
255 if !signal.dependent.get() {
257 result_signal
259 .source_signals
260 .borrow_mut()
261 .insert(WeakRef::new(signal));
262 signal
264 .dependent_signals
265 .borrow_mut()
266 .insert(WeakRef::new(&*result_signal));
267 } else {
268 for source_signal_weak in signal.source_signals.borrow().iter() {
270 if let Some(source_signal) = source_signal_weak.root() {
271 assert!(!source_signal.aborted() && !source_signal.dependent.get());
273 result_signal
275 .source_signals
276 .borrow_mut()
277 .insert(WeakRef::new(&*source_signal));
278 source_signal
280 .dependent_signals
281 .borrow_mut()
282 .insert(WeakRef::new(&*result_signal));
283 }
284 }
285 }
286 }
287 global.register_dependent_abort_signal(&result_signal);
289 result_signal
290 }
291
292 fn prune_dead_weak_refs(&self) {
294 self.source_signals.borrow_mut().retain(|w| w.is_alive());
295 self.dependent_signals.borrow_mut().retain(|w| w.is_alive());
296 }
297
298 fn has_abort_algorithms(&self) -> bool {
300 !self.abort_algorithms.borrow().is_empty()
301 }
302
303 fn has_abort_listeners(&self) -> bool {
305 self.upcast::<EventTarget>()
306 .has_listeners_for(&atom!("abort"))
307 }
308
309 pub(crate) fn must_keep_alive_for_gc(&self) -> bool {
315 self.prune_dead_weak_refs();
317
318 if self.aborted() {
320 return false;
321 }
322
323 if self.source_signals.borrow().is_empty() {
325 return false;
326 }
327
328 let has_algos = self.has_abort_algorithms();
329 let has_listeners = self.has_abort_listeners();
330
331 has_algos || has_listeners
332 }
333}
334
335impl AbortSignalMethods<crate::DomTypeHolder> for AbortSignal {
336 fn Aborted(&self) -> bool {
338 self.aborted()
340 }
341
342 fn Abort(
344 cx: &mut JSContext,
345 global: &GlobalScope,
346 reason: HandleValue,
347 ) -> DomRoot<AbortSignal> {
348 let signal = AbortSignal::new_with_proto(cx, global, None);
350
351 let abort_reason = reason.get();
354 if !abort_reason.is_undefined() {
355 signal.abort_reason.set(abort_reason);
356 } else {
357 rooted!(&in(cx) let mut rooted_error = UndefinedValue());
358 Error::Abort(None).to_jsval(cx, global, rooted_error.handle_mut());
359 signal.abort_reason.set(rooted_error.get())
360 }
361
362 signal
364 }
365
366 fn Timeout(
368 cx: &mut JSContext,
369 global: &GlobalScope,
370 milliseconds: u64,
371 ) -> DomRoot<AbortSignal> {
372 let signal = AbortSignal::new_with_proto(cx, global, None);
374
375 let signal_keepalive: Trusted<AbortSignal> = Trusted::new(&signal);
379
380 let ms_i64 = if milliseconds > i64::MAX as u64 {
381 i64::MAX
382 } else {
383 milliseconds as i64
384 };
385
386 global.run_steps_after_a_timeout(
388 DOMString::from("AbortSignal-timeout"),
389 ms_i64,
390 move |_cx, global| {
391 let task_source = global.task_manager().timer_task_source().to_sendable();
392
393 task_source.queue(task!(abortsignal_timeout: move |cx| {
397 let signal_for_task = signal_keepalive.root();
398
399 rooted!(&in(cx) let mut reason = UndefinedValue());
400 Error::Timeout(None).to_jsval(
401 cx,
402 &signal_for_task.global(),
403 reason.handle_mut(),
404 );
405
406 let mut realm = enter_auto_realm(cx, &*signal_for_task.global());
407 let mut realm = realm.current_realm();
408
409 signal_for_task.signal_abort(
411 &mut realm,
412 reason.handle(),
413 );
414 }));
415 },
416 );
417
418 signal
420 }
421
422 fn Any(
424 cx: &mut JSContext,
425 global: &GlobalScope,
426 signals: Vec<DomRoot<AbortSignal>>,
427 ) -> DomRoot<AbortSignal> {
428 Self::create_dependent_abort_signal(cx, signals, global)
431 }
432
433 fn Reason(&self, _cx: SafeJSContext, mut rval: MutableHandleValue) {
435 rval.set(self.abort_reason.get());
437 }
438
439 #[expect(unsafe_code)]
441 fn ThrowIfAborted(&self, cx: &mut JSContext) -> Fallible<()> {
442 if self.aborted() {
444 unsafe {
445 JS_SetPendingException(
446 cx,
447 HandleValue::from_raw(self.abort_reason.handle()),
448 ExceptionStackBehavior::Capture,
449 )
450 };
451 return Err(Error::JSFailed);
452 }
453 Ok(())
454 }
455
456 event_handler!(abort, GetOnabort, SetOnabort);
458}