From ac156463c30d72ede1ad1d293a4a5a5a0297b0bc Mon Sep 17 00:00:00 2001 From: Yuta Saito Date: Wed, 19 Jun 2024 04:43:22 +0000 Subject: [PATCH] dep: Optionalize `wasm-opt` crate dependency The `wasm-opt` crate brings in C++ dependencies (binaryen) and it makes the build process more complex and slower (also setting up cross-compilation for C++ is not trivial unlike Rust). Also the DCE process doesn't make much difference if we allow all WASI subsystems. This commit makes the `wasm-opt` crate dependency optional so that crate users can choose whether to use it or not. --- Cargo.toml | 8 ++++++- src/lib.rs | 66 +++++++++++++++++++++++++++++++++--------------------- 2 files changed, 48 insertions(+), 26 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index abfd62f..3bfb728 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -33,7 +33,7 @@ toml = "0.8" walrus = "0.20.3" wasm-compose = "0.5.5" wasm-metadata = "0.10.20" -wasm-opt = "0.116.0" +wasm-opt = { version = "0.116.0", optional = true } wit-component = "0.21.0" [build-dependencies] @@ -51,3 +51,9 @@ wasmparser = "0.121.2" [workspace.dependencies] anyhow = "1" wit-bindgen = "0.25.0" + +[features] +default = ["wasm-opt"] +# Optimize the generated virtual adapter with wasm-opt to reduce its size. +# If you allow all WASI subsystems, this feature doesn't make much difference. +wasm-opt = ["dep:wasm-opt"] diff --git a/src/lib.rs b/src/lib.rs index da8f717..9f6f104 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1,6 +1,6 @@ use anyhow::{Context, Result}; use serde::Deserialize; -use std::{env, fs, time::SystemTime}; +use std::env; use virt_deny::{ deny_clocks_virt, deny_exit_virt, deny_http_virt, deny_random_virt, deny_sockets_virt, }; @@ -8,7 +8,6 @@ use virt_env::{create_env_virt, strip_env_virt}; use virt_io::{create_io_virt, VirtStdio}; use walrus_ops::strip_virt; use wasm_metadata::Producers; -use wasm_opt::{Feature, OptimizationOptions, ShrinkLevel}; use wit_component::{metadata, ComponentEncoder, DecodedWasm, StringEncoding}; mod data; @@ -130,7 +129,8 @@ impl WasiVirt { } pub fn finish(&mut self) -> Result { - let config = walrus::ModuleConfig::new(); + let mut config = walrus::ModuleConfig::new(); + config.generate_name_section(self.debug); let mut module = if self.debug { config.parse(VIRT_ADAPTER_DEBUG) } else { @@ -294,24 +294,7 @@ impl WasiVirt { // we save into a temporary file and run wasm-opt before returning // this can be disabled with wasm_opt: false if self.wasm_opt.unwrap_or(true) { - let dir = env::temp_dir(); - let tmp_input = dir.join(format!("virt.core.input.{}.wasm", timestamp())); - let tmp_output = dir.join(format!("virt.core.output.{}.wasm", timestamp())); - fs::write(&tmp_input, bytes) - .with_context(|| "Unable to write temporary file for wasm-opt call on adapter")?; - OptimizationOptions::new_opt_level_2() - .shrink_level(ShrinkLevel::Level1) - .enable_feature(Feature::All) - .debug_info(self.debug) - .run(&tmp_input, &tmp_output) - .with_context(|| "Unable to apply wasm-opt optimization to virt. This can be disabled with wasm_opt: false.") - .or_else(|e| { - fs::remove_file(&tmp_input)?; - Err(e) - })?; - bytes = fs::read(&tmp_output)?; - fs::remove_file(&tmp_input)?; - fs::remove_file(&tmp_output)?; + bytes = apply_wasm_opt(bytes)?; } // now adapt the virtualized component @@ -325,9 +308,42 @@ impl WasiVirt { } } -fn timestamp() -> u64 { - match SystemTime::now().duration_since(SystemTime::UNIX_EPOCH) { - Ok(n) => n.as_secs(), - Err(_) => panic!(), +fn apply_wasm_opt(bytes: Vec) -> Result> { + #[cfg(not(feature = "wasm-opt"))] + { + return Ok(bytes); + } + + #[cfg(feature = "wasm-opt")] + { + use std::{fs, time::SystemTime}; + use wasm_opt::{Feature, OptimizationOptions, ShrinkLevel}; + + fn timestamp() -> u64 { + match SystemTime::now().duration_since(SystemTime::UNIX_EPOCH) { + Ok(n) => n.as_secs(), + Err(_) => panic!(), + } + } + + let dir = env::temp_dir(); + let tmp_input = dir.join(format!("virt.core.input.{}.wasm", timestamp())); + let tmp_output = dir.join(format!("virt.core.output.{}.wasm", timestamp())); + fs::write(&tmp_input, bytes) + .with_context(|| "Unable to write temporary file for wasm-opt call on adapter")?; + OptimizationOptions::new_opt_level_2() + .shrink_level(ShrinkLevel::Level1) + .enable_feature(Feature::All) + .debug_info(self.debug) + .run(&tmp_input, &tmp_output) + .with_context(|| "Unable to apply wasm-opt optimization to virt. This can be disabled with wasm_opt: false.") + .or_else(|e| { + fs::remove_file(&tmp_input)?; + Err(e) + })?; + bytes = fs::read(&tmp_output)?; + fs::remove_file(&tmp_input)?; + fs::remove_file(&tmp_output)?; + Ok(bytes) } }