mirror of
https://github.com/fluencelabs/wasm-bindgen
synced 2025-06-21 00:36:33 +00:00
webidl: add features necessary to support Event.initEvent
1. Adds support for binding `DOMString` arguments to `&str`. 2. Ignore whether an argument is optional, and just always emit bindings for it.
This commit is contained in:
@ -68,6 +68,34 @@ fn compile_ast(ast: &backend::ast::Program) -> String {
|
|||||||
tokens.to_string()
|
tokens.to_string()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn is_rust_keyword(name: &str) -> bool {
|
||||||
|
match name {
|
||||||
|
"abstract" | "alignof" | "as" | "become" | "box" | "break" | "const" | "continue"
|
||||||
|
| "crate" | "do" | "else" | "enum" | "extern" | "false" | "final" | "fn" | "for" | "if"
|
||||||
|
| "impl" | "in" | "let" | "loop" | "macro" | "match" | "mod" | "move" | "mut"
|
||||||
|
| "offsetof" | "override" | "priv" | "proc" | "pub" | "pure" | "ref" | "return"
|
||||||
|
| "Self" | "self" | "sizeof" | "static" | "struct" | "super" | "trait" | "true"
|
||||||
|
| "type" | "typeof" | "unsafe" | "unsized" | "use" | "virtual" | "where" | "while"
|
||||||
|
| "yield" | "bool" | "_" => true,
|
||||||
|
_ => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create an `Ident`, possibly mangling it if it conflicts with a Rust keyword.
|
||||||
|
fn rust_ident(name: &str) -> Ident {
|
||||||
|
if is_rust_keyword(name) {
|
||||||
|
Ident::new(&format!("{}_", name), proc_macro2::Span::call_site())
|
||||||
|
} else {
|
||||||
|
raw_ident(name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create an `Ident` without checking to see if it conflicts with a Rust
|
||||||
|
// keyword.
|
||||||
|
fn raw_ident(name: &str) -> Ident {
|
||||||
|
Ident::new(name, proc_macro2::Span::call_site())
|
||||||
|
}
|
||||||
|
|
||||||
trait WebidlParse<'a> {
|
trait WebidlParse<'a> {
|
||||||
type Extra;
|
type Extra;
|
||||||
|
|
||||||
@ -138,7 +166,7 @@ impl<'a> WebidlParse<'a> for webidl::ast::NonPartialInterface {
|
|||||||
vis: syn::Visibility::Public(syn::VisPublic {
|
vis: syn::Visibility::Public(syn::VisPublic {
|
||||||
pub_token: Default::default(),
|
pub_token: Default::default(),
|
||||||
}),
|
}),
|
||||||
name: Ident::new(&self.name, proc_macro2::Span::call_site()),
|
name: rust_ident(&self.name),
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -188,13 +216,12 @@ impl<'a> WebidlParse<'a> for webidl::ast::Operation {
|
|||||||
|
|
||||||
fn simple_path_ty<I>(segments: I) -> syn::Type
|
fn simple_path_ty<I>(segments: I) -> syn::Type
|
||||||
where
|
where
|
||||||
I: IntoIterator,
|
I: IntoIterator<Item = Ident>,
|
||||||
I::Item: AsRef<str>,
|
|
||||||
{
|
{
|
||||||
let segments: Vec<_> = segments
|
let segments: Vec<_> = segments
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|s| syn::PathSegment {
|
.map(|i| syn::PathSegment {
|
||||||
ident: syn::Ident::new(s.as_ref(), proc_macro2::Span::call_site()),
|
ident: i,
|
||||||
arguments: syn::PathArguments::None,
|
arguments: syn::PathArguments::None,
|
||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
@ -208,6 +235,10 @@ where
|
|||||||
}.into()
|
}.into()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn ident_ty(ident: Ident) -> syn::Type {
|
||||||
|
simple_path_ty(Some(ident))
|
||||||
|
}
|
||||||
|
|
||||||
fn shared_ref(ty: syn::Type) -> syn::Type {
|
fn shared_ref(ty: syn::Type) -> syn::Type {
|
||||||
syn::TypeReference {
|
syn::TypeReference {
|
||||||
and_token: Default::default(),
|
and_token: Default::default(),
|
||||||
@ -217,37 +248,51 @@ fn shared_ref(ty: syn::Type) -> syn::Type {
|
|||||||
}.into()
|
}.into()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn webidl_ty_to_syn_ty(ty: &webidl::ast::Type) -> Option<syn::Type> {
|
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
||||||
|
enum TypePosition {
|
||||||
|
Argument,
|
||||||
|
Return,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn webidl_ty_to_syn_ty(ty: &webidl::ast::Type, pos: TypePosition) -> Option<syn::Type> {
|
||||||
Some(match ty.kind {
|
Some(match ty.kind {
|
||||||
// `any` becomes `::wasm_bindgen::JsValue`.
|
// `any` becomes `::wasm_bindgen::JsValue`.
|
||||||
webidl::ast::TypeKind::Any => simple_path_ty(&["wasm_bindgen", "JsValue"]),
|
webidl::ast::TypeKind::Any => {
|
||||||
|
simple_path_ty(vec![rust_ident("wasm_bindgen"), rust_ident("JsValue")])
|
||||||
|
}
|
||||||
|
|
||||||
// A reference to a type by name becomes the same thing in the
|
// A reference to a type by name becomes the same thing in the
|
||||||
// bindings.
|
// bindings.
|
||||||
webidl::ast::TypeKind::Identifier(ref id) => simple_path_ty(Some(id)),
|
webidl::ast::TypeKind::Identifier(ref id) => ident_ty(rust_ident(id)),
|
||||||
|
|
||||||
// Scalars.
|
// Scalars.
|
||||||
webidl::ast::TypeKind::Boolean => simple_path_ty(Some("bool")),
|
webidl::ast::TypeKind::Boolean => ident_ty(raw_ident("bool")),
|
||||||
webidl::ast::TypeKind::Byte => simple_path_ty(Some("i8")),
|
webidl::ast::TypeKind::Byte => ident_ty(raw_ident("i8")),
|
||||||
webidl::ast::TypeKind::Octet => simple_path_ty(Some("u8")),
|
webidl::ast::TypeKind::Octet => ident_ty(raw_ident("u8")),
|
||||||
webidl::ast::TypeKind::RestrictedDouble | webidl::ast::TypeKind::UnrestrictedDouble => {
|
webidl::ast::TypeKind::RestrictedDouble | webidl::ast::TypeKind::UnrestrictedDouble => {
|
||||||
simple_path_ty(Some("f64"))
|
ident_ty(raw_ident("f64"))
|
||||||
}
|
}
|
||||||
webidl::ast::TypeKind::RestrictedFloat | webidl::ast::TypeKind::UnrestrictedFloat => {
|
webidl::ast::TypeKind::RestrictedFloat | webidl::ast::TypeKind::UnrestrictedFloat => {
|
||||||
simple_path_ty(Some("f32"))
|
ident_ty(raw_ident("f32"))
|
||||||
}
|
}
|
||||||
webidl::ast::TypeKind::SignedLong => simple_path_ty(Some("i32")),
|
webidl::ast::TypeKind::SignedLong => ident_ty(raw_ident("i32")),
|
||||||
webidl::ast::TypeKind::SignedLongLong => simple_path_ty(Some("i64")),
|
webidl::ast::TypeKind::SignedLongLong => ident_ty(raw_ident("i64")),
|
||||||
webidl::ast::TypeKind::SignedShort => simple_path_ty(Some("i16")),
|
webidl::ast::TypeKind::SignedShort => ident_ty(raw_ident("i16")),
|
||||||
webidl::ast::TypeKind::UnsignedLong => simple_path_ty(Some("u32")),
|
webidl::ast::TypeKind::UnsignedLong => ident_ty(raw_ident("u32")),
|
||||||
webidl::ast::TypeKind::UnsignedLongLong => simple_path_ty(Some("u64")),
|
webidl::ast::TypeKind::UnsignedLongLong => ident_ty(raw_ident("u64")),
|
||||||
webidl::ast::TypeKind::UnsignedShort => simple_path_ty(Some("u16")),
|
webidl::ast::TypeKind::UnsignedShort => ident_ty(raw_ident("u16")),
|
||||||
|
|
||||||
|
// `DOMString -> `&str` for arguments
|
||||||
|
webidl::ast::TypeKind::DOMString if pos == TypePosition::Argument => {
|
||||||
|
shared_ref(ident_ty(raw_ident("str")))
|
||||||
|
}
|
||||||
|
// `DOMString` is not supported yet in other positions.
|
||||||
|
webidl::ast::TypeKind::DOMString => return None,
|
||||||
|
|
||||||
// Support for these types is not yet implemented, so skip
|
// Support for these types is not yet implemented, so skip
|
||||||
// generating any bindings for this function.
|
// generating any bindings for this function.
|
||||||
webidl::ast::TypeKind::ArrayBuffer
|
webidl::ast::TypeKind::ArrayBuffer
|
||||||
| webidl::ast::TypeKind::ByteString
|
| webidl::ast::TypeKind::ByteString
|
||||||
| webidl::ast::TypeKind::DOMString
|
|
||||||
| webidl::ast::TypeKind::DataView
|
| webidl::ast::TypeKind::DataView
|
||||||
| webidl::ast::TypeKind::Error
|
| webidl::ast::TypeKind::Error
|
||||||
| webidl::ast::TypeKind::Float32Array
|
| webidl::ast::TypeKind::Float32Array
|
||||||
@ -272,7 +317,7 @@ fn webidl_ty_to_syn_ty(ty: &webidl::ast::Type) -> Option<syn::Type> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn simple_fn_arg(ident: proc_macro2::Ident, ty: syn::Type) -> syn::FnArg {
|
fn simple_fn_arg(ident: Ident, ty: syn::Type) -> syn::FnArg {
|
||||||
syn::FnArg::Captured(syn::ArgCaptured {
|
syn::FnArg::Captured(syn::ArgCaptured {
|
||||||
pat: syn::Pat::Ident(syn::PatIdent {
|
pat: syn::Pat::Ident(syn::PatIdent {
|
||||||
by_ref: None,
|
by_ref: None,
|
||||||
@ -297,12 +342,13 @@ impl<'a> WebidlParse<'a> for webidl::ast::RegularOperation {
|
|||||||
);
|
);
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
Some(ref name) => Ident::new(name, proc_macro2::Span::call_site()),
|
Some(ref name) => rust_ident(name),
|
||||||
};
|
};
|
||||||
|
|
||||||
let (output, ret) = match self.return_type {
|
let (output, ret) = match self.return_type {
|
||||||
webidl::ast::ReturnType::Void => (syn::ReturnType::Default, None),
|
webidl::ast::ReturnType::Void => (syn::ReturnType::Default, None),
|
||||||
webidl::ast::ReturnType::NonVoid(ref ty) => match webidl_ty_to_syn_ty(ty) {
|
webidl::ast::ReturnType::NonVoid(ref ty) => {
|
||||||
|
match webidl_ty_to_syn_ty(ty, TypePosition::Return) {
|
||||||
None => {
|
None => {
|
||||||
warn!(
|
warn!(
|
||||||
"Operation's return type is not yet supported: {:?}. Skipping bindings for {:?}",
|
"Operation's return type is not yet supported: {:?}. Skipping bindings for {:?}",
|
||||||
@ -314,29 +360,19 @@ impl<'a> WebidlParse<'a> for webidl::ast::RegularOperation {
|
|||||||
syn::ReturnType::Type(Default::default(), Box::new(ty.clone())),
|
syn::ReturnType::Type(Default::default(), Box::new(ty.clone())),
|
||||||
Some(ty),
|
Some(ty),
|
||||||
),
|
),
|
||||||
},
|
}
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let mut inputs = Vec::with_capacity(self.arguments.len() + 1);
|
let mut inputs = Vec::with_capacity(self.arguments.len() + 1);
|
||||||
let mut arguments = Vec::with_capacity(self.arguments.len() + 1);
|
let mut arguments = Vec::with_capacity(self.arguments.len() + 1);
|
||||||
|
|
||||||
let self_ty = simple_path_ty(Some(self_name));
|
let self_ty = ident_ty(rust_ident(self_name));
|
||||||
let self_ref_ty = shared_ref(self_ty.clone());
|
let self_ref_ty = shared_ref(self_ty.clone());
|
||||||
inputs.push(simple_fn_arg(
|
inputs.push(simple_fn_arg(raw_ident("self_"), self_ref_ty.clone()));
|
||||||
proc_macro2::Ident::new("self_", proc_macro2::Span::call_site()),
|
|
||||||
self_ref_ty.clone(),
|
|
||||||
));
|
|
||||||
arguments.push(self_ref_ty);
|
arguments.push(self_ref_ty);
|
||||||
|
|
||||||
for arg in &self.arguments {
|
for arg in &self.arguments {
|
||||||
if arg.optional {
|
|
||||||
warn!(
|
|
||||||
"Optional arguments are not supported yet. Skipping bindings for {:?}",
|
|
||||||
self
|
|
||||||
);
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
if arg.variadic {
|
if arg.variadic {
|
||||||
warn!(
|
warn!(
|
||||||
"Variadic arguments are not supported yet. Skipping bindings for {:?}",
|
"Variadic arguments are not supported yet. Skipping bindings for {:?}",
|
||||||
@ -345,7 +381,7 @@ impl<'a> WebidlParse<'a> for webidl::ast::RegularOperation {
|
|||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
match webidl_ty_to_syn_ty(&arg.type_) {
|
match webidl_ty_to_syn_ty(&arg.type_, TypePosition::Argument) {
|
||||||
None => {
|
None => {
|
||||||
warn!(
|
warn!(
|
||||||
"Argument's type is not yet supported: {:?}. Skipping bindings for {:?}",
|
"Argument's type is not yet supported: {:?}. Skipping bindings for {:?}",
|
||||||
@ -354,20 +390,14 @@ impl<'a> WebidlParse<'a> for webidl::ast::RegularOperation {
|
|||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
Some(ty) => {
|
Some(ty) => {
|
||||||
inputs.push(simple_fn_arg(
|
inputs.push(simple_fn_arg(rust_ident(&arg.name), ty.clone()));
|
||||||
proc_macro2::Ident::new(&arg.name, proc_macro2::Span::call_site()),
|
|
||||||
ty.clone(),
|
|
||||||
));
|
|
||||||
arguments.push(ty);
|
arguments.push(ty);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let rust_name = fn_name.clone();
|
let rust_name = fn_name.clone();
|
||||||
let shim = proc_macro2::Ident::new(
|
let shim = rust_ident(&format!("__wbg_f_{}_{}_{}", fn_name, fn_name, self_name));
|
||||||
&format!("__wbg_f_{}_{}_{}", fn_name, fn_name, self_name),
|
|
||||||
proc_macro2::Span::call_site(),
|
|
||||||
);
|
|
||||||
|
|
||||||
program.imports.push(backend::ast::Import {
|
program.imports.push(backend::ast::Import {
|
||||||
module: None,
|
module: None,
|
||||||
|
@ -171,6 +171,59 @@ impl Event {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#[no_mangle]
|
||||||
|
#[allow(non_snake_case)]
|
||||||
|
pub extern "C" fn __wbindgen_describe___wbg_f_initEvent_initEvent_Event() {
|
||||||
|
use wasm_bindgen::describe::*;
|
||||||
|
inform(FUNCTION);
|
||||||
|
inform(4u32);
|
||||||
|
<&Event as WasmDescribe>::describe();
|
||||||
|
<&str as WasmDescribe>::describe();
|
||||||
|
<bool as WasmDescribe>::describe();
|
||||||
|
<bool as WasmDescribe>::describe();
|
||||||
|
inform(0);
|
||||||
|
}
|
||||||
|
impl Event {
|
||||||
|
#[allow(bad_style)]
|
||||||
|
#[cfg(all(target_arch = "wasm32", not(target_os = "emscripten")))]
|
||||||
|
pub extern "C" fn initEvent(&self, type_: &str, bubbles: bool, cancelable: bool) {
|
||||||
|
::wasm_bindgen::__rt::link_this_library();
|
||||||
|
#[wasm_import_module = "__wbindgen_placeholder__"]
|
||||||
|
extern "C" {
|
||||||
|
fn __wbg_f_initEvent_initEvent_Event(
|
||||||
|
self_: <&Event as ::wasm_bindgen::convert::IntoWasmAbi>::Abi,
|
||||||
|
type_: <&str as ::wasm_bindgen::convert::IntoWasmAbi>::Abi,
|
||||||
|
bubbles: <bool as ::wasm_bindgen::convert::IntoWasmAbi>::Abi,
|
||||||
|
cancelable: <bool as ::wasm_bindgen::convert::IntoWasmAbi>::Abi,
|
||||||
|
) -> ();
|
||||||
|
}
|
||||||
|
unsafe {
|
||||||
|
let _ret = {
|
||||||
|
let mut __stack = ::wasm_bindgen::convert::GlobalStack::new();
|
||||||
|
let self_ =
|
||||||
|
<&Event as ::wasm_bindgen::convert::IntoWasmAbi>::into_abi(self, &mut __stack);
|
||||||
|
let type_ =
|
||||||
|
<&str as ::wasm_bindgen::convert::IntoWasmAbi>::into_abi(type_, &mut __stack);
|
||||||
|
let bubbles =
|
||||||
|
<bool as ::wasm_bindgen::convert::IntoWasmAbi>::into_abi(bubbles, &mut __stack);
|
||||||
|
let cancelable = <bool as ::wasm_bindgen::convert::IntoWasmAbi>::into_abi(
|
||||||
|
cancelable,
|
||||||
|
&mut __stack,
|
||||||
|
);
|
||||||
|
__wbg_f_initEvent_initEvent_Event(self_, type_, bubbles, cancelable)
|
||||||
|
};
|
||||||
|
()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#[allow(bad_style, unused_variables)]
|
||||||
|
#[cfg(not(all(target_arch = "wasm32", not(target_os = "emscripten"))))]
|
||||||
|
pub extern "C" fn initEvent(&self, type_: &str, bubbles: bool, cancelable: bool) {
|
||||||
|
panic!(
|
||||||
|
"cannot call wasm-bindgen imported functions on \
|
||||||
|
non-wasm targets"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
#[allow(non_upper_case_globals)]
|
#[allow(non_upper_case_globals)]
|
||||||
#[wasm_custom_section = "__wasm_bindgen_unstable"]
|
#[wasm_custom_section = "__wasm_bindgen_unstable"]
|
||||||
const __WASM_BINDGEN_GENERATED_wasm_bindgen_webidl_0_1_0_0 : [ u8 ; 180usize ] = * b"\xB0\0\0\0{\"exports\":[],\"enums\":[],\"imports\":[{\"module\":null,\"version\":null,\"js_namespace\":null,\"kind\":{\"kind\":\"type\"}}],\"structs\":[],\"version\":\"0.2.11 (3879f6f42)\",\"schema_version\":\"4\"}" ;
|
const __WASM_BINDGEN_GENERATED_wasm_bindgen_webidl_0_1_0_0 : [ u8 ; 180usize ] = * b"\xB0\0\0\0{\"exports\":[],\"enums\":[],\"imports\":[{\"module\":null,\"version\":null,\"js_namespace\":null,\"kind\":{\"kind\":\"type\"}}],\"structs\":[],\"version\":\"0.2.11 (3879f6f42)\",\"schema_version\":\"4\"}" ;
|
||||||
|
Reference in New Issue
Block a user