#![doc(html_root_url = "https://docs.rs/wasm-bindgen-cli-support/0.2")]
use anyhow::{bail, Context, Error};
use std::collections::{BTreeMap, HashMap, HashSet};
use std::env;
use std::fs;
use std::mem;
use std::path::{Path, PathBuf};
use std::str;
use walrus::Module;
pub(crate) const PLACEHOLDER_MODULE: &str = "__wbindgen_placeholder__";
mod decode;
mod descriptor;
mod descriptors;
mod externref;
mod intrinsic;
mod js;
mod multivalue;
pub mod wasm2es6js;
mod wit;
pub struct Bindgen {
input: Input,
out_name: Option<String>,
mode: OutputMode,
debug: bool,
typescript: bool,
omit_imports: bool,
demangle: bool,
keep_lld_exports: bool,
keep_debug: bool,
remove_name_section: bool,
remove_producers_section: bool,
omit_default_module_path: bool,
emit_start: bool,
threads: wasm_bindgen_threads_xform::Config,
externref: bool,
multi_value: bool,
encode_into: EncodeInto,
split_linked_modules: bool,
symbol_dispose: bool,
}
pub struct Output {
module: walrus::Module,
stem: String,
generated: Generated,
}
struct Generated {
mode: OutputMode,
js: String,
ts: String,
start: Option<String>,
snippets: HashMap<String, Vec<String>>,
local_modules: HashMap<String, String>,
npm_dependencies: HashMap<String, (PathBuf, String)>,
typescript: bool,
}
#[derive(Clone)]
enum OutputMode {
Bundler { browser_only: bool },
Web,
NoModules { global: String },
Node { module: bool },
Deno,
}
enum Input {
Path(PathBuf),
Module(Module, String),
Bytes(Vec<u8>, String),
None,
}
pub enum EncodeInto {
Test,
Always,
Never,
}
impl Bindgen {
pub fn new() -> Bindgen {
let externref =
env::var("WASM_BINDGEN_ANYREF").is_ok() || env::var("WASM_BINDGEN_EXTERNREF").is_ok();
let multi_value = env::var("WASM_BINDGEN_MULTI_VALUE").is_ok();
let symbol_dispose = env::var("WASM_BINDGEN_EXPERIMENTAL_SYMBOL_DISPOSE").is_ok();
Bindgen {
input: Input::None,
out_name: None,
mode: OutputMode::Bundler {
browser_only: false,
},
debug: false,
typescript: false,
omit_imports: false,
demangle: true,
keep_lld_exports: false,
keep_debug: false,
remove_name_section: false,
remove_producers_section: false,
emit_start: true,
threads: threads_config(),
externref,
multi_value,
encode_into: EncodeInto::Test,
omit_default_module_path: true,
split_linked_modules: false,
symbol_dispose,
}
}
pub fn input_path<P: AsRef<Path>>(&mut self, path: P) -> &mut Bindgen {
self.input = Input::Path(path.as_ref().to_path_buf());
self
}
pub fn out_name(&mut self, name: &str) -> &mut Bindgen {
self.out_name = Some(name.to_string());
self
}
#[deprecated = "automatically detected via `-Ctarget-feature=+reference-types`"]
pub fn reference_types(&mut self, enable: bool) -> &mut Bindgen {
self.externref = enable;
self
}
pub fn input_module(&mut self, name: &str, module: Module) -> &mut Bindgen {
let name = name.to_string();
self.input = Input::Module(module, name);
self
}
pub fn input_bytes(&mut self, name: &str, bytes: Vec<u8>) -> &mut Bindgen {
let name = name.to_string();
self.input = Input::Bytes(bytes, name);
self
}
fn switch_mode(&mut self, mode: OutputMode, flag: &str) -> Result<(), Error> {
match self.mode {
OutputMode::Bundler { .. } => self.mode = mode,
_ => bail!(
"cannot specify `{}` with another output mode already specified",
flag
),
}
Ok(())
}
pub fn nodejs(&mut self, node: bool) -> Result<&mut Bindgen, Error> {
if node {
self.switch_mode(OutputMode::Node { module: false }, "--target nodejs")?;
}
Ok(self)
}
pub fn nodejs_module(&mut self, node: bool) -> Result<&mut Bindgen, Error> {
if node {
self.switch_mode(
OutputMode::Node { module: true },
"--target experimental-nodejs-module",
)?;
}
Ok(self)
}
pub fn bundler(&mut self, bundler: bool) -> Result<&mut Bindgen, Error> {
if bundler {
self.switch_mode(
OutputMode::Bundler {
browser_only: false,
},
"--target bundler",
)?;
}
Ok(self)
}
pub fn web(&mut self, web: bool) -> Result<&mut Bindgen, Error> {
if web {
self.switch_mode(OutputMode::Web, "--target web")?;
}
Ok(self)
}
pub fn no_modules(&mut self, no_modules: bool) -> Result<&mut Bindgen, Error> {
if no_modules {
self.switch_mode(
OutputMode::NoModules {
global: "wasm_bindgen".to_string(),
},
"--target no-modules",
)?;
}
Ok(self)
}
pub fn browser(&mut self, browser: bool) -> Result<&mut Bindgen, Error> {
if browser {
match &mut self.mode {
OutputMode::Bundler { browser_only } => *browser_only = true,
_ => bail!("cannot specify `--browser` with other output types"),
}
}
Ok(self)
}
pub fn deno(&mut self, deno: bool) -> Result<&mut Bindgen, Error> {
if deno {
self.switch_mode(OutputMode::Deno, "--target deno")?;
self.encode_into(EncodeInto::Always);
}
Ok(self)
}
pub fn no_modules_global(&mut self, name: &str) -> Result<&mut Bindgen, Error> {
match &mut self.mode {
OutputMode::NoModules { global } => *global = name.to_string(),
_ => bail!("can only specify `--no-modules-global` with `--target no-modules`"),
}
Ok(self)
}
pub fn debug(&mut self, debug: bool) -> &mut Bindgen {
self.debug = debug;
self
}
pub fn typescript(&mut self, typescript: bool) -> &mut Bindgen {
self.typescript = typescript;
self
}
pub fn omit_imports(&mut self, omit_imports: bool) -> &mut Bindgen {
self.omit_imports = omit_imports;
self
}
pub fn demangle(&mut self, demangle: bool) -> &mut Bindgen {
self.demangle = demangle;
self
}
pub fn keep_lld_exports(&mut self, keep_lld_exports: bool) -> &mut Bindgen {
self.keep_lld_exports = keep_lld_exports;
self
}
pub fn keep_debug(&mut self, keep_debug: bool) -> &mut Bindgen {
self.keep_debug = keep_debug;
self
}
pub fn remove_name_section(&mut self, remove: bool) -> &mut Bindgen {
self.remove_name_section = remove;
self
}
pub fn remove_producers_section(&mut self, remove: bool) -> &mut Bindgen {
self.remove_producers_section = remove;
self
}
pub fn emit_start(&mut self, emit: bool) -> &mut Bindgen {
self.emit_start = emit;
self
}
pub fn encode_into(&mut self, mode: EncodeInto) -> &mut Bindgen {
self.encode_into = mode;
self
}
pub fn omit_default_module_path(&mut self, omit_default_module_path: bool) -> &mut Bindgen {
self.omit_default_module_path = omit_default_module_path;
self
}
pub fn split_linked_modules(&mut self, split_linked_modules: bool) -> &mut Bindgen {
self.split_linked_modules = split_linked_modules;
self
}
pub fn generate<P: AsRef<Path>>(&mut self, path: P) -> Result<(), Error> {
self.generate_output()?.emit(path.as_ref())
}
pub fn stem(&self) -> Result<&str, Error> {
Ok(match &self.input {
Input::None => bail!("must have an input by now"),
Input::Module(_, name) | Input::Bytes(_, name) => name,
Input::Path(path) => match &self.out_name {
Some(name) => name,
None => path.file_stem().unwrap().to_str().unwrap(),
},
})
}
pub fn generate_output(&mut self) -> Result<Output, Error> {
let mut module = match self.input {
Input::None => bail!("must have an input by now"),
Input::Module(ref mut m, _) => {
let blank_module = Module::default();
mem::replace(m, blank_module)
}
Input::Path(ref path) => {
let bytes = std::fs::read(path)
.with_context(|| format!("failed reading '{}'", path.display()))?;
self.module_from_bytes(&bytes).with_context(|| {
format!("failed getting Wasm module for '{}'", path.display())
})?
}
Input::Bytes(ref bytes, _) => self
.module_from_bytes(bytes)
.context("failed getting Wasm module")?,
};
if let Ok(true) = wasm_bindgen_wasm_conventions::target_feature(&module, "reference-types")
{
self.externref = true;
}
if let Ok(true) = wasm_bindgen_wasm_conventions::target_feature(&module, "multivalue") {
self.multi_value = true;
}
if matches!(self.mode, OutputMode::Web)
&& module.exports.iter().any(|export| export.name == "default")
{
bail!("exported symbol \"default\" not allowed for --target web")
}
let thread_count = self
.threads
.run(&mut module)
.with_context(|| "failed to prepare module for threading")?;
if self.demangle {
demangle(&mut module);
}
if !self.keep_lld_exports {
unexported_unused_lld_things(&mut module);
}
module
.producers
.add_processed_by("wasm-bindgen", &wasm_bindgen_shared::version());
let mut storage = Vec::new();
let programs = wit::extract_programs(&mut module, &mut storage)?;
descriptors::execute(&mut module)?;
wit::process(self, &mut module, programs, thread_count)?;
if self.externref {
externref::process(&mut module)?;
} else {
let ids = module
.exports
.iter()
.filter(|e| e.name.starts_with("__externref"))
.map(|e| e.id())
.collect::<Vec<_>>();
for id in ids {
module.exports.delete(id);
}
externref::force_contiguous_elements(&mut module)?;
}
if self.multi_value {
multivalue::run(&mut module)
.context("failed to transform return pointers into multi-value Wasm")?;
}
gc_module_and_adapters(&mut module);
let stem = self.stem()?;
let aux = module
.customs
.delete_typed::<wit::WasmBindgenAux>()
.expect("aux section should be present");
let adapters = module
.customs
.delete_typed::<wit::NonstandardWitSection>()
.unwrap();
let mut cx = js::Context::new(&mut module, self, &adapters, &aux)?;
cx.generate()?;
let (js, ts, start) = cx.finalize(stem)?;
let generated = Generated {
snippets: aux.snippets.clone(),
local_modules: aux.local_modules.clone(),
mode: self.mode.clone(),
typescript: self.typescript,
npm_dependencies: cx.npm_dependencies.clone(),
js,
ts,
start,
};
Ok(Output {
module,
stem: stem.to_string(),
generated,
})
}
fn module_from_bytes(&self, bytes: &[u8]) -> Result<Module, Error> {
walrus::ModuleConfig::new()
.strict_validate(false)
.generate_dwarf(self.keep_debug)
.generate_name_section(!self.remove_name_section)
.generate_producers_section(!self.remove_producers_section)
.parse(bytes)
.context("failed to parse input as wasm")
}
fn local_module_name(&self, module: &str) -> String {
format!("./snippets/{}", module)
}
fn inline_js_module_name(
&self,
unique_crate_identifier: &str,
snippet_idx_in_crate: usize,
) -> String {
format!(
"./snippets/{}/inline{}.js",
unique_crate_identifier, snippet_idx_in_crate,
)
}
}
fn reset_indentation(s: &str) -> String {
let mut indent: u32 = 0;
let mut dst = String::new();
fn is_doc_comment(line: &str) -> bool {
line.starts_with("*")
}
for line in s.lines() {
let line = line.trim();
if is_doc_comment(line) {
for _ in 0..indent {
dst.push_str(" ");
}
dst.push(' ');
dst.push_str(line);
dst.push('\n');
continue;
}
if line.starts_with('}') {
indent = indent.saturating_sub(1);
}
let extra = if line.starts_with(':') || line.starts_with('?') {
1
} else {
0
};
if !line.is_empty() {
for _ in 0..indent + extra {
dst.push_str(" ");
}
dst.push_str(line);
}
dst.push('\n');
if line.ends_with('{') {
indent += 1;
}
}
dst
}
fn threads_config() -> wasm_bindgen_threads_xform::Config {
let mut cfg = wasm_bindgen_threads_xform::Config::new();
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());
}
cfg
}
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());
}
}
}
impl OutputMode {
fn uses_es_modules(&self) -> bool {
matches!(
self,
OutputMode::Bundler { .. }
| OutputMode::Web
| OutputMode::Node { module: true }
| OutputMode::Deno
)
}
fn nodejs(&self) -> bool {
matches!(self, OutputMode::Node { .. })
}
fn no_modules(&self) -> bool {
matches!(self, OutputMode::NoModules { .. })
}
fn web(&self) -> bool {
matches!(self, OutputMode::Web)
}
fn esm_integration(&self) -> bool {
matches!(
self,
OutputMode::Bundler { .. } | OutputMode::Node { module: true }
)
}
}
fn unexported_unused_lld_things(module: &mut Module) {
let mut to_remove = Vec::new();
for export in module.exports.iter() {
match export.name.as_str() {
"__heap_base" | "__data_end" | "__indirect_function_table" => {
to_remove.push(export.id());
}
_ => {}
}
}
for id in to_remove {
module.exports.delete(id);
}
}
impl Output {
pub fn js(&self) -> &str {
&self.generated.js
}
pub fn ts(&self) -> Option<&str> {
if self.generated.typescript {
Some(&self.generated.ts)
} else {
None
}
}
pub fn start(&self) -> Option<&String> {
self.generated.start.as_ref()
}
pub fn snippets(&self) -> &HashMap<String, Vec<String>> {
&self.generated.snippets
}
pub fn local_modules(&self) -> &HashMap<String, String> {
&self.generated.local_modules
}
pub fn npm_dependencies(&self) -> &HashMap<String, (PathBuf, String)> {
&self.generated.npm_dependencies
}
pub fn wasm(&self) -> &walrus::Module {
&self.module
}
pub fn wasm_mut(&mut self) -> &mut walrus::Module {
&mut self.module
}
pub fn emit(&mut self, out_dir: impl AsRef<Path>) -> Result<(), Error> {
self._emit(out_dir.as_ref())
}
fn _emit(&mut self, out_dir: &Path) -> Result<(), Error> {
let wasm_name = format!("{}_bg", self.stem);
let wasm_path = out_dir.join(&wasm_name).with_extension("wasm");
fs::create_dir_all(out_dir)?;
let wasm_bytes = self.module.emit_wasm();
fs::write(&wasm_path, wasm_bytes)
.with_context(|| format!("failed to write `{}`", wasm_path.display()))?;
let gen = &self.generated;
for (identifier, list) in gen.snippets.iter() {
for (i, js) in list.iter().enumerate() {
let name = format!("inline{}.js", i);
let path = out_dir.join("snippets").join(identifier).join(name);
fs::create_dir_all(path.parent().unwrap())?;
fs::write(&path, js)
.with_context(|| format!("failed to write `{}`", path.display()))?;
}
}
for (path, contents) in gen.local_modules.iter() {
let path = out_dir.join("snippets").join(path);
fs::create_dir_all(path.parent().unwrap())?;
fs::write(&path, contents)
.with_context(|| format!("failed to write `{}`", path.display()))?;
}
let is_genmode_nodemodule = matches!(gen.mode, OutputMode::Node { module: true });
if !gen.npm_dependencies.is_empty() || is_genmode_nodemodule {
#[derive(serde::Serialize)]
struct PackageJson<'a> {
#[serde(rename = "type", skip_serializing_if = "Option::is_none")]
ty: Option<&'static str>,
dependencies: BTreeMap<&'a str, &'a str>,
}
let pj = PackageJson {
ty: is_genmode_nodemodule.then_some("module"),
dependencies: gen
.npm_dependencies
.iter()
.map(|(k, v)| (k.as_str(), v.1.as_str()))
.collect(),
};
let json = serde_json::to_string_pretty(&pj)?;
fs::write(out_dir.join("package.json"), json)?;
}
let extension = "js";
fn write<P, C>(path: P, contents: C) -> Result<(), anyhow::Error>
where
P: AsRef<Path>,
C: AsRef<[u8]>,
{
fs::write(&path, contents)
.with_context(|| format!("failed to write `{}`", path.as_ref().display()))
}
let js_path = out_dir.join(&self.stem).with_extension(extension);
if gen.mode.esm_integration() {
let js_name = format!("{}_bg.{}", self.stem, extension);
let start = gen.start.as_deref().unwrap_or("");
if matches!(gen.mode, OutputMode::Node { .. }) {
write(
&js_path,
format!(
"\
{start}
export * from \"./{js_name}\";",
),
)?;
} else {
write(
&js_path,
format!(
"\
import * as wasm from \"./{wasm_name}.wasm\";
export * from \"./{js_name}\";
{start}"
),
)?;
}
write(out_dir.join(&js_name), reset_indentation(&gen.js))?;
} else {
write(&js_path, reset_indentation(&gen.js))?;
}
if gen.typescript {
let ts_path = js_path.with_extension("d.ts");
fs::write(&ts_path, &gen.ts)
.with_context(|| format!("failed to write `{}`", ts_path.display()))?;
}
if gen.typescript {
let ts_path = wasm_path.with_extension("wasm.d.ts");
let ts = wasm2es6js::typescript(&self.module)?;
fs::write(&ts_path, ts)
.with_context(|| format!("failed to write `{}`", ts_path.display()))?;
}
Ok(())
}
}
fn gc_module_and_adapters(module: &mut Module) {
loop {
walrus::passes::gc::run(module);
let imports_remaining = module
.imports
.iter()
.map(|i| i.id())
.collect::<HashSet<_>>();
let mut section = module
.customs
.delete_typed::<wit::NonstandardWitSection>()
.unwrap();
section
.implements
.retain(|pair| imports_remaining.contains(&pair.0));
let aux = module.customs.get_typed::<wit::WasmBindgenAux>().unwrap();
let any_removed = section.gc(aux);
module.customs.add(*section);
if !any_removed {
break;
}
}
}
fn sorted_iter<K, V>(map: &HashMap<K, V>) -> impl Iterator<Item = (&K, &V)>
where
K: Ord,
{
let mut pairs = map.iter().collect::<Vec<_>>();
pairs.sort_by_key(|(k, _)| *k);
pairs.into_iter()
}