2018-07-19 14:57:04 -05:00
|
|
|
#![doc(html_root_url = "https://docs.rs/wasm-bindgen-cli-support/0.2")]
|
|
|
|
|
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};
|
2018-03-29 01:47:44 -07:00
|
|
|
use std::collections::BTreeSet;
|
2018-08-15 17:52:26 -07:00
|
|
|
use std::env;
|
2018-07-25 16:06:47 -07:00
|
|
|
use std::fs;
|
2018-07-25 15:49:50 -07:00
|
|
|
use std::mem;
|
2017-12-24 15:32:40 -08:00
|
|
|
use std::path::{Path, PathBuf};
|
2018-08-26 15:43:33 -07:00
|
|
|
use std::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
|
|
|
use walrus::Module;
|
2017-12-14 19:31:01 -08:00
|
|
|
|
2018-08-26 15:43:33 -07:00
|
|
|
mod decode;
|
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
|
|
|
mod descriptor;
|
2018-06-27 22:42:34 -07:00
|
|
|
mod js;
|
2018-01-29 21:20:38 -08:00
|
|
|
pub mod wasm2es6js;
|
2017-12-18 12:39:14 -08:00
|
|
|
|
2017-12-14 19:31:01 -08:00
|
|
|
pub struct Bindgen {
|
2018-07-25 15:49:50 -07:00
|
|
|
input: Input,
|
2018-12-04 21:35:05 -05:00
|
|
|
out_name: Option<String>,
|
2017-12-14 21:55:21 -08:00
|
|
|
nodejs: bool,
|
2018-07-04 22:37:09 -05:00
|
|
|
nodejs_experimental_modules: bool,
|
2018-03-28 07:37:56 -07:00
|
|
|
browser: bool,
|
2018-04-11 13:59:58 +05:45
|
|
|
no_modules: bool,
|
2018-04-19 13:33:58 -07:00
|
|
|
no_modules_global: Option<String>,
|
2017-12-19 09:25:41 -08:00
|
|
|
debug: bool,
|
2018-01-29 21:20:38 -08:00
|
|
|
typescript: bool,
|
2018-04-11 11:43:18 -07:00
|
|
|
demangle: bool,
|
2018-07-13 08:10:51 -07:00
|
|
|
keep_debug: bool,
|
2018-11-09 07:45:19 -08:00
|
|
|
remove_name_section: bool,
|
2019-02-14 10:08:24 -08:00
|
|
|
remove_producers_section: bool,
|
2018-11-28 09:25:51 -08:00
|
|
|
emit_start: bool,
|
2018-08-15 17:52:26 -07:00
|
|
|
// Experimental support for `WeakRefGroup`, an upcoming ECMAScript feature.
|
|
|
|
// Currently only enable-able through an env var.
|
|
|
|
weak_refs: bool,
|
2018-10-04 20:00:23 -07:00
|
|
|
// Experimental support for the wasm threads proposal, transforms the wasm
|
|
|
|
// module to be "ready to be instantiated on any thread"
|
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
|
|
|
threads: Option<wasm_bindgen_threads_xform::Config>,
|
2018-10-18 08:43:36 -07:00
|
|
|
anyref: bool,
|
2019-02-20 08:39:46 -08:00
|
|
|
encode_into: EncodeInto,
|
2017-12-14 19:31:01 -08:00
|
|
|
}
|
|
|
|
|
2018-07-25 15:49:50 -07:00
|
|
|
enum Input {
|
|
|
|
Path(PathBuf),
|
|
|
|
Module(Module, String),
|
|
|
|
None,
|
|
|
|
}
|
|
|
|
|
2019-02-20 08:39:46 -08:00
|
|
|
pub enum EncodeInto {
|
|
|
|
Test,
|
|
|
|
Always,
|
|
|
|
Never,
|
|
|
|
}
|
|
|
|
|
2017-12-14 19:31:01 -08:00
|
|
|
impl Bindgen {
|
|
|
|
pub fn new() -> Bindgen {
|
|
|
|
Bindgen {
|
2018-07-25 15:49:50 -07:00
|
|
|
input: Input::None,
|
2018-12-04 21:35:05 -05:00
|
|
|
out_name: None,
|
2017-12-14 21:55:21 -08:00
|
|
|
nodejs: false,
|
2018-07-04 22:37:09 -05:00
|
|
|
nodejs_experimental_modules: false,
|
2018-03-28 07:37:56 -07:00
|
|
|
browser: false,
|
2018-04-11 13:59:58 +05:45
|
|
|
no_modules: false,
|
2018-04-19 13:33:58 -07:00
|
|
|
no_modules_global: None,
|
2017-12-19 09:25:41 -08:00
|
|
|
debug: false,
|
2018-01-29 21:20:38 -08:00
|
|
|
typescript: false,
|
2018-04-11 11:43:18 -07:00
|
|
|
demangle: true,
|
2018-07-13 08:10:51 -07:00
|
|
|
keep_debug: false,
|
2018-11-09 07:45:19 -08:00
|
|
|
remove_name_section: false,
|
2019-02-15 11:22:46 -08:00
|
|
|
remove_producers_section: false,
|
2018-11-28 09:25:51 -08:00
|
|
|
emit_start: true,
|
2018-08-15 17:52:26 -07:00
|
|
|
weak_refs: env::var("WASM_BINDGEN_WEAKREF").is_ok(),
|
2018-10-04 20:00:23 -07:00
|
|
|
threads: threads_config(),
|
2018-10-18 08:43:36 -07:00
|
|
|
anyref: env::var("WASM_BINDGEN_ANYREF").is_ok(),
|
2019-02-20 08:39:46 -08:00
|
|
|
encode_into: EncodeInto::Test,
|
2017-12-14 19:31:01 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn input_path<P: AsRef<Path>>(&mut self, path: P) -> &mut Bindgen {
|
2018-07-25 15:49:50 -07:00
|
|
|
self.input = Input::Path(path.as_ref().to_path_buf());
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-12-04 21:35:05 -05:00
|
|
|
pub fn out_name(&mut self, name: &str) -> &mut Bindgen {
|
|
|
|
self.out_name = Some(name.to_string());
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-07-25 15:49:50 -07:00
|
|
|
/// Explicitly specify the already parsed input module.
|
2018-10-08 10:01:53 -07:00
|
|
|
pub fn input_module(&mut self, name: &str, module: Module) -> &mut Bindgen {
|
2018-07-25 15:49:50 -07:00
|
|
|
let name = name.to_string();
|
2018-10-08 10:01:53 -07:00
|
|
|
self.input = Input::Module(module, name);
|
|
|
|
return self;
|
2017-12-14 19:31:01 -08:00
|
|
|
}
|
|
|
|
|
2017-12-14 21:55:21 -08:00
|
|
|
pub fn nodejs(&mut self, node: bool) -> &mut Bindgen {
|
|
|
|
self.nodejs = node;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-07-04 22:37:09 -05:00
|
|
|
pub fn nodejs_experimental_modules(&mut self, node: bool) -> &mut Bindgen {
|
|
|
|
self.nodejs_experimental_modules = node;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-03-28 07:37:56 -07:00
|
|
|
pub fn browser(&mut self, browser: bool) -> &mut Bindgen {
|
|
|
|
self.browser = browser;
|
2018-03-07 08:50:56 -08:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-04-11 13:59:58 +05:45
|
|
|
pub fn no_modules(&mut self, no_modules: bool) -> &mut Bindgen {
|
|
|
|
self.no_modules = no_modules;
|
2018-04-04 20:06:53 +05:45
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-04-19 13:33:58 -07:00
|
|
|
pub fn no_modules_global(&mut self, name: &str) -> &mut Bindgen {
|
|
|
|
self.no_modules_global = Some(name.to_string());
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2017-12-19 09:25:41 -08:00
|
|
|
pub fn debug(&mut self, debug: bool) -> &mut Bindgen {
|
|
|
|
self.debug = debug;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-01-29 21:20:38 -08:00
|
|
|
pub fn typescript(&mut self, typescript: bool) -> &mut Bindgen {
|
|
|
|
self.typescript = typescript;
|
2017-12-24 15:32:40 -08:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-04-11 11:43:18 -07:00
|
|
|
pub fn demangle(&mut self, demangle: bool) -> &mut Bindgen {
|
|
|
|
self.demangle = demangle;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-07-13 08:10:51 -07:00
|
|
|
pub fn keep_debug(&mut self, keep_debug: bool) -> &mut Bindgen {
|
|
|
|
self.keep_debug = keep_debug;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-11-09 07:45:19 -08:00
|
|
|
pub fn remove_name_section(&mut self, remove: bool) -> &mut Bindgen {
|
|
|
|
self.remove_name_section = remove;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2019-02-14 10:08:24 -08:00
|
|
|
pub fn remove_producers_section(&mut self, remove: bool) -> &mut Bindgen {
|
|
|
|
self.remove_producers_section = remove;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-11-28 09:25:51 -08:00
|
|
|
pub fn emit_start(&mut self, emit: bool) -> &mut Bindgen {
|
|
|
|
self.emit_start = emit;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2019-02-20 08:39:46 -08:00
|
|
|
pub fn encode_into(&mut self, mode: EncodeInto) -> &mut Bindgen {
|
|
|
|
self.encode_into = mode;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-01-29 21:20:38 -08:00
|
|
|
pub fn generate<P: AsRef<Path>>(&mut self, path: P) -> Result<(), Error> {
|
|
|
|
self._generate(path.as_ref())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn _generate(&mut self, out_dir: &Path) -> Result<(), Error> {
|
2018-07-25 15:49:50 -07:00
|
|
|
let (mut module, stem) = match self.input {
|
|
|
|
Input::None => bail!("must have an input by now"),
|
|
|
|
Input::Module(ref mut m, ref name) => {
|
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 blank_module = Module::default();
|
2018-07-25 15:49:50 -07:00
|
|
|
(mem::replace(m, blank_module), &name[..])
|
|
|
|
}
|
|
|
|
Input::Path(ref path) => {
|
2018-07-25 16:06:47 -07:00
|
|
|
let contents = fs::read(&path)
|
2018-07-25 15:49:50 -07:00
|
|
|
.with_context(|_| format!("failed to read `{}`", path.display()))?;
|
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 module = walrus::ModuleConfig::new()
|
|
|
|
// Skip validation of the module as LLVM's output is
|
|
|
|
// generally already well-formed and so we won't gain much
|
|
|
|
// from re-validating. Additionally LLVM's current output
|
|
|
|
// for threads includes atomic instructions but doesn't
|
|
|
|
// include shared memory, so it fails that part of
|
|
|
|
// validation!
|
|
|
|
.strict_validate(false)
|
2019-02-14 06:35:40 -08:00
|
|
|
.generate_dwarf(self.keep_debug)
|
|
|
|
.generate_name_section(!self.remove_name_section)
|
2019-02-14 10:08:24 -08:00
|
|
|
.generate_producers_section(!self.remove_producers_section)
|
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
|
|
|
.parse(&contents)
|
2018-07-25 15:49:50 -07:00
|
|
|
.context("failed to parse input file as wasm")?;
|
2018-12-04 21:35:05 -05:00
|
|
|
let stem = match &self.out_name {
|
|
|
|
Some(name) => &name,
|
|
|
|
None => path.file_stem().unwrap().to_str().unwrap(),
|
|
|
|
};
|
2018-07-25 15:49:50 -07:00
|
|
|
(module, stem)
|
|
|
|
}
|
2017-12-14 19:31:01 -08:00
|
|
|
};
|
2018-10-18 08:43:36 -07:00
|
|
|
|
|
|
|
// This isn't the hardest thing in the world too support but we
|
|
|
|
// basically don't know how to rationalize #[wasm_bindgen(start)] and
|
|
|
|
// the actual `start` function if present. Figure this out later if it
|
|
|
|
// comes up, but otherwise we should continue to be compatible with
|
|
|
|
// LLVM's output today.
|
|
|
|
//
|
|
|
|
// Note that start function handling in `js/mod.rs` will need to be
|
|
|
|
// updated as well, because `#[wasm_bindgen(start)]` is inserted *after*
|
|
|
|
// a module's start function, if any, because we assume start functions
|
|
|
|
// only show up when injected on behalf of wasm-bindgen's passes.
|
|
|
|
if module.start.is_some() {
|
|
|
|
bail!("wasm-bindgen is currently incompatible with modules that \
|
|
|
|
already have a start function");
|
|
|
|
}
|
|
|
|
|
2018-08-26 15:43:33 -07:00
|
|
|
let mut program_storage = Vec::new();
|
|
|
|
let programs = extract_programs(&mut module, &mut program_storage)
|
2018-04-25 11:42:22 -07:00
|
|
|
.with_context(|_| "failed to extract wasm-bindgen custom sections")?;
|
2018-02-06 15:52:44 -08:00
|
|
|
|
2018-10-04 20:00:23 -07:00
|
|
|
if let Some(cfg) = &self.threads {
|
|
|
|
cfg.run(&mut module)
|
|
|
|
.with_context(|_| "failed to prepare module for threading")?;
|
|
|
|
}
|
|
|
|
|
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.demangle {
|
|
|
|
demangle(&mut module);
|
|
|
|
}
|
|
|
|
|
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
|
|
|
// Here we're actually instantiating the module we've parsed above for
|
|
|
|
// execution. Why, you might be asking, are we executing wasm code? A
|
|
|
|
// good question!
|
|
|
|
//
|
|
|
|
// Transmitting information from `#[wasm_bindgen]` here to the CLI tool
|
|
|
|
// is pretty tricky. Specifically information about the types involved
|
|
|
|
// with a function signature (especially generic ones) can be hefty to
|
|
|
|
// translate over. As a result, the macro emits a bunch of shims which,
|
|
|
|
// when executed, will describe to us what the types look like.
|
|
|
|
//
|
|
|
|
// This means that whenever we encounter an import or export we'll
|
|
|
|
// execute a shim function which informs us about its type so we can
|
|
|
|
// then generate the appropriate bindings.
|
2019-02-19 13:07:00 -08:00
|
|
|
let mut instance = wasm_bindgen_wasm_interpreter::Interpreter::new(&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
|
|
|
|
|
|
|
let mut memories = module.memories.iter().map(|m| m.id());
|
|
|
|
let memory = memories.next();
|
|
|
|
if memories.next().is_some() {
|
|
|
|
bail!("multiple memories currently not supported");
|
|
|
|
}
|
|
|
|
drop(memories);
|
|
|
|
let memory = memory.unwrap_or_else(|| module.memories.add_local(false, 1, None));
|
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-02-06 15:52:44 -08:00
|
|
|
let (js, ts) = {
|
|
|
|
let mut cx = js::Context {
|
|
|
|
globals: String::new(),
|
|
|
|
imports: String::new(),
|
2018-09-28 13:17:37 -07:00
|
|
|
imports_post: String::new(),
|
2018-03-29 01:47:44 -07:00
|
|
|
footer: String::new(),
|
2018-02-06 15:52:44 -08:00
|
|
|
typescript: format!("/* tslint:disable */\n"),
|
2019-01-14 13:11:07 -08:00
|
|
|
exposed_globals: Some(Default::default()),
|
2018-02-06 15:52:44 -08:00
|
|
|
required_internal_exports: Default::default(),
|
2018-02-07 16:41:33 -08:00
|
|
|
imported_names: Default::default(),
|
2018-07-30 10:50:43 -07:00
|
|
|
imported_identifiers: Default::default(),
|
2019-01-14 13:11:07 -08:00
|
|
|
exported_classes: Some(Default::default()),
|
2018-02-06 15:52:44 -08:00
|
|
|
config: &self,
|
|
|
|
module: &mut module,
|
2018-04-04 08:24:19 -07:00
|
|
|
function_table_needed: false,
|
2018-08-19 17:07:30 -07:00
|
|
|
interpreter: &mut instance,
|
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,
|
2018-08-27 09:59:47 -07:00
|
|
|
imported_functions: Default::default(),
|
|
|
|
imported_statics: Default::default(),
|
2018-11-12 11:59:13 -08:00
|
|
|
direct_imports: Default::default(),
|
2018-11-28 09:25:51 -08:00
|
|
|
start: None,
|
2018-10-18 08:43:36 -07:00
|
|
|
anyref: Default::default(),
|
2018-02-06 15:52:44 -08:00
|
|
|
};
|
2018-10-18 08:43:36 -07:00
|
|
|
cx.anyref.enabled = self.anyref;
|
|
|
|
cx.anyref.prepare(cx.module)?;
|
2018-02-06 15:52:44 -08:00
|
|
|
for program in programs.iter() {
|
|
|
|
js::SubContext {
|
|
|
|
program,
|
|
|
|
cx: &mut cx,
|
2018-10-01 12:33:33 -07:00
|
|
|
vendor_prefixes: Default::default(),
|
2018-11-27 12:07:59 -08:00
|
|
|
}
|
|
|
|
.generate()?;
|
2018-02-06 15:52:44 -08:00
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
cx.finalize(stem)?
|
2018-02-06 15:52:44 -08:00
|
|
|
};
|
2018-01-29 21:20:38 -08:00
|
|
|
|
2018-09-26 08:26:00 -07:00
|
|
|
let extension = if self.nodejs_experimental_modules {
|
|
|
|
"mjs"
|
|
|
|
} else {
|
|
|
|
"js"
|
|
|
|
};
|
2018-07-04 22:37:09 -05:00
|
|
|
let js_path = out_dir.join(stem).with_extension(extension);
|
2018-07-25 16:06:47 -07:00
|
|
|
fs::write(&js_path, reset_indentation(&js))
|
2018-04-25 11:42:22 -07:00
|
|
|
.with_context(|_| format!("failed to write `{}`", js_path.display()))?;
|
2018-01-29 21:20:38 -08:00
|
|
|
|
|
|
|
if self.typescript {
|
2018-11-27 12:36:09 -08:00
|
|
|
let ts_path = js_path.with_extension("d.ts");
|
2018-07-25 16:06:47 -07:00
|
|
|
fs::write(&ts_path, ts)
|
2018-04-25 11:42:22 -07:00
|
|
|
.with_context(|_| format!("failed to write `{}`", ts_path.display()))?;
|
2018-01-29 21:20:38 -08:00
|
|
|
}
|
2017-12-14 19:31:01 -08:00
|
|
|
|
2018-03-05 22:25:14 +01:00
|
|
|
let wasm_path = out_dir.join(format!("{}_bg", stem)).with_extension("wasm");
|
2018-03-29 01:47:44 -07:00
|
|
|
|
|
|
|
if self.nodejs {
|
2018-07-04 22:37:09 -05:00
|
|
|
let js_path = wasm_path.with_extension(extension);
|
2018-03-29 01:47:44 -07:00
|
|
|
let shim = self.generate_node_wasm_import(&module, &wasm_path);
|
2018-07-25 16:06:47 -07:00
|
|
|
fs::write(&js_path, shim)
|
2018-04-25 11:42:22 -07:00
|
|
|
.with_context(|_| format!("failed to write `{}`", js_path.display()))?;
|
2018-03-29 01:47:44 -07:00
|
|
|
}
|
|
|
|
|
2018-11-27 12:36:09 -08:00
|
|
|
if self.typescript {
|
|
|
|
let ts_path = wasm_path.with_extension("d.ts");
|
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 ts = wasm2es6js::typescript(&module)?;
|
2018-11-27 12:36:09 -08:00
|
|
|
fs::write(&ts_path, ts)
|
|
|
|
.with_context(|_| format!("failed to write `{}`", ts_path.display()))?;
|
|
|
|
}
|
|
|
|
|
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_bytes = module.emit_wasm()?;
|
2018-07-25 16:06:47 -07:00
|
|
|
fs::write(&wasm_path, wasm_bytes)
|
2018-04-25 11:42:22 -07:00
|
|
|
.with_context(|_| format!("failed to write `{}`", wasm_path.display()))?;
|
2018-11-27 12:36:09 -08:00
|
|
|
|
2017-12-14 19:31:01 -08:00
|
|
|
Ok(())
|
|
|
|
}
|
2018-03-29 01:47:44 -07:00
|
|
|
|
|
|
|
fn generate_node_wasm_import(&self, m: &Module, path: &Path) -> String {
|
|
|
|
let mut imports = BTreeSet::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 m.imports.iter() {
|
|
|
|
imports.insert(&import.module);
|
2018-03-29 01:47:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut shim = String::new();
|
2018-07-04 22:37:09 -05:00
|
|
|
|
|
|
|
if self.nodejs_experimental_modules {
|
|
|
|
for (i, module) in imports.iter().enumerate() {
|
2018-09-26 08:26:00 -07:00
|
|
|
shim.push_str(&format!("import * as import{} from '{}';\n", i, module));
|
2018-07-04 22:37:09 -05:00
|
|
|
}
|
|
|
|
// On windows skip the leading `/` which comes out when we parse a
|
|
|
|
// url to use `C:\...` instead of `\C:\...`
|
2018-09-26 08:26:00 -07:00
|
|
|
shim.push_str(&format!(
|
|
|
|
"
|
2018-07-04 22:37:09 -05:00
|
|
|
import * as path from 'path';
|
|
|
|
import * as fs from 'fs';
|
|
|
|
import * as url from 'url';
|
|
|
|
import * as process from 'process';
|
|
|
|
|
|
|
|
let file = path.dirname(url.parse(import.meta.url).pathname);
|
|
|
|
if (process.platform === 'win32') {{
|
|
|
|
file = file.substring(1);
|
|
|
|
}}
|
|
|
|
const bytes = fs.readFileSync(path.join(file, '{}'));
|
2018-09-26 08:26:00 -07:00
|
|
|
",
|
|
|
|
path.file_name().unwrap().to_str().unwrap()
|
|
|
|
));
|
2018-07-04 22:37:09 -05:00
|
|
|
} else {
|
2018-09-26 08:26:00 -07:00
|
|
|
shim.push_str(&format!(
|
|
|
|
"
|
2018-07-04 22:37:09 -05:00
|
|
|
const path = require('path').join(__dirname, '{}');
|
|
|
|
const bytes = require('fs').readFileSync(path);
|
2018-09-26 08:26:00 -07:00
|
|
|
",
|
|
|
|
path.file_name().unwrap().to_str().unwrap()
|
|
|
|
));
|
2018-07-04 22:37:09 -05:00
|
|
|
}
|
2018-03-29 01:47:44 -07:00
|
|
|
shim.push_str("let imports = {};\n");
|
2018-07-04 22:37:09 -05:00
|
|
|
for (i, module) in imports.iter().enumerate() {
|
|
|
|
if self.nodejs_experimental_modules {
|
|
|
|
shim.push_str(&format!("imports['{}'] = import{};\n", module, i));
|
|
|
|
} else {
|
|
|
|
shim.push_str(&format!("imports['{0}'] = require('{0}');\n", module));
|
|
|
|
}
|
2018-03-29 01:47:44 -07:00
|
|
|
}
|
|
|
|
|
2018-06-27 22:42:34 -07:00
|
|
|
shim.push_str(&format!(
|
|
|
|
"
|
2018-07-04 22:37:09 -05:00
|
|
|
const wasmModule = new WebAssembly.Module(bytes);
|
|
|
|
const wasmInstance = new WebAssembly.Instance(wasmModule, imports);
|
2018-06-15 23:39:51 -07:00
|
|
|
",
|
2018-06-27 22:42:34 -07:00
|
|
|
));
|
2018-03-29 01:47:44 -07:00
|
|
|
|
2018-07-04 22:37:09 -05:00
|
|
|
if self.nodejs_experimental_modules {
|
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 entry in m.exports.iter() {
|
|
|
|
shim.push_str("export const ");
|
|
|
|
shim.push_str(&entry.name);
|
|
|
|
shim.push_str(" = wasmInstance.exports.");
|
|
|
|
shim.push_str(&entry.name);
|
|
|
|
shim.push_str(";\n");
|
2018-07-04 22:37:09 -05:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
shim.push_str("module.exports = wasmInstance.exports;\n");
|
|
|
|
}
|
|
|
|
|
2018-06-15 12:55:37 -05:00
|
|
|
reset_indentation(&shim)
|
2018-03-29 01:47:44 -07:00
|
|
|
}
|
2017-12-14 19:31:01 -08:00
|
|
|
}
|
|
|
|
|
2018-08-26 15:43:33 -07:00
|
|
|
fn extract_programs<'a>(
|
|
|
|
module: &mut Module,
|
|
|
|
program_storage: &'a mut Vec<Vec<u8>>,
|
|
|
|
) -> Result<Vec<decode::Program<'a>>, 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 my_version = wasm_bindgen_shared::version();
|
2018-04-25 11:42:22 -07:00
|
|
|
let mut to_remove = Vec::new();
|
2018-08-26 15:43:33 -07:00
|
|
|
assert!(program_storage.is_empty());
|
2017-12-14 19:31:01 -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
|
|
|
for (i, custom) in module.custom.iter_mut().enumerate() {
|
|
|
|
if custom.name != "__wasm_bindgen_unstable" {
|
2018-06-27 22:42:34 -07:00
|
|
|
continue;
|
2018-03-14 14:33:53 -07:00
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
to_remove.push(i);
|
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
|
|
|
program_storage.push(mem::replace(&mut custom.value, Vec::new()));
|
2018-08-26 15:43:33 -07:00
|
|
|
}
|
2018-03-14 14:33:53 -07:00
|
|
|
|
2018-08-26 15:43:33 -07:00
|
|
|
for i in to_remove.into_iter().rev() {
|
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
|
|
|
module.custom.remove(i);
|
2018-08-26 15:43:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut ret = Vec::new();
|
|
|
|
for program in program_storage.iter() {
|
|
|
|
let mut payload = &program[..];
|
|
|
|
while let Some(data) = get_remaining(&mut payload) {
|
|
|
|
// Historical versions of wasm-bindgen have used JSON as the custom
|
|
|
|
// data section format. Newer versions, however, are using a custom
|
|
|
|
// serialization protocol that looks much more like the wasm spec.
|
|
|
|
//
|
|
|
|
// We, however, want a sanity check to ensure that if we're running
|
|
|
|
// against the wrong wasm-bindgen we get a nicer error than an
|
|
|
|
// internal decode error. To that end we continue to verify a tiny
|
|
|
|
// bit of json at the beginning of each blob before moving to the
|
|
|
|
// next blob. This should keep us compatible with older wasm-bindgen
|
|
|
|
// instances as well as forward-compatible for now.
|
|
|
|
//
|
|
|
|
// Note, though, that as `wasm-pack` picks up steam it's hoped we
|
|
|
|
// can just delete this entirely. The `wasm-pack` project already
|
|
|
|
// manages versions for us, so we in theory should need this check
|
|
|
|
// less and less over time.
|
|
|
|
if let Some(their_version) = verify_schema_matches(data)? {
|
2018-06-27 22:42:34 -07:00
|
|
|
bail!(
|
|
|
|
"
|
2018-03-01 19:32:05 -08:00
|
|
|
|
|
|
|
it looks like the Rust project used to create this wasm file was linked against
|
|
|
|
a different version of wasm-bindgen than this binary:
|
|
|
|
|
|
|
|
rust wasm file: {}
|
|
|
|
this binary: {}
|
|
|
|
|
|
|
|
Currently the bindgen format is unstable enough that these two version must
|
|
|
|
exactly match, so it's required that these two version are kept in sync by
|
|
|
|
either updating the wasm-bindgen dependency or this binary. You should be able
|
|
|
|
to update the wasm-bindgen dependency with:
|
|
|
|
|
|
|
|
cargo update -p wasm-bindgen
|
|
|
|
|
|
|
|
or you can update the binary with
|
|
|
|
|
2018-04-20 19:07:56 -07:00
|
|
|
cargo install -f wasm-bindgen-cli
|
2018-03-01 19:32:05 -08:00
|
|
|
|
|
|
|
if this warning fails to go away though and you're not sure what to do feel free
|
2018-07-19 14:57:04 -05:00
|
|
|
to open an issue at https://github.com/rustwasm/wasm-bindgen/issues!
|
2018-03-01 19:32:05 -08:00
|
|
|
",
|
2018-08-26 15:43:33 -07:00
|
|
|
their_version,
|
|
|
|
my_version,
|
2018-06-27 22:42:34 -07:00
|
|
|
);
|
2018-03-01 19:32:05 -08:00
|
|
|
}
|
2018-08-26 15:43:33 -07:00
|
|
|
let next = get_remaining(&mut payload).unwrap();
|
|
|
|
ret.push(<decode::Program as decode::Decode>::decode_all(next));
|
2018-03-09 16:09:07 -08:00
|
|
|
}
|
2018-04-25 11:42:22 -07:00
|
|
|
}
|
2018-08-26 15:43:33 -07:00
|
|
|
Ok(ret)
|
|
|
|
}
|
2018-03-14 10:50:12 -07:00
|
|
|
|
2018-08-26 15:43:33 -07:00
|
|
|
fn get_remaining<'a>(data: &mut &'a [u8]) -> Option<&'a [u8]> {
|
|
|
|
if data.len() == 0 {
|
2018-11-27 12:07:59 -08:00
|
|
|
return None;
|
2018-04-25 11:42:22 -07:00
|
|
|
}
|
2018-08-26 15:43:33 -07:00
|
|
|
let len = ((data[0] as usize) << 0)
|
|
|
|
| ((data[1] as usize) << 8)
|
|
|
|
| ((data[2] as usize) << 16)
|
|
|
|
| ((data[3] as usize) << 24);
|
|
|
|
let (a, b) = data[4..].split_at(len);
|
|
|
|
*data = b;
|
|
|
|
Some(a)
|
|
|
|
}
|
|
|
|
|
2018-11-27 12:07:59 -08:00
|
|
|
fn verify_schema_matches<'a>(data: &'a [u8]) -> Result<Option<&'a str>, Error> {
|
2018-08-26 15:43:33 -07:00
|
|
|
macro_rules! bad {
|
2018-11-27 12:07:59 -08:00
|
|
|
() => {
|
|
|
|
bail!("failed to decode what looked like wasm-bindgen data")
|
|
|
|
};
|
2018-08-26 15:43:33 -07:00
|
|
|
}
|
|
|
|
let data = match str::from_utf8(data) {
|
|
|
|
Ok(s) => s,
|
|
|
|
Err(_) => bad!(),
|
|
|
|
};
|
|
|
|
if !data.starts_with("{") || !data.ends_with("}") {
|
|
|
|
bad!()
|
|
|
|
}
|
|
|
|
let needle = "\"schema_version\":\"";
|
|
|
|
let rest = match data.find(needle) {
|
|
|
|
Some(i) => &data[i + needle.len()..],
|
|
|
|
None => bad!(),
|
|
|
|
};
|
|
|
|
let their_schema_version = match rest.find("\"") {
|
|
|
|
Some(i) => &rest[..i],
|
|
|
|
None => bad!(),
|
|
|
|
};
|
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 their_schema_version == wasm_bindgen_shared::SCHEMA_VERSION {
|
2018-11-27 12:07:59 -08:00
|
|
|
return Ok(None);
|
2018-08-26 15:43:33 -07:00
|
|
|
}
|
|
|
|
let needle = "\"version\":\"";
|
|
|
|
let rest = match data.find(needle) {
|
|
|
|
Some(i) => &data[i + needle.len()..],
|
|
|
|
None => bad!(),
|
|
|
|
};
|
|
|
|
let their_version = match rest.find("\"") {
|
|
|
|
Some(i) => &rest[..i],
|
|
|
|
None => bad!(),
|
|
|
|
};
|
|
|
|
Ok(Some(their_version))
|
2018-02-06 16:06:21 -08:00
|
|
|
}
|
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-06-15 12:55:37 -05:00
|
|
|
fn reset_indentation(s: &str) -> String {
|
2018-07-09 07:44:41 -07:00
|
|
|
let mut indent: u32 = 0;
|
|
|
|
let mut dst = String::new();
|
2018-06-15 12:55:37 -05:00
|
|
|
|
2018-07-09 07:44:41 -07:00
|
|
|
for line in s.lines() {
|
|
|
|
let line = line.trim();
|
2018-07-09 14:55:25 -05:00
|
|
|
if line.starts_with('}') || (line.ends_with('}') && !line.starts_with('*')) {
|
2018-07-09 07:44:41 -07:00
|
|
|
indent = indent.saturating_sub(1);
|
2018-06-15 12:55:37 -05:00
|
|
|
}
|
2018-09-26 08:26:00 -07:00
|
|
|
let extra = if line.starts_with(':') || line.starts_with('?') {
|
|
|
|
1
|
|
|
|
} else {
|
|
|
|
0
|
|
|
|
};
|
2018-09-13 22:13:07 -07:00
|
|
|
if !line.is_empty() {
|
|
|
|
for _ in 0..indent + extra {
|
|
|
|
dst.push_str(" ");
|
|
|
|
}
|
|
|
|
dst.push_str(line);
|
2018-06-15 12:55:37 -05:00
|
|
|
}
|
2018-07-09 07:44:41 -07:00
|
|
|
dst.push_str("\n");
|
|
|
|
if line.ends_with('{') {
|
|
|
|
indent += 1;
|
2018-06-15 12:55:37 -05:00
|
|
|
}
|
|
|
|
}
|
2018-09-26 08:26:00 -07:00
|
|
|
return dst;
|
2018-06-15 12:55:37 -05:00
|
|
|
}
|
2018-10-04 20:00:23 -07:00
|
|
|
|
|
|
|
// Eventually these will all be CLI options, but while they're unstable features
|
|
|
|
// they're left as environment variables. We don't guarantee anything about
|
|
|
|
// backwards-compatibility with these options.
|
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 threads_config() -> Option<wasm_bindgen_threads_xform::Config> {
|
2018-10-04 20:00:23 -07:00
|
|
|
if env::var("WASM_BINDGEN_THREADS").is_err() {
|
2018-11-27 12:07:59 -08:00
|
|
|
return None;
|
2018-10-04 20:00:23 -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 mut cfg = wasm_bindgen_threads_xform::Config::new();
|
2018-10-04 20:00:23 -07:00
|
|
|
if let Ok(s) = env::var("WASM_BINDGEN_THREADS_MAX_MEMORY") {
|
|
|
|
cfg.maximum_memory(s.parse().unwrap());
|
|
|
|
}
|
|
|
|
if let Ok(s) = env::var("WASM_BINDGEN_THREADS_STACK_SIZE") {
|
|
|
|
cfg.thread_stack_size(s.parse().unwrap());
|
|
|
|
}
|
|
|
|
Some(cfg)
|
|
|
|
}
|
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 demangle(module: &mut Module) {
|
|
|
|
for func in module.funcs.iter_mut() {
|
|
|
|
let name = match &func.name {
|
|
|
|
Some(name) => name,
|
|
|
|
None => continue,
|
|
|
|
};
|
|
|
|
if let Ok(sym) = rustc_demangle::try_demangle(name) {
|
|
|
|
func.name = Some(sym.to_string());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|