diff options
Diffstat (limited to 'vendor/wasm-bindgen-backend/src/ast.rs')
-rw-r--r-- | vendor/wasm-bindgen-backend/src/ast.rs | 453 |
1 files changed, 453 insertions, 0 deletions
diff --git a/vendor/wasm-bindgen-backend/src/ast.rs b/vendor/wasm-bindgen-backend/src/ast.rs new file mode 100644 index 000000000..7703e2570 --- /dev/null +++ b/vendor/wasm-bindgen-backend/src/ast.rs @@ -0,0 +1,453 @@ +//! A representation of the Abstract Syntax Tree of a Rust program, +//! with all the added metadata necessary to generate WASM bindings +//! for it. + +use crate::Diagnostic; +use proc_macro2::{Ident, Span}; +use std::hash::{Hash, Hasher}; +use wasm_bindgen_shared as shared; + +/// An abstract syntax tree representing a rust program. Contains +/// extra information for joining up this rust code with javascript. +#[cfg_attr(feature = "extra-traits", derive(Debug))] +#[derive(Default, Clone)] +pub struct Program { + /// rust -> js interfaces + pub exports: Vec<Export>, + /// js -> rust interfaces + pub imports: Vec<Import>, + /// rust enums + pub enums: Vec<Enum>, + /// rust structs + pub structs: Vec<Struct>, + /// custom typescript sections to be included in the definition file + pub typescript_custom_sections: Vec<String>, + /// Inline JS snippets + pub inline_js: Vec<String>, +} + +impl Program { + /// Returns true if the Program is empty + pub fn is_empty(&self) -> bool { + self.exports.is_empty() + && self.imports.is_empty() + && self.enums.is_empty() + && self.structs.is_empty() + && self.typescript_custom_sections.is_empty() + && self.inline_js.is_empty() + } +} + +/// A rust to js interface. Allows interaction with rust objects/functions +/// from javascript. +#[cfg_attr(feature = "extra-traits", derive(Debug))] +#[derive(Clone)] +pub struct Export { + /// Comments extracted from the rust source. + pub comments: Vec<String>, + /// The rust function + pub function: Function, + /// The class name in JS this is attached to + pub js_class: Option<String>, + /// The kind (static, named, regular) + pub method_kind: MethodKind, + /// The type of `self` (either `self`, `&self`, or `&mut self`) + pub method_self: Option<MethodSelf>, + /// The struct name, in Rust, this is attached to + pub rust_class: Option<Ident>, + /// The name of the rust function/method on the rust side. + pub rust_name: Ident, + /// Whether or not this function should be flagged as the wasm start + /// function. + pub start: bool, +} + +/// The 3 types variations of `self`. +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub enum MethodSelf { + /// `self` + ByValue, + /// `&mut self` + RefMutable, + /// `&self` + RefShared, +} + +/// Things imported from a JS module (in an `extern` block) +#[cfg_attr(feature = "extra-traits", derive(Debug))] +#[derive(Clone)] +pub struct Import { + /// The type of module being imported from, if any + pub module: Option<ImportModule>, + /// The namespace to access the item through, if any + pub js_namespace: Option<Vec<String>>, + /// The type of item being imported + pub kind: ImportKind, +} + +/// The possible types of module to import from +#[cfg_attr(feature = "extra-traits", derive(Debug))] +#[derive(Clone)] +pub enum ImportModule { + /// Import from the named module, with relative paths interpreted + Named(String, Span), + /// Import from the named module, without interpreting paths + RawNamed(String, Span), + /// Import from an inline JS snippet + Inline(usize, Span), +} + +impl Hash for ImportModule { + fn hash<H: Hasher>(&self, h: &mut H) { + match self { + ImportModule::Named(name, _) => (1u8, name).hash(h), + ImportModule::Inline(idx, _) => (2u8, idx).hash(h), + ImportModule::RawNamed(name, _) => (3u8, name).hash(h), + } + } +} + +/// The type of item being imported +#[cfg_attr(feature = "extra-traits", derive(Debug))] +#[derive(Clone)] +pub enum ImportKind { + /// Importing a function + Function(ImportFunction), + /// Importing a static value + Static(ImportStatic), + /// Importing a type/class + Type(ImportType), + /// Importing a JS enum + Enum(ImportEnum), +} + +/// A function being imported from JS +#[cfg_attr(feature = "extra-traits", derive(Debug))] +#[derive(Clone)] +pub struct ImportFunction { + /// The full signature of the function + pub function: Function, + /// The name rust code will use + pub rust_name: Ident, + /// The type being returned + pub js_ret: Option<syn::Type>, + /// Whether to catch JS exceptions + pub catch: bool, + /// Whether the function is variadic on the JS side + pub variadic: bool, + /// Whether the function should use structural type checking + pub structural: bool, + /// Causes the Builder (See cli-support::js::binding::Builder) to error out if + /// it finds itself generating code for a function with this signature + pub assert_no_shim: bool, + /// The kind of function being imported + pub kind: ImportFunctionKind, + /// The shim name to use in the generated code. The 'shim' is a function that appears in + /// the generated JS as a wrapper around the actual function to import, performing any + /// necessary conversions (EG adding a try/catch to change a thrown error into a Result) + pub shim: Ident, + /// The doc comment on this import, if one is provided + pub doc_comment: String, +} + +/// The type of a function being imported +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub enum ImportFunctionKind { + /// A class method + Method { + /// The name of the class for this method, in JS + class: String, + /// The type of the class for this method, in Rust + ty: syn::Type, + /// The kind of method this is + kind: MethodKind, + }, + /// A standard function + Normal, +} + +/// The type of a method +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub enum MethodKind { + /// A class constructor + Constructor, + /// Any other kind of method + Operation(Operation), +} + +/// The operation performed by a class method +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub struct Operation { + /// Whether this method is static + pub is_static: bool, + /// The internal kind of this Operation + pub kind: OperationKind, +} + +/// The kind of operation performed by a method +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub enum OperationKind { + /// A standard method, nothing special + Regular, + /// A method for getting the value of the provided Ident + Getter(Option<Ident>), + /// A method for setting the value of the provided Ident + Setter(Option<Ident>), + /// A dynamically intercepted getter + IndexingGetter, + /// A dynamically intercepted setter + IndexingSetter, + /// A dynamically intercepted deleter + IndexingDeleter, +} + +/// The type of a static being imported +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub struct ImportStatic { + /// The visibility of this static in Rust + pub vis: syn::Visibility, + /// The type of static being imported + pub ty: syn::Type, + /// The name of the shim function used to access this static + pub shim: Ident, + /// The name of this static on the Rust side + pub rust_name: Ident, + /// The name of this static on the JS side + pub js_name: String, +} + +/// The metadata for a type being imported +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub struct ImportType { + /// The visibility of this type in Rust + pub vis: syn::Visibility, + /// The name of this type on the Rust side + pub rust_name: Ident, + /// The name of this type on the JS side + pub js_name: String, + /// The custom attributes to apply to this type + pub attrs: Vec<syn::Attribute>, + /// The TS definition to generate for this type + pub typescript_type: Option<String>, + /// The doc comment applied to this type, if one exists + pub doc_comment: Option<String>, + /// The name of the shim to check instanceof for this type + pub instanceof_shim: String, + /// The name of the remote function to use for the generated is_type_of + pub is_type_of: Option<syn::Expr>, + /// The list of classes this extends, if any + pub extends: Vec<syn::Path>, + /// A custom prefix to add and attempt to fall back to, if the type isn't found + pub vendor_prefixes: Vec<Ident>, + /// If present, don't generate a `Deref` impl + pub no_deref: bool, +} + +/// The metadata for an Enum being imported +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub struct ImportEnum { + /// The Rust enum's visibility + pub vis: syn::Visibility, + /// The Rust enum's identifiers + pub name: Ident, + /// The Rust identifiers for the variants + pub variants: Vec<Ident>, + /// The JS string values of the variants + pub variant_values: Vec<String>, + /// Attributes to apply to the Rust enum + pub rust_attrs: Vec<syn::Attribute>, +} + +/// Information about a function being imported or exported +#[cfg_attr(feature = "extra-traits", derive(Debug))] +#[derive(Clone)] +pub struct Function { + /// The name of the function + pub name: String, + /// The span of the function's name in Rust code + pub name_span: Span, + /// Whether the function has a js_name attribute + pub renamed_via_js_name: bool, + /// The arguments to the function + pub arguments: Vec<syn::PatType>, + /// The return type of the function, if provided + pub ret: Option<syn::Type>, + /// Any custom attributes being applied to the function + pub rust_attrs: Vec<syn::Attribute>, + /// The visibility of this function in Rust + pub rust_vis: syn::Visibility, + /// Whether this is an `async` function + pub r#async: bool, + /// Whether to generate a typescript definition for this function + pub generate_typescript: bool, + /// Whether this is a function with a variadict parameter + pub variadic: bool, +} + +/// Information about a Struct being exported +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub struct Struct { + /// The name of the struct in Rust code + pub rust_name: Ident, + /// The name of the struct in JS code + pub js_name: String, + /// All the fields of this struct to export + pub fields: Vec<StructField>, + /// The doc comments on this struct, if provided + pub comments: Vec<String>, + /// Whether this struct is inspectable (provides toJSON/toString properties to JS) + pub is_inspectable: bool, + /// Whether to generate a typescript definition for this struct + pub generate_typescript: bool, +} + +/// The field of a struct +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub struct StructField { + /// The name of the field in Rust code + pub rust_name: syn::Member, + /// The name of the field in JS code + pub js_name: String, + /// The name of the struct this field is part of + pub struct_name: Ident, + /// Whether this value is read-only to JS + pub readonly: bool, + /// The type of this field + pub ty: syn::Type, + /// The name of the getter shim for this field + pub getter: Ident, + /// The name of the setter shim for this field + pub setter: Ident, + /// The doc comments on this field, if any + pub comments: Vec<String>, + /// Whether to generate a typescript definition for this field + pub generate_typescript: bool, + /// Whether to use .clone() in the auto-generated getter for this field + pub getter_with_clone: bool, +} + +/// Information about an Enum being exported +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub struct Enum { + /// The name of this enum in Rust code + pub rust_name: Ident, + /// The name of this enum in JS code + pub js_name: String, + /// The variants provided by this enum + pub variants: Vec<Variant>, + /// The doc comments on this enum, if any + pub comments: Vec<String>, + /// The value to use for a `none` variant of the enum + pub hole: u32, + /// Whether to generate a typescript definition for this enum + pub generate_typescript: bool, +} + +/// The variant of an enum +#[cfg_attr(feature = "extra-traits", derive(Debug, PartialEq, Eq))] +#[derive(Clone)] +pub struct Variant { + /// The name of this variant + pub name: Ident, + /// The backing value of this variant + pub value: u32, + /// The doc comments on this variant, if any + pub comments: Vec<String>, +} + +/// Unused, the type of an argument to / return from a function +#[derive(Copy, Clone, Debug, PartialEq, Eq)] +pub enum TypeKind { + /// A by-reference arg, EG `&T` + ByRef, + /// A by-mutable-reference arg, EG `&mut T` + ByMutRef, + /// A by-value arg, EG `T` + ByValue, +} + +/// Unused, the location of a type for a function argument (import/export, argument/ret) +#[derive(Copy, Clone, Debug, PartialEq, Eq)] +pub enum TypeLocation { + /// An imported argument (JS side type) + ImportArgument, + /// An imported return + ImportRet, + /// An exported argument (Rust side type) + ExportArgument, + /// An exported return + ExportRet, +} + +impl Export { + /// Mangles a rust -> javascript export, so that the created Ident will be unique over function + /// name and class name, if the function belongs to a javascript class. + pub(crate) fn rust_symbol(&self) -> Ident { + let mut generated_name = String::from("__wasm_bindgen_generated"); + if let Some(class) = &self.js_class { + generated_name.push_str("_"); + generated_name.push_str(class); + } + generated_name.push_str("_"); + generated_name.push_str(&self.function.name.to_string()); + Ident::new(&generated_name, Span::call_site()) + } + + /// This is the name of the shim function that gets exported and takes the raw + /// ABI form of its arguments and converts them back into their normal, + /// "high level" form before calling the actual function. + pub(crate) fn export_name(&self) -> String { + let fn_name = self.function.name.to_string(); + match &self.js_class { + Some(class) => shared::struct_function_export_name(class, &fn_name), + None => shared::free_function_export_name(&fn_name), + } + } +} + +impl ImportKind { + /// Whether this type can be inside an `impl` block. + pub fn fits_on_impl(&self) -> bool { + match *self { + ImportKind::Function(_) => true, + ImportKind::Static(_) => false, + ImportKind::Type(_) => false, + ImportKind::Enum(_) => false, + } + } +} + +impl Function { + /// If the rust object has a `fn xxx(&self) -> MyType` method, get the name for a getter in + /// javascript (in this case `xxx`, so you can write `val = obj.xxx`) + pub fn infer_getter_property(&self) -> &str { + &self.name + } + + /// If the rust object has a `fn set_xxx(&mut self, MyType)` style method, get the name + /// for a setter in javascript (in this case `xxx`, so you can write `obj.xxx = val`) + pub fn infer_setter_property(&self) -> Result<String, Diagnostic> { + let name = self.name.to_string(); + + // Otherwise we infer names based on the Rust function name. + if !name.starts_with("set_") { + bail_span!( + syn::token::Pub(self.name_span), + "setters must start with `set_`, found: {}", + name, + ); + } + Ok(name[4..].to_string()) + } +} |