我写了一个简单的waser -wasi示例,它读取目录条目,但总是失败。
wasi_fs_example/src/lib.rs
:
#[no_mangle]
pub fn start() {
std::fs::read_dir("/").unwrap();
}
runner/src/main.rs
:
use wasmer::{Instance, Module, Store};
use wasmer_wasi::WasiState;
fn main() -> Result<(), Box<dyn std::error::Error>> {
let wasm = std::fs::read("target/wasm32-wasi/debug/wasi_fs_example.wasm")?;
let store = Store::default();
let module = Module::new(&store, wasm)?;
let mut wasi_env = WasiState::new("wasi_fs_example")
.preopen_dir("target")?
.finalize()?;
let import_object = wasi_env.import_object(&module)?;
let instance = Instance::new(&module, &import_object)?;
let start = instance.exports.get_function("start")?;
start.call(&[])?;
Ok(())
}
在运行时产生错误:
cargo build -p wasi_fs_example --target wasm32-wasi
cargo run -p runner
thread '<unnamed>' panicked at 'called `Result::unwrap()` on an `Err` value: Custom { kind: Other, error: "failed to find a pre-opened file descriptor through which "/" could be opened" }', wasi_fs_example/src/lib.rs:6:10
note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace
示例的完整源代码位于此处。为什么它不起作用?
目前在Rust中我们只能创建WASI二进制文件,而不能创建库。WASI仅在需要的主函数期间有效。在其他地方调用wasi函数会出现segfault,因为libpreopen没有初始化。
那么总结一下,目前如何解决这个问题:
- 根本没有
lib.crate-type
- 文件为
src/main.rs
,而不是src/lib.rs
- 将
#![no_main]
置于src/main.rs
的顶部 - 使用
RUSTFLAGS="-Z wasi-exec-model=reactor" cargo +nightly build --target wasm32-wasi
- 确保在调用
_initialize
之前调用其他
https://github.com/WebAssembly/WASI/issues/24
https://github.com/rust-lang/rust/pull/79997
在我的例子中:
wasi_fs_example/src/main.rs
:
#![no_main]
#[no_mangle]
pub extern "C" fn start() {
std::fs::read_dir("/").unwrap();
}
runner/src/main.rs
:
...
let start = instance.exports.get_function("_initialize")?;
start.call(&[])?;
let start = instance.exports.get_function("start")?;
start.call(&[])?;
...
RUSTFLAGS="-Z wasi-exec-model=reactor" cargo +nightly build -p wasi_fs_example --target wasm32-wasi