Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
use crate::decode;
|
|
|
|
use crate::descriptor::{Descriptor, VectorKind};
|
2019-02-25 11:11:30 -08:00
|
|
|
use crate::{Bindgen, EncodeInto, OutputMode};
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
use failure::{bail, Error, ResultExt};
|
2019-03-26 08:00:16 -07:00
|
|
|
use std::collections::{BTreeMap, HashMap, HashSet};
|
2019-02-26 08:44:04 -08:00
|
|
|
use std::env;
|
2019-02-27 12:20:33 -08:00
|
|
|
use std::fs;
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
use walrus::{MemoryId, Module};
|
|
|
|
use wasm_bindgen_wasm_interpreter::Interpreter;
|
2018-04-14 09:13:07 -07:00
|
|
|
|
|
|
|
mod js2rust;
|
2018-10-18 08:43:36 -07:00
|
|
|
use self::js2rust::{ExportedShim, Js2Rust};
|
2018-04-16 13:31:56 -07:00
|
|
|
mod rust2js;
|
|
|
|
use self::rust2js::Rust2Js;
|
2018-09-05 23:59:49 -07:00
|
|
|
mod closures;
|
2018-01-29 21:20:38 -08:00
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
pub struct Context<'a> {
|
2018-01-29 21:20:38 -08:00
|
|
|
pub globals: String,
|
|
|
|
pub imports: String,
|
2018-09-28 13:17:37 -07:00
|
|
|
pub imports_post: String,
|
2018-03-29 01:47:44 -07:00
|
|
|
pub footer: String,
|
2018-01-29 21:20:38 -08:00
|
|
|
pub typescript: String,
|
2019-01-14 13:11:07 -08:00
|
|
|
pub exposed_globals: Option<HashSet<&'static str>>,
|
2018-02-06 08:58:15 -08:00
|
|
|
pub required_internal_exports: HashSet<&'static str>,
|
2018-08-26 15:43:33 -07:00
|
|
|
pub imported_functions: HashSet<&'a str>,
|
|
|
|
pub imported_statics: HashSet<&'a str>,
|
2018-01-29 21:20:38 -08:00
|
|
|
pub config: &'a Bindgen,
|
|
|
|
pub module: &'a mut Module,
|
2018-11-28 09:25:51 -08:00
|
|
|
pub start: Option<String>,
|
2018-07-30 10:50:43 -07:00
|
|
|
|
|
|
|
/// A map which maintains a list of what identifiers we've imported and what
|
|
|
|
/// they're named locally.
|
|
|
|
///
|
|
|
|
/// The `Option<String>` key is the module that identifiers were imported
|
|
|
|
/// from, `None` being the global module. The second key is a map of
|
|
|
|
/// identifiers we've already imported from the module to what they're
|
|
|
|
/// called locally.
|
2019-02-25 11:11:30 -08:00
|
|
|
pub imported_names: HashMap<ImportModule<'a>, HashMap<&'a str, String>>,
|
2018-07-30 10:50:43 -07:00
|
|
|
|
|
|
|
/// A set of all imported identifiers to the number of times they've been
|
|
|
|
/// imported, used to generate new identifiers.
|
|
|
|
pub imported_identifiers: HashMap<String, usize>,
|
|
|
|
|
2018-11-12 11:59:13 -08:00
|
|
|
/// A map of all imported shim functions which can actually be directly
|
|
|
|
/// imported from the containing module. The mapping here maps to a tuple,
|
|
|
|
/// where the first element is the module to import from and the second
|
|
|
|
/// element is the name to import from the module.
|
|
|
|
///
|
|
|
|
/// Note that for `direct_imports` no shims are generated in JS that
|
|
|
|
/// wasm-bindgen emits.
|
|
|
|
pub direct_imports: HashMap<&'a str, (&'a str, &'a str)>,
|
|
|
|
|
2019-03-21 14:37:44 -07:00
|
|
|
pub exported_classes: Option<BTreeMap<String, ExportedClass>>,
|
2018-04-04 08:24:19 -07:00
|
|
|
pub function_table_needed: bool,
|
2018-08-19 17:07:30 -07:00
|
|
|
pub interpreter: &'a mut Interpreter,
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
pub memory: MemoryId,
|
2018-10-18 08:43:36 -07:00
|
|
|
|
2019-02-25 11:11:30 -08:00
|
|
|
/// A map of all local modules we've found, from the identifier they're
|
|
|
|
/// known as to their actual JS contents.
|
|
|
|
pub local_modules: HashMap<&'a str, &'a str>,
|
|
|
|
|
2019-03-05 14:53:14 -08:00
|
|
|
/// A map of how many snippets we've seen from each unique crate identifier,
|
|
|
|
/// used to number snippets correctly when writing them to the filesystem
|
|
|
|
/// when there's multiple snippets within one crate that aren't all part of
|
|
|
|
/// the same `Program`.
|
|
|
|
pub snippet_offsets: HashMap<&'a str, usize>,
|
2019-02-25 11:11:30 -08:00
|
|
|
|
2019-02-27 12:20:33 -08:00
|
|
|
/// All package.json dependencies we've learned about so far
|
|
|
|
pub package_json_read: HashSet<&'a str>,
|
2019-02-28 11:37:08 -08:00
|
|
|
|
|
|
|
/// A map of the name of npm dependencies we've loaded so far to the path
|
|
|
|
/// they're defined in as well as their version specification.
|
2019-02-27 12:20:33 -08:00
|
|
|
pub npm_dependencies: HashMap<String, (&'a str, String)>,
|
|
|
|
|
2018-10-18 08:43:36 -07:00
|
|
|
pub anyref: wasm_bindgen_anyref_xform::Context,
|
2018-02-07 16:41:33 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct ExportedClass {
|
2018-06-15 11:20:56 -05:00
|
|
|
comments: String,
|
2018-04-20 10:56:10 -07:00
|
|
|
contents: String,
|
|
|
|
typescript: String,
|
2018-09-21 15:45:31 -07:00
|
|
|
has_constructor: bool,
|
|
|
|
wrap_needed: bool,
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
pub struct SubContext<'a, 'b: 'a> {
|
2018-08-26 15:43:33 -07:00
|
|
|
pub program: &'b decode::Program<'b>,
|
2018-02-06 15:52:44 -08:00
|
|
|
pub cx: &'a mut Context<'b>,
|
2018-08-26 15:43:33 -07:00
|
|
|
pub vendor_prefixes: HashMap<&'b str, Vec<&'b str>>,
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-01-29 21:20:38 -08:00
|
|
|
|
2018-11-08 12:31:39 -08:00
|
|
|
pub enum ImportTarget {
|
|
|
|
Function(String),
|
|
|
|
Method(String),
|
|
|
|
Constructor(String),
|
|
|
|
StructuralMethod(String),
|
|
|
|
StructuralGetter(Option<String>, String),
|
|
|
|
StructuralSetter(Option<String>, String),
|
|
|
|
StructuralIndexingGetter(Option<String>),
|
|
|
|
StructuralIndexingSetter(Option<String>),
|
|
|
|
StructuralIndexingDeleter(Option<String>),
|
|
|
|
}
|
|
|
|
|
2018-11-12 11:59:13 -08:00
|
|
|
/// Return value of `determine_import` which is where we look at an imported
|
|
|
|
/// function AST and figure out where it's actually being imported from
|
|
|
|
/// (performing some validation checks and whatnot).
|
|
|
|
enum Import<'a> {
|
|
|
|
/// An item is imported from the global scope. The `name` is what's imported
|
|
|
|
/// and the optional `field` is the field on that item we're importing.
|
|
|
|
Global {
|
|
|
|
name: &'a str,
|
|
|
|
field: Option<&'a str>,
|
|
|
|
},
|
|
|
|
/// Same as `Global`, except the `name` is imported via an ESM import from
|
|
|
|
/// the specified `module` path.
|
|
|
|
Module {
|
|
|
|
module: &'a str,
|
|
|
|
name: &'a str,
|
|
|
|
field: Option<&'a str>,
|
|
|
|
},
|
2019-02-25 11:11:30 -08:00
|
|
|
/// Same as `Module`, except we're importing from a local module defined in
|
|
|
|
/// a local JS snippet.
|
|
|
|
LocalModule {
|
|
|
|
module: &'a str,
|
|
|
|
name: &'a str,
|
|
|
|
field: Option<&'a str>,
|
|
|
|
},
|
|
|
|
/// Same as `Module`, except we're importing from an `inline_js` attribute
|
|
|
|
InlineJs {
|
2019-03-05 14:53:14 -08:00
|
|
|
unique_crate_identifier: &'a str,
|
|
|
|
snippet_idx_in_crate: usize,
|
2019-02-25 11:11:30 -08:00
|
|
|
name: &'a str,
|
|
|
|
field: Option<&'a str>,
|
|
|
|
},
|
2018-11-12 11:59:13 -08:00
|
|
|
/// A global import which may have a number of vendor prefixes associated
|
|
|
|
/// with it, like `webkitAudioPrefix`. The `name` is the name to test
|
|
|
|
/// whether it's prefixed.
|
|
|
|
VendorPrefixed {
|
|
|
|
name: &'a str,
|
|
|
|
prefixes: Vec<&'a str>,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2018-11-29 18:15:36 -08:00
|
|
|
const INITIAL_HEAP_VALUES: &[&str] = &["undefined", "null", "true", "false"];
|
|
|
|
// Must be kept in sync with `src/lib.rs` of the `wasm-bindgen` crate
|
|
|
|
const INITIAL_HEAP_OFFSET: usize = 32;
|
2018-07-19 14:44:23 -05:00
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
impl<'a> Context<'a> {
|
2019-01-14 13:09:05 -08:00
|
|
|
fn should_write_global(&mut self, name: &'static str) -> bool {
|
2019-01-14 13:11:07 -08:00
|
|
|
self.exposed_globals.as_mut().unwrap().insert(name)
|
2019-01-14 13:09:05 -08:00
|
|
|
}
|
|
|
|
|
2018-06-15 11:20:56 -05:00
|
|
|
fn export(&mut self, name: &str, contents: &str, comments: Option<String>) {
|
2018-04-03 13:12:28 -07:00
|
|
|
let contents = contents.trim();
|
2018-06-15 11:20:56 -05:00
|
|
|
if let Some(ref c) = comments {
|
|
|
|
self.globals.push_str(c);
|
2019-02-27 20:09:28 +00:00
|
|
|
self.typescript.push_str(c);
|
2018-06-15 11:20:56 -05:00
|
|
|
}
|
2019-02-25 11:11:30 -08:00
|
|
|
let global = match self.config.mode {
|
|
|
|
OutputMode::Node {
|
|
|
|
experimental_modules: false,
|
|
|
|
} => {
|
|
|
|
if contents.starts_with("class") {
|
|
|
|
format!("{1}\nmodule.exports.{0} = {0};\n", name, contents)
|
|
|
|
} else {
|
|
|
|
format!("module.exports.{} = {};\n", name, contents)
|
|
|
|
}
|
2018-05-06 18:00:14 +02:00
|
|
|
}
|
2019-02-25 11:11:30 -08:00
|
|
|
OutputMode::NoModules { .. } => {
|
|
|
|
if contents.starts_with("class") {
|
|
|
|
format!("{1}\n__exports.{0} = {0};\n", name, contents)
|
|
|
|
} else {
|
|
|
|
format!("__exports.{} = {};\n", name, contents)
|
|
|
|
}
|
2018-05-06 18:00:14 +02:00
|
|
|
}
|
2019-03-07 07:33:40 -08:00
|
|
|
OutputMode::Bundler { .. }
|
2019-02-25 11:11:30 -08:00
|
|
|
| OutputMode::Node {
|
|
|
|
experimental_modules: true,
|
|
|
|
} => {
|
|
|
|
if contents.starts_with("function") {
|
|
|
|
format!("export function {}{}\n", name, &contents[8..])
|
|
|
|
} else if contents.starts_with("class") {
|
|
|
|
format!("export {}\n", contents)
|
|
|
|
} else {
|
|
|
|
format!("export const {} = {};\n", name, contents)
|
|
|
|
}
|
|
|
|
}
|
2019-03-07 07:33:40 -08:00
|
|
|
OutputMode::Web => {
|
2019-03-19 11:25:13 -07:00
|
|
|
// In web mode there's no need to export the internals of
|
2019-02-25 11:11:30 -08:00
|
|
|
// wasm-bindgen as we're not using the module itself as the
|
|
|
|
// import object but rather the `__exports` map we'll be
|
|
|
|
// initializing below.
|
|
|
|
let export = if name.starts_with("__wbindgen")
|
|
|
|
|| name.starts_with("__wbg_")
|
|
|
|
|| name.starts_with("__widl_")
|
|
|
|
{
|
|
|
|
""
|
|
|
|
} else {
|
|
|
|
"export "
|
|
|
|
};
|
|
|
|
if contents.starts_with("function") {
|
|
|
|
format!("{}function {}{}\n", export, name, &contents[8..])
|
|
|
|
} else if contents.starts_with("class") {
|
|
|
|
format!("{}{}\n", export, contents)
|
|
|
|
} else {
|
|
|
|
format!("{}const {} = {};\n", export, name, contents)
|
|
|
|
}
|
2018-04-03 13:12:28 -07:00
|
|
|
}
|
|
|
|
};
|
2018-04-14 09:37:49 -07:00
|
|
|
self.global(&global);
|
2019-02-25 11:11:30 -08:00
|
|
|
|
2019-03-07 07:33:40 -08:00
|
|
|
if self.config.mode.web() {
|
2019-02-25 11:11:30 -08:00
|
|
|
self.global(&format!("__exports.{} = {0};", name));
|
|
|
|
}
|
2018-04-03 13:12:28 -07:00
|
|
|
}
|
|
|
|
|
2018-06-27 22:42:34 -07:00
|
|
|
fn require_internal_export(&mut self, name: &'static str) -> Result<(), Error> {
|
2018-04-19 13:08:54 -07:00
|
|
|
if !self.required_internal_exports.insert(name) {
|
2018-06-27 22:42:34 -07:00
|
|
|
return Ok(());
|
2018-04-19 13:08:54 -07:00
|
|
|
}
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
|
|
|
|
if self.module.exports.iter().any(|e| e.name == name) {
|
|
|
|
return Ok(());
|
2018-04-19 13:08:54 -07:00
|
|
|
}
|
|
|
|
|
2018-06-27 22:42:34 -07:00
|
|
|
bail!(
|
|
|
|
"the exported function `{}` is required to generate bindings \
|
|
|
|
but it was not found in the wasm file, perhaps the `std` feature \
|
|
|
|
of the `wasm-bindgen` crate needs to be enabled?",
|
|
|
|
name
|
|
|
|
);
|
2018-04-19 13:08:54 -07:00
|
|
|
}
|
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
pub fn finalize(&mut self, module_name: &str) -> Result<(String, String), Error> {
|
2019-03-05 14:40:05 -08:00
|
|
|
// Wire up all default intrinsics, those which don't have any sort of
|
|
|
|
// dependency on the clsoure/anyref/etc passes. This is where almost all
|
|
|
|
// intrinsics are wired up.
|
|
|
|
self.wire_up_initial_intrinsics()?;
|
|
|
|
|
|
|
|
// Next up, perform our closure rewriting pass. This is where we'll
|
|
|
|
// update invocations of the closure intrinsics we have to instead call
|
|
|
|
// appropriate JS functions which actually create closures.
|
|
|
|
closures::rewrite(self).with_context(|_| "failed to generate internal closure shims")?;
|
|
|
|
|
|
|
|
// Finalize all bindings for JS classes. This is where we'll generate JS
|
|
|
|
// glue for all classes as well as finish up a few final imports like
|
|
|
|
// `__wrap` and such.
|
|
|
|
self.write_classes()?;
|
|
|
|
|
|
|
|
// And now that we're almost ready, run the final "anyref" pass. This is
|
|
|
|
// where we transform a wasm module which doesn't actually use `anyref`
|
|
|
|
// anywhere to using the type internally. The transformation here is
|
|
|
|
// based off all the previous data we've collected so far for each
|
|
|
|
// import/export listed.
|
|
|
|
self.anyref.run(self.module)?;
|
|
|
|
|
|
|
|
// With our transforms finished, we can now wire up the final list of
|
|
|
|
// intrinsics which may depend on the passes run above.
|
|
|
|
self.wire_up_late_intrinsics()?;
|
|
|
|
|
|
|
|
// We're almost done here, so we can delete any internal exports (like
|
|
|
|
// `__wbindgen_malloc`) if none of our JS glue actually needed it.
|
|
|
|
self.unexport_unused_internal_exports();
|
|
|
|
|
|
|
|
// Handle the `start` function, if one was specified. If we're in a
|
|
|
|
// --test mode (such as wasm-bindgen-test-runner) then we skip this
|
|
|
|
// entirely. Otherwise we want to first add a start function to the
|
|
|
|
// `start` section if one is specified.
|
|
|
|
//
|
|
|
|
// Note that once a start function is added, if any, we immediately
|
|
|
|
// un-start it. This is done because we require that the JS glue
|
|
|
|
// initializes first, so we execute wasm startup manually once the JS
|
|
|
|
// glue is all in place.
|
|
|
|
let mut needs_manual_start = false;
|
|
|
|
if self.config.emit_start {
|
|
|
|
self.add_start_function()?;
|
|
|
|
needs_manual_start = self.unstart_start_function();
|
|
|
|
}
|
|
|
|
|
|
|
|
// If our JS glue needs to access the function table, then do so here.
|
|
|
|
// JS closure shim generation may access the function table as an
|
|
|
|
// example, but if there's no closures in the module there's no need to
|
|
|
|
// export it!
|
|
|
|
self.export_table()?;
|
|
|
|
|
|
|
|
// After all we've done, especially
|
|
|
|
// `unexport_unused_internal_exports()`, we probably have a bunch of
|
|
|
|
// garbage in the module that's no longer necessary, so delete
|
|
|
|
// everything that we don't actually need.
|
|
|
|
walrus::passes::gc::run(self.module);
|
|
|
|
|
|
|
|
// Almost there, but before we're done make sure to rewrite the `module`
|
|
|
|
// field of all imports in the wasm module. The field is currently
|
|
|
|
// always `__wbindgen_placeholder__` coming out of rustc, but we need to
|
|
|
|
// update that here to the shim file or an actual ES module.
|
|
|
|
self.rewrite_imports(module_name);
|
|
|
|
|
|
|
|
// We likely made a ton of modifications, so add ourselves to the
|
|
|
|
// producers section!
|
|
|
|
self.update_producers_section();
|
|
|
|
|
|
|
|
// Cause any future calls to `should_write_global` to panic, making sure
|
|
|
|
// we don't ask for items which we can no longer emit.
|
|
|
|
drop(self.exposed_globals.take().unwrap());
|
|
|
|
|
|
|
|
Ok(self.finalize_js(module_name, needs_manual_start))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Performs the task of actually generating the final JS module, be it
|
2019-03-19 11:25:13 -07:00
|
|
|
/// `--target no-modules`, `--target web`, or for bundlers. This is the very
|
|
|
|
/// last step performed in `finalize`.
|
2019-03-05 14:40:05 -08:00
|
|
|
fn finalize_js(&mut self, module_name: &str, needs_manual_start: bool) -> (String, String) {
|
2019-04-01 19:45:53 -03:00
|
|
|
let mut ts = self.typescript.clone();
|
2019-03-05 14:40:05 -08:00
|
|
|
let mut js = String::new();
|
|
|
|
if self.config.mode.no_modules() {
|
|
|
|
js.push_str("(function() {\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Depending on the output mode, generate necessary glue to actually
|
|
|
|
// import the wasm file in one way or another.
|
2019-04-01 19:45:53 -03:00
|
|
|
let mut init = (String::new(), String::new());
|
2019-03-05 14:40:05 -08:00
|
|
|
match &self.config.mode {
|
2019-03-19 11:25:13 -07:00
|
|
|
// In `--target no-modules` mode we need to both expose a name on
|
|
|
|
// the global object as well as generate our own custom start
|
|
|
|
// function.
|
2019-03-05 14:40:05 -08:00
|
|
|
OutputMode::NoModules { global } => {
|
|
|
|
js.push_str("const __exports = {};\n");
|
|
|
|
js.push_str("let wasm;\n");
|
|
|
|
init = self.gen_init(&module_name, needs_manual_start);
|
|
|
|
self.footer.push_str(&format!(
|
|
|
|
"self.{} = Object.assign(init, __exports);\n",
|
|
|
|
global
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
// With normal CommonJS node we need to defer requiring the wasm
|
|
|
|
// until the end so most of our own exports are hooked up
|
|
|
|
OutputMode::Node {
|
|
|
|
experimental_modules: false,
|
|
|
|
} => {
|
|
|
|
self.footer
|
|
|
|
.push_str(&format!("wasm = require('./{}_bg');\n", module_name));
|
|
|
|
if needs_manual_start {
|
|
|
|
self.footer.push_str("wasm.__wbindgen_start();\n");
|
|
|
|
}
|
|
|
|
js.push_str("var wasm;\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
// With Bundlers and modern ES6 support in Node we can simply import
|
|
|
|
// the wasm file as if it were an ES module and let the
|
|
|
|
// bundler/runtime take care of it.
|
2019-03-07 07:33:40 -08:00
|
|
|
OutputMode::Bundler { .. }
|
2019-03-05 14:40:05 -08:00
|
|
|
| OutputMode::Node {
|
|
|
|
experimental_modules: true,
|
|
|
|
} => {
|
2019-04-04 15:53:43 -07:00
|
|
|
self.imports
|
|
|
|
.push_str(&format!("import * as wasm from './{}_bg';\n", module_name));
|
2019-03-05 14:40:05 -08:00
|
|
|
if needs_manual_start {
|
|
|
|
self.footer.push_str("wasm.__wbindgen_start();\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// With a browser-native output we're generating an ES module, but
|
|
|
|
// browsers don't support natively importing wasm right now so we
|
2019-03-19 11:25:13 -07:00
|
|
|
// expose the same initialization function as `--target no-modules`
|
|
|
|
// as the default export of the module.
|
2019-03-07 07:33:40 -08:00
|
|
|
OutputMode::Web => {
|
2019-04-04 15:53:43 -07:00
|
|
|
self.imports_post.push_str("const __exports = {};\n");
|
2019-03-05 14:40:05 -08:00
|
|
|
self.imports_post.push_str("let wasm;\n");
|
|
|
|
init = self.gen_init(&module_name, needs_manual_start);
|
|
|
|
self.footer.push_str("export default init;\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-01 19:45:53 -03:00
|
|
|
let (init_js, init_ts) = init;
|
|
|
|
|
|
|
|
ts.push_str(&init_ts);
|
|
|
|
|
2019-03-05 14:40:05 -08:00
|
|
|
// Emit all the JS for importing all our functionality
|
2019-04-04 15:53:43 -07:00
|
|
|
assert!(
|
|
|
|
!self.config.mode.uses_es_modules() || js.is_empty(),
|
|
|
|
"ES modules require imports to be at the start of the file"
|
|
|
|
);
|
2019-03-05 14:40:05 -08:00
|
|
|
js.push_str(&self.imports);
|
|
|
|
js.push_str("\n");
|
|
|
|
js.push_str(&self.imports_post);
|
|
|
|
js.push_str("\n");
|
|
|
|
|
|
|
|
// Emit all our exports from this module
|
|
|
|
js.push_str(&self.globals);
|
|
|
|
js.push_str("\n");
|
|
|
|
|
|
|
|
// Generate the initialization glue, if there was any
|
2019-04-01 19:45:53 -03:00
|
|
|
js.push_str(&init_js);
|
2019-03-05 14:40:05 -08:00
|
|
|
js.push_str("\n");
|
|
|
|
js.push_str(&self.footer);
|
|
|
|
js.push_str("\n");
|
|
|
|
if self.config.mode.no_modules() {
|
|
|
|
js.push_str("})();\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
while js.contains("\n\n\n") {
|
|
|
|
js = js.replace("\n\n\n", "\n\n");
|
|
|
|
}
|
|
|
|
|
2019-04-01 19:45:53 -03:00
|
|
|
(js, ts)
|
2019-03-05 14:40:05 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
fn wire_up_initial_intrinsics(&mut self) -> Result<(), Error> {
|
2018-04-26 19:03:46 -07:00
|
|
|
self.bind("__wbindgen_string_new", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_string_new",
|
|
|
|
&[],
|
|
|
|
true,
|
|
|
|
);
|
2018-04-26 19:03:46 -07:00
|
|
|
me.expose_get_string_from_wasm();
|
2018-10-18 08:43:36 -07:00
|
|
|
Ok(format!(
|
|
|
|
"function(p, l) {{ return {}; }}",
|
|
|
|
me.add_heap_object("getStringFromWasm(p, l)")
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_number_new", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_number_new",
|
|
|
|
&[],
|
|
|
|
true,
|
|
|
|
);
|
|
|
|
Ok(format!(
|
|
|
|
"function(i) {{ return {}; }}",
|
|
|
|
me.add_heap_object("i")
|
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_number_get", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_number_get",
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
2018-04-26 19:03:46 -07:00
|
|
|
me.expose_uint8_memory();
|
2018-10-18 08:43:36 -07:00
|
|
|
Ok(format!(
|
2018-06-27 22:42:34 -07:00
|
|
|
"
|
2018-10-18 08:43:36 -07:00
|
|
|
function(n, invalid) {{
|
|
|
|
let obj = {};
|
2018-06-15 12:55:37 -05:00
|
|
|
if (typeof(obj) === 'number') return obj;
|
2018-04-26 19:03:46 -07:00
|
|
|
getUint8Memory()[invalid] = 1;
|
|
|
|
return 0;
|
2018-10-18 08:43:36 -07:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-10-18 08:43:36 -07:00
|
|
|
me.get_object("n"),
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_is_null", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_is_null",
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
|
|
|
Ok(format!(
|
|
|
|
"function(i) {{ return {} === null ? 1 : 0; }}",
|
|
|
|
me.get_object("i")
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_is_undefined", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_is_undefined",
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
|
|
|
Ok(format!(
|
|
|
|
"function(i) {{ return {} === undefined ? 1 : 0; }}",
|
|
|
|
me.get_object("i")
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_boolean_get", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_boolean_get",
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
|
|
|
Ok(format!(
|
2018-06-27 22:42:34 -07:00
|
|
|
"
|
2018-10-18 08:43:36 -07:00
|
|
|
function(i) {{
|
|
|
|
let v = {};
|
|
|
|
return typeof(v) === 'boolean' ? (v ? 1 : 0) : 2;
|
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-10-18 08:43:36 -07:00
|
|
|
me.get_object("i"),
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_symbol_new", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_symbol_new",
|
|
|
|
&[],
|
|
|
|
true,
|
|
|
|
);
|
2018-04-26 19:03:46 -07:00
|
|
|
me.expose_get_string_from_wasm();
|
2018-10-18 08:43:36 -07:00
|
|
|
let expr = "ptr === 0 ? Symbol() : Symbol(getStringFromWasm(ptr, len))";
|
|
|
|
Ok(format!(
|
|
|
|
"function(ptr, len) {{ return {}; }}",
|
|
|
|
me.add_heap_object(expr)
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_is_symbol", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_is_symbol",
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
|
|
|
Ok(format!(
|
|
|
|
"function(i) {{ return typeof({}) === 'symbol' ? 1 : 0; }}",
|
|
|
|
me.get_object("i")
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
2018-07-09 16:35:25 -07:00
|
|
|
self.bind("__wbindgen_is_object", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_is_object",
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
|
|
|
Ok(format!(
|
2018-07-09 16:35:25 -07:00
|
|
|
"
|
2018-10-18 08:43:36 -07:00
|
|
|
function(i) {{
|
|
|
|
const val = {};
|
2018-07-09 16:35:25 -07:00
|
|
|
return typeof(val) === 'object' && val !== null ? 1 : 0;
|
2018-10-18 08:43:36 -07:00
|
|
|
}}",
|
|
|
|
me.get_object("i"),
|
2018-07-09 16:35:25 -07:00
|
|
|
))
|
|
|
|
})?;
|
2019-01-09 17:10:24 +00:00
|
|
|
|
2018-07-09 16:35:25 -07:00
|
|
|
self.bind("__wbindgen_is_function", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_is_function",
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
|
|
|
Ok(format!(
|
|
|
|
"function(i) {{ return typeof({}) === 'function' ? 1 : 0; }}",
|
|
|
|
me.get_object("i")
|
2018-07-09 16:35:25 -07:00
|
|
|
))
|
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_is_string", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_is_string",
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
|
|
|
Ok(format!(
|
|
|
|
"function(i) {{ return typeof({}) === 'string' ? 1 : 0; }}",
|
|
|
|
me.get_object("i")
|
2018-07-09 16:35:25 -07:00
|
|
|
))
|
|
|
|
})?;
|
|
|
|
|
2018-04-26 19:03:46 -07:00
|
|
|
self.bind("__wbindgen_string_get", &|me| {
|
|
|
|
me.expose_pass_string_to_wasm()?;
|
|
|
|
me.expose_uint32_memory();
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_string_get",
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
|
|
|
Ok(format!(
|
2018-06-27 22:42:34 -07:00
|
|
|
"
|
2018-10-18 08:43:36 -07:00
|
|
|
function(i, len_ptr) {{
|
|
|
|
let obj = {};
|
2018-06-15 12:55:37 -05:00
|
|
|
if (typeof(obj) !== 'string') return 0;
|
2018-11-13 08:10:05 -08:00
|
|
|
const ptr = passStringToWasm(obj);
|
|
|
|
getUint32Memory()[len_ptr / 4] = WASM_VECTOR_LEN;
|
2018-04-26 19:03:46 -07:00
|
|
|
return ptr;
|
2018-10-18 08:43:36 -07:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-10-18 08:43:36 -07:00
|
|
|
me.get_object("i"),
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
2019-01-10 23:37:12 +00:00
|
|
|
self.bind("__wbindgen_debug_string", &|me| {
|
2019-01-09 17:42:56 +00:00
|
|
|
me.expose_pass_string_to_wasm()?;
|
|
|
|
me.expose_uint32_memory();
|
2018-10-18 08:43:36 -07:00
|
|
|
|
|
|
|
let debug_str = "
|
|
|
|
val => {
|
|
|
|
// primitive types
|
|
|
|
const type = typeof val;
|
|
|
|
if (type == 'number' || type == 'boolean' || val == null) {
|
|
|
|
return `${val}`;
|
|
|
|
}
|
|
|
|
if (type == 'string') {
|
|
|
|
return `\"${val}\"`;
|
|
|
|
}
|
|
|
|
if (type == 'symbol') {
|
|
|
|
const description = val.description;
|
|
|
|
if (description == null) {
|
|
|
|
return 'Symbol';
|
|
|
|
} else {
|
|
|
|
return `Symbol(${description})`;
|
2019-01-10 23:37:12 +00:00
|
|
|
}
|
2018-10-18 08:43:36 -07:00
|
|
|
}
|
|
|
|
if (type == 'function') {
|
|
|
|
const name = val.name;
|
|
|
|
if (typeof name == 'string' && name.length > 0) {
|
|
|
|
return `Function(${name})`;
|
2019-01-10 23:37:12 +00:00
|
|
|
} else {
|
2018-10-18 08:43:36 -07:00
|
|
|
return 'Function';
|
2019-01-10 23:37:12 +00:00
|
|
|
}
|
2018-10-18 08:43:36 -07:00
|
|
|
}
|
|
|
|
// objects
|
|
|
|
if (Array.isArray(val)) {
|
|
|
|
const length = val.length;
|
|
|
|
let debug = '[';
|
|
|
|
if (length > 0) {
|
|
|
|
debug += debug_str(val[0]);
|
2019-01-10 23:37:12 +00:00
|
|
|
}
|
2018-10-18 08:43:36 -07:00
|
|
|
for(let i = 1; i < length; i++) {
|
|
|
|
debug += ', ' + debug_str(val[i]);
|
2019-01-16 10:46:26 +00:00
|
|
|
}
|
2018-10-18 08:43:36 -07:00
|
|
|
debug += ']';
|
|
|
|
return debug;
|
|
|
|
}
|
|
|
|
// Test for built-in
|
|
|
|
const builtInMatches = /\\[object ([^\\]]+)\\]/.exec(toString.call(val));
|
|
|
|
let className;
|
|
|
|
if (builtInMatches.length > 1) {
|
|
|
|
className = builtInMatches[1];
|
|
|
|
} else {
|
|
|
|
// Failed to match the standard '[object ClassName]'
|
|
|
|
return toString.call(val);
|
|
|
|
}
|
|
|
|
if (className == 'Object') {
|
|
|
|
// we're a user defined class or Object
|
|
|
|
// JSON.stringify avoids problems with cycles, and is generally much
|
|
|
|
// easier than looping through ownProperties of `val`.
|
|
|
|
try {
|
|
|
|
return 'Object(' + JSON.stringify(val) + ')';
|
|
|
|
} catch (_) {
|
|
|
|
return 'Object';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// errors
|
|
|
|
if (val instanceof Error) {
|
|
|
|
return `${val.name}: ${val.message}\n${val.stack}`;
|
|
|
|
}
|
|
|
|
// TODO we could test for more things here, like `Set`s and `Map`s.
|
|
|
|
return className;
|
|
|
|
}
|
|
|
|
";
|
|
|
|
Ok(format!(
|
|
|
|
"
|
|
|
|
function(i, len_ptr) {{
|
|
|
|
const debug_str = {};
|
|
|
|
const toString = Object.prototype.toString;
|
|
|
|
const val = {};
|
2019-01-10 23:37:12 +00:00
|
|
|
const debug = debug_str(val);
|
|
|
|
const ptr = passStringToWasm(debug);
|
2019-01-09 17:42:56 +00:00
|
|
|
getUint32Memory()[len_ptr / 4] = WASM_VECTOR_LEN;
|
|
|
|
return ptr;
|
2018-10-18 08:43:36 -07:00
|
|
|
}}
|
2019-01-09 17:42:56 +00:00
|
|
|
",
|
2018-10-18 08:43:36 -07:00
|
|
|
debug_str,
|
|
|
|
me.get_object("i"),
|
2019-01-09 17:42:56 +00:00
|
|
|
))
|
|
|
|
})?;
|
|
|
|
|
2018-04-26 19:03:46 -07:00
|
|
|
self.bind("__wbindgen_cb_drop", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_cb_drop",
|
|
|
|
&[(0, true)],
|
|
|
|
false,
|
|
|
|
);
|
|
|
|
Ok(format!(
|
2018-06-27 22:42:34 -07:00
|
|
|
"
|
2018-10-18 08:43:36 -07:00
|
|
|
function(i) {{
|
|
|
|
const obj = {}.original;
|
|
|
|
if (obj.cnt-- == 1) {{
|
2018-09-24 15:10:58 -07:00
|
|
|
obj.a = 0;
|
|
|
|
return 1;
|
2018-10-18 08:43:36 -07:00
|
|
|
}}
|
2018-09-24 15:10:58 -07:00
|
|
|
return 0;
|
2018-10-18 08:43:36 -07:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-10-18 08:43:36 -07:00
|
|
|
me.take_object("i"),
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_cb_forget", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
Ok(if me.config.anyref {
|
|
|
|
// TODO: we should rewrite this in the anyref xform to not even
|
|
|
|
// call into JS
|
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_cb_drop",
|
|
|
|
&[(0, true)],
|
|
|
|
false,
|
|
|
|
);
|
|
|
|
String::from("function(obj) {}")
|
|
|
|
} else {
|
|
|
|
me.expose_drop_ref();
|
|
|
|
"dropObject".to_string()
|
|
|
|
})
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_json_parse", &|me| {
|
|
|
|
me.expose_get_string_from_wasm();
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_json_parse",
|
|
|
|
&[],
|
|
|
|
true,
|
|
|
|
);
|
|
|
|
let expr = "JSON.parse(getStringFromWasm(ptr, len))";
|
|
|
|
let expr = me.add_heap_object(expr);
|
|
|
|
Ok(format!("function(ptr, len) {{ return {}; }}", expr))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
|
|
|
|
|
|
|
self.bind("__wbindgen_json_serialize", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
me.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
"__wbindgen_json_serialize",
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
2018-04-26 19:03:46 -07:00
|
|
|
me.expose_pass_string_to_wasm()?;
|
|
|
|
me.expose_uint32_memory();
|
2018-10-18 08:43:36 -07:00
|
|
|
Ok(format!(
|
2018-06-27 22:42:34 -07:00
|
|
|
"
|
2018-10-18 08:43:36 -07:00
|
|
|
function(idx, ptrptr) {{
|
|
|
|
const ptr = passStringToWasm(JSON.stringify({}));
|
2018-04-26 19:03:46 -07:00
|
|
|
getUint32Memory()[ptrptr / 4] = ptr;
|
2018-11-13 08:10:05 -08:00
|
|
|
return WASM_VECTOR_LEN;
|
2018-10-18 08:43:36 -07:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-10-18 08:43:36 -07:00
|
|
|
me.get_object("idx"),
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-04-26 19:03:46 -07:00
|
|
|
})?;
|
2018-01-29 21:20:38 -08:00
|
|
|
|
2018-06-01 16:47:45 -05:00
|
|
|
self.bind("__wbindgen_jsval_eq", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
Ok(format!(
|
|
|
|
"function(a, b) {{ return {} === {} ? 1 : 0; }}",
|
|
|
|
me.get_object("a"),
|
|
|
|
me.get_object("b")
|
2018-06-27 22:42:34 -07:00
|
|
|
))
|
2018-06-01 16:47:45 -05:00
|
|
|
})?;
|
|
|
|
|
2018-08-21 10:42:52 -07:00
|
|
|
self.bind("__wbindgen_memory", &|me| {
|
|
|
|
let mem = me.memory();
|
2018-10-18 08:43:36 -07:00
|
|
|
Ok(format!(
|
|
|
|
"function() {{ return {}; }}",
|
|
|
|
me.add_heap_object(mem)
|
|
|
|
))
|
2018-08-21 10:42:52 -07:00
|
|
|
})?;
|
|
|
|
|
2018-10-04 20:00:23 -07:00
|
|
|
self.bind("__wbindgen_module", &|me| {
|
2019-03-07 07:33:40 -08:00
|
|
|
if !me.config.mode.no_modules() && !me.config.mode.web() {
|
2018-11-27 12:07:59 -08:00
|
|
|
bail!(
|
|
|
|
"`wasm_bindgen::module` is currently only supported with \
|
2019-03-19 11:25:13 -07:00
|
|
|
`--target no-modules` and `--target web`"
|
2018-11-27 12:07:59 -08:00
|
|
|
);
|
2018-10-04 20:00:23 -07:00
|
|
|
}
|
|
|
|
Ok(format!(
|
2018-10-18 08:43:36 -07:00
|
|
|
"function() {{ return {}; }}",
|
|
|
|
me.add_heap_object("init.__wbindgen_wasm_module")
|
2018-10-04 20:00:23 -07:00
|
|
|
))
|
|
|
|
})?;
|
|
|
|
|
2019-04-08 07:43:38 -07:00
|
|
|
self.bind("__wbindgen_function_table", &|me| {
|
|
|
|
me.function_table_needed = true;
|
|
|
|
Ok(format!(
|
|
|
|
"function() {{ return {}; }}",
|
|
|
|
me.add_heap_object("wasm.__wbg_function_table")
|
|
|
|
))
|
|
|
|
})?;
|
|
|
|
|
2018-09-17 18:26:45 -07:00
|
|
|
self.bind("__wbindgen_rethrow", &|me| {
|
2018-10-18 08:43:36 -07:00
|
|
|
Ok(format!(
|
|
|
|
"function(idx) {{ throw {}; }}",
|
|
|
|
me.take_object("idx")
|
|
|
|
))
|
2018-09-17 18:26:45 -07:00
|
|
|
})?;
|
|
|
|
|
2019-03-05 14:40:05 -08:00
|
|
|
self.bind("__wbindgen_throw", &|me| {
|
|
|
|
me.expose_get_string_from_wasm();
|
|
|
|
Ok(String::from(
|
|
|
|
"
|
|
|
|
function(ptr, len) {
|
|
|
|
throw new Error(getStringFromWasm(ptr, len));
|
|
|
|
}
|
|
|
|
",
|
|
|
|
))
|
|
|
|
})?;
|
2018-10-18 08:43:36 -07:00
|
|
|
|
2019-03-05 14:40:05 -08:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Provide implementations of remaining intrinsics after initial passes
|
|
|
|
/// have been run on the wasm module.
|
|
|
|
///
|
|
|
|
/// The intrinsics implemented here are added very late in the process or
|
|
|
|
/// otherwise may be overwritten by passes (such as the anyref pass). As a
|
|
|
|
/// result they don't go into the initial list of intrinsics but go just at
|
|
|
|
/// the end.
|
|
|
|
fn wire_up_late_intrinsics(&mut self) -> Result<(), Error> {
|
2018-10-18 08:43:36 -07:00
|
|
|
// After the anyref pass has executed, if this intrinsic is needed then
|
|
|
|
// we expose a function which initializes it
|
|
|
|
self.bind("__wbindgen_init_anyref_table", &|me| {
|
|
|
|
me.expose_anyref_table();
|
|
|
|
Ok(String::from(
|
|
|
|
"function() {
|
|
|
|
const table = wasm.__wbg_anyref_table;
|
|
|
|
const offset = table.grow(4);
|
|
|
|
table.set(offset + 0, undefined);
|
|
|
|
table.set(offset + 1, null);
|
|
|
|
table.set(offset + 2, true);
|
|
|
|
table.set(offset + 3, false);
|
|
|
|
}",
|
|
|
|
))
|
|
|
|
})?;
|
|
|
|
|
|
|
|
// make sure that the anyref pass runs before binding this as anyref may
|
|
|
|
// remove calls to this import and then gc would remove it
|
|
|
|
self.bind("__wbindgen_object_clone_ref", &|me| {
|
|
|
|
me.expose_get_object();
|
|
|
|
me.expose_add_heap_object();
|
|
|
|
Ok(String::from(
|
|
|
|
"
|
|
|
|
function(idx) {
|
|
|
|
return addHeapObject(getObject(idx));
|
|
|
|
}
|
|
|
|
",
|
|
|
|
))
|
|
|
|
})?;
|
|
|
|
|
|
|
|
// like above, make sure anyref runs first and the anyref pass may
|
|
|
|
// remove usages of this.
|
|
|
|
self.bind("__wbindgen_object_drop_ref", &|me| {
|
|
|
|
me.expose_drop_ref();
|
|
|
|
Ok(String::from("function(i) { dropObject(i); }"))
|
|
|
|
})?;
|
|
|
|
|
2019-03-05 14:40:05 -08:00
|
|
|
Ok(())
|
2019-02-25 11:11:30 -08:00
|
|
|
}
|
|
|
|
|
2019-04-01 19:45:53 -03:00
|
|
|
fn ts_for_init_fn(has_memory: bool) -> String {
|
|
|
|
let (memory_doc, memory_param) = if has_memory {
|
|
|
|
(
|
|
|
|
"* @param {WebAssembly.Memory} maybe_memory\n",
|
|
|
|
", maybe_memory: WebAssembly.Memory",
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
("", "")
|
|
|
|
};
|
|
|
|
format!(
|
|
|
|
"\n\
|
|
|
|
/**\n\
|
|
|
|
* If `module_or_path` is {{RequestInfo}}, makes a request and\n\
|
|
|
|
* for everything else, calls `WebAssembly.instantiate` directly.\n\
|
|
|
|
*\n\
|
|
|
|
* @param {{RequestInfo | BufferSource | WebAssembly.Module}} module_or_path\n\
|
|
|
|
{}\
|
|
|
|
*\n\
|
|
|
|
* @returns {{Promise<any>}}\n\
|
|
|
|
*/\n\
|
|
|
|
export function init \
|
|
|
|
(module_or_path: RequestInfo | BufferSource | WebAssembly.Module{}): Promise<any>;
|
|
|
|
",
|
|
|
|
memory_doc, memory_param
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn gen_init(&mut self, module_name: &str, needs_manual_start: bool) -> (String, String) {
|
2019-02-25 11:11:30 -08:00
|
|
|
let mem = self.module.memories.get(self.memory);
|
|
|
|
let (init_memory1, init_memory2) = if mem.import.is_some() {
|
2018-10-04 20:00:23 -07:00
|
|
|
let mut memory = String::from("new WebAssembly.Memory({");
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
memory.push_str(&format!("initial:{}", mem.initial));
|
|
|
|
if let Some(max) = mem.maximum {
|
2018-10-04 20:00:23 -07:00
|
|
|
memory.push_str(&format!(",maximum:{}", max));
|
|
|
|
}
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
if mem.shared {
|
2018-10-04 20:00:23 -07:00
|
|
|
memory.push_str(",shared:true");
|
|
|
|
}
|
|
|
|
memory.push_str("})");
|
2019-02-25 11:11:30 -08:00
|
|
|
self.imports_post.push_str("let memory;\n");
|
|
|
|
(
|
|
|
|
format!("memory = __exports.memory = maybe_memory;"),
|
|
|
|
format!("memory = __exports.memory = {};", memory),
|
2018-04-05 19:50:26 +05:45
|
|
|
)
|
|
|
|
} else {
|
2019-02-25 11:11:30 -08:00
|
|
|
(String::new(), String::new())
|
2018-04-05 19:50:26 +05:45
|
|
|
};
|
2019-04-01 19:45:53 -03:00
|
|
|
let init_memory_arg = if mem.import.is_some() {
|
|
|
|
", maybe_memory"
|
|
|
|
} else {
|
|
|
|
""
|
|
|
|
};
|
2018-01-29 21:20:38 -08:00
|
|
|
|
2019-04-01 19:45:53 -03:00
|
|
|
let ts = Self::ts_for_init_fn(mem.import.is_some());
|
|
|
|
let js = format!(
|
2019-02-25 11:11:30 -08:00
|
|
|
"\
|
2019-04-02 14:15:42 -07:00
|
|
|
function init(module{init_memory_arg}) {{
|
2019-02-25 11:11:30 -08:00
|
|
|
let result;
|
|
|
|
const imports = {{ './{module}': __exports }};
|
2019-04-02 14:15:42 -07:00
|
|
|
if (module instanceof URL || typeof module === 'string' || module instanceof Request) {{
|
2019-02-25 11:11:30 -08:00
|
|
|
{init_memory2}
|
2019-04-02 14:15:42 -07:00
|
|
|
const response = fetch(module);
|
2019-02-25 11:11:30 -08:00
|
|
|
if (typeof WebAssembly.instantiateStreaming === 'function') {{
|
|
|
|
result = WebAssembly.instantiateStreaming(response, imports)
|
|
|
|
.catch(e => {{
|
|
|
|
console.warn(\"`WebAssembly.instantiateStreaming` failed. Assuming this is \
|
|
|
|
because your server does not serve wasm with \
|
|
|
|
`application/wasm` MIME type. Falling back to \
|
|
|
|
`WebAssembly.instantiate` which is slower. Original \
|
|
|
|
error:\\n\", e);
|
|
|
|
return response
|
|
|
|
.then(r => r.arrayBuffer())
|
|
|
|
.then(bytes => WebAssembly.instantiate(bytes, imports));
|
|
|
|
}});
|
|
|
|
}} else {{
|
|
|
|
result = response
|
|
|
|
.then(r => r.arrayBuffer())
|
|
|
|
.then(bytes => WebAssembly.instantiate(bytes, imports));
|
|
|
|
}}
|
2019-03-21 16:39:03 -05:00
|
|
|
}} else {{
|
|
|
|
{init_memory1}
|
2019-04-02 14:15:42 -07:00
|
|
|
result = WebAssembly.instantiate(module, imports)
|
|
|
|
.then(result => {{
|
|
|
|
if (result instanceof WebAssembly.Instance) {{
|
|
|
|
return {{ instance: result, module }};
|
|
|
|
}} else {{
|
|
|
|
return result;
|
|
|
|
}}
|
2019-03-21 16:39:03 -05:00
|
|
|
}});
|
2019-02-25 11:11:30 -08:00
|
|
|
}}
|
|
|
|
return result.then(({{instance, module}}) => {{
|
|
|
|
wasm = instance.exports;
|
|
|
|
init.__wbindgen_wasm_module = module;
|
|
|
|
{start}
|
|
|
|
return wasm;
|
|
|
|
}});
|
|
|
|
}}
|
|
|
|
",
|
2019-04-01 19:45:53 -03:00
|
|
|
init_memory_arg = init_memory_arg,
|
2019-02-25 11:11:30 -08:00
|
|
|
module = module_name,
|
|
|
|
init_memory1 = init_memory1,
|
|
|
|
init_memory2 = init_memory2,
|
|
|
|
start = if needs_manual_start {
|
|
|
|
"wasm.__wbindgen_start();"
|
|
|
|
} else {
|
|
|
|
""
|
|
|
|
},
|
2019-04-01 19:45:53 -03:00
|
|
|
);
|
|
|
|
|
|
|
|
(js, ts)
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-06-27 22:42:34 -07:00
|
|
|
fn bind(
|
|
|
|
&mut self,
|
|
|
|
name: &str,
|
|
|
|
f: &Fn(&mut Self) -> Result<String, Error>,
|
|
|
|
) -> Result<(), Error> {
|
2018-04-26 19:03:46 -07:00
|
|
|
if !self.wasm_import_needed(name) {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
let contents = f(self)
|
2018-06-27 22:42:34 -07:00
|
|
|
.with_context(|_| format!("failed to generate internal JS function `{}`", name))?;
|
2018-06-15 11:20:56 -05:00
|
|
|
self.export(name, &contents, None);
|
2018-04-26 19:03:46 -07:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn write_classes(&mut self) -> Result<(), Error> {
|
2019-01-14 13:11:07 -08:00
|
|
|
for (class, exports) in self.exported_classes.take().unwrap() {
|
2018-04-25 11:42:22 -07:00
|
|
|
self.write_class(&class, &exports)?;
|
2018-04-19 16:49:46 -07:00
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
Ok(())
|
2018-04-19 16:49:46 -07:00
|
|
|
}
|
2018-04-15 01:29:09 +02:00
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn write_class(&mut self, name: &str, class: &ExportedClass) -> Result<(), Error> {
|
2018-04-19 16:49:46 -07:00
|
|
|
let mut dst = format!("class {} {{\n", name);
|
|
|
|
let mut ts_dst = format!("export {}", dst);
|
2018-04-15 01:29:09 +02:00
|
|
|
|
2018-08-15 17:52:26 -07:00
|
|
|
let (mkweakref, freeref) = if self.config.weak_refs {
|
|
|
|
// When weak refs are enabled we use them to automatically free the
|
|
|
|
// contents of an exported rust class when it's gc'd. Note that a
|
|
|
|
// manual `free` function still exists for deterministic
|
|
|
|
// destruction.
|
|
|
|
//
|
|
|
|
// This is implemented by using a `WeakRefGroup` to run finalizers
|
|
|
|
// for all `WeakRef` objects that it creates. Upon construction of
|
|
|
|
// a new wasm object we use `makeRef` with "holdings" of a thunk to
|
|
|
|
// free the wasm instance. Once the `this` (the instance we're
|
|
|
|
// creating) is gc'd then the finalizer will run with the
|
|
|
|
// `WeakRef`, and we'll pull out the `holdings`, our pointer.
|
|
|
|
//
|
|
|
|
// Note, though, that if manual finalization happens we want to
|
|
|
|
// cancel the `WeakRef`-generated finalization, so we retain the
|
|
|
|
// `WeakRef` in a global map. This global map is then used to
|
|
|
|
// `drop()` the `WeakRef` (cancel finalization) whenever it is
|
|
|
|
// finalized.
|
|
|
|
self.expose_cleanup_groups();
|
2018-09-21 15:45:31 -07:00
|
|
|
let mk = format!("addCleanup(this, this.ptr, free{});", name);
|
2018-08-15 17:52:26 -07:00
|
|
|
let free = "
|
|
|
|
CLEANUPS_MAP.get(ptr).drop();
|
|
|
|
CLEANUPS_MAP.delete(ptr);
|
|
|
|
";
|
|
|
|
(mk, free)
|
|
|
|
} else {
|
|
|
|
(String::new(), "")
|
|
|
|
};
|
|
|
|
|
2018-09-21 15:45:31 -07:00
|
|
|
if self.config.debug && !class.has_constructor {
|
|
|
|
dst.push_str(
|
2018-06-27 22:42:34 -07:00
|
|
|
"
|
2018-09-21 15:45:31 -07:00
|
|
|
constructor() {
|
|
|
|
throw new Error('cannot invoke `new` directly');
|
|
|
|
}
|
2018-09-26 08:26:00 -07:00
|
|
|
",
|
2018-09-21 15:45:31 -07:00
|
|
|
);
|
2018-04-19 16:49:46 -07:00
|
|
|
}
|
2018-02-07 16:41:33 -08:00
|
|
|
|
2018-09-21 15:45:31 -07:00
|
|
|
let mut wrap_needed = class.wrap_needed;
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
let new_name = wasm_bindgen_shared::new_function(&name);
|
2018-04-19 16:49:46 -07:00
|
|
|
if self.wasm_import_needed(&new_name) {
|
2018-09-21 15:45:31 -07:00
|
|
|
wrap_needed = true;
|
2018-10-18 08:43:36 -07:00
|
|
|
self.anyref
|
|
|
|
.import_xform("__wbindgen_placeholder__", &new_name, &[], true);
|
|
|
|
let expr = format!("{}.__wrap(ptr)", name);
|
|
|
|
let expr = self.add_heap_object(&expr);
|
|
|
|
let body = format!("function(ptr) {{ return {}; }}", expr);
|
|
|
|
self.export(&new_name, &body, None);
|
2018-02-07 16:41:33 -08:00
|
|
|
}
|
2018-04-19 16:49:46 -07:00
|
|
|
|
2018-09-21 15:45:31 -07:00
|
|
|
if wrap_needed {
|
|
|
|
dst.push_str(&format!(
|
|
|
|
"
|
|
|
|
static __wrap(ptr) {{
|
|
|
|
const obj = Object.create({}.prototype);
|
|
|
|
obj.ptr = ptr;
|
|
|
|
{}
|
|
|
|
return obj;
|
|
|
|
}}
|
|
|
|
",
|
|
|
|
name,
|
|
|
|
mkweakref.replace("this", "obj"),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2018-08-15 17:52:26 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"
|
|
|
|
function free{}(ptr) {{
|
|
|
|
{}
|
|
|
|
wasm.{}(ptr);
|
|
|
|
}}
|
|
|
|
",
|
|
|
|
name,
|
|
|
|
freeref,
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
wasm_bindgen_shared::free_function(&name)
|
2018-08-15 17:52:26 -07:00
|
|
|
));
|
2018-06-27 22:42:34 -07:00
|
|
|
dst.push_str(&format!(
|
|
|
|
"
|
2018-04-19 16:49:46 -07:00
|
|
|
free() {{
|
|
|
|
const ptr = this.ptr;
|
|
|
|
this.ptr = 0;
|
2018-08-20 14:14:55 -07:00
|
|
|
free{}(ptr);
|
2018-04-19 16:49:46 -07:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-08-15 17:52:26 -07:00
|
|
|
name,
|
2018-06-27 22:42:34 -07:00
|
|
|
));
|
2019-02-01 16:07:31 +09:00
|
|
|
ts_dst.push_str(" free(): void;");
|
2018-04-19 16:49:46 -07:00
|
|
|
dst.push_str(&class.contents);
|
|
|
|
ts_dst.push_str(&class.typescript);
|
|
|
|
dst.push_str("}\n");
|
|
|
|
ts_dst.push_str("}\n");
|
|
|
|
|
2018-06-15 11:20:56 -05:00
|
|
|
self.export(&name, &dst, Some(class.comments.clone()));
|
2018-04-19 16:49:46 -07:00
|
|
|
self.typescript.push_str(&ts_dst);
|
2018-04-25 11:42:22 -07:00
|
|
|
|
|
|
|
Ok(())
|
2018-02-07 16:41:33 -08:00
|
|
|
}
|
|
|
|
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
fn export_table(&mut self) -> Result<(), Error> {
|
2018-04-04 08:24:19 -07:00
|
|
|
if !self.function_table_needed {
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
return Ok(());
|
2018-04-04 08:24:19 -07:00
|
|
|
}
|
2019-02-19 13:07:00 -08:00
|
|
|
let id = match self.module.tables.main_function_table()? {
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
Some(id) => id,
|
2019-02-19 13:07:00 -08:00
|
|
|
None => bail!("no function table found in module"),
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
};
|
|
|
|
self.module.exports.add("__wbg_function_table", id);
|
|
|
|
Ok(())
|
2018-04-04 08:24:19 -07:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn rewrite_imports(&mut self, module_name: &str) {
|
2018-04-03 13:29:26 -07:00
|
|
|
for (name, contents) in self._rewrite_imports(module_name) {
|
2018-06-15 11:20:56 -05:00
|
|
|
self.export(&name, &contents, None);
|
2018-04-03 13:29:26 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-27 22:42:34 -07:00
|
|
|
fn _rewrite_imports(&mut self, module_name: &str) -> Vec<(String, String)> {
|
2018-04-03 13:29:26 -07:00
|
|
|
let mut math_imports = Vec::new();
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
for import in self.module.imports.iter_mut() {
|
|
|
|
if import.module == "__wbindgen_placeholder__" {
|
|
|
|
import.module.truncate(0);
|
|
|
|
if let Some((module, name)) = self.direct_imports.get(import.name.as_str()) {
|
|
|
|
import.name.truncate(0);
|
|
|
|
import.module.push_str(module);
|
|
|
|
import.name.push_str(name);
|
2018-11-12 11:59:13 -08:00
|
|
|
} else {
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
import.module.push_str("./");
|
|
|
|
import.module.push_str(module_name);
|
2018-11-12 11:59:13 -08:00
|
|
|
}
|
2018-04-07 13:06:36 +05:45
|
|
|
continue;
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-02-17 16:43:58 -08:00
|
|
|
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
if import.module != "env" {
|
2018-04-07 13:06:36 +05:45
|
|
|
continue;
|
2018-02-17 16:43:58 -08:00
|
|
|
}
|
|
|
|
|
2018-08-20 23:33:29 -07:00
|
|
|
// If memory is imported we'll have exported it from the shim module
|
|
|
|
// so let's import it from there.
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
//
|
|
|
|
// TODO: we should track this is in a more first-class fashion
|
|
|
|
// rather than just matching on strings.
|
|
|
|
if import.name == "memory" {
|
|
|
|
import.module.truncate(0);
|
|
|
|
import.module.push_str("./");
|
|
|
|
import.module.push_str(module_name);
|
2018-09-26 08:26:00 -07:00
|
|
|
continue;
|
2018-08-20 23:33:29 -07:00
|
|
|
}
|
|
|
|
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
let renamed_import = format!("__wbindgen_{}", import.name);
|
2018-02-17 16:43:58 -08:00
|
|
|
let mut bind_math = |expr: &str| {
|
2018-06-27 22:42:34 -07:00
|
|
|
math_imports.push((renamed_import.clone(), format!("function{}", expr)));
|
2018-02-17 16:43:58 -08:00
|
|
|
};
|
|
|
|
|
2018-10-18 08:43:36 -07:00
|
|
|
// Note that since Rust 1.32.0 this is no longer necessary. Imports
|
|
|
|
// of these functions were fixed in rust-lang/rust#54257 and we're
|
|
|
|
// just waiting until pre-1.32.0 compilers are basically no longer
|
|
|
|
// in use to remove this.
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
match import.name.as_str() {
|
2018-03-02 19:20:14 -08:00
|
|
|
"Math_acos" => bind_math("(x) { return Math.acos(x); }"),
|
|
|
|
"Math_asin" => bind_math("(x) { return Math.asin(x); }"),
|
|
|
|
"Math_atan" => bind_math("(x) { return Math.atan(x); }"),
|
|
|
|
"Math_atan2" => bind_math("(x, y) { return Math.atan2(x, y); }"),
|
|
|
|
"Math_cbrt" => bind_math("(x) { return Math.cbrt(x); }"),
|
|
|
|
"Math_cosh" => bind_math("(x) { return Math.cosh(x); }"),
|
|
|
|
"Math_expm1" => bind_math("(x) { return Math.expm1(x); }"),
|
|
|
|
"Math_hypot" => bind_math("(x, y) { return Math.hypot(x, y); }"),
|
|
|
|
"Math_log1p" => bind_math("(x) { return Math.log1p(x); }"),
|
|
|
|
"Math_sinh" => bind_math("(x) { return Math.sinh(x); }"),
|
|
|
|
"Math_tan" => bind_math("(x) { return Math.tan(x); }"),
|
|
|
|
"Math_tanh" => bind_math("(x) { return Math.tanh(x); }"),
|
2018-02-17 16:43:58 -08:00
|
|
|
_ => continue,
|
|
|
|
}
|
|
|
|
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
import.module.truncate(0);
|
|
|
|
import.module.push_str("./");
|
|
|
|
import.module.push_str(module_name);
|
|
|
|
import.name = renamed_import.clone();
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-04-03 13:29:26 -07:00
|
|
|
|
|
|
|
math_imports
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn unexport_unused_internal_exports(&mut self) {
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
let mut to_remove = Vec::new();
|
|
|
|
for export in self.module.exports.iter() {
|
|
|
|
match export.name.as_str() {
|
|
|
|
// These are some internal imports set by LLD but currently
|
|
|
|
// we've got no use case for continuing to export them, so
|
|
|
|
// blacklist them.
|
|
|
|
"__heap_base" | "__data_end" | "__indirect_function_table" => {
|
|
|
|
to_remove.push(export.id());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise only consider our special exports, which all start
|
|
|
|
// with the same prefix which hopefully only we're using.
|
|
|
|
n if n.starts_with("__wbindgen") => {
|
|
|
|
if !self.required_internal_exports.contains(n) {
|
|
|
|
to_remove.push(export.id());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for id in to_remove {
|
2019-02-19 13:07:00 -08:00
|
|
|
self.module.exports.delete(id);
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-01-29 21:20:38 -08:00
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_drop_ref(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("drop_ref") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-11-29 18:15:36 -08:00
|
|
|
self.expose_global_heap();
|
|
|
|
self.expose_global_heap_next();
|
|
|
|
|
|
|
|
// Note that here we check if `idx` shouldn't actually be dropped. This
|
|
|
|
// is due to the fact that `JsValue::null()` and friends can be passed
|
|
|
|
// by value to JS where we'll automatically call this method. Those
|
|
|
|
// constants, however, cannot be dropped. See #1054 for removing this
|
|
|
|
// branch.
|
|
|
|
//
|
|
|
|
// Otherwise the free operation here is pretty simple, just appending to
|
|
|
|
// the linked list of heap slots that are free.
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"
|
2018-11-29 18:15:36 -08:00
|
|
|
function dropObject(idx) {{
|
2018-07-19 14:44:23 -05:00
|
|
|
if (idx < {}) return;
|
2018-11-29 18:15:36 -08:00
|
|
|
heap[idx] = heap_next;
|
|
|
|
heap_next = idx;
|
2018-01-29 21:20:38 -08:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-11-29 18:15:36 -08:00
|
|
|
INITIAL_HEAP_OFFSET + INITIAL_HEAP_VALUES.len(),
|
2018-06-27 22:42:34 -07:00
|
|
|
));
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
|
|
|
|
2018-11-29 18:15:36 -08:00
|
|
|
fn expose_global_heap(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("heap") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-10-18 08:43:36 -07:00
|
|
|
assert!(!self.config.anyref);
|
2018-11-29 18:15:36 -08:00
|
|
|
self.global(&format!("const heap = new Array({});", INITIAL_HEAP_OFFSET));
|
2018-11-30 13:23:41 -08:00
|
|
|
self.global("heap.fill(undefined);");
|
2018-11-29 18:15:36 -08:00
|
|
|
self.global(&format!("heap.push({});", INITIAL_HEAP_VALUES.join(", ")));
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-01-29 21:20:38 -08:00
|
|
|
|
2018-11-29 18:15:36 -08:00
|
|
|
fn expose_global_heap_next(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("heap_next") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-11-29 18:15:36 -08:00
|
|
|
self.expose_global_heap();
|
|
|
|
self.global("let heap_next = heap.length;");
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_get_object(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("get_object") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-11-29 18:15:36 -08:00
|
|
|
self.expose_global_heap();
|
2018-01-29 21:20:38 -08:00
|
|
|
|
2018-11-29 18:15:36 -08:00
|
|
|
// Accessing a heap object is just a simple index operation due to how
|
|
|
|
// the stack/heap are laid out.
|
|
|
|
self.global("function getObject(idx) { return heap[idx]; }");
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_assert_num(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("assert_num") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"
|
2018-02-06 15:52:44 -08:00
|
|
|
function _assertNum(n) {{
|
2018-06-15 12:55:37 -05:00
|
|
|
if (typeof(n) !== 'number') throw new Error('expected a number argument');
|
2018-02-06 15:52:44 -08:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
"
|
2018-06-27 22:42:34 -07:00
|
|
|
));
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-01-29 21:20:38 -08:00
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_assert_bool(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("assert_bool") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"
|
2018-02-06 15:52:44 -08:00
|
|
|
function _assertBoolean(n) {{
|
2018-06-15 12:55:37 -05:00
|
|
|
if (typeof(n) !== 'boolean') {{
|
2018-02-06 15:52:44 -08:00
|
|
|
throw new Error('expected a boolean argument');
|
2018-06-15 12:55:37 -05:00
|
|
|
}}
|
2018-02-06 15:52:44 -08:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
"
|
2018-06-27 22:42:34 -07:00
|
|
|
));
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
|
|
|
|
2018-11-13 08:10:05 -08:00
|
|
|
fn expose_wasm_vector_len(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("wasm_vector_len") {
|
2018-11-13 08:10:05 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
self.global("let WASM_VECTOR_LEN = 0;");
|
|
|
|
}
|
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn expose_pass_string_to_wasm(&mut self) -> Result<(), Error> {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("pass_string_to_wasm") {
|
2018-04-25 11:42:22 -07:00
|
|
|
return Ok(());
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
self.require_internal_export("__wbindgen_malloc")?;
|
2018-11-13 08:10:05 -08:00
|
|
|
self.expose_wasm_vector_len();
|
2018-04-03 12:44:09 -07:00
|
|
|
let debug = if self.config.debug {
|
|
|
|
"
|
2018-06-15 12:55:37 -05:00
|
|
|
if (typeof(arg) !== 'string') throw new Error('expected a string argument');
|
2018-04-03 12:44:09 -07:00
|
|
|
"
|
|
|
|
} else {
|
|
|
|
""
|
|
|
|
};
|
2019-02-20 08:39:46 -08:00
|
|
|
|
2019-03-25 17:42:10 +00:00
|
|
|
// If we are targeting Node.js, it doesn't have `encodeInto` yet
|
|
|
|
// but it does have `Buffer::write` which has similar semantics but
|
|
|
|
// doesn't require creating intermediate view using `subarray`
|
|
|
|
// and also has `Buffer::byteLength` to calculate size upfront.
|
|
|
|
if self.config.mode.nodejs() {
|
|
|
|
self.expose_node_buffer_memory();
|
|
|
|
|
|
|
|
self.global(&format!(
|
|
|
|
"
|
|
|
|
function passStringToWasm(arg) {{
|
|
|
|
{}
|
|
|
|
const size = Buffer.byteLength(arg);
|
|
|
|
const ptr = wasm.__wbindgen_malloc(size);
|
|
|
|
getNodeBufferMemory().write(arg, ptr, size);
|
|
|
|
WASM_VECTOR_LEN = size;
|
|
|
|
return ptr;
|
|
|
|
}}
|
|
|
|
",
|
|
|
|
debug,
|
|
|
|
));
|
|
|
|
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
self.expose_text_encoder();
|
|
|
|
self.expose_uint8_memory();
|
|
|
|
|
2019-02-20 08:39:46 -08:00
|
|
|
// The first implementation we have for this is to use
|
|
|
|
// `TextEncoder#encode` which has been around for quite some time.
|
|
|
|
let use_encode = format!(
|
2018-06-27 22:42:34 -07:00
|
|
|
"
|
2018-04-03 12:44:09 -07:00
|
|
|
{}
|
2018-09-28 20:01:46 -07:00
|
|
|
const buf = cachedTextEncoder.encode(arg);
|
2018-04-03 12:44:09 -07:00
|
|
|
const ptr = wasm.__wbindgen_malloc(buf.length);
|
2018-04-03 13:20:56 -07:00
|
|
|
getUint8Memory().set(buf, ptr);
|
2018-11-13 08:10:05 -08:00
|
|
|
WASM_VECTOR_LEN = buf.length;
|
|
|
|
return ptr;
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-06-27 22:42:34 -07:00
|
|
|
debug
|
2019-02-20 08:39:46 -08:00
|
|
|
);
|
|
|
|
|
|
|
|
// Another possibility is to use `TextEncoder#encodeInto` which is much
|
|
|
|
// newer and isn't implemented everywhere yet. It's more efficient,
|
|
|
|
// however, becaues it allows us to elide an intermediate allocation.
|
|
|
|
let use_encode_into = format!(
|
|
|
|
"
|
|
|
|
{}
|
|
|
|
let size = arg.length;
|
|
|
|
let ptr = wasm.__wbindgen_malloc(size);
|
|
|
|
let writeOffset = 0;
|
|
|
|
while (true) {{
|
|
|
|
const view = getUint8Memory().subarray(ptr + writeOffset, ptr + size);
|
|
|
|
const {{ read, written }} = cachedTextEncoder.encodeInto(arg, view);
|
2019-04-01 13:08:08 +01:00
|
|
|
if (read === arg.length) {{
|
2019-02-20 08:39:46 -08:00
|
|
|
break;
|
|
|
|
}}
|
2019-04-01 13:08:08 +01:00
|
|
|
arg = arg.substring(read);
|
|
|
|
writeOffset += written;
|
|
|
|
ptr = wasm.__wbindgen_realloc(ptr, size, size += arg.length * 3);
|
2019-02-20 08:39:46 -08:00
|
|
|
}}
|
|
|
|
WASM_VECTOR_LEN = writeOffset;
|
|
|
|
return ptr;
|
|
|
|
",
|
|
|
|
debug
|
|
|
|
);
|
|
|
|
|
2019-02-28 14:57:55 -08:00
|
|
|
// Looks like `encodeInto` doesn't currently work when the memory passed
|
|
|
|
// in is backed by a `SharedArrayBuffer`, so force usage of `encode` if
|
|
|
|
// a `SharedArrayBuffer` is in use.
|
|
|
|
let shared = self.module.memories.get(self.memory).shared;
|
|
|
|
|
2019-02-20 08:39:46 -08:00
|
|
|
match self.config.encode_into {
|
2019-02-28 14:57:55 -08:00
|
|
|
EncodeInto::Always if !shared => {
|
2019-02-20 08:39:46 -08:00
|
|
|
self.require_internal_export("__wbindgen_realloc")?;
|
|
|
|
self.global(&format!(
|
|
|
|
"function passStringToWasm(arg) {{ {} }}",
|
|
|
|
use_encode_into,
|
|
|
|
));
|
|
|
|
}
|
2019-02-28 14:57:55 -08:00
|
|
|
EncodeInto::Test if !shared => {
|
2019-02-20 08:39:46 -08:00
|
|
|
self.require_internal_export("__wbindgen_realloc")?;
|
|
|
|
self.global(&format!(
|
|
|
|
"
|
|
|
|
let passStringToWasm;
|
|
|
|
if (typeof cachedTextEncoder.encodeInto === 'function') {{
|
|
|
|
passStringToWasm = function(arg) {{ {} }};
|
|
|
|
}} else {{
|
|
|
|
passStringToWasm = function(arg) {{ {} }};
|
|
|
|
}}
|
|
|
|
",
|
2019-03-05 14:40:05 -08:00
|
|
|
use_encode_into, use_encode,
|
2019-02-20 08:39:46 -08:00
|
|
|
));
|
|
|
|
}
|
2019-02-28 14:57:55 -08:00
|
|
|
_ => {
|
|
|
|
self.global(&format!(
|
|
|
|
"function passStringToWasm(arg) {{ {} }}",
|
|
|
|
use_encode,
|
|
|
|
));
|
|
|
|
}
|
2019-02-20 08:39:46 -08:00
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
Ok(())
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn expose_pass_array8_to_wasm(&mut self) -> Result<(), Error> {
|
2018-02-16 18:58:37 -08:00
|
|
|
self.expose_uint8_memory();
|
2018-05-05 14:10:25 -07:00
|
|
|
self.pass_array_to_wasm("passArray8ToWasm", "getUint8Memory", 1)
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn expose_pass_array16_to_wasm(&mut self) -> Result<(), Error> {
|
2018-02-16 18:58:37 -08:00
|
|
|
self.expose_uint16_memory();
|
2018-05-05 14:10:25 -07:00
|
|
|
self.pass_array_to_wasm("passArray16ToWasm", "getUint16Memory", 2)
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn expose_pass_array32_to_wasm(&mut self) -> Result<(), Error> {
|
2018-02-16 18:58:37 -08:00
|
|
|
self.expose_uint32_memory();
|
2018-05-05 14:10:25 -07:00
|
|
|
self.pass_array_to_wasm("passArray32ToWasm", "getUint32Memory", 4)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_pass_array64_to_wasm(&mut self) -> Result<(), Error> {
|
|
|
|
self.expose_uint64_memory();
|
|
|
|
self.pass_array_to_wasm("passArray64ToWasm", "getUint64Memory", 8)
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn expose_pass_array_f32_to_wasm(&mut self) -> Result<(), Error> {
|
2018-05-05 14:10:25 -07:00
|
|
|
self.expose_f32_memory();
|
|
|
|
self.pass_array_to_wasm("passArrayF32ToWasm", "getFloat32Memory", 4)
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn expose_pass_array_f64_to_wasm(&mut self) -> Result<(), Error> {
|
2018-05-05 14:10:25 -07:00
|
|
|
self.expose_f64_memory();
|
|
|
|
self.pass_array_to_wasm("passArrayF64ToWasm", "getFloat64Memory", 8)
|
|
|
|
}
|
|
|
|
|
2018-07-20 13:47:49 -05:00
|
|
|
fn expose_pass_array_jsvalue_to_wasm(&mut self) -> Result<(), Error> {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("pass_array_jsvalue") {
|
2018-07-20 13:47:49 -05:00
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
self.require_internal_export("__wbindgen_malloc")?;
|
|
|
|
self.expose_uint32_memory();
|
2018-10-18 08:43:36 -07:00
|
|
|
if self.config.anyref {
|
|
|
|
// TODO: using `addToAnyrefTable` goes back and forth between wasm
|
|
|
|
// and JS a lot, we should have a bulk operation for this.
|
|
|
|
self.expose_add_to_anyref_table()?;
|
|
|
|
self.global(
|
|
|
|
"
|
|
|
|
function passArrayJsValueToWasm(array) {
|
|
|
|
const ptr = wasm.__wbindgen_malloc(array.length * 4);
|
|
|
|
const mem = getUint32Memory();
|
|
|
|
for (let i = 0; i < array.length; i++) {
|
|
|
|
mem[ptr / 4 + i] = addToAnyrefTable(array[i]);
|
|
|
|
}
|
|
|
|
WASM_VECTOR_LEN = array.length;
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
self.expose_add_heap_object();
|
|
|
|
self.global(
|
|
|
|
"
|
|
|
|
function passArrayJsValueToWasm(array) {
|
|
|
|
const ptr = wasm.__wbindgen_malloc(array.length * 4);
|
|
|
|
const mem = getUint32Memory();
|
|
|
|
for (let i = 0; i < array.length; i++) {
|
|
|
|
mem[ptr / 4 + i] = addHeapObject(array[i]);
|
|
|
|
}
|
|
|
|
WASM_VECTOR_LEN = array.length;
|
|
|
|
return ptr;
|
2018-07-20 13:47:49 -05:00
|
|
|
}
|
|
|
|
|
2018-10-18 08:43:36 -07:00
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
2018-07-20 13:47:49 -05:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-06-27 22:42:34 -07:00
|
|
|
fn pass_array_to_wasm(
|
|
|
|
&mut self,
|
|
|
|
name: &'static str,
|
|
|
|
delegate: &str,
|
|
|
|
size: usize,
|
|
|
|
) -> Result<(), Error> {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global(name) {
|
2018-06-27 22:42:34 -07:00
|
|
|
return Ok(());
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
self.require_internal_export("__wbindgen_malloc")?;
|
2018-11-13 08:10:05 -08:00
|
|
|
self.expose_wasm_vector_len();
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"
|
2018-05-05 14:10:25 -07:00
|
|
|
function {}(arg) {{
|
|
|
|
const ptr = wasm.__wbindgen_malloc(arg.length * {size});
|
|
|
|
{}().set(arg, ptr / {size});
|
2018-11-13 08:10:05 -08:00
|
|
|
WASM_VECTOR_LEN = arg.length;
|
|
|
|
return ptr;
|
2018-02-16 18:58:37 -08:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-06-27 22:42:34 -07:00
|
|
|
name,
|
|
|
|
delegate,
|
|
|
|
size = size
|
|
|
|
));
|
2018-04-25 11:42:22 -07:00
|
|
|
Ok(())
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_text_encoder(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("text_encoder") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-09-28 20:01:46 -07:00
|
|
|
self.expose_text_processor("TextEncoder");
|
2018-02-05 16:39:11 -08:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_text_decoder(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("text_decoder") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-02-05 16:39:11 -08:00
|
|
|
}
|
2018-09-28 20:01:46 -07:00
|
|
|
self.expose_text_processor("TextDecoder");
|
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_text_processor(&mut self, s: &str) {
|
2019-02-25 11:11:30 -08:00
|
|
|
if self.config.mode.nodejs_experimental_modules() {
|
2018-11-27 12:07:59 -08:00
|
|
|
self.imports
|
|
|
|
.push_str(&format!("import {{ {} }} from 'util';\n", s));
|
2018-09-28 20:01:46 -07:00
|
|
|
self.global(&format!("let cached{0} = new {0}('utf-8');", s));
|
2019-02-25 11:11:30 -08:00
|
|
|
} else if self.config.mode.nodejs() {
|
2018-09-28 20:01:46 -07:00
|
|
|
self.global(&format!("const {0} = require('util').{0};", s));
|
|
|
|
self.global(&format!("let cached{0} = new {0}('utf-8');", s));
|
2019-02-25 11:11:30 -08:00
|
|
|
} else if !self.config.mode.always_run_in_browser() {
|
2018-11-27 12:07:59 -08:00
|
|
|
self.global(&format!(
|
|
|
|
"
|
2018-09-28 20:01:46 -07:00
|
|
|
const l{0} = typeof {0} === 'undefined' ? \
|
|
|
|
require('util').{0} : {0};\
|
2018-11-27 12:07:59 -08:00
|
|
|
",
|
|
|
|
s
|
|
|
|
));
|
2018-09-28 20:01:46 -07:00
|
|
|
self.global(&format!("let cached{0} = new l{0}('utf-8');", s));
|
2018-10-03 15:54:57 -07:00
|
|
|
} else {
|
|
|
|
self.global(&format!("let cached{0} = new {0}('utf-8');", s));
|
2018-03-28 07:37:56 -07:00
|
|
|
}
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-02-05 16:39:11 -08:00
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_get_string_from_wasm(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("get_string_from_wasm") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-03-28 07:37:56 -07:00
|
|
|
self.expose_text_decoder();
|
|
|
|
self.expose_uint8_memory();
|
2018-08-19 21:33:42 -07:00
|
|
|
|
|
|
|
// Typically we try to give a raw view of memory out to `TextDecoder` to
|
|
|
|
// avoid copying too much data. If, however, a `SharedArrayBuffer` is
|
|
|
|
// being used it looks like that is rejected by `TextDecoder` or
|
|
|
|
// otherwise doesn't work with it. When we detect a shared situation we
|
|
|
|
// use `slice` which creates a new array instead of `subarray` which
|
|
|
|
// creates just a view. That way in shared mode we copy more data but in
|
|
|
|
// non-shared mode there's no need to copy the data except for the
|
|
|
|
// string itself.
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
let is_shared = self.module.memories.get(self.memory).shared;
|
2018-08-19 21:33:42 -07:00
|
|
|
let method = if is_shared { "slice" } else { "subarray" };
|
|
|
|
|
2018-09-26 08:26:00 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"
|
2018-08-19 21:33:42 -07:00
|
|
|
function getStringFromWasm(ptr, len) {{
|
2018-09-28 20:01:46 -07:00
|
|
|
return cachedTextDecoder.decode(getUint8Memory().{}(ptr, ptr + len));
|
2018-08-19 21:33:42 -07:00
|
|
|
}}
|
2018-09-26 08:26:00 -07:00
|
|
|
",
|
|
|
|
method
|
|
|
|
));
|
2018-02-05 16:39:11 -08:00
|
|
|
}
|
|
|
|
|
2018-10-18 08:43:36 -07:00
|
|
|
fn expose_get_array_js_value_from_wasm(&mut self) -> Result<(), Error> {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("get_array_js_value_from_wasm") {
|
2018-10-18 08:43:36 -07:00
|
|
|
return Ok(());
|
2018-02-28 10:56:56 +01:00
|
|
|
}
|
2018-07-04 08:30:18 -07:00
|
|
|
self.expose_uint32_memory();
|
2018-10-18 08:43:36 -07:00
|
|
|
if self.config.anyref {
|
|
|
|
self.expose_anyref_table();
|
|
|
|
self.global(
|
|
|
|
"
|
|
|
|
function getArrayJsValueFromWasm(ptr, len) {
|
|
|
|
const mem = getUint32Memory();
|
|
|
|
const slice = mem.subarray(ptr / 4, ptr / 4 + len);
|
|
|
|
const result = [];
|
|
|
|
for (let i = 0; i < slice.length; i++) {
|
|
|
|
result.push(wasm.__wbg_anyref_table.get(slice[i]));
|
|
|
|
}
|
|
|
|
wasm.__wbindgen_drop_anyref_slice(ptr, len);
|
|
|
|
return result;
|
2018-06-15 23:39:51 -07:00
|
|
|
}
|
2018-10-18 08:43:36 -07:00
|
|
|
",
|
|
|
|
);
|
|
|
|
self.require_internal_export("__wbindgen_drop_anyref_slice")?;
|
|
|
|
} else {
|
|
|
|
self.expose_take_object();
|
|
|
|
self.global(
|
|
|
|
"
|
|
|
|
function getArrayJsValueFromWasm(ptr, len) {
|
|
|
|
const mem = getUint32Memory();
|
|
|
|
const slice = mem.subarray(ptr / 4, ptr / 4 + len);
|
|
|
|
const result = [];
|
|
|
|
for (let i = 0; i < slice.length; i++) {
|
|
|
|
result.push(takeObject(slice[i]));
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
Ok(())
|
2018-02-28 10:56:56 +01:00
|
|
|
}
|
|
|
|
|
2018-02-16 18:58:37 -08:00
|
|
|
fn expose_get_array_i8_from_wasm(&mut self) {
|
2018-05-01 10:06:35 -07:00
|
|
|
self.expose_int8_memory();
|
|
|
|
self.arrayget("getArrayI8FromWasm", "getInt8Memory", 1);
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_get_array_u8_from_wasm(&mut self) {
|
2018-02-18 23:55:34 +01:00
|
|
|
self.expose_uint8_memory();
|
2018-05-01 10:06:35 -07:00
|
|
|
self.arrayget("getArrayU8FromWasm", "getUint8Memory", 1);
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
2018-09-24 13:49:12 -07:00
|
|
|
fn expose_get_clamped_array_u8_from_wasm(&mut self) {
|
|
|
|
self.expose_clamped_uint8_memory();
|
|
|
|
self.arrayget("getClampedArrayU8FromWasm", "getUint8ClampedMemory", 1);
|
|
|
|
}
|
|
|
|
|
2018-02-16 18:58:37 -08:00
|
|
|
fn expose_get_array_i16_from_wasm(&mut self) {
|
2018-05-01 10:06:35 -07:00
|
|
|
self.expose_int16_memory();
|
|
|
|
self.arrayget("getArrayI16FromWasm", "getInt16Memory", 2);
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_get_array_u16_from_wasm(&mut self) {
|
2018-02-18 23:55:34 +01:00
|
|
|
self.expose_uint16_memory();
|
2018-05-01 10:06:35 -07:00
|
|
|
self.arrayget("getArrayU16FromWasm", "getUint16Memory", 2);
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_get_array_i32_from_wasm(&mut self) {
|
2018-05-01 10:06:35 -07:00
|
|
|
self.expose_int32_memory();
|
|
|
|
self.arrayget("getArrayI32FromWasm", "getInt32Memory", 4);
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_get_array_u32_from_wasm(&mut self) {
|
2018-02-18 23:55:34 +01:00
|
|
|
self.expose_uint32_memory();
|
2018-05-01 10:06:35 -07:00
|
|
|
self.arrayget("getArrayU32FromWasm", "getUint32Memory", 4);
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
2018-05-05 14:10:25 -07:00
|
|
|
fn expose_get_array_i64_from_wasm(&mut self) {
|
|
|
|
self.expose_int64_memory();
|
|
|
|
self.arrayget("getArrayI64FromWasm", "getInt64Memory", 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_get_array_u64_from_wasm(&mut self) {
|
|
|
|
self.expose_uint64_memory();
|
|
|
|
self.arrayget("getArrayU64FromWasm", "getUint64Memory", 8);
|
|
|
|
}
|
|
|
|
|
2018-02-16 18:58:37 -08:00
|
|
|
fn expose_get_array_f32_from_wasm(&mut self) {
|
2018-05-01 10:06:35 -07:00
|
|
|
self.expose_f32_memory();
|
|
|
|
self.arrayget("getArrayF32FromWasm", "getFloat32Memory", 4);
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_get_array_f64_from_wasm(&mut self) {
|
2018-05-01 10:06:35 -07:00
|
|
|
self.expose_f64_memory();
|
|
|
|
self.arrayget("getArrayF64FromWasm", "getFloat64Memory", 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn arrayget(&mut self, name: &'static str, mem: &'static str, size: usize) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global(name) {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"
|
2018-05-01 10:06:35 -07:00
|
|
|
function {name}(ptr, len) {{
|
|
|
|
return {mem}().subarray(ptr / {size}, ptr / {size} + len);
|
2018-02-16 18:58:37 -08:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-05-01 10:06:35 -07:00
|
|
|
name = name,
|
|
|
|
mem = mem,
|
|
|
|
size = size,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2019-03-25 17:42:10 +00:00
|
|
|
fn expose_node_buffer_memory(&mut self) {
|
|
|
|
self.memview("getNodeBufferMemory", "Buffer.from");
|
|
|
|
}
|
|
|
|
|
2018-05-01 10:06:35 -07:00
|
|
|
fn expose_int8_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getInt8Memory", "new Int8Array");
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_uint8_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getUint8Memory", "new Uint8Array");
|
2018-05-01 10:06:35 -07:00
|
|
|
}
|
|
|
|
|
2018-09-24 13:49:12 -07:00
|
|
|
fn expose_clamped_uint8_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getUint8ClampedMemory", "new Uint8ClampedArray");
|
2018-09-24 13:49:12 -07:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:06:35 -07:00
|
|
|
fn expose_int16_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getInt16Memory", "new Int16Array");
|
2018-02-06 08:58:15 -08:00
|
|
|
}
|
|
|
|
|
2018-02-16 18:58:37 -08:00
|
|
|
fn expose_uint16_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getUint16Memory", "new Uint16Array");
|
2018-05-01 10:06:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_int32_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getInt32Memory", "new Int32Array");
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_uint32_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getUint32Memory", "new Uint32Array");
|
2018-05-01 10:06:35 -07:00
|
|
|
}
|
|
|
|
|
2018-05-05 14:10:25 -07:00
|
|
|
fn expose_int64_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getInt64Memory", "new BigInt64Array");
|
2018-05-05 14:10:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_uint64_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getUint64Memory", "new BigUint64Array");
|
2018-05-05 14:10:25 -07:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:06:35 -07:00
|
|
|
fn expose_f32_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getFloat32Memory", "new Float32Array");
|
2018-05-01 10:06:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_f64_memory(&mut self) {
|
2019-03-25 17:42:10 +00:00
|
|
|
self.memview("getFloat64Memory", "new Float64Array");
|
2018-05-01 10:06:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fn memview_function(&mut self, t: VectorKind) -> &'static str {
|
|
|
|
match t {
|
|
|
|
VectorKind::String => {
|
|
|
|
self.expose_uint8_memory();
|
|
|
|
"getUint8Memory"
|
|
|
|
}
|
|
|
|
VectorKind::I8 => {
|
|
|
|
self.expose_int8_memory();
|
|
|
|
"getInt8Memory"
|
|
|
|
}
|
|
|
|
VectorKind::U8 => {
|
|
|
|
self.expose_uint8_memory();
|
|
|
|
"getUint8Memory"
|
|
|
|
}
|
2018-09-24 13:49:12 -07:00
|
|
|
VectorKind::ClampedU8 => {
|
|
|
|
self.expose_clamped_uint8_memory();
|
|
|
|
"getUint8ClampedMemory"
|
|
|
|
}
|
2018-05-01 10:06:35 -07:00
|
|
|
VectorKind::I16 => {
|
|
|
|
self.expose_int16_memory();
|
|
|
|
"getInt16Memory"
|
|
|
|
}
|
|
|
|
VectorKind::U16 => {
|
|
|
|
self.expose_uint16_memory();
|
|
|
|
"getUint16Memory"
|
|
|
|
}
|
|
|
|
VectorKind::I32 => {
|
|
|
|
self.expose_int32_memory();
|
|
|
|
"getInt32Memory"
|
|
|
|
}
|
|
|
|
VectorKind::U32 => {
|
|
|
|
self.expose_uint32_memory();
|
|
|
|
"getUint32Memory"
|
|
|
|
}
|
2018-05-05 14:10:25 -07:00
|
|
|
VectorKind::I64 => {
|
|
|
|
self.expose_int64_memory();
|
|
|
|
"getInt64Memory"
|
|
|
|
}
|
|
|
|
VectorKind::U64 => {
|
|
|
|
self.expose_uint64_memory();
|
|
|
|
"getUint64Memory"
|
|
|
|
}
|
2018-05-01 10:06:35 -07:00
|
|
|
VectorKind::F32 => {
|
|
|
|
self.expose_f32_memory();
|
|
|
|
"getFloat32Memory"
|
|
|
|
}
|
|
|
|
VectorKind::F64 => {
|
|
|
|
self.expose_f64_memory();
|
|
|
|
"getFloat64Memory"
|
|
|
|
}
|
|
|
|
VectorKind::Anyref => {
|
|
|
|
self.expose_uint32_memory();
|
|
|
|
"getUint32Memory"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn memview(&mut self, name: &'static str, js: &str) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global(name) {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-08-20 23:33:29 -07:00
|
|
|
let mem = self.memory();
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"
|
2018-05-01 10:06:35 -07:00
|
|
|
let cache{name} = null;
|
|
|
|
function {name}() {{
|
2018-08-20 23:33:29 -07:00
|
|
|
if (cache{name} === null || cache{name}.buffer !== {mem}.buffer) {{
|
2019-03-25 17:42:10 +00:00
|
|
|
cache{name} = {js}({mem}.buffer);
|
2018-06-15 12:55:37 -05:00
|
|
|
}}
|
2018-05-01 10:06:35 -07:00
|
|
|
return cache{name};
|
2018-01-29 21:20:38 -08:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-05-01 10:06:35 -07:00
|
|
|
name = name,
|
|
|
|
js = js,
|
2018-08-20 23:33:29 -07:00
|
|
|
mem = mem,
|
2018-05-01 10:06:35 -07:00
|
|
|
));
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_assert_class(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("assert_class") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-06-15 23:39:51 -07:00
|
|
|
self.global(
|
2018-06-27 22:42:34 -07:00
|
|
|
"
|
2018-06-15 23:39:51 -07:00
|
|
|
function _assertClass(instance, klass) {
|
|
|
|
if (!(instance instanceof klass)) {
|
|
|
|
throw new Error(`expected instance of ${klass.name}`);
|
|
|
|
}
|
2018-02-06 15:52:44 -08:00
|
|
|
return instance.ptr;
|
2018-06-15 23:39:51 -07:00
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-11-29 18:15:36 -08:00
|
|
|
fn expose_global_stack_pointer(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("stack_pointer") {
|
2018-11-29 18:15:36 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
self.global(&format!("let stack_pointer = {};", INITIAL_HEAP_OFFSET));
|
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_borrowed_objects(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("borrowed_objects") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-11-29 18:15:36 -08:00
|
|
|
self.expose_global_heap();
|
|
|
|
self.expose_global_stack_pointer();
|
|
|
|
// Our `stack_pointer` points to where we should start writing stack
|
|
|
|
// objects, and the `stack_pointer` is incremented in a `finally` block
|
|
|
|
// after executing this. Once we've reserved stack space we write the
|
|
|
|
// value. Eventually underflow will throw an exception, but JS sort of
|
|
|
|
// just handles it today...
|
2018-06-15 23:39:51 -07:00
|
|
|
self.global(
|
2018-06-27 22:42:34 -07:00
|
|
|
"
|
2018-06-15 23:39:51 -07:00
|
|
|
function addBorrowedObject(obj) {
|
2018-11-29 18:15:36 -08:00
|
|
|
if (stack_pointer == 1) throw new Error('out of js stack');
|
|
|
|
heap[--stack_pointer] = obj;
|
|
|
|
return stack_pointer;
|
2018-06-15 23:39:51 -07:00
|
|
|
}
|
2018-11-30 13:04:05 -08:00
|
|
|
",
|
2018-06-15 23:39:51 -07:00
|
|
|
);
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_take_object(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("take_object") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-02-06 15:52:44 -08:00
|
|
|
self.expose_get_object();
|
|
|
|
self.expose_drop_ref();
|
2018-06-15 23:39:51 -07:00
|
|
|
self.global(
|
2018-06-27 22:42:34 -07:00
|
|
|
"
|
2018-06-15 23:39:51 -07:00
|
|
|
function takeObject(idx) {
|
2018-02-06 15:52:44 -08:00
|
|
|
const ret = getObject(idx);
|
2018-11-29 18:15:36 -08:00
|
|
|
dropObject(idx);
|
2018-02-06 15:52:44 -08:00
|
|
|
return ret;
|
2018-06-15 23:39:51 -07:00
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn expose_add_heap_object(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("add_heap_object") {
|
2018-04-07 13:06:36 +05:45
|
|
|
return;
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-11-29 18:15:36 -08:00
|
|
|
self.expose_global_heap();
|
|
|
|
self.expose_global_heap_next();
|
|
|
|
let set_heap_next = if self.config.debug {
|
2018-06-27 22:42:34 -07:00
|
|
|
String::from(
|
|
|
|
"
|
2018-11-29 18:15:36 -08:00
|
|
|
if (typeof(heap_next) !== 'number') throw new Error('corrupt heap');
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-06-27 22:42:34 -07:00
|
|
|
)
|
2018-01-29 21:20:38 -08:00
|
|
|
} else {
|
2018-11-29 18:15:36 -08:00
|
|
|
String::new()
|
2018-01-29 21:20:38 -08:00
|
|
|
};
|
2018-11-29 18:15:36 -08:00
|
|
|
|
|
|
|
// Allocating a slot on the heap first goes through the linked list
|
|
|
|
// (starting at `heap_next`). Once that linked list is exhausted we'll
|
|
|
|
// be pointing beyond the end of the array, at which point we'll reserve
|
|
|
|
// one more slot and use that.
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"
|
2018-02-06 15:52:44 -08:00
|
|
|
function addHeapObject(obj) {{
|
2018-11-29 18:15:36 -08:00
|
|
|
if (heap_next === heap.length) heap.push(heap.length + 1);
|
|
|
|
const idx = heap_next;
|
|
|
|
heap_next = heap[idx];
|
2018-02-06 15:52:44 -08:00
|
|
|
{}
|
2018-11-29 18:15:36 -08:00
|
|
|
heap[idx] = obj;
|
|
|
|
return idx;
|
2018-01-29 21:20:38 -08:00
|
|
|
}}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-11-29 18:15:36 -08:00
|
|
|
set_heap_next
|
2018-06-27 22:42:34 -07:00
|
|
|
));
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-10-18 08:43:36 -07:00
|
|
|
fn expose_handle_error(&mut self) -> Result<(), Error> {
|
2019-02-04 02:08:08 +01:00
|
|
|
if !self.should_write_global("handle_error") {
|
2018-10-18 08:43:36 -07:00
|
|
|
return Ok(());
|
2019-02-04 02:08:08 +01:00
|
|
|
}
|
|
|
|
self.expose_uint32_memory();
|
2018-10-18 08:43:36 -07:00
|
|
|
if self.config.anyref {
|
|
|
|
self.expose_add_to_anyref_table()?;
|
|
|
|
self.global(
|
|
|
|
"
|
|
|
|
function handleError(exnptr, e) {
|
|
|
|
const idx = addToAnyrefTable(e);
|
|
|
|
const view = getUint32Memory();
|
|
|
|
view[exnptr / 4] = 1;
|
|
|
|
view[exnptr / 4 + 1] = idx;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
self.expose_add_heap_object();
|
|
|
|
self.global(
|
|
|
|
"
|
|
|
|
function handleError(exnptr, e) {
|
|
|
|
const view = getUint32Memory();
|
|
|
|
view[exnptr / 4] = 1;
|
|
|
|
view[exnptr / 4 + 1] = addHeapObject(e);
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
Ok(())
|
2019-02-04 02:08:08 +01:00
|
|
|
}
|
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
fn wasm_import_needed(&self, name: &str) -> bool {
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
self.module
|
|
|
|
.imports
|
2018-06-27 22:42:34 -07:00
|
|
|
.iter()
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
.any(|i| i.module == "__wbindgen_placeholder__" && i.name == name)
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-02-16 13:49:53 -08:00
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn pass_to_wasm_function(&mut self, t: VectorKind) -> Result<&'static str, Error> {
|
|
|
|
let s = match t {
|
2018-02-16 18:58:37 -08:00
|
|
|
VectorKind::String => {
|
2018-04-25 11:42:22 -07:00
|
|
|
self.expose_pass_string_to_wasm()?;
|
2018-02-16 18:58:37 -08:00
|
|
|
"passStringToWasm"
|
|
|
|
}
|
2018-09-24 13:49:12 -07:00
|
|
|
VectorKind::I8 | VectorKind::U8 | VectorKind::ClampedU8 => {
|
2018-04-25 11:42:22 -07:00
|
|
|
self.expose_pass_array8_to_wasm()?;
|
2018-02-16 18:58:37 -08:00
|
|
|
"passArray8ToWasm"
|
|
|
|
}
|
2018-06-27 22:42:34 -07:00
|
|
|
VectorKind::U16 | VectorKind::I16 => {
|
2018-04-25 11:42:22 -07:00
|
|
|
self.expose_pass_array16_to_wasm()?;
|
2018-02-16 18:58:37 -08:00
|
|
|
"passArray16ToWasm"
|
|
|
|
}
|
2018-06-27 22:42:34 -07:00
|
|
|
VectorKind::I32 | VectorKind::U32 => {
|
2018-04-25 11:42:22 -07:00
|
|
|
self.expose_pass_array32_to_wasm()?;
|
2018-02-16 18:58:37 -08:00
|
|
|
"passArray32ToWasm"
|
|
|
|
}
|
2018-06-27 22:42:34 -07:00
|
|
|
VectorKind::I64 | VectorKind::U64 => {
|
2018-05-05 14:10:25 -07:00
|
|
|
self.expose_pass_array64_to_wasm()?;
|
|
|
|
"passArray64ToWasm"
|
|
|
|
}
|
2018-02-16 18:58:37 -08:00
|
|
|
VectorKind::F32 => {
|
2018-04-25 11:42:22 -07:00
|
|
|
self.expose_pass_array_f32_to_wasm()?;
|
2018-02-16 18:58:37 -08:00
|
|
|
"passArrayF32ToWasm"
|
|
|
|
}
|
|
|
|
VectorKind::F64 => {
|
2018-04-25 11:42:22 -07:00
|
|
|
self.expose_pass_array_f64_to_wasm()?;
|
2018-02-16 18:58:37 -08:00
|
|
|
"passArrayF64ToWasm"
|
|
|
|
}
|
2018-07-20 13:47:49 -05:00
|
|
|
VectorKind::Anyref => {
|
|
|
|
self.expose_pass_array_jsvalue_to_wasm()?;
|
|
|
|
"passArrayJsValueToWasm"
|
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
};
|
|
|
|
Ok(s)
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
|
|
|
|
2018-10-18 08:43:36 -07:00
|
|
|
fn expose_get_vector_from_wasm(&mut self, ty: VectorKind) -> Result<&'static str, Error> {
|
|
|
|
Ok(match ty {
|
2018-02-16 18:58:37 -08:00
|
|
|
VectorKind::String => {
|
|
|
|
self.expose_get_string_from_wasm();
|
|
|
|
"getStringFromWasm"
|
|
|
|
}
|
|
|
|
VectorKind::I8 => {
|
|
|
|
self.expose_get_array_i8_from_wasm();
|
|
|
|
"getArrayI8FromWasm"
|
|
|
|
}
|
|
|
|
VectorKind::U8 => {
|
|
|
|
self.expose_get_array_u8_from_wasm();
|
|
|
|
"getArrayU8FromWasm"
|
|
|
|
}
|
2018-09-24 13:49:12 -07:00
|
|
|
VectorKind::ClampedU8 => {
|
|
|
|
self.expose_get_clamped_array_u8_from_wasm();
|
|
|
|
"getClampedArrayU8FromWasm"
|
|
|
|
}
|
2018-02-16 18:58:37 -08:00
|
|
|
VectorKind::I16 => {
|
|
|
|
self.expose_get_array_i16_from_wasm();
|
|
|
|
"getArrayI16FromWasm"
|
|
|
|
}
|
|
|
|
VectorKind::U16 => {
|
|
|
|
self.expose_get_array_u16_from_wasm();
|
|
|
|
"getArrayU16FromWasm"
|
|
|
|
}
|
|
|
|
VectorKind::I32 => {
|
|
|
|
self.expose_get_array_i32_from_wasm();
|
|
|
|
"getArrayI32FromWasm"
|
|
|
|
}
|
|
|
|
VectorKind::U32 => {
|
|
|
|
self.expose_get_array_u32_from_wasm();
|
|
|
|
"getArrayU32FromWasm"
|
|
|
|
}
|
2018-05-05 14:10:25 -07:00
|
|
|
VectorKind::I64 => {
|
|
|
|
self.expose_get_array_i64_from_wasm();
|
|
|
|
"getArrayI64FromWasm"
|
|
|
|
}
|
|
|
|
VectorKind::U64 => {
|
|
|
|
self.expose_get_array_u64_from_wasm();
|
|
|
|
"getArrayU64FromWasm"
|
|
|
|
}
|
2018-02-16 18:58:37 -08:00
|
|
|
VectorKind::F32 => {
|
|
|
|
self.expose_get_array_f32_from_wasm();
|
|
|
|
"getArrayF32FromWasm"
|
|
|
|
}
|
|
|
|
VectorKind::F64 => {
|
|
|
|
self.expose_get_array_f64_from_wasm();
|
|
|
|
"getArrayF64FromWasm"
|
|
|
|
}
|
Overhaul how type information gets to the CLI
This commit is a complete overhaul of how the `#[wasm_bindgen]` macro
communicates type information to the CLI tool, and it's done in a somewhat...
unconventional fashion.
Today we've got a problem where the generated JS needs to understand the types
of each function exported or imported. This understanding is what enables it to
generate the appropriate JS wrappers and such. We want to, however, be quite
flexible and extensible in types that are supported across the boundary, which
means that internally we rely on the trait system to resolve what's what.
Communicating the type information historically was done by creating a four byte
"descriptor" and using associated type projections to communicate that to the
CLI tool. Unfortunately four bytes isn't a lot of space to cram information like
arguments to a generic function, tuple types, etc. In general this just wasn't
flexible enough and the way custom references were treated was also already a
bit of a hack.
This commit takes a radical step of creating a **descriptor function** for each
function imported/exported. The really crazy part is that the `wasm-bindgen` CLI
tool now embeds a wasm interpreter and executes these functions when the CLI
tool is invoked. By allowing arbitrary functions to get executed it's now *much*
easier to inform `wasm-bindgen` about complicated structures of types. Rest
assured though that all these descriptor functions are automatically unexported
and gc'd away, so this should not have any impact on binary sizes
A new internal trait, `WasmDescribe`, is added to represent a description of all
types, sort of like a serialization of the structure of a type that
`wasm-bindgen` can understand. This works by calling a special exported function
with a `u32` value a bunch of times. This means that when we run a descriptor we
effectively get a `Vec<u32>` in the `wasm-bindgen` CLI tool. This list of
integers can then be parsed into a rich `enum` for the JS generation to work
with.
This commit currently only retains feature parity with the previous
implementation. I hope to soon solve issues like #123, #104, and #111 with this
support.
2018-04-13 07:33:46 -07:00
|
|
|
VectorKind::Anyref => {
|
2018-10-18 08:43:36 -07:00
|
|
|
self.expose_get_array_js_value_from_wasm()?;
|
2018-02-28 10:56:56 +01:00
|
|
|
"getArrayJsValueFromWasm"
|
|
|
|
}
|
2018-10-18 08:43:36 -07:00
|
|
|
})
|
2018-02-16 18:58:37 -08:00
|
|
|
}
|
2018-03-31 07:57:47 -07:00
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn expose_global_argument_ptr(&mut self) -> Result<(), Error> {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("global_argument_ptr") {
|
2018-04-25 11:42:22 -07:00
|
|
|
return Ok(());
|
2018-03-31 07:57:47 -07:00
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
self.require_internal_export("__wbindgen_global_argument_ptr")?;
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(
|
|
|
|
"
|
2018-03-31 07:57:47 -07:00
|
|
|
let cachedGlobalArgumentPtr = null;
|
|
|
|
function globalArgumentPtr() {
|
2018-06-15 23:39:51 -07:00
|
|
|
if (cachedGlobalArgumentPtr === null) {
|
2018-03-31 07:57:47 -07:00
|
|
|
cachedGlobalArgumentPtr = wasm.__wbindgen_global_argument_ptr();
|
2018-06-15 23:39:51 -07:00
|
|
|
}
|
2018-03-31 07:57:47 -07:00
|
|
|
return cachedGlobalArgumentPtr;
|
|
|
|
}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-06-27 22:42:34 -07:00
|
|
|
);
|
2018-04-25 11:42:22 -07:00
|
|
|
Ok(())
|
2018-03-31 07:57:47 -07:00
|
|
|
}
|
2018-04-09 06:16:41 -07:00
|
|
|
|
|
|
|
fn expose_get_inherited_descriptor(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("get_inherited_descriptor") {
|
2018-06-27 22:42:34 -07:00
|
|
|
return;
|
2018-04-09 06:16:41 -07:00
|
|
|
}
|
|
|
|
// It looks like while rare some browsers will move descriptors up the
|
|
|
|
// property chain which runs the risk of breaking wasm-bindgen-generated
|
|
|
|
// code because we're looking for precise descriptor functions rather
|
|
|
|
// than relying on the prototype chain like most "normal JS" projects
|
|
|
|
// do.
|
|
|
|
//
|
|
|
|
// As a result we have a small helper here which will walk the prototype
|
|
|
|
// chain looking for a descriptor. For some more information on this see
|
|
|
|
// #109
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(
|
|
|
|
"
|
2018-04-09 06:16:41 -07:00
|
|
|
function GetOwnOrInheritedPropertyDescriptor(obj, id) {
|
|
|
|
while (obj) {
|
|
|
|
let desc = Object.getOwnPropertyDescriptor(obj, id);
|
|
|
|
if (desc) return desc;
|
|
|
|
obj = Object.getPrototypeOf(obj);
|
|
|
|
}
|
2018-10-10 16:10:24 -07:00
|
|
|
return {}
|
2018-04-09 06:16:41 -07:00
|
|
|
}
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-06-27 22:42:34 -07:00
|
|
|
);
|
2018-04-09 06:16:41 -07:00
|
|
|
}
|
2018-04-05 18:30:37 -07:00
|
|
|
|
2018-05-05 14:10:25 -07:00
|
|
|
fn expose_u32_cvt_shim(&mut self) -> &'static str {
|
|
|
|
let name = "u32CvtShim";
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global(name) {
|
2018-06-27 22:42:34 -07:00
|
|
|
return name;
|
2018-05-05 14:10:25 -07:00
|
|
|
}
|
|
|
|
self.global(&format!("const {} = new Uint32Array(2);", name));
|
|
|
|
name
|
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_int64_cvt_shim(&mut self) -> &'static str {
|
|
|
|
let name = "int64CvtShim";
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global(name) {
|
2018-06-27 22:42:34 -07:00
|
|
|
return name;
|
2018-05-05 14:10:25 -07:00
|
|
|
}
|
|
|
|
let n = self.expose_u32_cvt_shim();
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"const {} = new BigInt64Array({}.buffer);",
|
|
|
|
name, n
|
|
|
|
));
|
2018-05-05 14:10:25 -07:00
|
|
|
name
|
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_uint64_cvt_shim(&mut self) -> &'static str {
|
|
|
|
let name = "uint64CvtShim";
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global(name) {
|
2018-06-27 22:42:34 -07:00
|
|
|
return name;
|
2018-05-05 14:10:25 -07:00
|
|
|
}
|
|
|
|
let n = self.expose_u32_cvt_shim();
|
2018-06-27 22:42:34 -07:00
|
|
|
self.global(&format!(
|
|
|
|
"const {} = new BigUint64Array({}.buffer);",
|
|
|
|
name, n
|
|
|
|
));
|
2018-05-05 14:10:25 -07:00
|
|
|
name
|
|
|
|
}
|
|
|
|
|
2018-07-19 14:44:23 -05:00
|
|
|
fn expose_is_like_none(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("is_like_none") {
|
2018-09-26 08:26:00 -07:00
|
|
|
return;
|
2018-07-19 14:44:23 -05:00
|
|
|
}
|
2018-09-26 08:26:00 -07:00
|
|
|
self.global(
|
|
|
|
"
|
2018-07-19 14:44:23 -05:00
|
|
|
function isLikeNone(x) {
|
|
|
|
return x === undefined || x === null;
|
|
|
|
}
|
2018-09-26 08:26:00 -07:00
|
|
|
",
|
|
|
|
);
|
2018-07-19 14:44:23 -05:00
|
|
|
}
|
|
|
|
|
2018-08-15 17:52:26 -07:00
|
|
|
fn expose_cleanup_groups(&mut self) {
|
2019-01-14 13:09:05 -08:00
|
|
|
if !self.should_write_global("cleanup_groups") {
|
2018-09-26 08:26:00 -07:00
|
|
|
return;
|
2018-08-15 17:52:26 -07:00
|
|
|
}
|
|
|
|
self.global(
|
|
|
|
"
|
|
|
|
const CLEANUPS = new WeakRefGroup(x => x.holdings());
|
|
|
|
const CLEANUPS_MAP = new Map();
|
2018-09-21 15:45:31 -07:00
|
|
|
|
|
|
|
function addCleanup(obj, ptr, free) {
|
|
|
|
const ref = CLEANUPS.makeRef(obj, () => free(ptr));
|
|
|
|
CLEANUPS_MAP.set(ptr, ref);
|
|
|
|
}
|
2018-09-26 08:26:00 -07:00
|
|
|
",
|
2018-08-15 17:52:26 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-08-19 17:07:30 -07:00
|
|
|
fn describe(&mut self, name: &str) -> Option<Descriptor> {
|
Overhaul how type information gets to the CLI
This commit is a complete overhaul of how the `#[wasm_bindgen]` macro
communicates type information to the CLI tool, and it's done in a somewhat...
unconventional fashion.
Today we've got a problem where the generated JS needs to understand the types
of each function exported or imported. This understanding is what enables it to
generate the appropriate JS wrappers and such. We want to, however, be quite
flexible and extensible in types that are supported across the boundary, which
means that internally we rely on the trait system to resolve what's what.
Communicating the type information historically was done by creating a four byte
"descriptor" and using associated type projections to communicate that to the
CLI tool. Unfortunately four bytes isn't a lot of space to cram information like
arguments to a generic function, tuple types, etc. In general this just wasn't
flexible enough and the way custom references were treated was also already a
bit of a hack.
This commit takes a radical step of creating a **descriptor function** for each
function imported/exported. The really crazy part is that the `wasm-bindgen` CLI
tool now embeds a wasm interpreter and executes these functions when the CLI
tool is invoked. By allowing arbitrary functions to get executed it's now *much*
easier to inform `wasm-bindgen` about complicated structures of types. Rest
assured though that all these descriptor functions are automatically unexported
and gc'd away, so this should not have any impact on binary sizes
A new internal trait, `WasmDescribe`, is added to represent a description of all
types, sort of like a serialization of the structure of a type that
`wasm-bindgen` can understand. This works by calling a special exported function
with a `u32` value a bunch of times. This means that when we run a descriptor we
effectively get a `Vec<u32>` in the `wasm-bindgen` CLI tool. This list of
integers can then be parsed into a rich `enum` for the JS generation to work
with.
This commit currently only retains feature parity with the previous
implementation. I hope to soon solve issues like #123, #104, and #111 with this
support.
2018-04-13 07:33:46 -07:00
|
|
|
let name = format!("__wbindgen_describe_{}", name);
|
2018-09-05 23:59:49 -07:00
|
|
|
let descriptor = self.interpreter.interpret_descriptor(&name, self.module)?;
|
|
|
|
Some(Descriptor::decode(descriptor))
|
Overhaul how type information gets to the CLI
This commit is a complete overhaul of how the `#[wasm_bindgen]` macro
communicates type information to the CLI tool, and it's done in a somewhat...
unconventional fashion.
Today we've got a problem where the generated JS needs to understand the types
of each function exported or imported. This understanding is what enables it to
generate the appropriate JS wrappers and such. We want to, however, be quite
flexible and extensible in types that are supported across the boundary, which
means that internally we rely on the trait system to resolve what's what.
Communicating the type information historically was done by creating a four byte
"descriptor" and using associated type projections to communicate that to the
CLI tool. Unfortunately four bytes isn't a lot of space to cram information like
arguments to a generic function, tuple types, etc. In general this just wasn't
flexible enough and the way custom references were treated was also already a
bit of a hack.
This commit takes a radical step of creating a **descriptor function** for each
function imported/exported. The really crazy part is that the `wasm-bindgen` CLI
tool now embeds a wasm interpreter and executes these functions when the CLI
tool is invoked. By allowing arbitrary functions to get executed it's now *much*
easier to inform `wasm-bindgen` about complicated structures of types. Rest
assured though that all these descriptor functions are automatically unexported
and gc'd away, so this should not have any impact on binary sizes
A new internal trait, `WasmDescribe`, is added to represent a description of all
types, sort of like a serialization of the structure of a type that
`wasm-bindgen` can understand. This works by calling a special exported function
with a `u32` value a bunch of times. This means that when we run a descriptor we
effectively get a `Vec<u32>` in the `wasm-bindgen` CLI tool. This list of
integers can then be parsed into a rich `enum` for the JS generation to work
with.
This commit currently only retains feature parity with the previous
implementation. I hope to soon solve issues like #123, #104, and #111 with this
support.
2018-04-13 07:33:46 -07:00
|
|
|
}
|
|
|
|
|
2018-04-14 09:37:49 -07:00
|
|
|
fn global(&mut self, s: &str) {
|
2018-04-16 13:31:56 -07:00
|
|
|
let s = s.trim();
|
|
|
|
|
|
|
|
// Ensure a blank line between adjacent items, and ensure everything is
|
|
|
|
// terminated with a newline.
|
2018-06-15 11:20:56 -05:00
|
|
|
while !self.globals.ends_with("\n\n\n") && !self.globals.ends_with("*/\n") {
|
2018-04-14 09:37:49 -07:00
|
|
|
self.globals.push_str("\n");
|
|
|
|
}
|
2018-04-16 13:31:56 -07:00
|
|
|
self.globals.push_str(s);
|
|
|
|
self.globals.push_str("\n");
|
2018-04-14 09:37:49 -07:00
|
|
|
}
|
2018-04-25 07:26:33 -07:00
|
|
|
|
2018-07-04 22:37:09 -05:00
|
|
|
fn use_node_require(&self) -> bool {
|
2019-02-25 11:11:30 -08:00
|
|
|
self.config.mode.nodejs() && !self.config.mode.nodejs_experimental_modules()
|
2018-07-04 22:37:09 -05:00
|
|
|
}
|
2018-08-20 23:33:29 -07:00
|
|
|
|
|
|
|
fn memory(&mut self) -> &'static str {
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
if self.module.memories.get(self.memory).import.is_some() {
|
|
|
|
"memory"
|
|
|
|
} else {
|
|
|
|
"wasm.memory"
|
2018-08-20 23:33:29 -07:00
|
|
|
}
|
|
|
|
}
|
2018-09-21 15:45:31 -07:00
|
|
|
|
|
|
|
fn require_class_wrap(&mut self, class: &str) {
|
|
|
|
self.exported_classes
|
2019-01-14 13:11:07 -08:00
|
|
|
.as_mut()
|
|
|
|
.expect("classes already written")
|
2018-09-21 15:45:31 -07:00
|
|
|
.entry(class.to_string())
|
|
|
|
.or_insert_with(ExportedClass::default)
|
|
|
|
.wrap_needed = true;
|
|
|
|
}
|
2018-11-12 11:59:13 -08:00
|
|
|
|
|
|
|
fn import_identifier(&mut self, import: Import<'a>) -> String {
|
|
|
|
// Here's where it's a bit tricky. We need to make sure that importing
|
|
|
|
// the same identifier from two different modules works, and they're
|
|
|
|
// named uniquely below. Additionally if we've already imported the same
|
|
|
|
// identifier from the module in question then we'd like to reuse the
|
|
|
|
// one that was previously imported.
|
|
|
|
//
|
|
|
|
// Our `imported_names` map keeps track of all imported identifiers from
|
|
|
|
// modules, mapping the imported names onto names actually available for
|
|
|
|
// use in our own module. If our identifier isn't present then we
|
|
|
|
// generate a new identifier and are sure to generate the appropriate JS
|
|
|
|
// import for our new identifier.
|
|
|
|
let use_node_require = self.use_node_require();
|
|
|
|
let imported_identifiers = &mut self.imported_identifiers;
|
|
|
|
let imports = &mut self.imports;
|
|
|
|
let imports_post = &mut self.imports_post;
|
|
|
|
let identifier = self
|
|
|
|
.imported_names
|
|
|
|
.entry(import.module())
|
|
|
|
.or_insert_with(Default::default)
|
|
|
|
.entry(import.name())
|
|
|
|
.or_insert_with(|| {
|
|
|
|
let name = generate_identifier(import.name(), imported_identifiers);
|
|
|
|
match &import {
|
2019-02-25 11:11:30 -08:00
|
|
|
Import::Module { .. }
|
|
|
|
| Import::LocalModule { .. }
|
|
|
|
| Import::InlineJs { .. } => {
|
|
|
|
// When doing a modular import local snippets (either
|
|
|
|
// inline or not) are routed to a local `./snippets`
|
|
|
|
// directory which the rest of `wasm-bindgen` will fill
|
|
|
|
// in.
|
|
|
|
let path = match import {
|
|
|
|
Import::Module { module, .. } => module.to_string(),
|
|
|
|
Import::LocalModule { module, .. } => format!("./snippets/{}", module),
|
2019-03-05 14:53:14 -08:00
|
|
|
Import::InlineJs {
|
|
|
|
unique_crate_identifier,
|
|
|
|
snippet_idx_in_crate,
|
|
|
|
..
|
|
|
|
} => format!(
|
|
|
|
"./snippets/{}/inline{}.js",
|
|
|
|
unique_crate_identifier, snippet_idx_in_crate
|
|
|
|
),
|
2019-02-25 11:11:30 -08:00
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
2018-11-12 11:59:13 -08:00
|
|
|
if use_node_require {
|
|
|
|
imports.push_str(&format!(
|
|
|
|
"const {} = require(String.raw`{}`).{};\n",
|
2018-11-27 12:07:59 -08:00
|
|
|
name,
|
2019-02-25 11:11:30 -08:00
|
|
|
path,
|
2018-11-27 12:07:59 -08:00
|
|
|
import.name()
|
2018-11-12 11:59:13 -08:00
|
|
|
));
|
|
|
|
} else if import.name() == name {
|
2019-02-25 11:11:30 -08:00
|
|
|
imports.push_str(&format!("import {{ {} }} from '{}';\n", name, path));
|
2018-11-12 11:59:13 -08:00
|
|
|
} else {
|
|
|
|
imports.push_str(&format!(
|
|
|
|
"import {{ {} as {} }} from '{}';\n",
|
2018-11-27 12:07:59 -08:00
|
|
|
import.name(),
|
|
|
|
name,
|
2019-02-25 11:11:30 -08:00
|
|
|
path
|
2018-11-12 11:59:13 -08:00
|
|
|
));
|
|
|
|
}
|
|
|
|
name
|
|
|
|
}
|
|
|
|
|
|
|
|
Import::VendorPrefixed { prefixes, .. } => {
|
|
|
|
imports_post.push_str("const l");
|
|
|
|
imports_post.push_str(&name);
|
|
|
|
imports_post.push_str(" = ");
|
|
|
|
switch(imports_post, &name, "", prefixes);
|
|
|
|
imports_post.push_str(";\n");
|
|
|
|
|
|
|
|
fn switch(dst: &mut String, name: &str, prefix: &str, left: &[&str]) {
|
|
|
|
if left.len() == 0 {
|
|
|
|
dst.push_str(prefix);
|
|
|
|
return dst.push_str(name);
|
|
|
|
}
|
|
|
|
dst.push_str("(typeof ");
|
|
|
|
dst.push_str(prefix);
|
|
|
|
dst.push_str(name);
|
|
|
|
dst.push_str(" == 'undefined' ? ");
|
|
|
|
match left.len() {
|
|
|
|
1 => {
|
|
|
|
dst.push_str(&left[0]);
|
|
|
|
dst.push_str(name);
|
|
|
|
}
|
|
|
|
_ => switch(dst, name, &left[0], &left[1..]),
|
|
|
|
}
|
|
|
|
dst.push_str(" : ");
|
|
|
|
dst.push_str(prefix);
|
|
|
|
dst.push_str(name);
|
|
|
|
dst.push_str(")");
|
|
|
|
}
|
|
|
|
format!("l{}", name)
|
|
|
|
}
|
|
|
|
|
|
|
|
Import::Global { .. } => name,
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// If there's a namespace we didn't actually import `item` but rather
|
|
|
|
// the namespace, so access through that.
|
|
|
|
match import.field() {
|
|
|
|
Some(field) => format!("{}.{}", identifier, field),
|
|
|
|
None => identifier.clone(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn generated_import_target(
|
|
|
|
&mut self,
|
|
|
|
name: Import<'a>,
|
|
|
|
import: &decode::ImportFunction<'a>,
|
|
|
|
) -> Result<ImportTarget, Error> {
|
|
|
|
let method_data = match &import.method {
|
|
|
|
Some(data) => data,
|
|
|
|
None => {
|
|
|
|
let name = self.import_identifier(name);
|
|
|
|
if import.structural || !name.contains(".") {
|
2018-11-27 12:07:59 -08:00
|
|
|
return Ok(ImportTarget::Function(name));
|
2018-11-12 11:59:13 -08:00
|
|
|
}
|
|
|
|
self.global(&format!("const {}_target = {};", import.shim, name));
|
|
|
|
let target = format!("{}_target", import.shim);
|
2018-11-27 12:07:59 -08:00
|
|
|
return Ok(ImportTarget::Function(target));
|
2018-11-12 11:59:13 -08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
let class = self.import_identifier(name);
|
|
|
|
let op = match &method_data.kind {
|
|
|
|
decode::MethodKind::Constructor => {
|
2018-12-11 20:42:37 +02:00
|
|
|
return Ok(ImportTarget::Constructor(class.to_string()));
|
2018-11-12 11:59:13 -08:00
|
|
|
}
|
|
|
|
decode::MethodKind::Operation(op) => op,
|
|
|
|
};
|
|
|
|
if import.structural {
|
2018-11-27 12:07:59 -08:00
|
|
|
let class = if op.is_static {
|
|
|
|
Some(class.clone())
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
2018-11-12 11:59:13 -08:00
|
|
|
|
|
|
|
return Ok(match &op.kind {
|
|
|
|
decode::OperationKind::Regular => {
|
|
|
|
let name = import.function.name.to_string();
|
|
|
|
match class {
|
|
|
|
Some(c) => ImportTarget::Function(format!("{}.{}", c, name)),
|
|
|
|
None => ImportTarget::StructuralMethod(name),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
decode::OperationKind::Getter(g) => {
|
|
|
|
ImportTarget::StructuralGetter(class, g.to_string())
|
|
|
|
}
|
|
|
|
decode::OperationKind::Setter(s) => {
|
|
|
|
ImportTarget::StructuralSetter(class, s.to_string())
|
|
|
|
}
|
|
|
|
decode::OperationKind::IndexingGetter => {
|
|
|
|
ImportTarget::StructuralIndexingGetter(class)
|
|
|
|
}
|
|
|
|
decode::OperationKind::IndexingSetter => {
|
|
|
|
ImportTarget::StructuralIndexingSetter(class)
|
|
|
|
}
|
|
|
|
decode::OperationKind::IndexingDeleter => {
|
|
|
|
ImportTarget::StructuralIndexingDeleter(class)
|
|
|
|
}
|
2018-11-27 12:07:59 -08:00
|
|
|
});
|
2018-11-12 11:59:13 -08:00
|
|
|
}
|
|
|
|
|
2018-11-27 12:07:59 -08:00
|
|
|
let target = format!(
|
|
|
|
"typeof {0} === 'undefined' ? null : {}{}",
|
|
|
|
class,
|
|
|
|
if op.is_static { "" } else { ".prototype" }
|
|
|
|
);
|
2018-11-12 11:59:13 -08:00
|
|
|
let (mut target, name) = match &op.kind {
|
2018-11-27 12:07:59 -08:00
|
|
|
decode::OperationKind::Regular => (
|
|
|
|
format!("{}.{}", target, import.function.name),
|
|
|
|
&import.function.name,
|
|
|
|
),
|
2018-11-12 11:59:13 -08:00
|
|
|
decode::OperationKind::Getter(g) => {
|
|
|
|
self.expose_get_inherited_descriptor();
|
2018-11-27 12:07:59 -08:00
|
|
|
(
|
|
|
|
format!(
|
|
|
|
"GetOwnOrInheritedPropertyDescriptor({}, '{}').get",
|
|
|
|
target, g,
|
|
|
|
),
|
|
|
|
g,
|
|
|
|
)
|
2018-11-12 11:59:13 -08:00
|
|
|
}
|
|
|
|
decode::OperationKind::Setter(s) => {
|
|
|
|
self.expose_get_inherited_descriptor();
|
2018-11-27 12:07:59 -08:00
|
|
|
(
|
|
|
|
format!(
|
|
|
|
"GetOwnOrInheritedPropertyDescriptor({}, '{}').set",
|
|
|
|
target, s,
|
|
|
|
),
|
|
|
|
s,
|
|
|
|
)
|
2018-11-12 11:59:13 -08:00
|
|
|
}
|
2018-11-27 12:07:59 -08:00
|
|
|
decode::OperationKind::IndexingGetter => panic!("indexing getter should be structural"),
|
|
|
|
decode::OperationKind::IndexingSetter => panic!("indexing setter should be structural"),
|
2018-11-12 11:59:13 -08:00
|
|
|
decode::OperationKind::IndexingDeleter => {
|
|
|
|
panic!("indexing deleter should be structural")
|
|
|
|
}
|
|
|
|
};
|
2018-11-27 12:07:59 -08:00
|
|
|
target.push_str(&format!(
|
|
|
|
" || function() {{
|
2018-11-12 11:59:13 -08:00
|
|
|
throw new Error(`wasm-bindgen: {}.{} does not exist`);
|
2018-11-27 12:07:59 -08:00
|
|
|
}}",
|
|
|
|
class, name
|
|
|
|
));
|
2018-11-12 11:59:13 -08:00
|
|
|
if op.is_static {
|
|
|
|
target.insert(0, '(');
|
|
|
|
target.push_str(").bind(");
|
|
|
|
target.push_str(&class);
|
|
|
|
target.push_str(")");
|
|
|
|
}
|
|
|
|
|
|
|
|
self.global(&format!("const {}_target = {};", import.shim, target));
|
|
|
|
Ok(if op.is_static {
|
|
|
|
ImportTarget::Function(format!("{}_target", import.shim))
|
|
|
|
} else {
|
|
|
|
ImportTarget::Method(format!("{}_target", import.shim))
|
|
|
|
})
|
|
|
|
}
|
2018-11-19 11:21:05 -08:00
|
|
|
|
|
|
|
/// Update the wasm file's `producers` section to include information about
|
|
|
|
/// the wasm-bindgen tool.
|
|
|
|
///
|
|
|
|
/// Specified at:
|
|
|
|
/// https://github.com/WebAssembly/tool-conventions/blob/master/ProducersSection.md
|
|
|
|
fn update_producers_section(&mut self) {
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
self.module
|
|
|
|
.producers
|
|
|
|
.add_processed_by("wasm-bindgen", &wasm_bindgen_shared::version());
|
2018-11-19 11:21:05 -08:00
|
|
|
}
|
2018-11-28 09:25:51 -08:00
|
|
|
|
|
|
|
fn add_start_function(&mut self) -> Result<(), Error> {
|
|
|
|
let start = match &self.start {
|
|
|
|
Some(name) => name.clone(),
|
|
|
|
None => return Ok(()),
|
|
|
|
};
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
let export = match self.module.exports.iter().find(|e| e.name == start) {
|
|
|
|
Some(export) => export,
|
|
|
|
None => bail!("export `{}` not found", start),
|
|
|
|
};
|
|
|
|
let id = match export.item {
|
|
|
|
walrus::ExportItem::Function(i) => i,
|
|
|
|
_ => bail!("export `{}` wasn't a function", start),
|
2018-11-28 09:25:51 -08:00
|
|
|
};
|
|
|
|
|
2018-10-18 08:43:36 -07:00
|
|
|
let prev_start = match self.module.start {
|
|
|
|
Some(f) => f,
|
|
|
|
None => {
|
|
|
|
self.module.start = Some(id);
|
|
|
|
return Ok(());
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
}
|
2018-10-18 08:43:36 -07:00
|
|
|
};
|
2018-11-28 09:25:51 -08:00
|
|
|
|
2018-10-18 08:43:36 -07:00
|
|
|
// Note that we call the previous start function, if any, first. This is
|
|
|
|
// because the start function currently only shows up when it's injected
|
|
|
|
// through thread/anyref transforms. These injected start functions need
|
|
|
|
// to happen before user code, so we always schedule them first.
|
|
|
|
let mut builder = walrus::FunctionBuilder::new();
|
|
|
|
let call1 = builder.call(prev_start, Box::new([]));
|
|
|
|
let call2 = builder.call(id, Box::new([]));
|
|
|
|
let ty = self.module.funcs.get(id).ty();
|
|
|
|
let new_start = builder.finish(ty, Vec::new(), vec![call1, call2], self.module);
|
|
|
|
self.module.start = Some(new_start);
|
2018-11-28 09:25:51 -08:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// If a start function is present, it removes it from the `start` section
|
|
|
|
/// of the wasm module and then moves it to an exported function, named
|
|
|
|
/// `__wbindgen_start`.
|
|
|
|
fn unstart_start_function(&mut self) -> bool {
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
let start = match self.module.start.take() {
|
|
|
|
Some(id) => id,
|
|
|
|
None => return false,
|
|
|
|
};
|
|
|
|
self.module.exports.add("__wbindgen_start", start);
|
|
|
|
true
|
2018-11-28 09:25:51 -08:00
|
|
|
}
|
|
|
|
|
2018-10-18 08:43:36 -07:00
|
|
|
fn expose_anyref_table(&mut self) {
|
|
|
|
assert!(self.config.anyref);
|
|
|
|
if !self.should_write_global("anyref_table") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
self.module
|
|
|
|
.exports
|
|
|
|
.add("__wbg_anyref_table", self.anyref.anyref_table_id());
|
|
|
|
}
|
|
|
|
|
|
|
|
fn expose_add_to_anyref_table(&mut self) -> Result<(), Error> {
|
|
|
|
assert!(self.config.anyref);
|
|
|
|
if !self.should_write_global("add_to_anyref_table") {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
self.expose_anyref_table();
|
|
|
|
self.require_internal_export("__wbindgen_anyref_table_alloc")?;
|
|
|
|
self.global(
|
|
|
|
"
|
|
|
|
function addToAnyrefTable(obj) {
|
|
|
|
const idx = wasm.__wbindgen_anyref_table_alloc();
|
|
|
|
wasm.__wbg_anyref_table.set(idx, obj);
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn add_heap_object(&mut self, expr: &str) -> String {
|
|
|
|
if self.config.anyref {
|
|
|
|
expr.to_string()
|
|
|
|
} else {
|
|
|
|
self.expose_add_heap_object();
|
|
|
|
format!("addHeapObject({})", expr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn take_object(&mut self, expr: &str) -> String {
|
|
|
|
if self.config.anyref {
|
|
|
|
expr.to_string()
|
|
|
|
} else {
|
|
|
|
self.expose_take_object();
|
|
|
|
format!("takeObject({})", expr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_object(&mut self, expr: &str) -> String {
|
|
|
|
if self.config.anyref {
|
|
|
|
expr.to_string()
|
|
|
|
} else {
|
|
|
|
self.expose_get_object();
|
|
|
|
format!("getObject({})", expr)
|
|
|
|
}
|
|
|
|
}
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-01-29 21:20:38 -08:00
|
|
|
|
2018-02-06 15:52:44 -08:00
|
|
|
impl<'a, 'b> SubContext<'a, 'b> {
|
2018-04-25 11:42:22 -07:00
|
|
|
pub fn generate(&mut self) -> Result<(), Error> {
|
2019-02-25 11:11:30 -08:00
|
|
|
for m in self.program.local_modules.iter() {
|
2019-02-26 08:30:59 -08:00
|
|
|
// All local modules we find should be unique, but the same module
|
|
|
|
// may have showed up in a few different blocks. If that's the case
|
|
|
|
// all the same identifiers should have the same contents.
|
|
|
|
if let Some(prev) = self.cx.local_modules.insert(m.identifier, m.contents) {
|
|
|
|
assert_eq!(prev, m.contents);
|
|
|
|
}
|
2019-02-25 11:11:30 -08:00
|
|
|
}
|
2018-02-07 16:41:33 -08:00
|
|
|
for f in self.program.exports.iter() {
|
2018-06-27 22:42:34 -07:00
|
|
|
self.generate_export(f).with_context(|_| {
|
|
|
|
format!(
|
|
|
|
"failed to generate bindings for Rust export `{}`",
|
|
|
|
f.function.name
|
|
|
|
)
|
|
|
|
})?;
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-09-28 13:17:37 -07:00
|
|
|
for f in self.program.imports.iter() {
|
2018-08-26 15:43:33 -07:00
|
|
|
if let decode::ImportKind::Type(ty) = &f.kind {
|
2018-10-01 12:33:33 -07:00
|
|
|
self.register_vendor_prefix(ty);
|
2018-09-28 13:17:37 -07:00
|
|
|
}
|
|
|
|
}
|
2018-03-21 09:55:16 -07:00
|
|
|
for f in self.program.imports.iter() {
|
2018-04-25 11:42:22 -07:00
|
|
|
self.generate_import(f)?;
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-02-22 12:01:38 +01:00
|
|
|
for e in self.program.enums.iter() {
|
|
|
|
self.generate_enum(e);
|
|
|
|
}
|
2018-04-16 08:05:18 -07:00
|
|
|
for s in self.program.structs.iter() {
|
2018-11-29 12:01:16 -08:00
|
|
|
self.generate_struct(s).with_context(|_| {
|
2018-11-30 13:04:05 -08:00
|
|
|
format!("failed to generate bindings for Rust struct `{}`", s.name,)
|
2018-11-29 12:01:16 -08:00
|
|
|
})?;
|
2018-04-16 08:05:18 -07:00
|
|
|
}
|
2018-11-17 23:04:19 -05:00
|
|
|
for s in self.program.typescript_custom_sections.iter() {
|
|
|
|
self.cx.typescript.push_str(s);
|
|
|
|
self.cx.typescript.push_str("\n\n");
|
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
|
2019-02-27 12:20:33 -08:00
|
|
|
if let Some(path) = self.program.package_json {
|
|
|
|
self.add_package_json(path)?;
|
|
|
|
}
|
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
Ok(())
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-08-26 15:43:33 -07:00
|
|
|
fn generate_export(&mut self, export: &decode::Export<'b>) -> Result<(), Error> {
|
2018-02-07 16:41:33 -08:00
|
|
|
if let Some(ref class) = export.class {
|
2018-11-28 09:25:51 -08:00
|
|
|
assert!(!export.start);
|
2018-04-07 13:06:36 +05:45
|
|
|
return self.generate_export_for_class(class, export);
|
2018-02-07 16:41:33 -08:00
|
|
|
}
|
2018-06-18 13:48:57 -07:00
|
|
|
|
|
|
|
let descriptor = match self.cx.describe(&export.function.name) {
|
|
|
|
None => return Ok(()),
|
|
|
|
Some(d) => d,
|
|
|
|
};
|
|
|
|
|
2018-11-28 09:25:51 -08:00
|
|
|
if export.start {
|
|
|
|
self.set_start_function(export.function.name)?;
|
|
|
|
}
|
|
|
|
|
2018-07-09 11:07:57 -05:00
|
|
|
let (js, ts, js_doc) = Js2Rust::new(&export.function.name, self.cx)
|
2019-03-14 08:46:42 -03:00
|
|
|
.process(descriptor.unwrap_function(), &export.function.arg_names)?
|
2018-10-18 08:43:36 -07:00
|
|
|
.finish(
|
|
|
|
"function",
|
|
|
|
&format!("wasm.{}", export.function.name),
|
|
|
|
ExportedShim::Named(&export.function.name),
|
|
|
|
);
|
2018-06-27 22:42:34 -07:00
|
|
|
self.cx.export(
|
|
|
|
&export.function.name,
|
|
|
|
&js,
|
2018-07-09 11:07:57 -05:00
|
|
|
Some(format_doc_comments(&export.comments, Some(js_doc))),
|
2018-06-27 22:42:34 -07:00
|
|
|
);
|
2018-02-06 15:52:44 -08:00
|
|
|
self.cx.globals.push_str("\n");
|
|
|
|
self.cx.typescript.push_str("export ");
|
|
|
|
self.cx.typescript.push_str(&ts);
|
|
|
|
self.cx.typescript.push_str("\n");
|
2018-04-25 11:42:22 -07:00
|
|
|
Ok(())
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
|
|
|
|
2018-11-28 09:25:51 -08:00
|
|
|
fn set_start_function(&mut self, start: &str) -> Result<(), Error> {
|
|
|
|
if let Some(prev) = &self.cx.start {
|
2018-11-30 13:04:05 -08:00
|
|
|
bail!(
|
|
|
|
"cannot flag `{}` as start function as `{}` is \
|
|
|
|
already the start function",
|
|
|
|
start,
|
|
|
|
prev
|
|
|
|
);
|
2018-11-28 09:25:51 -08:00
|
|
|
}
|
|
|
|
self.cx.start = Some(start.to_string());
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn generate_export_for_class(
|
|
|
|
&mut self,
|
2018-08-26 15:43:33 -07:00
|
|
|
class_name: &'b str,
|
|
|
|
export: &decode::Export,
|
2018-04-25 11:42:22 -07:00
|
|
|
) -> Result<(), Error> {
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
let wasm_name =
|
|
|
|
wasm_bindgen_shared::struct_function_export_name(class_name, &export.function.name);
|
2018-06-18 13:48:57 -07:00
|
|
|
|
|
|
|
let descriptor = match self.cx.describe(&wasm_name) {
|
|
|
|
None => return Ok(()),
|
|
|
|
Some(d) => d,
|
|
|
|
};
|
|
|
|
|
2018-09-21 15:45:31 -07:00
|
|
|
let function_name = if export.is_constructor {
|
|
|
|
"constructor"
|
|
|
|
} else {
|
|
|
|
&export.function.name
|
|
|
|
};
|
|
|
|
let (js, ts, js_doc) = Js2Rust::new(function_name, self.cx)
|
2018-06-28 20:09:11 -05:00
|
|
|
.method(export.method, export.consumed)
|
2018-09-26 08:26:00 -07:00
|
|
|
.constructor(if export.is_constructor {
|
|
|
|
Some(class_name)
|
|
|
|
} else {
|
|
|
|
None
|
2018-11-27 12:07:59 -08:00
|
|
|
})
|
2019-03-14 08:46:42 -03:00
|
|
|
.process(descriptor.unwrap_function(), &export.function.arg_names)?
|
2018-10-18 08:43:36 -07:00
|
|
|
.finish(
|
|
|
|
"",
|
|
|
|
&format!("wasm.{}", wasm_name),
|
|
|
|
ExportedShim::Named(&wasm_name),
|
|
|
|
);
|
2018-07-09 11:07:57 -05:00
|
|
|
|
2018-06-27 22:42:34 -07:00
|
|
|
let class = self
|
|
|
|
.cx
|
|
|
|
.exported_classes
|
2019-01-14 13:11:07 -08:00
|
|
|
.as_mut()
|
|
|
|
.expect("classes already written")
|
2018-06-27 22:42:34 -07:00
|
|
|
.entry(class_name.to_string())
|
2018-02-07 16:41:33 -08:00
|
|
|
.or_insert(ExportedClass::default());
|
2018-06-27 22:42:34 -07:00
|
|
|
class
|
|
|
|
.contents
|
2018-07-09 11:07:57 -05:00
|
|
|
.push_str(&format_doc_comments(&export.comments, Some(js_doc)));
|
2018-09-21 15:45:31 -07:00
|
|
|
|
2019-02-01 15:44:42 +09:00
|
|
|
class.typescript.push_str(" "); // Indentation
|
|
|
|
|
2018-09-21 15:45:31 -07:00
|
|
|
if export.is_constructor {
|
|
|
|
if class.has_constructor {
|
2018-09-26 08:26:00 -07:00
|
|
|
bail!("found duplicate constructor `{}`", export.function.name);
|
2018-09-21 15:45:31 -07:00
|
|
|
}
|
|
|
|
class.has_constructor = true;
|
|
|
|
} else if !export.method {
|
2018-04-03 13:20:56 -07:00
|
|
|
class.contents.push_str("static ");
|
|
|
|
class.typescript.push_str("static ");
|
|
|
|
}
|
2018-04-14 15:34:11 +02:00
|
|
|
|
2018-09-21 15:45:31 -07:00
|
|
|
class.contents.push_str(function_name);
|
2018-02-07 16:41:33 -08:00
|
|
|
class.contents.push_str(&js);
|
|
|
|
class.contents.push_str("\n");
|
|
|
|
class.typescript.push_str(&ts);
|
|
|
|
class.typescript.push_str("\n");
|
2018-04-25 11:42:22 -07:00
|
|
|
Ok(())
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
|
|
|
|
2018-08-26 15:43:33 -07:00
|
|
|
fn generate_import(&mut self, import: &decode::Import<'b>) -> Result<(), Error> {
|
2018-03-21 09:55:16 -07:00
|
|
|
match import.kind {
|
2018-08-26 15:43:33 -07:00
|
|
|
decode::ImportKind::Function(ref f) => {
|
2018-06-27 22:42:34 -07:00
|
|
|
self.generate_import_function(import, f).with_context(|_| {
|
|
|
|
format!(
|
|
|
|
"failed to generate bindings for JS import `{}`",
|
|
|
|
f.function.name
|
|
|
|
)
|
|
|
|
})?;
|
2018-03-21 09:55:16 -07:00
|
|
|
}
|
2018-08-26 15:43:33 -07:00
|
|
|
decode::ImportKind::Static(ref s) => {
|
2018-06-27 22:42:34 -07:00
|
|
|
self.generate_import_static(import, s).with_context(|_| {
|
|
|
|
format!("failed to generate bindings for JS import `{}`", s.name)
|
|
|
|
})?;
|
2018-03-21 09:55:16 -07:00
|
|
|
}
|
2018-08-26 15:43:33 -07:00
|
|
|
decode::ImportKind::Type(ref ty) => {
|
2018-08-04 09:41:59 -07:00
|
|
|
self.generate_import_type(import, ty).with_context(|_| {
|
2018-09-26 08:26:00 -07:00
|
|
|
format!("failed to generate bindings for JS import `{}`", ty.name,)
|
2018-08-04 09:41:59 -07:00
|
|
|
})?;
|
|
|
|
}
|
2018-08-26 15:43:33 -07:00
|
|
|
decode::ImportKind::Enum(_) => {}
|
2018-03-21 08:09:59 -07:00
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
Ok(())
|
2018-03-21 09:55:16 -07:00
|
|
|
}
|
2018-03-21 08:09:59 -07:00
|
|
|
|
2018-04-25 11:42:22 -07:00
|
|
|
fn generate_import_static(
|
|
|
|
&mut self,
|
2018-08-26 15:43:33 -07:00
|
|
|
info: &decode::Import<'b>,
|
|
|
|
import: &decode::ImportStatic<'b>,
|
2018-06-27 22:42:34 -07:00
|
|
|
) -> Result<(), Error> {
|
2018-08-27 09:59:47 -07:00
|
|
|
// The same static can be imported in multiple locations, so only
|
|
|
|
// generate bindings once for it.
|
2018-08-26 15:43:33 -07:00
|
|
|
if !self.cx.imported_statics.insert(import.shim) {
|
2018-09-26 08:26:00 -07:00
|
|
|
return Ok(());
|
2018-08-27 09:59:47 -07:00
|
|
|
}
|
|
|
|
|
2018-03-21 09:55:16 -07:00
|
|
|
// TODO: should support more types to import here
|
2018-04-25 11:42:22 -07:00
|
|
|
let obj = self.import_name(info, &import.name)?;
|
2018-10-18 08:43:36 -07:00
|
|
|
self.cx
|
|
|
|
.anyref
|
|
|
|
.import_xform("__wbindgen_placeholder__", &import.shim, &[], true);
|
|
|
|
let body = format!("function() {{ return {}; }}", self.cx.add_heap_object(&obj));
|
|
|
|
self.cx.export(&import.shim, &body, None);
|
2018-04-25 11:42:22 -07:00
|
|
|
Ok(())
|
2018-03-21 08:09:59 -07:00
|
|
|
}
|
|
|
|
|
2018-06-27 22:42:34 -07:00
|
|
|
fn generate_import_function(
|
|
|
|
&mut self,
|
2018-08-26 15:43:33 -07:00
|
|
|
info: &decode::Import<'b>,
|
|
|
|
import: &decode::ImportFunction<'b>,
|
2018-06-27 22:42:34 -07:00
|
|
|
) -> Result<(), Error> {
|
2018-06-29 15:52:31 -07:00
|
|
|
if !self.cx.wasm_import_needed(&import.shim) {
|
2018-07-07 10:20:31 -07:00
|
|
|
return Ok(());
|
2018-06-29 15:52:31 -07:00
|
|
|
}
|
|
|
|
|
2018-08-27 09:59:47 -07:00
|
|
|
// It's possible for the same function to be imported in two locations,
|
|
|
|
// but we only want to generate one.
|
2018-08-26 15:43:33 -07:00
|
|
|
if !self.cx.imported_functions.insert(import.shim) {
|
2018-08-27 09:59:47 -07:00
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
2018-06-18 13:48:57 -07:00
|
|
|
let descriptor = match self.cx.describe(&import.shim) {
|
|
|
|
None => return Ok(()),
|
|
|
|
Some(d) => d,
|
|
|
|
};
|
2018-02-06 19:04:12 -08:00
|
|
|
|
2018-11-12 11:59:13 -08:00
|
|
|
// Figure out the name that we're importing to dangle further references
|
|
|
|
// off of. This is the function name if there's no method all here, or
|
|
|
|
// the class if there's a method call.
|
|
|
|
let name = match &import.method {
|
|
|
|
Some(data) => self.determine_import(info, &data.class)?,
|
|
|
|
None => self.determine_import(info, &import.function.name)?,
|
2018-08-28 15:19:31 -07:00
|
|
|
};
|
|
|
|
|
2018-11-12 11:59:13 -08:00
|
|
|
// Build up our shim's state, and we'll use that to guide whether we
|
|
|
|
// actually emit an import here or not.
|
|
|
|
let mut shim = Rust2Js::new(self.cx);
|
2019-01-14 15:59:31 -08:00
|
|
|
if shim.cx.config.debug {
|
|
|
|
shim.catch_and_rethrow(true);
|
|
|
|
}
|
2018-11-12 11:59:13 -08:00
|
|
|
shim.catch(import.catch)
|
|
|
|
.variadic(import.variadic)
|
|
|
|
.process(descriptor.unwrap_function())?;
|
2018-11-08 12:31:39 -08:00
|
|
|
|
2018-11-12 11:59:13 -08:00
|
|
|
// If this is a bare function import and the shim doesn't actually do
|
|
|
|
// anything (all argument/return conversions are noops) then we can wire
|
|
|
|
// up the wasm import directly to the destination. We don't actually
|
|
|
|
// wire up anything here, but we record it to get wired up later.
|
|
|
|
if import.method.is_none() && shim.is_noop() {
|
2018-11-27 12:07:59 -08:00
|
|
|
if let Import::Module {
|
|
|
|
module,
|
|
|
|
name,
|
|
|
|
field: None,
|
|
|
|
} = name
|
|
|
|
{
|
2018-11-12 11:59:13 -08:00
|
|
|
shim.cx.direct_imports.insert(import.shim, (module, name));
|
2018-10-18 08:43:36 -07:00
|
|
|
|
|
|
|
if shim.ret_anyref || shim.anyref_args.len() > 0 {
|
|
|
|
shim.cx.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
&import.shim,
|
|
|
|
&shim.anyref_args,
|
|
|
|
shim.ret_anyref,
|
|
|
|
);
|
|
|
|
}
|
2018-11-27 12:07:59 -08:00
|
|
|
return Ok(());
|
2018-11-08 12:31:39 -08:00
|
|
|
}
|
|
|
|
}
|
2018-08-28 15:19:31 -07:00
|
|
|
|
2018-11-12 11:59:13 -08:00
|
|
|
// If the above optimization fails then we actually generate the import
|
|
|
|
// here (possibly emitting some glue in our JS module) and then emit the
|
|
|
|
// shim as the wasm will be importing the shim.
|
|
|
|
let target = shim.cx.generated_import_target(name, import)?;
|
2018-10-18 08:43:36 -07:00
|
|
|
let js = shim.finish(&target, &import.shim)?;
|
2018-11-12 11:59:13 -08:00
|
|
|
shim.cx.export(&import.shim, &js, None);
|
|
|
|
Ok(())
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-02-22 12:01:38 +01:00
|
|
|
|
2018-08-04 09:41:59 -07:00
|
|
|
fn generate_import_type(
|
|
|
|
&mut self,
|
2018-08-26 15:43:33 -07:00
|
|
|
info: &decode::Import<'b>,
|
2018-11-12 11:59:13 -08:00
|
|
|
import: &decode::ImportType<'b>,
|
2018-08-04 09:41:59 -07:00
|
|
|
) -> Result<(), Error> {
|
|
|
|
if !self.cx.wasm_import_needed(&import.instanceof_shim) {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
let name = self.import_name(info, &import.name)?;
|
2018-10-18 08:43:36 -07:00
|
|
|
self.cx.anyref.import_xform(
|
|
|
|
"__wbindgen_placeholder__",
|
|
|
|
&import.instanceof_shim,
|
|
|
|
&[(0, false)],
|
|
|
|
false,
|
|
|
|
);
|
2018-09-26 08:26:00 -07:00
|
|
|
let body = format!(
|
2018-10-18 08:43:36 -07:00
|
|
|
"function(idx) {{ return {} instanceof {} ? 1 : 0; }}",
|
|
|
|
self.cx.get_object("idx"),
|
|
|
|
name
|
2018-08-04 09:41:59 -07:00
|
|
|
);
|
|
|
|
self.cx.export(&import.instanceof_shim, &body, None);
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-08-26 15:43:33 -07:00
|
|
|
fn generate_enum(&mut self, enum_: &decode::Enum) {
|
2018-02-22 12:01:38 +01:00
|
|
|
let mut variants = String::new();
|
|
|
|
|
|
|
|
for variant in enum_.variants.iter() {
|
2018-02-23 14:17:53 +01:00
|
|
|
variants.push_str(&format!("{}:{},", variant.name, variant.value));
|
2018-02-22 12:01:38 +01:00
|
|
|
}
|
2018-06-27 22:42:34 -07:00
|
|
|
self.cx.export(
|
|
|
|
&enum_.name,
|
|
|
|
&format!("Object.freeze({{ {} }})", variants),
|
2018-07-09 11:07:57 -05:00
|
|
|
Some(format_doc_comments(&enum_.comments, None)),
|
2018-06-27 22:42:34 -07:00
|
|
|
);
|
|
|
|
self.cx
|
|
|
|
.typescript
|
|
|
|
.push_str(&format!("export enum {} {{", enum_.name));
|
2018-02-23 17:30:18 +01:00
|
|
|
|
|
|
|
for variant in enum_.variants.iter() {
|
2019-02-01 16:25:25 +09:00
|
|
|
self.cx
|
|
|
|
.typescript
|
|
|
|
.push_str(&format!("\n {},", variant.name));
|
2018-02-23 17:30:18 +01:00
|
|
|
}
|
2019-02-01 16:25:25 +09:00
|
|
|
self.cx.typescript.push_str("\n}\n");
|
2018-02-22 12:01:38 +01:00
|
|
|
}
|
2018-03-21 09:55:16 -07:00
|
|
|
|
2018-08-26 15:43:33 -07:00
|
|
|
fn generate_struct(&mut self, struct_: &decode::Struct) -> Result<(), Error> {
|
2018-10-10 10:21:19 -07:00
|
|
|
let mut dst = String::new();
|
|
|
|
let mut ts_dst = String::new();
|
|
|
|
for field in struct_.fields.iter() {
|
Migrate `wasm-bindgen` to using `walrus`
This commit moves `wasm-bindgen` the CLI tool from internally using
`parity-wasm` for wasm parsing/serialization to instead use `walrus`.
The `walrus` crate is something we've been working on recently with an
aim to replace the usage of `parity-wasm` in `wasm-bindgen` to make the
current CLI tool more maintainable as well as more future-proof.
The `walrus` crate provides a much nicer AST to work with as well as a
structured `Module`, whereas `parity-wasm` provides a very raw interface
to the wasm module which isn't really appropriate for our use case. The
many transformations and tweaks that wasm-bindgen does have a huge
amount of ad-hoc index management to carefully craft a final wasm
binary, but this is all entirely taken care for us with the `walrus`
crate.
Additionally, `wasm-bindgen` will ingest and rewrite the wasm file,
often changing the binary offsets of functions. Eventually with DWARF
debug information we'll need to be sure to preserve the debug
information throughout the transformations that `wasm-bindgen` does
today. This is practically impossible to do with the `parity-wasm`
architecture, but `walrus` was designed from the get-go to solve this
problem transparently in the `walrus` crate itself. (it doesn't today,
but this is planned work)
It is the intention that this does not end up regressing any
`wasm-bindgen` use cases, neither in functionality or in speed. As a
large change and refactoring, however, it's likely that at least
something will arise! We'll want to continue to remain vigilant to any
issues that come up with this commit.
Note that the `gc` crate has been deleted as part of this change, as the
`gc` crate is no longer necessary since `walrus` does it automatically.
Additionally the `gc` crate was one of the main problems with preserving
debug information as it often deletes wasm items!
Finally, this also starts moving crates to the 2018 edition where
necessary since `walrus` requires the 2018 edition, and in general it's
more pleasant to work within the 2018 edition!
2019-01-31 09:54:23 -08:00
|
|
|
let wasm_getter = wasm_bindgen_shared::struct_field_get(&struct_.name, &field.name);
|
|
|
|
let wasm_setter = wasm_bindgen_shared::struct_field_set(&struct_.name, &field.name);
|
2018-10-10 10:21:19 -07:00
|
|
|
let descriptor = match self.cx.describe(&wasm_getter) {
|
|
|
|
None => continue,
|
|
|
|
Some(d) => d,
|
|
|
|
};
|
|
|
|
|
|
|
|
let set = {
|
2018-10-18 08:43:36 -07:00
|
|
|
let setter = ExportedShim::Named(&wasm_setter);
|
2018-10-10 10:21:19 -07:00
|
|
|
let mut cx = Js2Rust::new(&field.name, self.cx);
|
|
|
|
cx.method(true, false)
|
2019-03-14 08:46:42 -03:00
|
|
|
.argument(&descriptor, None)?
|
2018-10-10 10:21:19 -07:00
|
|
|
.ret(&Descriptor::Unit)?;
|
|
|
|
ts_dst.push_str(&format!(
|
2019-02-01 16:07:31 +09:00
|
|
|
"\n {}{}: {};",
|
2018-10-10 10:21:19 -07:00
|
|
|
if field.readonly { "readonly " } else { "" },
|
|
|
|
field.name,
|
|
|
|
&cx.js_arguments[0].1
|
|
|
|
));
|
2018-10-18 08:43:36 -07:00
|
|
|
cx.finish("", &format!("wasm.{}", wasm_setter), setter).0
|
2018-10-10 10:21:19 -07:00
|
|
|
};
|
2018-10-18 08:43:36 -07:00
|
|
|
let getter = ExportedShim::Named(&wasm_getter);
|
2018-10-10 10:21:19 -07:00
|
|
|
let (get, _ts, js_doc) = Js2Rust::new(&field.name, self.cx)
|
|
|
|
.method(true, false)
|
|
|
|
.ret(&descriptor)?
|
2018-10-18 08:43:36 -07:00
|
|
|
.finish("", &format!("wasm.{}", wasm_getter), getter);
|
2018-10-10 10:21:19 -07:00
|
|
|
if !dst.ends_with("\n") {
|
|
|
|
dst.push_str("\n");
|
|
|
|
}
|
|
|
|
dst.push_str(&format_doc_comments(&field.comments, Some(js_doc)));
|
|
|
|
dst.push_str("get ");
|
|
|
|
dst.push_str(&field.name);
|
|
|
|
dst.push_str(&get);
|
|
|
|
dst.push_str("\n");
|
|
|
|
if !field.readonly {
|
|
|
|
dst.push_str("set ");
|
|
|
|
dst.push_str(&field.name);
|
|
|
|
dst.push_str(&set);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let class = self
|
|
|
|
.cx
|
|
|
|
.exported_classes
|
2019-01-14 13:11:07 -08:00
|
|
|
.as_mut()
|
|
|
|
.expect("classes already written")
|
2018-08-26 15:43:33 -07:00
|
|
|
.entry(struct_.name.to_string())
|
2018-10-10 10:21:19 -07:00
|
|
|
.or_insert_with(Default::default);
|
|
|
|
class.comments = format_doc_comments(&struct_.comments, None);
|
|
|
|
class.contents.push_str(&dst);
|
|
|
|
class.contents.push_str("\n");
|
|
|
|
class.typescript.push_str(&ts_dst);
|
|
|
|
class.typescript.push_str("\n");
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-11-27 12:07:59 -08:00
|
|
|
fn register_vendor_prefix(&mut self, info: &decode::ImportType<'b>) {
|
2018-10-01 12:33:33 -07:00
|
|
|
if info.vendor_prefixes.len() == 0 {
|
2018-11-27 12:07:59 -08:00
|
|
|
return;
|
2018-09-28 13:17:37 -07:00
|
|
|
}
|
2018-10-01 12:33:33 -07:00
|
|
|
self.vendor_prefixes
|
2018-08-26 15:43:33 -07:00
|
|
|
.entry(info.name)
|
2018-09-28 13:17:37 -07:00
|
|
|
.or_insert(Vec::new())
|
2018-10-01 12:33:33 -07:00
|
|
|
.extend(info.vendor_prefixes.iter().cloned());
|
2018-09-28 13:17:37 -07:00
|
|
|
}
|
|
|
|
|
2018-11-27 12:07:59 -08:00
|
|
|
fn determine_import(
|
|
|
|
&self,
|
|
|
|
import: &decode::Import<'b>,
|
|
|
|
item: &'b str,
|
|
|
|
) -> Result<Import<'b>, Error> {
|
2019-03-19 11:25:13 -07:00
|
|
|
// First up, imports don't work at all in `--target no-modules` mode as
|
|
|
|
// we're not sure how to import them.
|
2019-02-25 11:11:30 -08:00
|
|
|
let is_local_snippet = match import.module {
|
|
|
|
decode::ImportModule::Named(s) => self.cx.local_modules.contains_key(s),
|
2019-03-15 08:04:25 -07:00
|
|
|
decode::ImportModule::RawNamed(_) => false,
|
2019-02-25 11:11:30 -08:00
|
|
|
decode::ImportModule::Inline(_) => true,
|
|
|
|
decode::ImportModule::None => false,
|
|
|
|
};
|
|
|
|
if self.cx.config.mode.no_modules() {
|
|
|
|
if is_local_snippet {
|
|
|
|
bail!(
|
2019-03-19 11:25:13 -07:00
|
|
|
"local JS snippets are not supported with `--target no-modules`; \
|
|
|
|
use `--target web` or no flag instead",
|
2019-02-25 11:11:30 -08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
if let decode::ImportModule::Named(module) = &import.module {
|
2018-06-27 22:42:34 -07:00
|
|
|
bail!(
|
2019-03-19 11:25:13 -07:00
|
|
|
"import from `{}` module not allowed with `--target no-modules`; \
|
|
|
|
use `nodejs`, `web`, or `bundler` target instead",
|
2018-06-27 22:42:34 -07:00
|
|
|
module
|
|
|
|
);
|
2018-04-11 14:22:20 +05:45
|
|
|
}
|
2018-07-30 10:50:43 -07:00
|
|
|
}
|
2018-04-11 14:22:20 +05:45
|
|
|
|
2019-02-25 11:11:30 -08:00
|
|
|
// FIXME: currently we require that local JS snippets are written in ES
|
|
|
|
// module syntax for imports/exports, but nodejs uses CommonJS to handle
|
|
|
|
// this meaning that local JS snippets are basically guaranteed to be
|
|
|
|
// incompatible. We need to implement a pass that translates the ES
|
|
|
|
// module syntax in the snippet to a CommonJS module, which is in theory
|
|
|
|
// not that hard but is a chunk of work to do.
|
|
|
|
if is_local_snippet && self.cx.config.mode.nodejs() {
|
2019-02-26 08:44:04 -08:00
|
|
|
// have a small unergonomic escape hatch for our webidl-tests tests
|
|
|
|
if env::var("WBINDGEN_I_PROMISE_JS_SYNTAX_WORKS_IN_NODE").is_err() {
|
|
|
|
bail!(
|
2019-03-19 11:25:13 -07:00
|
|
|
"local JS snippets are not supported with `--target nodejs`; \
|
2019-02-26 08:44:04 -08:00
|
|
|
see rustwasm/rfcs#6 for more details, but this restriction \
|
|
|
|
will be lifted in the future"
|
|
|
|
);
|
|
|
|
}
|
2019-02-25 11:11:30 -08:00
|
|
|
}
|
|
|
|
|
2019-03-19 11:25:13 -07:00
|
|
|
// Similar to `--target no-modules`, only allow vendor prefixes
|
|
|
|
// basically for web apis, shouldn't be necessary for things like npm
|
|
|
|
// packages or other imported items.
|
2018-10-01 12:33:33 -07:00
|
|
|
let vendor_prefixes = self.vendor_prefixes.get(item);
|
|
|
|
if let Some(vendor_prefixes) = vendor_prefixes {
|
|
|
|
assert!(vendor_prefixes.len() > 0);
|
2018-09-28 13:17:37 -07:00
|
|
|
|
2019-02-25 11:11:30 -08:00
|
|
|
if is_local_snippet {
|
|
|
|
bail!(
|
|
|
|
"local JS snippets do not support vendor prefixes for \
|
|
|
|
the import of `{}` with a polyfill of `{}`",
|
|
|
|
item,
|
|
|
|
&vendor_prefixes[0]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if let decode::ImportModule::Named(module) = &import.module {
|
2018-09-28 13:17:37 -07:00
|
|
|
bail!(
|
|
|
|
"import of `{}` from `{}` has a polyfill of `{}` listed, but
|
2018-10-01 12:33:33 -07:00
|
|
|
vendor prefixes aren't supported when importing from modules",
|
2018-09-28 13:17:37 -07:00
|
|
|
item,
|
|
|
|
module,
|
2018-10-01 12:33:33 -07:00
|
|
|
&vendor_prefixes[0],
|
2018-09-28 13:17:37 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
if let Some(ns) = &import.js_namespace {
|
2018-11-27 12:07:59 -08:00
|
|
|
bail!(
|
|
|
|
"import of `{}` through js namespace `{}` isn't supported \
|
|
|
|
right now when it lists a polyfill",
|
|
|
|
item,
|
|
|
|
ns
|
|
|
|
);
|
2018-09-28 13:17:37 -07:00
|
|
|
}
|
2018-11-12 11:59:13 -08:00
|
|
|
return Ok(Import::VendorPrefixed {
|
|
|
|
name: item,
|
|
|
|
prefixes: vendor_prefixes.clone(),
|
2018-11-27 12:07:59 -08:00
|
|
|
});
|
2018-09-28 13:17:37 -07:00
|
|
|
}
|
|
|
|
|
2019-02-25 11:11:30 -08:00
|
|
|
let (name, field) = match import.js_namespace {
|
|
|
|
Some(ns) => (ns, Some(item)),
|
|
|
|
None => (item, None),
|
2018-11-27 12:07:59 -08:00
|
|
|
};
|
2018-07-30 10:50:43 -07:00
|
|
|
|
2018-11-12 11:59:13 -08:00
|
|
|
Ok(match import.module {
|
2019-02-25 11:11:30 -08:00
|
|
|
decode::ImportModule::Named(module) if is_local_snippet => Import::LocalModule {
|
|
|
|
module,
|
|
|
|
name,
|
|
|
|
field,
|
|
|
|
},
|
2019-03-15 08:04:25 -07:00
|
|
|
decode::ImportModule::Named(module) | decode::ImportModule::RawNamed(module) => {
|
|
|
|
Import::Module {
|
|
|
|
module,
|
|
|
|
name,
|
|
|
|
field,
|
|
|
|
}
|
|
|
|
}
|
2019-03-05 14:53:14 -08:00
|
|
|
decode::ImportModule::Inline(idx) => {
|
|
|
|
let offset = *self
|
|
|
|
.cx
|
|
|
|
.snippet_offsets
|
|
|
|
.get(self.program.unique_crate_identifier)
|
|
|
|
.unwrap_or(&0);
|
|
|
|
Import::InlineJs {
|
|
|
|
unique_crate_identifier: self.program.unique_crate_identifier,
|
|
|
|
snippet_idx_in_crate: idx as usize + offset,
|
|
|
|
name,
|
|
|
|
field,
|
|
|
|
}
|
|
|
|
}
|
2019-02-25 11:11:30 -08:00
|
|
|
decode::ImportModule::None => Import::Global { name, field },
|
2018-11-12 11:59:13 -08:00
|
|
|
})
|
|
|
|
}
|
2018-07-30 10:50:43 -07:00
|
|
|
|
2018-11-27 12:07:59 -08:00
|
|
|
fn import_name(&mut self, import: &decode::Import<'b>, item: &'b str) -> Result<String, Error> {
|
2018-11-12 11:59:13 -08:00
|
|
|
let import = self.determine_import(import, item)?;
|
|
|
|
Ok(self.cx.import_identifier(import))
|
|
|
|
}
|
2019-02-27 12:20:33 -08:00
|
|
|
|
|
|
|
fn add_package_json(&mut self, path: &'b str) -> Result<(), Error> {
|
|
|
|
if !self.cx.package_json_read.insert(path) {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
if !self.cx.config.mode.nodejs() && !self.cx.config.mode.bundler() {
|
2019-03-26 08:00:16 -07:00
|
|
|
bail!(
|
|
|
|
"NPM dependencies have been specified in `{}` but \
|
|
|
|
this is only compatible with the `bundler` and `nodejs` targets"
|
|
|
|
);
|
2019-02-27 12:20:33 -08:00
|
|
|
}
|
|
|
|
let contents = fs::read_to_string(path).context(format!("failed to read `{}`", path))?;
|
|
|
|
let json: serde_json::Value = serde_json::from_str(&contents)?;
|
|
|
|
let object = match json.as_object() {
|
|
|
|
Some(s) => s,
|
|
|
|
None => bail!(
|
|
|
|
"expected `package.json` to have an JSON object in `{}`",
|
|
|
|
path
|
|
|
|
),
|
|
|
|
};
|
|
|
|
let mut iter = object.iter();
|
|
|
|
let (key, value) = match iter.next() {
|
|
|
|
Some(pair) => pair,
|
|
|
|
None => return Ok(()),
|
|
|
|
};
|
|
|
|
if key != "dependencies" || iter.next().is_some() {
|
|
|
|
bail!(
|
|
|
|
"NPM manifest found at `{}` can currently only have one key, \
|
|
|
|
`dependencies`, and no other fields",
|
|
|
|
path
|
|
|
|
);
|
|
|
|
}
|
|
|
|
let value = match value.as_object() {
|
|
|
|
Some(s) => s,
|
|
|
|
None => bail!("expected `dependencies` to be a JSON object in `{}`", path),
|
|
|
|
};
|
|
|
|
|
|
|
|
for (name, value) in value.iter() {
|
|
|
|
let value = match value.as_str() {
|
|
|
|
Some(s) => s,
|
|
|
|
None => bail!(
|
|
|
|
"keys in `dependencies` are expected to be strings in `{}`",
|
|
|
|
path
|
|
|
|
),
|
|
|
|
};
|
|
|
|
if let Some((prev, _prev_version)) = self.cx.npm_dependencies.get(name) {
|
|
|
|
bail!(
|
|
|
|
"dependency on NPM package `{}` specified in two `package.json` files, \
|
|
|
|
which at the time is not allowed:\n * {}\n * {}",
|
|
|
|
name,
|
|
|
|
path,
|
|
|
|
prev
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
self.cx
|
|
|
|
.npm_dependencies
|
|
|
|
.insert(name.to_string(), (path, value.to_string()));
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2018-11-12 11:59:13 -08:00
|
|
|
}
|
|
|
|
|
2019-02-25 11:11:30 -08:00
|
|
|
#[derive(Hash, Eq, PartialEq)]
|
|
|
|
pub enum ImportModule<'a> {
|
|
|
|
Named(&'a str),
|
2019-03-05 14:53:14 -08:00
|
|
|
Inline(&'a str, usize),
|
2019-02-25 11:11:30 -08:00
|
|
|
None,
|
|
|
|
}
|
|
|
|
|
2018-11-12 11:59:13 -08:00
|
|
|
impl<'a> Import<'a> {
|
2019-02-25 11:11:30 -08:00
|
|
|
fn module(&self) -> ImportModule<'a> {
|
2018-11-12 11:59:13 -08:00
|
|
|
match self {
|
2019-02-25 11:11:30 -08:00
|
|
|
Import::Module { module, .. } | Import::LocalModule { module, .. } => {
|
|
|
|
ImportModule::Named(module)
|
|
|
|
}
|
2019-03-05 14:53:14 -08:00
|
|
|
Import::InlineJs {
|
|
|
|
unique_crate_identifier,
|
|
|
|
snippet_idx_in_crate,
|
|
|
|
..
|
|
|
|
} => ImportModule::Inline(unique_crate_identifier, *snippet_idx_in_crate),
|
2019-02-25 11:11:30 -08:00
|
|
|
Import::Global { .. } | Import::VendorPrefixed { .. } => ImportModule::None,
|
2018-11-12 11:59:13 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn field(&self) -> Option<&'a str> {
|
|
|
|
match self {
|
2019-02-25 11:11:30 -08:00
|
|
|
Import::Module { field, .. }
|
|
|
|
| Import::LocalModule { field, .. }
|
|
|
|
| Import::InlineJs { field, .. }
|
|
|
|
| Import::Global { field, .. } => *field,
|
2018-11-12 11:59:13 -08:00
|
|
|
Import::VendorPrefixed { .. } => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn name(&self) -> &'a str {
|
|
|
|
match self {
|
2018-11-27 12:07:59 -08:00
|
|
|
Import::Module { name, .. }
|
2019-02-25 11:11:30 -08:00
|
|
|
| Import::LocalModule { name, .. }
|
|
|
|
| Import::InlineJs { name, .. }
|
2018-11-27 12:07:59 -08:00
|
|
|
| Import::Global { name, .. }
|
|
|
|
| Import::VendorPrefixed { name, .. } => *name,
|
2018-03-21 09:55:16 -07:00
|
|
|
}
|
2018-07-30 10:50:43 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn generate_identifier(name: &str, used_names: &mut HashMap<String, usize>) -> String {
|
|
|
|
let cnt = used_names.entry(name.to_string()).or_insert(0);
|
|
|
|
*cnt += 1;
|
2018-12-11 20:42:37 +02:00
|
|
|
// We want to mangle `default` at once, so we can support default exports and don't generate
|
|
|
|
// invalid glue code like this: `import { default } from './module';`.
|
|
|
|
if *cnt == 1 && name != "default" {
|
2018-07-30 10:50:43 -07:00
|
|
|
name.to_string()
|
|
|
|
} else {
|
|
|
|
format!("{}{}", name, cnt)
|
2018-03-21 09:55:16 -07:00
|
|
|
}
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2018-04-16 13:31:56 -07:00
|
|
|
|
2018-08-26 15:43:33 -07:00
|
|
|
fn format_doc_comments(comments: &[&str], js_doc_comments: Option<String>) -> String {
|
2018-06-27 22:42:34 -07:00
|
|
|
let body: String = comments
|
|
|
|
.iter()
|
|
|
|
.map(|c| format!("*{}\n", c.trim_matches('"')))
|
|
|
|
.collect();
|
2018-07-09 11:07:57 -05:00
|
|
|
let doc = if let Some(docs) = js_doc_comments {
|
2018-07-13 22:36:51 -07:00
|
|
|
docs.lines().map(|l| format!("* {} \n", l)).collect()
|
2018-07-09 11:07:57 -05:00
|
|
|
} else {
|
|
|
|
String::new()
|
|
|
|
};
|
|
|
|
format!("/**\n{}{}*/\n", body, doc)
|
2018-06-27 22:42:34 -07:00
|
|
|
}
|
2018-12-11 20:42:37 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_generate_identifier() {
|
|
|
|
let mut used_names: HashMap<String, usize> = HashMap::new();
|
|
|
|
assert_eq!(
|
|
|
|
generate_identifier("someVar", &mut used_names),
|
|
|
|
"someVar".to_string()
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
generate_identifier("someVar", &mut used_names),
|
|
|
|
"someVar2".to_string()
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
generate_identifier("default", &mut used_names),
|
|
|
|
"default1".to_string()
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
generate_identifier("default", &mut used_names),
|
|
|
|
"default2".to_string()
|
|
|
|
);
|
|
|
|
}
|