/* 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/. */ //! Base classes to work with IDL callbacks. use crate::dom::bindings::codegen::Bindings::WindowBinding::WindowBinding::WindowMethods; use crate::dom::bindings::error::{report_pending_exception, Error, Fallible}; use crate::dom::bindings::inheritance::Castable; use crate::dom::bindings::reflector::DomObject; use crate::dom::bindings::root::{Dom, DomRoot}; use crate::dom::bindings::settings_stack::{AutoEntryScript, AutoIncumbentScript}; use crate::dom::bindings::utils::AsCCharPtrPtr; use crate::dom::globalscope::GlobalScope; use crate::dom::window::Window; use crate::realms::{enter_realm, InRealm}; use crate::script_runtime::JSContext; use js::jsapi::Heap; use js::jsapi::{AddRawValueRoot, IsCallable, JSObject}; use js::jsapi::{EnterRealm, LeaveRealm, Realm, RemoveRawValueRoot}; use js::jsval::{JSVal, ObjectValue, UndefinedValue}; use js::rust::wrappers::{JS_GetProperty, JS_WrapObject}; use js::rust::{MutableHandleObject, Runtime}; use std::default::Default; use std::ffi::CString; use std::mem::drop; use std::ops::Deref; use std::ptr; use std::rc::Rc; /// The exception handling used for a call. #[derive(Clone, Copy, PartialEq)] pub enum ExceptionHandling { /// Report any exception and don't throw it to the caller code. Report, /// Throw any exception to the caller code. Rethrow, } /// A common base class for representing IDL callback function and /// callback interface types. #[derive(JSTraceable)] #[unrooted_must_root_lint::must_root] pub struct CallbackObject { /// The underlying `JSObject`. callback: Heap<*mut JSObject>, permanent_js_root: Heap, /// The ["callback context"], that is, the global to use as incumbent /// global when calling the callback. /// /// Looking at the WebIDL standard, it appears as though there would always /// be a value here, but [sometimes] callback functions are created by /// hand-waving without defining the value of the callback context, and /// without any JavaScript code on the stack to grab an incumbent global /// from. /// /// ["callback context"]: https://heycam.github.io/webidl/#dfn-callback-context /// [sometimes]: https://github.com/whatwg/html/issues/2248 incumbent: Option>, } impl Default for CallbackObject { #[allow(unrooted_must_root)] fn default() -> CallbackObject { CallbackObject::new() } } impl CallbackObject { #[allow(unrooted_must_root)] fn new() -> CallbackObject { CallbackObject { callback: Heap::default(), permanent_js_root: Heap::default(), incumbent: GlobalScope::incumbent().map(|i| Dom::from_ref(&*i)), } } pub fn get(&self) -> *mut JSObject { self.callback.get() } #[allow(unsafe_code)] unsafe fn init(&mut self, cx: JSContext, callback: *mut JSObject) { self.callback.set(callback); self.permanent_js_root.set(ObjectValue(callback)); assert!(AddRawValueRoot( *cx, self.permanent_js_root.get_unsafe(), b"CallbackObject::root\n".as_c_char_ptr() )); } } impl Drop for CallbackObject { #[allow(unsafe_code)] fn drop(&mut self) { unsafe { let cx = Runtime::get(); RemoveRawValueRoot(cx, self.permanent_js_root.get_unsafe()); } } } impl PartialEq for CallbackObject { fn eq(&self, other: &CallbackObject) -> bool { self.callback.get() == other.callback.get() } } /// A trait to be implemented by concrete IDL callback function and /// callback interface types. pub trait CallbackContainer { /// Create a new CallbackContainer object for the given `JSObject`. unsafe fn new(cx: JSContext, callback: *mut JSObject) -> Rc; /// Returns the underlying `CallbackObject`. fn callback_holder(&self) -> &CallbackObject; /// Returns the underlying `JSObject`. fn callback(&self) -> *mut JSObject { self.callback_holder().get() } /// Returns the ["callback context"], that is, the global to use as /// incumbent global when calling the callback. /// /// ["callback context"]: https://heycam.github.io/webidl/#dfn-callback-context fn incumbent(&self) -> Option<&GlobalScope> { self.callback_holder().incumbent.as_ref().map(Dom::deref) } } /// A common base class for representing IDL callback function types. #[derive(JSTraceable, PartialEq)] #[unrooted_must_root_lint::must_root] pub struct CallbackFunction { object: CallbackObject, } impl CallbackFunction { /// Create a new `CallbackFunction` for this object. #[allow(unrooted_must_root)] pub fn new() -> CallbackFunction { CallbackFunction { object: CallbackObject::new(), } } /// Returns the underlying `CallbackObject`. pub fn callback_holder(&self) -> &CallbackObject { &self.object } /// Initialize the callback function with a value. /// Should be called once this object is done moving. pub unsafe fn init(&mut self, cx: JSContext, callback: *mut JSObject) { self.object.init(cx, callback); } } /// A common base class for representing IDL callback interface types. #[derive(JSTraceable, PartialEq)] #[unrooted_must_root_lint::must_root] pub struct CallbackInterface { object: CallbackObject, } impl CallbackInterface { /// Create a new CallbackInterface object for the given `JSObject`. pub fn new() -> CallbackInterface { CallbackInterface { object: CallbackObject::new(), } } /// Returns the underlying `CallbackObject`. pub fn callback_holder(&self) -> &CallbackObject { &self.object } /// Initialize the callback function with a value. /// Should be called once this object is done moving. pub unsafe fn init(&mut self, cx: JSContext, callback: *mut JSObject) { self.object.init(cx, callback); } /// Returns the property with the given `name`, if it is a callable object, /// or an error otherwise. pub fn get_callable_property(&self, cx: JSContext, name: &str) -> Fallible { rooted!(in(*cx) let mut callable = UndefinedValue()); rooted!(in(*cx) let obj = self.callback_holder().get()); unsafe { let c_name = CString::new(name).unwrap(); if !JS_GetProperty(*cx, obj.handle(), c_name.as_ptr(), callable.handle_mut()) { return Err(Error::JSFailed); } if !callable.is_object() || !IsCallable(callable.to_object()) { return Err(Error::Type(format!( "The value of the {} property is not callable", name ))); } } Ok(callable.get()) } } /// Wraps the reflector for `p` into the realm of `cx`. pub fn wrap_call_this_object(cx: JSContext, p: &T, mut rval: MutableHandleObject) { rval.set(p.reflector().get_jsobject().get()); assert!(!rval.get().is_null()); unsafe { if !JS_WrapObject(*cx, rval) { rval.set(ptr::null_mut()); } } } /// A class that performs whatever setup we need to safely make a call while /// this class is on the stack. After `new` returns, the call is safe to make. pub struct CallSetup { /// The global for reporting exceptions. This is the global object of the /// (possibly wrapped) callback object. exception_global: DomRoot, /// The `JSContext` used for the call. cx: JSContext, /// The realm we were in before the call. old_realm: *mut Realm, /// The exception handling used for the call. handling: ExceptionHandling, /// /// steps 8 and 18.2. entry_script: Option, /// /// steps 9 and 18.1. incumbent_script: Option, } impl CallSetup { /// Performs the setup needed to make a call. #[allow(unrooted_must_root)] pub fn new(callback: &T, handling: ExceptionHandling) -> CallSetup { let global = unsafe { GlobalScope::from_object(callback.callback()) }; if let Some(window) = global.downcast::() { window.Document().ensure_safe_to_run_script_or_layout(); } let cx = GlobalScope::get_cx(); let aes = AutoEntryScript::new(&global); let ais = callback.incumbent().map(AutoIncumbentScript::new); CallSetup { exception_global: global, cx: cx, old_realm: unsafe { EnterRealm(*cx, callback.callback()) }, handling: handling, entry_script: Some(aes), incumbent_script: ais, } } /// Returns the `JSContext` used for the call. pub fn get_context(&self) -> JSContext { self.cx } } impl Drop for CallSetup { fn drop(&mut self) { unsafe { LeaveRealm(*self.cx, self.old_realm); if self.handling == ExceptionHandling::Report { let ar = enter_realm(&*self.exception_global); report_pending_exception(*self.cx, true, InRealm::Entered(&ar)); } drop(self.incumbent_script.take()); drop(self.entry_script.take().unwrap()); } } }