mirror of
https://github.com/servo/servo.git
synced 2025-07-23 15:23:42 +01:00
Auto merge of #13183 - nox:cleanup-interface, r=Ms2ger
Clean up stuff in bindings::interface <!-- Reviewable:start --> This change is [<img src="https://reviewable.io/review_button.svg" height="34" align="absmiddle" alt="Reviewable"/>](https://reviewable.io/reviews/servo/servo/13183) <!-- Reviewable:end -->
This commit is contained in:
commit
628c644fee
4 changed files with 215 additions and 214 deletions
|
@ -1136,20 +1136,20 @@ def instantiateJSToNativeConversionTemplate(templateBody, replacements,
|
||||||
|
|
||||||
def convertConstIDLValueToJSVal(value):
|
def convertConstIDLValueToJSVal(value):
|
||||||
if isinstance(value, IDLNullValue):
|
if isinstance(value, IDLNullValue):
|
||||||
return "NullVal"
|
return "ConstantVal::NullVal"
|
||||||
tag = value.type.tag()
|
tag = value.type.tag()
|
||||||
if tag in [IDLType.Tags.int8, IDLType.Tags.uint8, IDLType.Tags.int16,
|
if tag in [IDLType.Tags.int8, IDLType.Tags.uint8, IDLType.Tags.int16,
|
||||||
IDLType.Tags.uint16, IDLType.Tags.int32]:
|
IDLType.Tags.uint16, IDLType.Tags.int32]:
|
||||||
return "IntVal(%s)" % (value.value)
|
return "ConstantVal::IntVal(%s)" % (value.value)
|
||||||
if tag == IDLType.Tags.uint32:
|
if tag == IDLType.Tags.uint32:
|
||||||
return "UintVal(%s)" % (value.value)
|
return "ConstantVal::UintVal(%s)" % (value.value)
|
||||||
if tag in [IDLType.Tags.int64, IDLType.Tags.uint64]:
|
if tag in [IDLType.Tags.int64, IDLType.Tags.uint64]:
|
||||||
return "DoubleVal(%s)" % (value.value)
|
return "ConstantVal::DoubleVal(%s)" % (value.value)
|
||||||
if tag == IDLType.Tags.bool:
|
if tag == IDLType.Tags.bool:
|
||||||
return "BoolVal(true)" if value.value else "BoolVal(false)"
|
return "ConstantVal::BoolVal(true)" if value.value else "ConstantVal::BoolVal(false)"
|
||||||
if tag in [IDLType.Tags.unrestricted_float, IDLType.Tags.float,
|
if tag in [IDLType.Tags.unrestricted_float, IDLType.Tags.float,
|
||||||
IDLType.Tags.unrestricted_double, IDLType.Tags.double]:
|
IDLType.Tags.unrestricted_double, IDLType.Tags.double]:
|
||||||
return "DoubleVal(%s)" % (value.value)
|
return "ConstantVal::DoubleVal(%s)" % (value.value)
|
||||||
raise TypeError("Const value of unhandled type: " + value.type)
|
raise TypeError("Const value of unhandled type: " + value.type)
|
||||||
|
|
||||||
|
|
||||||
|
@ -2072,12 +2072,9 @@ class CGInterfaceObjectJSClass(CGThing):
|
||||||
"depth": self.descriptor.prototypeDepth
|
"depth": self.descriptor.prototypeDepth
|
||||||
}
|
}
|
||||||
return """\
|
return """\
|
||||||
static INTERFACE_OBJECT_OPS: js::jsapi::ClassOps =
|
static INTERFACE_OBJECT_CLASS: NonCallbackInterfaceObjectClass =
|
||||||
NonCallbackInterfaceObjectClass::ops(%(constructorBehavior)s);
|
|
||||||
|
|
||||||
static InterfaceObjectClass: NonCallbackInterfaceObjectClass =
|
|
||||||
NonCallbackInterfaceObjectClass::new(
|
NonCallbackInterfaceObjectClass::new(
|
||||||
&INTERFACE_OBJECT_OPS,
|
&%(constructorBehavior)s,
|
||||||
%(representation)s,
|
%(representation)s,
|
||||||
PrototypeList::ID::%(id)s,
|
PrototypeList::ID::%(id)s,
|
||||||
%(depth)s);
|
%(depth)s);
|
||||||
|
@ -2751,7 +2748,7 @@ rooted!(in(cx) let mut interface = ptr::null_mut());
|
||||||
create_noncallback_interface_object(cx,
|
create_noncallback_interface_object(cx,
|
||||||
global,
|
global,
|
||||||
interface_proto.handle(),
|
interface_proto.handle(),
|
||||||
&InterfaceObjectClass,
|
&INTERFACE_OBJECT_CLASS,
|
||||||
%(static_methods)s,
|
%(static_methods)s,
|
||||||
%(static_attrs)s,
|
%(static_attrs)s,
|
||||||
%(consts)s,
|
%(consts)s,
|
||||||
|
@ -2820,13 +2817,13 @@ assert!((*cache)[PrototypeList::Constructor::%(id)s as usize].is_null());
|
||||||
|
|
||||||
constructors = self.descriptor.interface.namedConstructors
|
constructors = self.descriptor.interface.namedConstructors
|
||||||
if constructors:
|
if constructors:
|
||||||
decl = "let named_constructors: [(NonNullJSNative, &'static [u8], u32); %d]" % len(constructors)
|
decl = "let named_constructors: [(ConstructorClassHook, &'static [u8], u32); %d]" % len(constructors)
|
||||||
specs = []
|
specs = []
|
||||||
for constructor in constructors:
|
for constructor in constructors:
|
||||||
hook = CONSTRUCT_HOOK_NAME + "_" + constructor.identifier.name
|
hook = CONSTRUCT_HOOK_NAME + "_" + constructor.identifier.name
|
||||||
name = str_to_const_array(constructor.identifier.name)
|
name = str_to_const_array(constructor.identifier.name)
|
||||||
length = methodLength(constructor)
|
length = methodLength(constructor)
|
||||||
specs.append(CGGeneric("(%s as NonNullJSNative, %s, %d)" % (hook, name, length)))
|
specs.append(CGGeneric("(%s as ConstructorClassHook, %s, %d)" % (hook, name, length)))
|
||||||
values = CGIndenter(CGList(specs, "\n"), 4)
|
values = CGIndenter(CGList(specs, "\n"), 4)
|
||||||
code.append(CGWrapper(values, pre="%s = [\n" % decl, post="\n];"))
|
code.append(CGWrapper(values, pre="%s = [\n" % decl, post="\n];"))
|
||||||
code.append(CGGeneric("create_named_constructors(cx, global, &named_constructors, prototype.handle());"))
|
code.append(CGGeneric("create_named_constructors(cx, global, &named_constructors, prototype.handle());"))
|
||||||
|
@ -5432,15 +5429,14 @@ def generate_imports(config, cgthings, descriptors, callbacks=None, dictionaries
|
||||||
'dom',
|
'dom',
|
||||||
'dom::bindings',
|
'dom::bindings',
|
||||||
'dom::bindings::codegen::InterfaceObjectMap',
|
'dom::bindings::codegen::InterfaceObjectMap',
|
||||||
|
'dom::bindings::constant::ConstantSpec',
|
||||||
|
'dom::bindings::constant::ConstantVal',
|
||||||
'dom::bindings::global::GlobalRef',
|
'dom::bindings::global::GlobalRef',
|
||||||
'dom::bindings::global::global_root_from_object',
|
'dom::bindings::global::global_root_from_object',
|
||||||
'dom::bindings::global::global_root_from_reflector',
|
'dom::bindings::global::global_root_from_reflector',
|
||||||
'dom::bindings::interface::ConstantSpec',
|
'dom::bindings::interface::ConstructorClassHook',
|
||||||
'dom::bindings::interface::ConstantVal::IntVal',
|
|
||||||
'dom::bindings::interface::ConstantVal::UintVal',
|
|
||||||
'dom::bindings::interface::InterfaceConstructorBehavior',
|
'dom::bindings::interface::InterfaceConstructorBehavior',
|
||||||
'dom::bindings::interface::NonCallbackInterfaceObjectClass',
|
'dom::bindings::interface::NonCallbackInterfaceObjectClass',
|
||||||
'dom::bindings::interface::NonNullJSNative',
|
|
||||||
'dom::bindings::interface::create_callback_interface_object',
|
'dom::bindings::interface::create_callback_interface_object',
|
||||||
'dom::bindings::interface::create_global_object',
|
'dom::bindings::interface::create_global_object',
|
||||||
'dom::bindings::interface::create_interface_prototype_object',
|
'dom::bindings::interface::create_interface_prototype_object',
|
||||||
|
|
65
components/script/dom/bindings/constant.rs
Normal file
65
components/script/dom/bindings/constant.rs
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
/* 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 http://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
|
//! WebIDL constants.
|
||||||
|
|
||||||
|
use js::jsapi::{HandleObject, JSContext, JSPROP_ENUMERATE, JSPROP_PERMANENT};
|
||||||
|
use js::jsapi::{JSPROP_READONLY, JS_DefineProperty};
|
||||||
|
use js::jsval::{BooleanValue, DoubleValue, Int32Value, JSVal, NullValue, UInt32Value};
|
||||||
|
use libc;
|
||||||
|
|
||||||
|
/// Representation of an IDL constant.
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct ConstantSpec {
|
||||||
|
/// name of the constant.
|
||||||
|
pub name: &'static [u8],
|
||||||
|
/// value of the constant.
|
||||||
|
pub value: ConstantVal,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Representation of an IDL constant value.
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub enum ConstantVal {
|
||||||
|
/// `long` constant.
|
||||||
|
IntVal(i32),
|
||||||
|
/// `unsigned long` constant.
|
||||||
|
UintVal(u32),
|
||||||
|
/// `double` constant.
|
||||||
|
DoubleVal(f64),
|
||||||
|
/// `boolean` constant.
|
||||||
|
BoolVal(bool),
|
||||||
|
/// `null` constant.
|
||||||
|
NullVal,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ConstantSpec {
|
||||||
|
/// Returns a `JSVal` that represents the value of this `ConstantSpec`.
|
||||||
|
pub fn get_value(&self) -> JSVal {
|
||||||
|
match self.value {
|
||||||
|
ConstantVal::NullVal => NullValue(),
|
||||||
|
ConstantVal::IntVal(i) => Int32Value(i),
|
||||||
|
ConstantVal::UintVal(u) => UInt32Value(u),
|
||||||
|
ConstantVal::DoubleVal(d) => DoubleValue(d),
|
||||||
|
ConstantVal::BoolVal(b) => BooleanValue(b),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Defines constants on `obj`.
|
||||||
|
/// Fails on JSAPI failure.
|
||||||
|
pub unsafe fn define_constants(
|
||||||
|
cx: *mut JSContext,
|
||||||
|
obj: HandleObject,
|
||||||
|
constants: &[ConstantSpec]) {
|
||||||
|
for spec in constants {
|
||||||
|
rooted!(in(cx) let value = spec.get_value());
|
||||||
|
assert!(JS_DefineProperty(cx,
|
||||||
|
obj,
|
||||||
|
spec.name.as_ptr() as *const libc::c_char,
|
||||||
|
value.handle(),
|
||||||
|
JSPROP_ENUMERATE | JSPROP_READONLY | JSPROP_PERMANENT,
|
||||||
|
None,
|
||||||
|
None));
|
||||||
|
}
|
||||||
|
}
|
|
@ -6,6 +6,7 @@
|
||||||
|
|
||||||
use dom::bindings::codegen::InterfaceObjectMap::Globals;
|
use dom::bindings::codegen::InterfaceObjectMap::Globals;
|
||||||
use dom::bindings::codegen::PrototypeList;
|
use dom::bindings::codegen::PrototypeList;
|
||||||
|
use dom::bindings::constant::{ConstantSpec, define_constants};
|
||||||
use dom::bindings::conversions::{DOM_OBJECT_SLOT, get_dom_class};
|
use dom::bindings::conversions::{DOM_OBJECT_SLOT, get_dom_class};
|
||||||
use dom::bindings::guard::Guard;
|
use dom::bindings::guard::Guard;
|
||||||
use dom::bindings::utils::{DOM_PROTOTYPE_SLOT, ProtoOrIfaceArray, get_proto_or_iface_array};
|
use dom::bindings::utils::{DOM_PROTOTYPE_SLOT, ProtoOrIfaceArray, get_proto_or_iface_array};
|
||||||
|
@ -13,8 +14,8 @@ use js::error::throw_type_error;
|
||||||
use js::glue::{RUST_SYMBOL_TO_JSID, UncheckedUnwrapObject};
|
use js::glue::{RUST_SYMBOL_TO_JSID, UncheckedUnwrapObject};
|
||||||
use js::jsapi::{Class, ClassOps, CompartmentOptions, GetGlobalForObjectCrossCompartment};
|
use js::jsapi::{Class, ClassOps, CompartmentOptions, GetGlobalForObjectCrossCompartment};
|
||||||
use js::jsapi::{GetWellKnownSymbol, HandleObject, HandleValue, JSAutoCompartment};
|
use js::jsapi::{GetWellKnownSymbol, HandleObject, HandleValue, JSAutoCompartment};
|
||||||
use js::jsapi::{JSClass, JSContext, JSFUN_CONSTRUCTOR, JSFunctionSpec, JSNative, JSObject};
|
use js::jsapi::{JSClass, JSContext, JSFUN_CONSTRUCTOR, JSFunctionSpec, JSObject};
|
||||||
use js::jsapi::{JSPROP_ENUMERATE, JSPROP_PERMANENT, JSPROP_READONLY, JSPROP_RESOLVING};
|
use js::jsapi::{JSPROP_PERMANENT, JSPROP_READONLY, JSPROP_RESOLVING};
|
||||||
use js::jsapi::{JSPropertySpec, JSString, JSTracer, JSVersion, JS_AtomizeAndPinString};
|
use js::jsapi::{JSPropertySpec, JSString, JSTracer, JSVersion, JS_AtomizeAndPinString};
|
||||||
use js::jsapi::{JS_DefineProperty, JS_DefineProperty1, JS_DefineProperty2};
|
use js::jsapi::{JS_DefineProperty, JS_DefineProperty1, JS_DefineProperty2};
|
||||||
use js::jsapi::{JS_DefineProperty4, JS_DefinePropertyById3, JS_FireOnNewGlobalObject};
|
use js::jsapi::{JS_DefineProperty4, JS_DefinePropertyById3, JS_FireOnNewGlobalObject};
|
||||||
|
@ -24,99 +25,11 @@ use js::jsapi::{JS_NewObject, JS_NewObjectWithUniqueType, JS_NewPlainObject};
|
||||||
use js::jsapi::{JS_NewStringCopyN, JS_SetReservedSlot, MutableHandleObject};
|
use js::jsapi::{JS_NewStringCopyN, JS_SetReservedSlot, MutableHandleObject};
|
||||||
use js::jsapi::{MutableHandleValue, ObjectOps, OnNewGlobalHookOption, SymbolCode};
|
use js::jsapi::{MutableHandleValue, ObjectOps, OnNewGlobalHookOption, SymbolCode};
|
||||||
use js::jsapi::{TrueHandleValue, Value};
|
use js::jsapi::{TrueHandleValue, Value};
|
||||||
use js::jsval::{BooleanValue, DoubleValue, Int32Value, JSVal, NullValue};
|
use js::jsval::{JSVal, PrivateValue};
|
||||||
use js::jsval::{PrivateValue, UInt32Value};
|
|
||||||
use js::rust::{define_methods, define_properties};
|
use js::rust::{define_methods, define_properties};
|
||||||
use libc;
|
use libc;
|
||||||
use std::ptr;
|
use std::ptr;
|
||||||
|
|
||||||
/// Representation of an IDL constant value.
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub enum ConstantVal {
|
|
||||||
/// `long` constant.
|
|
||||||
IntVal(i32),
|
|
||||||
/// `unsigned long` constant.
|
|
||||||
UintVal(u32),
|
|
||||||
/// `double` constant.
|
|
||||||
DoubleVal(f64),
|
|
||||||
/// `boolean` constant.
|
|
||||||
BoolVal(bool),
|
|
||||||
/// `null` constant.
|
|
||||||
NullVal,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Representation of an IDL constant.
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct ConstantSpec {
|
|
||||||
/// name of the constant.
|
|
||||||
pub name: &'static [u8],
|
|
||||||
/// value of the constant.
|
|
||||||
pub value: ConstantVal,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ConstantSpec {
|
|
||||||
/// Returns a `JSVal` that represents the value of this `ConstantSpec`.
|
|
||||||
pub fn get_value(&self) -> JSVal {
|
|
||||||
match self.value {
|
|
||||||
ConstantVal::NullVal => NullValue(),
|
|
||||||
ConstantVal::IntVal(i) => Int32Value(i),
|
|
||||||
ConstantVal::UintVal(u) => UInt32Value(u),
|
|
||||||
ConstantVal::DoubleVal(d) => DoubleValue(d),
|
|
||||||
ConstantVal::BoolVal(b) => BooleanValue(b),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// A JSNative that cannot be null.
|
|
||||||
pub type NonNullJSNative =
|
|
||||||
unsafe extern "C" fn (arg1: *mut JSContext, arg2: libc::c_uint, arg3: *mut JSVal) -> bool;
|
|
||||||
|
|
||||||
/// Defines constants on `obj`.
|
|
||||||
/// Fails on JSAPI failure.
|
|
||||||
unsafe fn define_constants(
|
|
||||||
cx: *mut JSContext,
|
|
||||||
obj: HandleObject,
|
|
||||||
constants: &[ConstantSpec]) {
|
|
||||||
for spec in constants {
|
|
||||||
rooted!(in(cx) let value = spec.get_value());
|
|
||||||
assert!(JS_DefineProperty(cx,
|
|
||||||
obj,
|
|
||||||
spec.name.as_ptr() as *const libc::c_char,
|
|
||||||
value.handle(),
|
|
||||||
JSPROP_ENUMERATE | JSPROP_READONLY | JSPROP_PERMANENT,
|
|
||||||
None,
|
|
||||||
None));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
unsafe extern "C" fn fun_to_string_hook(cx: *mut JSContext,
|
|
||||||
obj: HandleObject,
|
|
||||||
_indent: u32)
|
|
||||||
-> *mut JSString {
|
|
||||||
let js_class = JS_GetClass(obj.get());
|
|
||||||
assert!(!js_class.is_null());
|
|
||||||
let repr = (*(js_class as *const NonCallbackInterfaceObjectClass)).representation;
|
|
||||||
assert!(!repr.is_empty());
|
|
||||||
let ret = JS_NewStringCopyN(cx, repr.as_ptr() as *const libc::c_char, repr.len());
|
|
||||||
assert!(!ret.is_null());
|
|
||||||
ret
|
|
||||||
}
|
|
||||||
|
|
||||||
const OBJECT_OPS: ObjectOps = ObjectOps {
|
|
||||||
lookupProperty: None,
|
|
||||||
defineProperty: None,
|
|
||||||
hasProperty: None,
|
|
||||||
getProperty: None,
|
|
||||||
setProperty: None,
|
|
||||||
getOwnPropertyDescriptor: None,
|
|
||||||
deleteProperty: None,
|
|
||||||
watch: None,
|
|
||||||
unwatch: None,
|
|
||||||
getElements: None,
|
|
||||||
enumerate: None,
|
|
||||||
funToString: Some(fun_to_string_hook),
|
|
||||||
};
|
|
||||||
|
|
||||||
/// The class of a non-callback interface object.
|
/// The class of a non-callback interface object.
|
||||||
#[derive(Copy, Clone)]
|
#[derive(Copy, Clone)]
|
||||||
pub struct NonCallbackInterfaceObjectClass {
|
pub struct NonCallbackInterfaceObjectClass {
|
||||||
|
@ -133,27 +46,8 @@ pub struct NonCallbackInterfaceObjectClass {
|
||||||
unsafe impl Sync for NonCallbackInterfaceObjectClass {}
|
unsafe impl Sync for NonCallbackInterfaceObjectClass {}
|
||||||
|
|
||||||
impl NonCallbackInterfaceObjectClass {
|
impl NonCallbackInterfaceObjectClass {
|
||||||
/// Create `ClassOps` for a `NonCallbackInterfaceObjectClass`.
|
|
||||||
pub const fn ops(constructor_behavior: InterfaceConstructorBehavior)
|
|
||||||
-> ClassOps {
|
|
||||||
ClassOps {
|
|
||||||
addProperty: None,
|
|
||||||
delProperty: None,
|
|
||||||
getProperty: None,
|
|
||||||
setProperty: None,
|
|
||||||
enumerate: None,
|
|
||||||
resolve: None,
|
|
||||||
mayResolve: None,
|
|
||||||
finalize: None,
|
|
||||||
call: constructor_behavior.call,
|
|
||||||
construct: constructor_behavior.construct,
|
|
||||||
hasInstance: Some(has_instance_hook),
|
|
||||||
trace: None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Create a new `NonCallbackInterfaceObjectClass` structure.
|
/// Create a new `NonCallbackInterfaceObjectClass` structure.
|
||||||
pub const fn new(ops: &'static ClassOps,
|
pub const fn new(constructor_behavior: &'static InterfaceConstructorBehavior,
|
||||||
string_rep: &'static [u8],
|
string_rep: &'static [u8],
|
||||||
proto_id: PrototypeList::ID,
|
proto_id: PrototypeList::ID,
|
||||||
proto_depth: u16)
|
proto_depth: u16)
|
||||||
|
@ -162,7 +56,7 @@ impl NonCallbackInterfaceObjectClass {
|
||||||
class: Class {
|
class: Class {
|
||||||
name: b"Function\0" as *const _ as *const libc::c_char,
|
name: b"Function\0" as *const _ as *const libc::c_char,
|
||||||
flags: 0,
|
flags: 0,
|
||||||
cOps: ops,
|
cOps: &constructor_behavior.0,
|
||||||
spec: ptr::null(),
|
spec: ptr::null(),
|
||||||
ext: ptr::null(),
|
ext: ptr::null(),
|
||||||
oOps: &OBJECT_OPS,
|
oOps: &OBJECT_OPS,
|
||||||
|
@ -186,26 +80,43 @@ pub type ConstructorClassHook =
|
||||||
unsafe extern "C" fn(cx: *mut JSContext, argc: u32, vp: *mut Value) -> bool;
|
unsafe extern "C" fn(cx: *mut JSContext, argc: u32, vp: *mut Value) -> bool;
|
||||||
|
|
||||||
/// The constructor behavior of a non-callback interface object.
|
/// The constructor behavior of a non-callback interface object.
|
||||||
pub struct InterfaceConstructorBehavior {
|
pub struct InterfaceConstructorBehavior(ClassOps);
|
||||||
call: JSNative,
|
|
||||||
construct: JSNative,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl InterfaceConstructorBehavior {
|
impl InterfaceConstructorBehavior {
|
||||||
/// An interface constructor that unconditionally throws a type error.
|
/// An interface constructor that unconditionally throws a type error.
|
||||||
pub const fn throw() -> InterfaceConstructorBehavior {
|
pub const fn throw() -> Self {
|
||||||
InterfaceConstructorBehavior {
|
InterfaceConstructorBehavior(ClassOps {
|
||||||
|
addProperty: None,
|
||||||
|
delProperty: None,
|
||||||
|
getProperty: None,
|
||||||
|
setProperty: None,
|
||||||
|
enumerate: None,
|
||||||
|
resolve: None,
|
||||||
|
mayResolve: None,
|
||||||
|
finalize: None,
|
||||||
call: Some(invalid_constructor),
|
call: Some(invalid_constructor),
|
||||||
construct: Some(invalid_constructor),
|
construct: Some(invalid_constructor),
|
||||||
}
|
hasInstance: Some(has_instance_hook),
|
||||||
|
trace: None,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/// An interface constructor that calls a native Rust function.
|
/// An interface constructor that calls a native Rust function.
|
||||||
pub const fn call(hook: ConstructorClassHook) -> InterfaceConstructorBehavior {
|
pub const fn call(hook: ConstructorClassHook) -> Self {
|
||||||
InterfaceConstructorBehavior {
|
InterfaceConstructorBehavior(ClassOps {
|
||||||
|
addProperty: None,
|
||||||
|
delProperty: None,
|
||||||
|
getProperty: None,
|
||||||
|
setProperty: None,
|
||||||
|
enumerate: None,
|
||||||
|
resolve: None,
|
||||||
|
mayResolve: None,
|
||||||
|
finalize: None,
|
||||||
call: Some(non_new_constructor),
|
call: Some(non_new_constructor),
|
||||||
construct: Some(hook),
|
construct: Some(hook),
|
||||||
}
|
hasInstance: Some(has_instance_hook),
|
||||||
|
trace: None,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -318,7 +229,7 @@ pub unsafe fn create_noncallback_interface_object(
|
||||||
pub unsafe fn create_named_constructors(
|
pub unsafe fn create_named_constructors(
|
||||||
cx: *mut JSContext,
|
cx: *mut JSContext,
|
||||||
global: HandleObject,
|
global: HandleObject,
|
||||||
named_constructors: &[(NonNullJSNative, &[u8], u32)],
|
named_constructors: &[(ConstructorClassHook, &[u8], u32)],
|
||||||
interface_prototype_object: HandleObject) {
|
interface_prototype_object: HandleObject) {
|
||||||
rooted!(in(cx) let mut constructor = ptr::null_mut());
|
rooted!(in(cx) let mut constructor = ptr::null_mut());
|
||||||
|
|
||||||
|
@ -346,6 +257,107 @@ pub unsafe fn create_named_constructors(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
unsafe fn create_object(
|
||||||
|
cx: *mut JSContext,
|
||||||
|
proto: HandleObject,
|
||||||
|
class: &'static JSClass,
|
||||||
|
methods: &[Guard<&'static [JSFunctionSpec]>],
|
||||||
|
properties: &[Guard<&'static [JSPropertySpec]>],
|
||||||
|
constants: &[Guard<&[ConstantSpec]>],
|
||||||
|
rval: MutableHandleObject) {
|
||||||
|
rval.set(JS_NewObjectWithUniqueType(cx, class, proto));
|
||||||
|
assert!(!rval.ptr.is_null());
|
||||||
|
define_guarded_methods(cx, rval.handle(), methods);
|
||||||
|
define_guarded_properties(cx, rval.handle(), properties);
|
||||||
|
define_guarded_constants(cx, rval.handle(), constants);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Conditionally define constants on an object.
|
||||||
|
pub unsafe fn define_guarded_constants(
|
||||||
|
cx: *mut JSContext,
|
||||||
|
obj: HandleObject,
|
||||||
|
constants: &[Guard<&[ConstantSpec]>]) {
|
||||||
|
for guard in constants {
|
||||||
|
if let Some(specs) = guard.expose(cx, obj) {
|
||||||
|
define_constants(cx, obj, specs);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Conditionally define methods on an object.
|
||||||
|
pub unsafe fn define_guarded_methods(
|
||||||
|
cx: *mut JSContext,
|
||||||
|
obj: HandleObject,
|
||||||
|
methods: &[Guard<&'static [JSFunctionSpec]>]) {
|
||||||
|
for guard in methods {
|
||||||
|
if let Some(specs) = guard.expose(cx, obj) {
|
||||||
|
define_methods(cx, obj, specs).unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Conditionally define properties on an object.
|
||||||
|
pub unsafe fn define_guarded_properties(
|
||||||
|
cx: *mut JSContext,
|
||||||
|
obj: HandleObject,
|
||||||
|
properties: &[Guard<&'static [JSPropertySpec]>]) {
|
||||||
|
for guard in properties {
|
||||||
|
if let Some(specs) = guard.expose(cx, obj) {
|
||||||
|
define_properties(cx, obj, specs).unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns whether an interface with exposure set given by `globals` should
|
||||||
|
/// be exposed in the global object `obj`.
|
||||||
|
pub unsafe fn is_exposed_in(object: HandleObject, globals: Globals) -> bool {
|
||||||
|
let unwrapped = UncheckedUnwrapObject(object.get(), /* stopAtWindowProxy = */ 0);
|
||||||
|
let dom_class = get_dom_class(unwrapped).unwrap();
|
||||||
|
globals.contains(dom_class.global)
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe fn define_on_global_object(
|
||||||
|
cx: *mut JSContext,
|
||||||
|
global: HandleObject,
|
||||||
|
name: &[u8],
|
||||||
|
obj: HandleObject) {
|
||||||
|
assert!(*name.last().unwrap() == b'\0');
|
||||||
|
assert!(JS_DefineProperty1(cx,
|
||||||
|
global,
|
||||||
|
name.as_ptr() as *const libc::c_char,
|
||||||
|
obj,
|
||||||
|
JSPROP_RESOLVING,
|
||||||
|
None, None));
|
||||||
|
}
|
||||||
|
|
||||||
|
const OBJECT_OPS: ObjectOps = ObjectOps {
|
||||||
|
lookupProperty: None,
|
||||||
|
defineProperty: None,
|
||||||
|
hasProperty: None,
|
||||||
|
getProperty: None,
|
||||||
|
setProperty: None,
|
||||||
|
getOwnPropertyDescriptor: None,
|
||||||
|
deleteProperty: None,
|
||||||
|
watch: None,
|
||||||
|
unwatch: None,
|
||||||
|
getElements: None,
|
||||||
|
enumerate: None,
|
||||||
|
funToString: Some(fun_to_string_hook),
|
||||||
|
};
|
||||||
|
|
||||||
|
unsafe extern "C" fn fun_to_string_hook(cx: *mut JSContext,
|
||||||
|
obj: HandleObject,
|
||||||
|
_indent: u32)
|
||||||
|
-> *mut JSString {
|
||||||
|
let js_class = JS_GetClass(obj.get());
|
||||||
|
assert!(!js_class.is_null());
|
||||||
|
let repr = (*(js_class as *const NonCallbackInterfaceObjectClass)).representation;
|
||||||
|
assert!(!repr.is_empty());
|
||||||
|
let ret = JS_NewStringCopyN(cx, repr.as_ptr() as *const libc::c_char, repr.len());
|
||||||
|
assert!(!ret.is_null());
|
||||||
|
ret
|
||||||
|
}
|
||||||
|
|
||||||
/// Hook for instanceof on interface objects.
|
/// Hook for instanceof on interface objects.
|
||||||
unsafe extern "C" fn has_instance_hook(cx: *mut JSContext,
|
unsafe extern "C" fn has_instance_hook(cx: *mut JSContext,
|
||||||
obj: HandleObject,
|
obj: HandleObject,
|
||||||
|
@ -405,21 +417,6 @@ unsafe fn has_instance(
|
||||||
Err(())
|
Err(())
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe fn create_object(
|
|
||||||
cx: *mut JSContext,
|
|
||||||
proto: HandleObject,
|
|
||||||
class: &'static JSClass,
|
|
||||||
methods: &[Guard<&'static [JSFunctionSpec]>],
|
|
||||||
properties: &[Guard<&'static [JSPropertySpec]>],
|
|
||||||
constants: &[Guard<&[ConstantSpec]>],
|
|
||||||
rval: MutableHandleObject) {
|
|
||||||
rval.set(JS_NewObjectWithUniqueType(cx, class, proto));
|
|
||||||
assert!(!rval.ptr.is_null());
|
|
||||||
define_guarded_methods(cx, rval.handle(), methods);
|
|
||||||
define_guarded_properties(cx, rval.handle(), properties);
|
|
||||||
define_guarded_constants(cx, rval.handle(), constants);
|
|
||||||
}
|
|
||||||
|
|
||||||
unsafe fn create_unscopable_object(
|
unsafe fn create_unscopable_object(
|
||||||
cx: *mut JSContext,
|
cx: *mut JSContext,
|
||||||
names: &[&[u8]],
|
names: &[&[u8]],
|
||||||
|
@ -436,42 +433,6 @@ unsafe fn create_unscopable_object(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Conditionally define constants on an object.
|
|
||||||
pub unsafe fn define_guarded_constants(
|
|
||||||
cx: *mut JSContext,
|
|
||||||
obj: HandleObject,
|
|
||||||
constants: &[Guard<&[ConstantSpec]>]) {
|
|
||||||
for guard in constants {
|
|
||||||
if let Some(specs) = guard.expose(cx, obj) {
|
|
||||||
define_constants(cx, obj, specs);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Conditionally define methods on an object.
|
|
||||||
pub unsafe fn define_guarded_methods(
|
|
||||||
cx: *mut JSContext,
|
|
||||||
obj: HandleObject,
|
|
||||||
methods: &[Guard<&'static [JSFunctionSpec]>]) {
|
|
||||||
for guard in methods {
|
|
||||||
if let Some(specs) = guard.expose(cx, obj) {
|
|
||||||
define_methods(cx, obj, specs).unwrap();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Conditionally define properties on an object.
|
|
||||||
pub unsafe fn define_guarded_properties(
|
|
||||||
cx: *mut JSContext,
|
|
||||||
obj: HandleObject,
|
|
||||||
properties: &[Guard<&'static [JSPropertySpec]>]) {
|
|
||||||
for guard in properties {
|
|
||||||
if let Some(specs) = guard.expose(cx, obj) {
|
|
||||||
define_properties(cx, obj, specs).unwrap();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
unsafe fn define_name(cx: *mut JSContext, obj: HandleObject, name: &[u8]) {
|
unsafe fn define_name(cx: *mut JSContext, obj: HandleObject, name: &[u8]) {
|
||||||
assert!(*name.last().unwrap() == b'\0');
|
assert!(*name.last().unwrap() == b'\0');
|
||||||
rooted!(in(cx) let name = JS_AtomizeAndPinString(cx, name.as_ptr() as *const libc::c_char));
|
rooted!(in(cx) let name = JS_AtomizeAndPinString(cx, name.as_ptr() as *const libc::c_char));
|
||||||
|
@ -493,20 +454,6 @@ unsafe fn define_length(cx: *mut JSContext, obj: HandleObject, length: u32) {
|
||||||
None, None));
|
None, None));
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe fn define_on_global_object(
|
|
||||||
cx: *mut JSContext,
|
|
||||||
global: HandleObject,
|
|
||||||
name: &[u8],
|
|
||||||
obj: HandleObject) {
|
|
||||||
assert!(*name.last().unwrap() == b'\0');
|
|
||||||
assert!(JS_DefineProperty1(cx,
|
|
||||||
global,
|
|
||||||
name.as_ptr() as *const libc::c_char,
|
|
||||||
obj,
|
|
||||||
JSPROP_RESOLVING,
|
|
||||||
None, None));
|
|
||||||
}
|
|
||||||
|
|
||||||
unsafe extern "C" fn invalid_constructor(
|
unsafe extern "C" fn invalid_constructor(
|
||||||
cx: *mut JSContext,
|
cx: *mut JSContext,
|
||||||
_argc: libc::c_uint,
|
_argc: libc::c_uint,
|
||||||
|
@ -524,11 +471,3 @@ unsafe extern "C" fn non_new_constructor(
|
||||||
throw_type_error(cx, "This constructor needs to be called with `new`.");
|
throw_type_error(cx, "This constructor needs to be called with `new`.");
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns whether an interface with exposure set given by `globals` should
|
|
||||||
/// be exposed in the global object `obj`.
|
|
||||||
pub unsafe fn is_exposed_in(object: HandleObject, globals: Globals) -> bool {
|
|
||||||
let unwrapped = UncheckedUnwrapObject(object.get(), /* stopAtWindowProxy = */ 0);
|
|
||||||
let dom_class = get_dom_class(unwrapped).unwrap();
|
|
||||||
globals.contains(dom_class.global)
|
|
||||||
}
|
|
||||||
|
|
|
@ -131,6 +131,7 @@
|
||||||
pub use style::domrefcell as cell;
|
pub use style::domrefcell as cell;
|
||||||
|
|
||||||
pub mod callback;
|
pub mod callback;
|
||||||
|
pub mod constant;
|
||||||
pub mod conversions;
|
pub mod conversions;
|
||||||
pub mod error;
|
pub mod error;
|
||||||
pub mod global;
|
pub mod global;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue