script/dom/bindings/
utils.rs1use std::cell::RefCell;
8use std::thread::LocalKey;
9
10use js::conversions::ToJSValConvertible;
11use js::glue::{IsWrapper, JSPrincipalsCallbacks, UnwrapObjectDynamic, UnwrapObjectStatic};
12use js::jsapi::{
13 CallArgs, DOMCallbacks, HandleObject as RawHandleObject, JS_FreezeObject, JSContext, JSObject,
14};
15use js::realm::CurrentRealm;
16use js::rust::{HandleObject, MutableHandleValue, get_object_class, is_dom_class};
17use script_bindings::conversions::SafeToJSValConvertible;
18use script_bindings::interfaces::{DomHelpers, Interface};
19use script_bindings::settings_stack::StackEntry;
20
21use crate::DomTypes;
22use crate::dom::bindings::codegen::{InterfaceObjectMap, PrototypeList};
23use crate::dom::bindings::constructor::call_html_constructor;
24use crate::dom::bindings::conversions::DerivedFrom;
25use crate::dom::bindings::error::{Error, report_pending_exception, throw_dom_exception};
26use crate::dom::bindings::principals::PRINCIPALS_CALLBACKS;
27use crate::dom::bindings::proxyhandler::is_platform_object_same_origin;
28use crate::dom::bindings::reflector::{DomObject, DomObjectWrap, reflect_dom_object};
29use crate::dom::bindings::root::DomRoot;
30use crate::dom::bindings::settings_stack;
31use crate::dom::globalscope::GlobalScope;
32use crate::dom::windowproxy::WindowProxyHandler;
33use crate::realms::InRealm;
34use crate::script_runtime::{CanGc, JSContext as SafeJSContext};
35use crate::script_thread::ScriptThread;
36
37#[derive(JSTraceable, MallocSizeOf)]
38pub(crate) struct GlobalStaticData {
40 #[ignore_malloc_size_of = "WindowProxyHandler does not properly implement it anyway"]
41 pub(crate) windowproxy_handler: &'static WindowProxyHandler,
43}
44
45impl GlobalStaticData {
46 pub(crate) fn new() -> GlobalStaticData {
48 GlobalStaticData {
49 windowproxy_handler: WindowProxyHandler::proxy_handler(),
50 }
51 }
52}
53
54pub(crate) use script_bindings::utils::*;
55
56pub(crate) fn to_frozen_array<T: ToJSValConvertible>(
58 convertibles: &[T],
59 cx: SafeJSContext,
60 mut rval: MutableHandleValue,
61 can_gc: CanGc,
62) {
63 convertibles.safe_to_jsval(cx, rval.reborrow(), can_gc);
64
65 rooted!(in(*cx) let obj = rval.to_object());
66 unsafe { JS_FreezeObject(*cx, RawHandleObject::from(obj.handle())) };
67}
68
69#[expect(dead_code)]
72pub(crate) fn is_platform_object_dynamic(obj: *mut JSObject, cx: *mut JSContext) -> bool {
73 is_platform_object(obj, &|o| unsafe {
74 UnwrapObjectDynamic(o, cx, false)
75 })
76}
77
78pub(crate) fn is_platform_object_static(obj: *mut JSObject) -> bool {
81 is_platform_object(obj, &|o| unsafe { UnwrapObjectStatic(o) })
82}
83
84fn is_platform_object(
85 obj: *mut JSObject,
86 unwrap_obj: &dyn Fn(*mut JSObject) -> *mut JSObject,
87) -> bool {
88 unsafe {
89 let mut clasp = get_object_class(obj);
91 if is_dom_class(&*clasp) {
92 return true;
93 }
94 if IsWrapper(obj) {
96 let unwrapped_obj = unwrap_obj(obj);
97 if unwrapped_obj.is_null() {
98 return false;
99 }
100 clasp = get_object_class(obj);
101 }
102 is_dom_class(&*clasp)
104 }
105}
106
107unsafe extern "C" fn instance_class_has_proto_at_depth(
108 clasp: *const js::jsapi::JSClass,
109 proto_id: u32,
110 depth: u32,
111) -> bool {
112 let domclass: *const DOMJSClass = clasp as *const _;
113 let domclass = unsafe { &*domclass };
114 domclass.dom_class.interface_chain[depth as usize] as u32 == proto_id
115}
116
117unsafe extern "C" fn instance_class_is_error(clasp: *const js::jsapi::JSClass) -> bool {
119 if !is_dom_class(unsafe { &*clasp }) {
120 return false;
121 }
122 let domclass: *const DOMJSClass = clasp as *const _;
123 let domclass = unsafe { &*domclass };
124 let root_interface = domclass.dom_class.interface_chain[0] as u32;
125 root_interface == PrototypeList::ID::DOMException as u32
127}
128
129#[allow(missing_docs)] pub(crate) const DOM_CALLBACKS: DOMCallbacks = DOMCallbacks {
131 instanceClassMatchesProto: Some(instance_class_has_proto_at_depth),
132 instanceClassIsError: Some(instance_class_is_error),
133};
134
135pub(crate) fn define_all_exposed_interfaces(
138 global: &GlobalScope,
139 _in_realm: InRealm,
140 _can_gc: CanGc,
141) {
142 let cx = GlobalScope::get_cx();
143 for (_, interface) in &InterfaceObjectMap::MAP {
144 (interface.define)(cx, global.reflector().get_jsobject());
145 }
146}
147
148impl DomHelpers<crate::DomTypeHolder> for crate::DomTypeHolder {
149 fn throw_dom_exception(
150 cx: SafeJSContext,
151 global: &<crate::DomTypeHolder as DomTypes>::GlobalScope,
152 result: Error,
153 can_gc: CanGc,
154 ) {
155 throw_dom_exception(cx, global, result, can_gc)
156 }
157
158 fn call_html_constructor<
159 T: DerivedFrom<<crate::DomTypeHolder as DomTypes>::Element> + DomObject,
160 >(
161 cx: SafeJSContext,
162 args: &CallArgs,
163 global: &<crate::DomTypeHolder as DomTypes>::GlobalScope,
164 proto_id: PrototypeList::ID,
165 creator: unsafe fn(SafeJSContext, HandleObject, *mut ProtoOrIfaceArray),
166 can_gc: CanGc,
167 ) -> bool {
168 call_html_constructor::<T>(cx, args, global, proto_id, creator, can_gc)
169 }
170
171 fn settings_stack() -> &'static LocalKey<RefCell<Vec<StackEntry<crate::DomTypeHolder>>>> {
172 &settings_stack::STACK
173 }
174
175 fn principals_callbacks() -> &'static JSPrincipalsCallbacks {
176 &PRINCIPALS_CALLBACKS
177 }
178
179 fn is_platform_object_same_origin(cx: &CurrentRealm, obj: RawHandleObject) -> bool {
180 unsafe { is_platform_object_same_origin(cx, obj) }
181 }
182
183 fn interface_map() -> &'static phf::Map<&'static [u8], Interface> {
184 &InterfaceObjectMap::MAP
185 }
186
187 fn push_new_element_queue() {
188 ScriptThread::custom_element_reaction_stack().push_new_element_queue()
189 }
190 fn pop_current_element_queue(can_gc: CanGc) {
191 ScriptThread::custom_element_reaction_stack().pop_current_element_queue(can_gc)
192 }
193
194 fn reflect_dom_object<T, U>(obj: Box<T>, global: &U, can_gc: CanGc) -> DomRoot<T>
195 where
196 T: DomObject + DomObjectWrap<crate::DomTypeHolder>,
197 U: DerivedFrom<GlobalScope>,
198 {
199 reflect_dom_object(obj, global, can_gc)
200 }
201
202 fn report_pending_exception(
203 cx: SafeJSContext,
204 dispatch_event: bool,
205 realm: InRealm,
206 can_gc: CanGc,
207 ) {
208 report_pending_exception(cx, dispatch_event, realm, can_gc)
209 }
210}