2018-08-09 21:51:41 +03:00
|
|
|
use std::collections::BTreeMap;
|
2018-08-29 15:00:58 -07:00
|
|
|
use std::iter::FromIterator;
|
|
|
|
use std::ptr;
|
2018-06-14 02:03:52 -07:00
|
|
|
|
|
|
|
use backend;
|
2018-08-04 15:16:02 -07:00
|
|
|
use backend::util::{ident_ty, leading_colon_path_ty, raw_ident, rust_ident};
|
2018-07-10 22:59:59 -07:00
|
|
|
use heck::{CamelCase, SnakeCase};
|
2018-06-25 10:41:33 -07:00
|
|
|
use proc_macro2::Ident;
|
2018-06-14 02:03:52 -07:00
|
|
|
use syn;
|
2018-08-03 14:39:33 -07:00
|
|
|
use weedle;
|
|
|
|
use weedle::attribute::{ExtendedAttributeList, ExtendedAttribute};
|
2018-08-11 23:46:33 +03:00
|
|
|
use weedle::argument::Argument;
|
2018-08-03 14:39:33 -07:00
|
|
|
use weedle::literal::{ConstValue, FloatLit, IntegerLit};
|
2018-06-14 02:03:52 -07:00
|
|
|
|
2018-08-29 15:00:58 -07:00
|
|
|
use first_pass::{self, FirstPassRecord, OperationId, OperationData2, Signature};
|
2018-08-11 23:46:33 +03:00
|
|
|
use idl_type::{IdlType, ToIdlType, flatten};
|
2018-07-13 21:46:36 -07:00
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Take a type and create an immutable shared reference to that type.
|
2018-08-11 23:46:33 +03:00
|
|
|
pub(crate) fn shared_ref(ty: syn::Type) -> syn::Type {
|
2018-06-14 02:03:52 -07:00
|
|
|
syn::TypeReference {
|
|
|
|
and_token: Default::default(),
|
|
|
|
lifetime: None,
|
|
|
|
mutability: None,
|
|
|
|
elem: Box::new(ty),
|
|
|
|
}.into()
|
|
|
|
}
|
|
|
|
|
2018-07-27 17:57:24 +01:00
|
|
|
/// Fix camelcase of identifiers like HTMLBRElement
|
|
|
|
pub fn camel_case_ident(identifier: &str) -> String {
|
|
|
|
identifier.replace("HTML", "HTML_").to_camel_case()
|
|
|
|
}
|
|
|
|
|
2018-07-29 17:12:36 +01:00
|
|
|
// Returns a link to MDN
|
|
|
|
pub fn mdn_doc(class: &str, method: Option<&str>) -> String {
|
|
|
|
let mut link = format!("https://developer.mozilla.org/en-US/docs/Web/API/{}", class);
|
|
|
|
if let Some(method) = method {
|
|
|
|
link.push_str(&format!("/{}", method));
|
|
|
|
}
|
|
|
|
format!("[Documentation]({})", link).into()
|
|
|
|
}
|
2018-07-27 17:57:24 +01:00
|
|
|
|
2018-08-03 14:39:33 -07:00
|
|
|
// Array type is borrowed for arguments (`&[T]`) and owned for return value (`Vec<T>`).
|
2018-08-11 23:46:33 +03:00
|
|
|
pub(crate) fn array(base_ty: &str, pos: TypePosition) -> syn::Type {
|
2018-08-03 14:39:33 -07:00
|
|
|
match pos {
|
|
|
|
TypePosition::Argument => {
|
|
|
|
shared_ref(slice_ty(ident_ty(raw_ident(base_ty))))
|
|
|
|
}
|
|
|
|
TypePosition::Return => {
|
|
|
|
vec_ty(ident_ty(raw_ident(base_ty)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Map a webidl const value to the correct wasm-bindgen const value
|
2018-08-03 14:39:33 -07:00
|
|
|
pub fn webidl_const_v_to_backend_const_v(v: &ConstValue) -> backend::ast::ConstValue {
|
|
|
|
use std::f64::{NEG_INFINITY, INFINITY, NAN};
|
|
|
|
use backend::ast;
|
|
|
|
|
2018-07-13 06:04:40 +02:00
|
|
|
match *v {
|
2018-08-03 14:39:33 -07:00
|
|
|
ConstValue::Boolean(b) => ast::ConstValue::BooleanLiteral(b.0),
|
|
|
|
ConstValue::Float(FloatLit::NegInfinity(_)) => {
|
|
|
|
ast::ConstValue::FloatLiteral(NEG_INFINITY)
|
|
|
|
}
|
|
|
|
ConstValue::Float(FloatLit::Infinity(_)) => {
|
|
|
|
ast::ConstValue::FloatLiteral(INFINITY)
|
|
|
|
}
|
|
|
|
ConstValue::Float(FloatLit::NaN(_)) => {
|
|
|
|
ast::ConstValue::FloatLiteral(NAN)
|
|
|
|
}
|
|
|
|
ConstValue::Float(FloatLit::Value(s)) => {
|
|
|
|
ast::ConstValue::FloatLiteral(s.0.parse().unwrap())
|
|
|
|
}
|
|
|
|
ConstValue::Integer(lit) => {
|
|
|
|
let mklit = |orig_text: &str, base: u32, offset: usize| {
|
|
|
|
let (negative, text) = if orig_text.starts_with("-") {
|
|
|
|
(true, &orig_text[1..])
|
|
|
|
} else {
|
|
|
|
(false, orig_text)
|
|
|
|
};
|
|
|
|
if text == "0" {
|
|
|
|
return ast::ConstValue::SignedIntegerLiteral(0)
|
|
|
|
}
|
|
|
|
let text = &text[offset..];
|
|
|
|
let n = u64::from_str_radix(text, base)
|
|
|
|
.unwrap_or_else(|_| panic!("literal too big: {}", orig_text));
|
|
|
|
if negative {
|
|
|
|
let n = if n > (i64::min_value() as u64).wrapping_neg() {
|
|
|
|
panic!("literal too big: {}", orig_text)
|
|
|
|
} else {
|
|
|
|
n.wrapping_neg() as i64
|
|
|
|
};
|
|
|
|
ast::ConstValue::SignedIntegerLiteral(n)
|
|
|
|
} else {
|
|
|
|
ast::ConstValue::UnsignedIntegerLiteral(n)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
match lit {
|
|
|
|
IntegerLit::Hex(h) => mklit(h.0, 16, 2), // leading 0x
|
|
|
|
IntegerLit::Oct(h) => mklit(h.0, 8, 1), // leading 0
|
|
|
|
IntegerLit::Dec(h) => mklit(h.0, 10, 0),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ConstValue::Null(_) => ast::ConstValue::Null,
|
2018-07-13 06:04:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// From `ident` and `Ty`, create `ident: Ty` for use in e.g. `fn(ident: Ty)`.
|
2018-06-14 02:03:52 -07:00
|
|
|
fn simple_fn_arg(ident: Ident, ty: syn::Type) -> syn::ArgCaptured {
|
|
|
|
syn::ArgCaptured {
|
|
|
|
pat: syn::Pat::Ident(syn::PatIdent {
|
|
|
|
by_ref: None,
|
|
|
|
mutability: None,
|
|
|
|
ident,
|
|
|
|
subpat: None,
|
|
|
|
}),
|
|
|
|
colon_token: Default::default(),
|
|
|
|
ty,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Create `()`.
|
2018-07-11 11:07:03 -07:00
|
|
|
fn unit_ty() -> syn::Type {
|
|
|
|
syn::Type::Tuple(syn::TypeTuple {
|
|
|
|
paren_token: Default::default(),
|
|
|
|
elems: syn::punctuated::Punctuated::new(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// From `T` create `Result<T, ::wasm_bindgen::JsValue>`.
|
2018-07-11 11:07:03 -07:00
|
|
|
fn result_ty(t: syn::Type) -> syn::Type {
|
|
|
|
let js_value = leading_colon_path_ty(vec![rust_ident("wasm_bindgen"), rust_ident("JsValue")]);
|
|
|
|
|
|
|
|
let arguments = syn::PathArguments::AngleBracketed(syn::AngleBracketedGenericArguments {
|
|
|
|
colon2_token: None,
|
|
|
|
lt_token: Default::default(),
|
|
|
|
args: FromIterator::from_iter(vec![
|
|
|
|
syn::GenericArgument::Type(t),
|
|
|
|
syn::GenericArgument::Type(js_value),
|
|
|
|
]),
|
|
|
|
gt_token: Default::default(),
|
|
|
|
});
|
|
|
|
|
|
|
|
let ident = raw_ident("Result");
|
|
|
|
let seg = syn::PathSegment { ident, arguments };
|
|
|
|
let path: syn::Path = seg.into();
|
|
|
|
let ty = syn::TypePath { qself: None, path };
|
|
|
|
ty.into()
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// From `T` create `[T]`.
|
2018-08-11 23:46:33 +03:00
|
|
|
pub(crate) fn slice_ty(t: syn::Type) -> syn::Type {
|
2018-07-18 17:59:24 -05:00
|
|
|
syn::TypeSlice {
|
|
|
|
bracket_token: Default::default(),
|
|
|
|
elem: Box::new(t),
|
|
|
|
}.into()
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// From `T` create `Vec<T>`.
|
2018-08-11 23:46:33 +03:00
|
|
|
pub(crate) fn vec_ty(t: syn::Type) -> syn::Type {
|
2018-07-18 17:59:24 -05:00
|
|
|
let arguments = syn::PathArguments::AngleBracketed(syn::AngleBracketedGenericArguments {
|
|
|
|
colon2_token: None,
|
|
|
|
lt_token: Default::default(),
|
|
|
|
args: FromIterator::from_iter(vec![
|
|
|
|
syn::GenericArgument::Type(t),
|
|
|
|
]),
|
|
|
|
gt_token: Default::default(),
|
|
|
|
});
|
|
|
|
|
|
|
|
let ident = raw_ident("Vec");
|
|
|
|
let seg = syn::PathSegment { ident, arguments };
|
|
|
|
let path: syn::Path = seg.into();
|
|
|
|
let ty = syn::TypePath { qself: None, path };
|
|
|
|
ty.into()
|
|
|
|
}
|
|
|
|
|
2018-08-03 14:39:33 -07:00
|
|
|
/// From `T` create `Option<T>`
|
2018-08-11 23:46:33 +03:00
|
|
|
pub(crate) fn option_ty(t: syn::Type) -> syn::Type {
|
2018-08-03 14:39:33 -07:00
|
|
|
let arguments = syn::PathArguments::AngleBracketed(syn::AngleBracketedGenericArguments {
|
|
|
|
colon2_token: None,
|
|
|
|
lt_token: Default::default(),
|
|
|
|
args: FromIterator::from_iter(vec![syn::GenericArgument::Type(t)]),
|
|
|
|
gt_token: Default::default(),
|
|
|
|
});
|
|
|
|
|
|
|
|
let ident = raw_ident("Option");
|
|
|
|
let seg = syn::PathSegment { ident, arguments };
|
|
|
|
let path: syn::Path = seg.into();
|
|
|
|
let ty = syn::TypePath { qself: None, path };
|
|
|
|
ty.into()
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Possible positions for a type in a function signature.
|
2018-07-10 22:59:59 -07:00
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
|
|
|
pub enum TypePosition {
|
|
|
|
Argument,
|
|
|
|
Return,
|
|
|
|
}
|
2018-06-14 02:03:52 -07:00
|
|
|
|
2018-08-03 14:39:33 -07:00
|
|
|
impl<'src> FirstPassRecord<'src> {
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Create a wasm-bindgen function, if possible.
|
2018-08-03 14:39:33 -07:00
|
|
|
pub fn create_function(
|
2018-07-10 22:59:59 -07:00
|
|
|
&self,
|
|
|
|
name: &str,
|
2018-07-30 17:41:22 +03:00
|
|
|
overloaded: bool,
|
|
|
|
same_argument_names: bool,
|
2018-08-11 23:46:33 +03:00
|
|
|
arguments: &[(&str, IdlType<'src>, bool)],
|
|
|
|
ret: IdlType<'src>,
|
2018-07-10 22:59:59 -07:00
|
|
|
kind: backend::ast::ImportFunctionKind,
|
|
|
|
structural: bool,
|
|
|
|
catch: bool,
|
2018-07-29 17:12:36 +01:00
|
|
|
doc_comment: Option<String>,
|
2018-08-11 23:46:33 +03:00
|
|
|
) -> Vec<backend::ast::ImportFunction> {
|
2018-08-09 19:24:33 +03:00
|
|
|
let rust_name = if overloaded && !arguments.is_empty() {
|
|
|
|
let mut argument_type_names = String::new();
|
2018-08-09 21:51:41 +03:00
|
|
|
for argument in arguments {
|
2018-08-09 19:24:33 +03:00
|
|
|
if argument_type_names.len() > 0 {
|
|
|
|
argument_type_names.push_str("_and_");
|
2018-08-03 14:39:33 -07:00
|
|
|
}
|
2018-08-09 19:24:33 +03:00
|
|
|
if same_argument_names {
|
2018-08-11 23:46:33 +03:00
|
|
|
argument.1.push_type_name(&mut argument_type_names);
|
2018-07-30 17:41:22 +03:00
|
|
|
} else {
|
2018-08-11 23:46:33 +03:00
|
|
|
argument_type_names.push_str(&argument.0.to_snake_case());
|
2018-07-30 17:41:22 +03:00
|
|
|
}
|
2018-08-09 19:24:33 +03:00
|
|
|
}
|
|
|
|
if name == "new" {
|
|
|
|
"with_".to_owned() + &argument_type_names
|
2018-07-30 17:41:22 +03:00
|
|
|
} else {
|
2018-08-09 19:24:33 +03:00
|
|
|
name.to_snake_case() + "_with_" + &argument_type_names
|
2018-07-30 17:41:22 +03:00
|
|
|
}
|
2018-08-09 19:24:33 +03:00
|
|
|
} else {
|
|
|
|
name.to_snake_case()
|
|
|
|
};
|
2018-07-10 22:59:59 -07:00
|
|
|
|
2018-08-11 23:46:33 +03:00
|
|
|
let converted_arguments = arguments
|
|
|
|
.iter()
|
|
|
|
.cloned()
|
|
|
|
.map(|(_name, idl_type, optional)| (idl_type, optional))
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
let possibilities = flatten(&converted_arguments);
|
|
|
|
let mut arguments_count_multiple = BTreeMap::new();
|
|
|
|
for idl_types in &possibilities {
|
|
|
|
arguments_count_multiple
|
|
|
|
.entry(idl_types.len())
|
|
|
|
.and_modify(|variants_count| { *variants_count = true; })
|
|
|
|
.or_insert(false);
|
|
|
|
}
|
|
|
|
let mut import_functions = Vec::new();
|
|
|
|
'outer: for idl_types in &possibilities {
|
|
|
|
let rust_name = if possibilities.len() > 1 {
|
2018-08-09 19:24:33 +03:00
|
|
|
let mut rust_name = rust_name.clone();
|
|
|
|
let mut first = true;
|
2018-08-16 23:13:34 -07:00
|
|
|
let iter = arguments.iter().zip(idl_types).enumerate();
|
|
|
|
for (i, ((argument_name, _, _), idl_type)) in iter {
|
|
|
|
if possibilities.iter().all(|p| p.get(i) == Some(idl_type)) {
|
|
|
|
continue
|
|
|
|
}
|
2018-08-11 23:46:33 +03:00
|
|
|
if first {
|
2018-08-16 23:13:34 -07:00
|
|
|
rust_name.push_str("_with_");
|
2018-08-11 23:46:33 +03:00
|
|
|
first = false;
|
|
|
|
} else {
|
|
|
|
rust_name.push_str("_and_");
|
|
|
|
}
|
|
|
|
if arguments_count_multiple[&idl_types.len()] {
|
|
|
|
idl_type.push_type_name(&mut rust_name);
|
|
|
|
} else {
|
|
|
|
rust_name.push_str(&argument_name.to_snake_case());
|
2018-08-09 19:24:33 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
rust_name
|
|
|
|
} else {
|
|
|
|
rust_name.clone()
|
2018-07-10 22:59:59 -07:00
|
|
|
};
|
2018-08-29 10:28:04 -07:00
|
|
|
let f = self.create_one_function(
|
|
|
|
name,
|
|
|
|
&rust_name,
|
|
|
|
arguments.iter().map(|s| s.0).zip(idl_types),
|
|
|
|
&ret,
|
|
|
|
kind.clone(),
|
|
|
|
structural,
|
|
|
|
catch,
|
|
|
|
doc_comment.clone(),
|
|
|
|
);
|
|
|
|
import_functions.extend(f);
|
|
|
|
}
|
|
|
|
import_functions
|
|
|
|
}
|
2018-07-10 22:59:59 -07:00
|
|
|
|
2018-08-29 10:28:04 -07:00
|
|
|
pub fn create_one_function<'a>(
|
|
|
|
&self,
|
|
|
|
js_name: &str,
|
|
|
|
rust_name: &str,
|
|
|
|
idl_arguments: impl Iterator<Item = (&'a str, &'a IdlType<'src>)>,
|
|
|
|
ret: &IdlType<'src>,
|
|
|
|
kind: backend::ast::ImportFunctionKind,
|
|
|
|
structural: bool,
|
|
|
|
catch: bool,
|
|
|
|
doc_comment: Option<String>,
|
|
|
|
) -> Option<backend::ast::ImportFunction> where 'src: 'a {
|
|
|
|
// Convert all of the arguments from their IDL type to a `syn` type,
|
|
|
|
// ready to pass to the backend.
|
|
|
|
//
|
|
|
|
// Note that for non-static methods we add a `&self` type placeholder,
|
|
|
|
// but this type isn't actually used so it's just here for show mostly.
|
|
|
|
let mut arguments = if let &backend::ast::ImportFunctionKind::Method {
|
|
|
|
ref ty,
|
|
|
|
kind: backend::ast::MethodKind::Operation(
|
|
|
|
backend::ast::Operation {
|
|
|
|
is_static: false, ..
|
|
|
|
}
|
|
|
|
),
|
|
|
|
..
|
|
|
|
} = &kind {
|
|
|
|
let mut res = Vec::with_capacity(idl_arguments.size_hint().0 + 1);
|
|
|
|
res.push(simple_fn_arg(raw_ident("self_"), shared_ref(ty.clone())));
|
|
|
|
res
|
|
|
|
} else {
|
|
|
|
Vec::with_capacity(idl_arguments.size_hint().0)
|
|
|
|
};
|
|
|
|
for (argument_name, idl_type) in idl_arguments {
|
|
|
|
let syn_type = match idl_type.to_syn_type(TypePosition::Argument) {
|
|
|
|
Some(t) => t,
|
|
|
|
None => {
|
2018-08-15 11:45:45 -07:00
|
|
|
warn!(
|
2018-08-15 14:24:09 -07:00
|
|
|
"Unsupported argument type: {:?} on {:?}",
|
2018-08-15 11:45:45 -07:00
|
|
|
idl_type,
|
|
|
|
rust_name
|
|
|
|
);
|
2018-08-29 10:28:04 -07:00
|
|
|
return None
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let argument_name = rust_ident(&argument_name.to_snake_case());
|
|
|
|
arguments.push(simple_fn_arg(argument_name, syn_type));
|
2018-08-09 19:24:33 +03:00
|
|
|
}
|
2018-08-29 10:28:04 -07:00
|
|
|
|
|
|
|
// Convert the return type to a `syn` type, handling the `catch`
|
|
|
|
// attribute here to use a `Result` in Rust.
|
|
|
|
let ret = match ret {
|
|
|
|
IdlType::Void => None,
|
|
|
|
ret @ _ => {
|
|
|
|
match ret.to_syn_type(TypePosition::Return) {
|
|
|
|
Some(ret) => Some(ret),
|
|
|
|
None => {
|
|
|
|
warn!(
|
|
|
|
"Unsupported return type: {:?} on {:?}",
|
|
|
|
ret,
|
|
|
|
rust_name
|
|
|
|
);
|
|
|
|
return None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
let js_ret = ret.clone();
|
|
|
|
let ret = if catch {
|
|
|
|
Some(ret.map_or_else(|| result_ty(unit_ty()), result_ty))
|
|
|
|
} else {
|
|
|
|
ret
|
|
|
|
};
|
|
|
|
|
|
|
|
Some(backend::ast::ImportFunction {
|
|
|
|
function: backend::ast::Function {
|
|
|
|
name: js_name.to_string(),
|
|
|
|
arguments,
|
|
|
|
ret: ret.clone(),
|
|
|
|
rust_attrs: vec![],
|
|
|
|
rust_vis: public(),
|
|
|
|
},
|
|
|
|
rust_name: rust_ident(rust_name),
|
|
|
|
js_ret: js_ret.clone(),
|
|
|
|
catch,
|
|
|
|
structural,
|
|
|
|
shim:{
|
|
|
|
let ns = match kind {
|
|
|
|
backend::ast::ImportFunctionKind::ScopedMethod { .. } |
|
|
|
|
backend::ast::ImportFunctionKind::Normal => "",
|
|
|
|
backend::ast::ImportFunctionKind::Method { ref class, .. } => class,
|
|
|
|
};
|
|
|
|
raw_ident(&format!("__widl_f_{}_{}", rust_name, ns))
|
|
|
|
},
|
|
|
|
kind,
|
|
|
|
doc_comment,
|
|
|
|
})
|
2018-08-11 23:46:33 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Convert arguments to ones suitable crating function
|
|
|
|
pub(crate) fn convert_arguments(
|
|
|
|
&self,
|
|
|
|
arguments: &[weedle::argument::Argument<'src>],
|
|
|
|
) -> Option<Vec<(&str, IdlType<'src>, bool)>> {
|
|
|
|
let mut converted_arguments = Vec::with_capacity(arguments.len());
|
|
|
|
for argument in arguments {
|
|
|
|
let name = match argument {
|
|
|
|
Argument::Single(single) => single.identifier.0,
|
|
|
|
Argument::Variadic(variadic) => variadic.identifier.0,
|
|
|
|
};
|
|
|
|
let ty = match argument {
|
|
|
|
Argument::Single(single) => &single.type_.type_,
|
|
|
|
Argument::Variadic(variadic) => &variadic.type_,
|
|
|
|
};
|
|
|
|
let idl_type = ty.to_idl_type(self)?;
|
|
|
|
let optional = match argument {
|
|
|
|
Argument::Single(single) => single.optional.is_some(),
|
|
|
|
Argument::Variadic(_variadic) => false,
|
|
|
|
};
|
|
|
|
converted_arguments.push((name, idl_type, optional));
|
|
|
|
}
|
|
|
|
Some(converted_arguments)
|
2018-07-10 22:59:59 -07:00
|
|
|
}
|
|
|
|
|
2018-07-30 17:41:22 +03:00
|
|
|
/// Whether operation is overloaded and
|
|
|
|
/// whether there overloads with same argument names for given argument types
|
|
|
|
pub fn get_operation_overloading(
|
|
|
|
&self,
|
2018-08-07 15:50:27 -07:00
|
|
|
arguments: &[Argument],
|
2018-08-13 18:59:52 +03:00
|
|
|
operation_id: &first_pass::OperationId,
|
2018-07-30 17:41:22 +03:00
|
|
|
self_name: &str,
|
2018-08-13 18:59:52 +03:00
|
|
|
namespace: bool,
|
2018-07-30 17:41:22 +03:00
|
|
|
) -> (bool, bool) {
|
2018-08-09 19:24:33 +03:00
|
|
|
fn get_operation_data<'src>(
|
|
|
|
record: &'src FirstPassRecord,
|
2018-08-13 18:59:52 +03:00
|
|
|
operation_id: &'src ::first_pass::OperationId,
|
2018-08-09 19:24:33 +03:00
|
|
|
self_name: &str,
|
|
|
|
mixin_name: &str,
|
|
|
|
) -> Option<&'src ::first_pass::OperationData<'src>> {
|
|
|
|
if let Some(mixin_data) = record.mixins.get(mixin_name) {
|
2018-08-13 18:59:52 +03:00
|
|
|
if let Some(operation_data) = mixin_data.operations.get(operation_id) {
|
2018-08-09 19:24:33 +03:00
|
|
|
return Some(operation_data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if let Some(mixin_names) = record.includes.get(mixin_name) {
|
|
|
|
for mixin_name in mixin_names {
|
2018-08-13 18:59:52 +03:00
|
|
|
if let Some(operation_data) = get_operation_data(record, operation_id, self_name, mixin_name) {
|
2018-08-09 19:24:33 +03:00
|
|
|
return Some(operation_data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
2018-08-13 18:59:52 +03:00
|
|
|
let operation_data = if !namespace {
|
|
|
|
self
|
|
|
|
.interfaces
|
|
|
|
.get(self_name)
|
|
|
|
.and_then(|interface_data| interface_data.operations.get(operation_id))
|
|
|
|
.unwrap_or_else(||
|
|
|
|
get_operation_data(self, operation_id, self_name, self_name)
|
|
|
|
.expect(&format!("not found operation {:?} in interface {}", operation_id, self_name))
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
self
|
|
|
|
.namespaces
|
|
|
|
.get(self_name)
|
|
|
|
.and_then(|interface_data| interface_data.operations.get(operation_id))
|
|
|
|
.expect(&format!("not found operation {:?} in namespace {}", operation_id, self_name))
|
2018-08-03 14:39:33 -07:00
|
|
|
};
|
2018-08-09 19:24:33 +03:00
|
|
|
|
2018-08-03 14:39:33 -07:00
|
|
|
let mut names = Vec::with_capacity(arguments.len());
|
2018-08-13 18:59:52 +03:00
|
|
|
for argument in arguments {
|
|
|
|
match argument {
|
|
|
|
Argument::Single(single) => names.push(single.identifier.0),
|
|
|
|
Argument::Variadic(variadic) => names.push(variadic.identifier.0),
|
2018-08-03 14:39:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
(
|
2018-08-09 19:24:33 +03:00
|
|
|
operation_data.overloaded,
|
|
|
|
*operation_data
|
2018-08-03 14:39:33 -07:00
|
|
|
.argument_names_same
|
|
|
|
.get(&names)
|
|
|
|
.unwrap_or(&false)
|
|
|
|
)
|
2018-07-30 17:41:22 +03:00
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Create a wasm-bindgen getter method, if possible.
|
2018-07-10 22:59:59 -07:00
|
|
|
pub fn create_getter(
|
|
|
|
&self,
|
|
|
|
name: &str,
|
2018-08-29 15:00:58 -07:00
|
|
|
ty: &weedle::types::Type<'src>,
|
2018-07-10 22:59:59 -07:00
|
|
|
self_name: &str,
|
|
|
|
is_static: bool,
|
|
|
|
is_structural: bool,
|
|
|
|
catch: bool,
|
2018-08-28 15:19:31 -07:00
|
|
|
global: bool,
|
2018-08-29 10:28:04 -07:00
|
|
|
) -> Option<backend::ast::ImportFunction> {
|
|
|
|
let kind = backend::ast::OperationKind::Getter(Some(raw_ident(name)));
|
|
|
|
let kind = self.import_function_kind(self_name, global, is_static, kind);
|
|
|
|
let ret = ty.to_idl_type(self)?;
|
|
|
|
self.create_one_function(
|
|
|
|
&name,
|
|
|
|
&name.to_snake_case(),
|
|
|
|
None.into_iter(),
|
|
|
|
&ret,
|
|
|
|
kind,
|
|
|
|
is_structural,
|
|
|
|
catch,
|
|
|
|
Some(format!("The `{}` getter\n\n{}", name, mdn_doc(self_name, Some(name))))
|
|
|
|
)
|
2018-07-10 22:59:59 -07:00
|
|
|
}
|
2018-06-14 22:48:32 -07:00
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Create a wasm-bindgen setter method, if possible.
|
2018-07-10 22:59:59 -07:00
|
|
|
pub fn create_setter(
|
|
|
|
&self,
|
|
|
|
name: &str,
|
2018-08-29 15:00:58 -07:00
|
|
|
field_ty: weedle::types::Type<'src>,
|
2018-07-10 22:59:59 -07:00
|
|
|
self_name: &str,
|
|
|
|
is_static: bool,
|
|
|
|
is_structural: bool,
|
|
|
|
catch: bool,
|
2018-08-28 15:19:31 -07:00
|
|
|
global: bool,
|
2018-08-29 10:28:04 -07:00
|
|
|
) -> Option<backend::ast::ImportFunction> {
|
|
|
|
let kind = backend::ast::OperationKind::Setter(Some(raw_ident(name)));
|
|
|
|
let kind = self.import_function_kind(self_name, global, is_static, kind);
|
|
|
|
let field_ty = field_ty.to_idl_type(self)?;
|
|
|
|
self.create_one_function(
|
|
|
|
&name,
|
|
|
|
&format!("set_{}", name).to_snake_case(),
|
|
|
|
Some((name, &field_ty)).into_iter(),
|
|
|
|
&IdlType::Void,
|
|
|
|
kind,
|
|
|
|
is_structural,
|
|
|
|
catch,
|
|
|
|
Some(format!("The `{}` setter\n\n{}", name, mdn_doc(self_name, Some(name))))
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2018-08-29 18:27:49 -07:00
|
|
|
pub fn import_function_kind(
|
2018-08-29 10:28:04 -07:00
|
|
|
&self,
|
|
|
|
self_name: &str,
|
|
|
|
global: bool,
|
|
|
|
is_static: bool,
|
|
|
|
operation_kind: backend::ast::OperationKind,
|
|
|
|
) -> backend::ast::ImportFunctionKind {
|
2018-08-28 15:19:31 -07:00
|
|
|
let operation = backend::ast::Operation {
|
|
|
|
is_static,
|
2018-08-29 10:28:04 -07:00
|
|
|
kind: operation_kind,
|
2018-08-28 15:19:31 -07:00
|
|
|
};
|
|
|
|
let ty = ident_ty(rust_ident(camel_case_ident(&self_name).as_str()));
|
2018-08-29 10:28:04 -07:00
|
|
|
if global {
|
2018-08-28 15:19:31 -07:00
|
|
|
backend::ast::ImportFunctionKind::ScopedMethod {
|
|
|
|
ty,
|
|
|
|
operation,
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
backend::ast::ImportFunctionKind::Method {
|
|
|
|
class: self_name.to_string(),
|
|
|
|
ty,
|
|
|
|
kind: backend::ast::MethodKind::Operation(operation),
|
|
|
|
}
|
2018-08-29 10:28:04 -07:00
|
|
|
}
|
2018-07-10 22:59:59 -07:00
|
|
|
}
|
2018-08-29 15:00:58 -07:00
|
|
|
|
|
|
|
pub fn create_imports(
|
|
|
|
&self,
|
|
|
|
kind: backend::ast::ImportFunctionKind,
|
|
|
|
id: &OperationId<'src>,
|
|
|
|
data: &OperationData2<'src>,
|
|
|
|
)
|
|
|
|
-> Vec<backend::ast::ImportFunction>
|
|
|
|
{
|
|
|
|
// First up expand all the signatures in `data` into all signatures that
|
|
|
|
// we're going to generate. These signatures will be used to determine
|
|
|
|
// the names for all the various functions.
|
|
|
|
#[derive(Clone)]
|
|
|
|
struct ExpandedSig<'a> {
|
|
|
|
orig: &'a Signature<'a>,
|
|
|
|
args: Vec<IdlType<'a>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut actual_signatures = Vec::new();
|
|
|
|
'outer:
|
|
|
|
for signature in data.signatures.iter() {
|
2018-08-29 17:33:35 -07:00
|
|
|
let real_start = actual_signatures.len();
|
|
|
|
let mut start = real_start;
|
2018-08-29 15:00:58 -07:00
|
|
|
|
|
|
|
// Start off with an empty signature, this'll handle zero-argument
|
|
|
|
// cases and otherwise the loop below will continue to add on to this.
|
|
|
|
actual_signatures.push(ExpandedSig {
|
|
|
|
orig: signature,
|
|
|
|
args: Vec::with_capacity(signature.args.len()),
|
|
|
|
});
|
|
|
|
|
|
|
|
for (i, arg) in signature.args.iter().enumerate() {
|
|
|
|
// If any argument in this signature can't be converted we have
|
|
|
|
// to throw out the entire signature, so revert back to the
|
|
|
|
// beginning and then keep going.
|
|
|
|
let idl_type = match arg.ty.to_idl_type(self) {
|
|
|
|
Some(t) => t,
|
|
|
|
None => {
|
2018-08-29 17:33:35 -07:00
|
|
|
actual_signatures.truncate(real_start);
|
2018-08-29 15:00:58 -07:00
|
|
|
continue 'outer
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-08-29 17:33:35 -07:00
|
|
|
if arg.optional {
|
|
|
|
assert!(signature.args[i..].iter().all(|a| a.optional));
|
|
|
|
let end = actual_signatures.len();
|
|
|
|
for j in start..end {
|
|
|
|
let sig = actual_signatures[j].clone();
|
|
|
|
actual_signatures.push(sig);
|
|
|
|
}
|
|
|
|
start = end;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert!(start < actual_signatures.len());
|
|
|
|
for sig in actual_signatures[start..].iter() {
|
|
|
|
assert_eq!(sig.args.len(), i);
|
|
|
|
}
|
|
|
|
|
2018-08-29 15:00:58 -07:00
|
|
|
// The first arugment gets pushed directly in-place, but all
|
|
|
|
// future expanded arguments will cause new signatures to be
|
|
|
|
// created. If we have an optional argument then we consider the
|
|
|
|
// already existing signature as the "none" case and the flatten
|
|
|
|
// below will produce the "some" case, so we've already
|
|
|
|
// processed the first argument effectively.
|
2018-08-29 17:33:35 -07:00
|
|
|
let mut first = true;
|
|
|
|
let cur = actual_signatures.len();
|
2018-08-29 15:00:58 -07:00
|
|
|
for idl_type in idl_type.flatten() {
|
|
|
|
for j in start..cur {
|
|
|
|
if first {
|
|
|
|
actual_signatures[j].args.push(idl_type.clone());
|
|
|
|
} else {
|
|
|
|
let mut sig = actual_signatures[j].clone();
|
2018-08-29 17:33:35 -07:00
|
|
|
assert_eq!(sig.args.len(), i + 1);
|
2018-08-29 15:00:58 -07:00
|
|
|
sig.args.truncate(i);
|
|
|
|
sig.args.push(idl_type.clone());
|
|
|
|
actual_signatures.push(sig);
|
|
|
|
}
|
|
|
|
}
|
2018-08-29 17:33:35 -07:00
|
|
|
first = false;
|
2018-08-29 15:00:58 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-29 18:27:49 -07:00
|
|
|
let (name, force_structural, force_throws) = match id {
|
|
|
|
// Constructors aren't annotated with `[Throws]` extended attributes
|
|
|
|
// (how could they be, since they themselves are extended
|
|
|
|
// attributes?) so we must conservatively assume that they can
|
|
|
|
// always throw.
|
|
|
|
//
|
|
|
|
// From https://heycam.github.io/webidl/#Constructor (emphasis
|
|
|
|
// mine):
|
|
|
|
//
|
|
|
|
// > The prose definition of a constructor must either return an IDL
|
|
|
|
// > value of a type corresponding to the interface the
|
|
|
|
// > `[Constructor]` extended attribute appears on, **or throw an
|
|
|
|
// > exception**.
|
|
|
|
OperationId::Constructor(_) => ("new", false, true),
|
|
|
|
OperationId::Operation(Some(s)) => (*s, false, false),
|
2018-08-29 15:00:58 -07:00
|
|
|
OperationId::Operation(None) => {
|
|
|
|
warn!("unsupported unnamed operation");
|
|
|
|
return Vec::new()
|
|
|
|
}
|
2018-08-29 18:27:49 -07:00
|
|
|
OperationId::IndexingGetter => ("get", true, false),
|
|
|
|
OperationId::IndexingSetter => ("set", true, false),
|
|
|
|
OperationId::IndexingDeleter => ("delete", true, false),
|
2018-08-29 15:00:58 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
let mut ret = Vec::new();
|
|
|
|
for signature in actual_signatures.iter() {
|
|
|
|
// Ignore signatures with invalid return types
|
|
|
|
let ret_ty = match signature.orig.ret.to_idl_type(self) {
|
|
|
|
Some(ty) => ty,
|
|
|
|
None => continue,
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut rust_name = name.to_snake_case();
|
|
|
|
let mut first = true;
|
|
|
|
for (i, arg) in signature.args.iter().enumerate() {
|
|
|
|
// Find out if any other known signature either has the same
|
|
|
|
// name for this argument or a different type for this argument.
|
|
|
|
let mut any_same_name = false;
|
|
|
|
let mut any_different_type = false;
|
2018-08-29 17:33:35 -07:00
|
|
|
let mut any_different = false;
|
2018-08-29 15:00:58 -07:00
|
|
|
let arg_name = signature.orig.args[i].name;
|
|
|
|
for other in actual_signatures.iter() {
|
|
|
|
if other.orig.args.get(i).map(|s| s.name) == Some(arg_name) {
|
|
|
|
if !ptr::eq(signature, other) {
|
|
|
|
any_same_name = true;
|
|
|
|
}
|
|
|
|
}
|
2018-08-29 17:33:35 -07:00
|
|
|
if let Some(other) = other.args.get(i) {
|
|
|
|
if other != arg {
|
|
|
|
any_different_type = true;
|
|
|
|
any_different = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
any_different = true;
|
2018-08-29 15:00:58 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If all signatures have the exact same type for this argument,
|
|
|
|
// then there's nothing to disambiguate so we don't modify the
|
|
|
|
// name.
|
2018-08-29 17:33:35 -07:00
|
|
|
if !any_different {
|
2018-08-29 15:00:58 -07:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
if first {
|
|
|
|
rust_name.push_str("_with_");
|
|
|
|
first = false;
|
|
|
|
} else {
|
|
|
|
rust_name.push_str("_and_");
|
|
|
|
}
|
|
|
|
|
|
|
|
// If this name of the argument for this signature is unique
|
|
|
|
// then that's a bit more human readable so we include it in the
|
|
|
|
// method name. Otherwise the type name should disambiguate
|
|
|
|
// correctly.
|
2018-08-29 17:33:35 -07:00
|
|
|
if !any_same_name || !any_different_type {
|
2018-08-29 15:00:58 -07:00
|
|
|
rust_name.push_str(&arg_name.to_snake_case());
|
|
|
|
} else {
|
|
|
|
arg.push_type_name(&mut rust_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ret.extend(self.create_one_function(
|
|
|
|
name,
|
|
|
|
&rust_name,
|
|
|
|
signature.args.iter()
|
|
|
|
.zip(&signature.orig.args)
|
|
|
|
.map(|(ty, orig_arg)| (orig_arg.name, ty)),
|
|
|
|
&ret_ty,
|
|
|
|
kind.clone(),
|
2018-08-29 17:33:35 -07:00
|
|
|
force_structural || is_structural(&signature.orig.attrs),
|
2018-08-29 18:27:49 -07:00
|
|
|
force_throws || throws(&signature.orig.attrs),
|
2018-08-29 15:00:58 -07:00
|
|
|
None,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2018-06-14 22:48:32 -07:00
|
|
|
}
|
2018-07-02 20:35:05 -07:00
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Search for an attribute by name in some webidl object's attributes.
|
2018-08-03 14:39:33 -07:00
|
|
|
fn has_named_attribute(list: &Option<ExtendedAttributeList>, attribute: &str) -> bool {
|
|
|
|
let list = match list {
|
|
|
|
Some(list) => list,
|
|
|
|
None => return false,
|
|
|
|
};
|
|
|
|
list.body.list.iter().any(|attr| match attr {
|
|
|
|
ExtendedAttribute::NoArgs(name) => (name.0).0 == attribute,
|
2018-07-10 22:59:59 -07:00
|
|
|
_ => false,
|
2018-07-02 20:35:05 -07:00
|
|
|
})
|
2018-07-07 10:20:31 -07:00
|
|
|
}
|
2018-07-09 16:35:25 -07:00
|
|
|
|
2018-07-17 22:23:17 -07:00
|
|
|
/// ChromeOnly is for things that are only exposed to privileged code in Firefox.
|
2018-08-03 14:39:33 -07:00
|
|
|
pub fn is_chrome_only(ext_attrs: &Option<ExtendedAttributeList>) -> bool {
|
2018-07-17 22:23:17 -07:00
|
|
|
has_named_attribute(ext_attrs, "ChromeOnly")
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Whether a webidl object is marked as a no interface object.
|
2018-08-03 14:39:33 -07:00
|
|
|
pub fn is_no_interface_object(ext_attrs: &Option<ExtendedAttributeList>) -> bool {
|
2018-07-17 22:23:17 -07:00
|
|
|
has_named_attribute(ext_attrs, "NoInterfaceObject")
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Whether a webidl object is marked as structural.
|
2018-08-03 14:39:33 -07:00
|
|
|
pub fn is_structural(attrs: &Option<ExtendedAttributeList>) -> bool {
|
|
|
|
has_named_attribute(attrs, "Unforgeable")
|
2018-07-09 16:35:25 -07:00
|
|
|
}
|
2018-07-11 11:07:03 -07:00
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Whether a webidl object is marked as throwing.
|
2018-08-03 14:39:33 -07:00
|
|
|
pub fn throws(attrs: &Option<ExtendedAttributeList>) -> bool {
|
|
|
|
has_named_attribute(attrs, "Throws")
|
2018-07-10 22:59:59 -07:00
|
|
|
}
|
|
|
|
|
2018-07-25 11:42:01 +01:00
|
|
|
/// Create a syn `pub` token
|
2018-07-10 22:59:59 -07:00
|
|
|
pub fn public() -> syn::Visibility {
|
|
|
|
syn::Visibility::Public(syn::VisPublic {
|
|
|
|
pub_token: Default::default(),
|
2018-07-11 11:07:03 -07:00
|
|
|
})
|
|
|
|
}
|