Pat Hickey
4 years ago
12 changed files with 0 additions and 394 deletions
@ -1,18 +0,0 @@ |
|||||
[package] |
|
||||
name = "wasi-virtfs" |
|
||||
version = "0.22.0" |
|
||||
authors = ["The Wasmtime Project Developers"] |
|
||||
description = "WASI implementation in Rust" |
|
||||
license = "Apache-2.0 WITH LLVM-exception" |
|
||||
categories = ["wasm"] |
|
||||
keywords = ["webassembly", "wasm"] |
|
||||
repository = "https://github.com/bytecodealliance/wasmtime" |
|
||||
readme = "README.md" |
|
||||
edition = "2018" |
|
||||
include = ["src/**/*", "LICENSE" ] |
|
||||
publish = false |
|
||||
|
|
||||
[dependencies] |
|
||||
wasi-common = { path = "../", version = "0.22.0" } |
|
||||
anyhow = "1.0" |
|
||||
cap-std = "0.12" |
|
@ -1,82 +0,0 @@ |
|||||
use crate::file::File; |
|
||||
use std::any::Any; |
|
||||
use std::path::{Path, PathBuf}; |
|
||||
use wasi_common::{ |
|
||||
dir::{ReaddirCursor, ReaddirEntity, WasiDir}, |
|
||||
file::{FdFlags, FileCaps, FileType, Filestat, OFlags, WasiFile}, |
|
||||
Error, ErrorExt, |
|
||||
}; |
|
||||
|
|
||||
pub struct Dir; |
|
||||
|
|
||||
impl Dir {} |
|
||||
|
|
||||
impl WasiDir for Dir { |
|
||||
fn as_any(&self) -> &dyn Any { |
|
||||
self |
|
||||
} |
|
||||
fn open_file( |
|
||||
&self, |
|
||||
symlink_follow: bool, |
|
||||
path: &str, |
|
||||
oflags: OFlags, |
|
||||
caps: FileCaps, |
|
||||
fdflags: FdFlags, |
|
||||
) -> Result<Box<dyn WasiFile>, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
|
|
||||
fn open_dir(&self, symlink_follow: bool, path: &str) -> Result<Box<dyn WasiDir>, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
|
|
||||
fn create_dir(&self, path: &str) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn readdir( |
|
||||
&self, |
|
||||
cursor: ReaddirCursor, |
|
||||
) -> Result<Box<dyn Iterator<Item = Result<(ReaddirEntity, String), Error>>>, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
|
|
||||
fn symlink(&self, src_path: &str, dest_path: &str) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn remove_dir(&self, path: &str) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
|
|
||||
fn unlink_file(&self, path: &str) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn read_link(&self, path: &str) -> Result<PathBuf, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn get_filestat(&self) -> Result<Filestat, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn get_path_filestat(&self, path: &str, follow_symlinks: bool) -> Result<Filestat, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn rename(&self, src_path: &str, dest_dir: &dyn WasiDir, dest_path: &str) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn hard_link( |
|
||||
&self, |
|
||||
src_path: &str, |
|
||||
target_dir: &dyn WasiDir, |
|
||||
target_path: &str, |
|
||||
) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn set_times( |
|
||||
&self, |
|
||||
path: &str, |
|
||||
atime: Option<wasi_common::SystemTimeSpec>, |
|
||||
mtime: Option<wasi_common::SystemTimeSpec>, |
|
||||
follow_symlinks: bool, |
|
||||
) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
} |
|
@ -1,73 +0,0 @@ |
|||||
use cap_std::Advice; |
|
||||
use std::any::Any; |
|
||||
use std::convert::TryInto; |
|
||||
use std::io; |
|
||||
use wasi_common::{ |
|
||||
file::{FdFlags, FileType, Filestat, WasiFile}, |
|
||||
Error, |
|
||||
}; |
|
||||
|
|
||||
pub struct File; |
|
||||
|
|
||||
impl File {} |
|
||||
|
|
||||
impl WasiFile for File { |
|
||||
fn as_any(&self) -> &dyn Any { |
|
||||
self |
|
||||
} |
|
||||
fn datasync(&self) -> Result<(), Error> { |
|
||||
Ok(()) |
|
||||
} |
|
||||
fn sync(&self) -> Result<(), Error> { |
|
||||
Ok(()) |
|
||||
} |
|
||||
fn get_filetype(&self) -> Result<FileType, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn get_fdflags(&self) -> Result<FdFlags, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn set_fdflags(&mut self, fdflags: FdFlags) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn get_filestat(&self) -> Result<Filestat, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn set_filestat_size(&self, size: u64) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn advise(&self, _offset: u64, _len: u64, _advice: Advice) -> Result<(), Error> { |
|
||||
Ok(()) |
|
||||
} |
|
||||
fn allocate(&self, offset: u64, len: u64) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn set_times( |
|
||||
&self, |
|
||||
atime: Option<wasi_common::SystemTimeSpec>, |
|
||||
mtime: Option<wasi_common::SystemTimeSpec>, |
|
||||
) -> Result<(), Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn read_vectored(&self, bufs: &mut [io::IoSliceMut]) -> Result<u64, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn read_vectored_at(&self, bufs: &mut [io::IoSliceMut], offset: u64) -> Result<u64, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn write_vectored(&self, bufs: &[io::IoSlice]) -> Result<u64, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn write_vectored_at(&self, bufs: &[io::IoSlice], offset: u64) -> Result<u64, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn seek(&self, pos: std::io::SeekFrom) -> Result<u64, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn peek(&self, buf: &mut [u8]) -> Result<u64, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
fn num_ready_bytes(&self) -> Result<u64, Error> { |
|
||||
todo!() |
|
||||
} |
|
||||
} |
|
@ -1,2 +0,0 @@ |
|||||
pub mod file; |
|
||||
pub mod dir; |
|
@ -1,21 +0,0 @@ |
|||||
# WASI |
|
||||
|
|
||||
You can also [browse this source code online][code] and clone the wasmtime |
|
||||
repository to run the example locally. |
|
||||
|
|
||||
[code]: https://github.com/bytecodealliance/wasmtime/blob/main/examples/wasi/main.rs |
|
||||
|
|
||||
This example shows off how to run a wasi binary with a memory filesystem. |
|
||||
|
|
||||
## Wasm Source code |
|
||||
|
|
||||
```rust,ignore |
|
||||
{{#include ../examples/wasi-fs/wasm/wasi-fs.rs}} |
|
||||
``` |
|
||||
|
|
||||
|
|
||||
## `wasi-fs.rs` |
|
||||
|
|
||||
```rust,ignore |
|
||||
{{#include ../examples/wasi-fs/main.rs}} |
|
||||
``` |
|
@ -1,122 +0,0 @@ |
|||||
/*
|
|
||||
Example of instantiating a WebAssembly which uses WASI imports. |
|
||||
|
|
||||
You can compile and run this example on Linux with: |
|
||||
|
|
||||
cargo build --release -p wasmtime-c-api |
|
||||
cc examples/wasi-fs/main.c \ |
|
||||
-I crates/c-api/include \ |
|
||||
-I crates/c-api/wasm-c-api/include \ |
|
||||
target/release/libwasmtime.a \ |
|
||||
-lpthread -ldl -lm \ |
|
||||
-o wasi-fs |
|
||||
./wasi-fs |
|
||||
|
|
||||
Note that on Windows and macOS the command will be similar, but you'll need |
|
||||
to tweak the `-lpthread` and such annotations. |
|
||||
*/ |
|
||||
|
|
||||
#include <assert.h> |
|
||||
#include <stdio.h> |
|
||||
#include <stdlib.h> |
|
||||
#include <wasm.h> |
|
||||
#include <wasi.h> |
|
||||
#include <wasmtime.h> |
|
||||
|
|
||||
#define MIN(a, b) ((a) < (b) ? (a) : (b)) |
|
||||
|
|
||||
static void exit_with_error(const char *message, wasmtime_error_t *error, wasm_trap_t *trap); |
|
||||
|
|
||||
int main() { |
|
||||
int ret = 0; |
|
||||
// Set up our context
|
|
||||
wasm_engine_t *engine = wasm_engine_new(); |
|
||||
assert(engine != NULL); |
|
||||
wasm_store_t *store = wasm_store_new(engine); |
|
||||
assert(store != NULL); |
|
||||
|
|
||||
wasm_byte_vec_t wasm; |
|
||||
// Load our input file to parse it next
|
|
||||
FILE* file = fopen("target/wasm32-wasi/debug/wasi-fs.wasm", "rb"); |
|
||||
if (!file) { |
|
||||
printf("> Error loading file!\n"); |
|
||||
exit(1); |
|
||||
} |
|
||||
fseek(file, 0L, SEEK_END); |
|
||||
size_t file_size = ftell(file); |
|
||||
wasm_byte_vec_new_uninitialized(&wasm, file_size); |
|
||||
fseek(file, 0L, SEEK_SET); |
|
||||
if (fread(wasm.data, file_size, 1, file) != 1) { |
|
||||
printf("> Error loading module!\n"); |
|
||||
exit(1); |
|
||||
} |
|
||||
fclose(file); |
|
||||
|
|
||||
// Compile our modules
|
|
||||
wasm_module_t *module = NULL; |
|
||||
wasmtime_error_t *error = wasmtime_module_new(engine, &wasm, &module); |
|
||||
if (!module) |
|
||||
exit_with_error("failed to compile module", error, NULL); |
|
||||
wasm_byte_vec_delete(&wasm); |
|
||||
|
|
||||
// Instantiate wasi
|
|
||||
wasi_config_t *wasi_config = wasi_config_new(); |
|
||||
assert(wasi_config); |
|
||||
wasi_config_inherit_argv(wasi_config); |
|
||||
wasi_config_inherit_env(wasi_config); |
|
||||
wasi_config_inherit_stdin(wasi_config); |
|
||||
wasi_config_inherit_stdout(wasi_config); |
|
||||
wasi_config_inherit_stderr(wasi_config); |
|
||||
wasi_config_preopen_dir(wasi_config, "examples/wasi-fs", "."); |
|
||||
wasm_trap_t *trap = NULL; |
|
||||
wasi_instance_t *wasi = wasi_instance_new(store, "wasi_snapshot_preview1", wasi_config, &trap); |
|
||||
if (wasi == NULL) |
|
||||
exit_with_error("failed to instantiate WASI", NULL, trap); |
|
||||
|
|
||||
wasmtime_linker_t *linker = wasmtime_linker_new(store); |
|
||||
error = wasmtime_linker_define_wasi(linker, wasi); |
|
||||
if (error != NULL) |
|
||||
exit_with_error("failed to link wasi", error, NULL); |
|
||||
|
|
||||
// Instantiate the module
|
|
||||
wasm_name_t empty; |
|
||||
wasm_name_new_from_string(&empty, ""); |
|
||||
wasm_instance_t *instance = NULL; |
|
||||
error = wasmtime_linker_module(linker, &empty, module); |
|
||||
if (error != NULL) |
|
||||
exit_with_error("failed to instantiate module", error, NULL); |
|
||||
|
|
||||
// Run it.
|
|
||||
wasm_func_t* func; |
|
||||
wasmtime_linker_get_default(linker, &empty, &func); |
|
||||
if (error != NULL) |
|
||||
exit_with_error("failed to locate default export for module", error, NULL); |
|
||||
|
|
||||
wasm_val_vec_t args_vec = WASM_EMPTY_VEC; |
|
||||
wasm_val_vec_t results_vec = WASM_EMPTY_VEC; |
|
||||
error = wasmtime_func_call(func, &args_vec, &results_vec, &trap); |
|
||||
if (error != NULL) |
|
||||
exit_with_error("error calling default export", error, trap); |
|
||||
|
|
||||
// Clean up after ourselves at this point
|
|
||||
wasm_name_delete(&empty); |
|
||||
wasm_module_delete(module); |
|
||||
wasm_store_delete(store); |
|
||||
wasm_engine_delete(engine); |
|
||||
return 0; |
|
||||
} |
|
||||
|
|
||||
static void exit_with_error(const char *message, wasmtime_error_t *error, wasm_trap_t *trap) { |
|
||||
fprintf(stderr, "error: %s\n", message); |
|
||||
wasm_byte_vec_t error_message; |
|
||||
if (error != NULL) { |
|
||||
wasmtime_error_message(error, &error_message); |
|
||||
wasmtime_error_delete(error); |
|
||||
} else { |
|
||||
wasm_trap_message(trap, &error_message); |
|
||||
wasm_trap_delete(trap); |
|
||||
} |
|
||||
fprintf(stderr, "%.*s\n", (int) error_message.size, error_message.data); |
|
||||
wasm_byte_vec_delete(&error_message); |
|
||||
exit(1); |
|
||||
} |
|
@ -1,46 +0,0 @@ |
|||||
//! Example of running a wasi binary in a memory filesystem
|
|
||||
|
|
||||
// The corresponding wasm binary can be built with:
|
|
||||
// `cargo build -p example-wasi-fs-wasm --target wasm32-wasi`
|
|
||||
//
|
|
||||
// then you can execute this example with `cargo run --example wasi-fs`
|
|
||||
|
|
||||
use anyhow::Result; |
|
||||
use std::collections::HashMap; |
|
||||
use wasmtime::*; |
|
||||
use wasmtime_wasi::virtfs::{VecFileContents, VirtualDirEntry}; |
|
||||
use wasmtime_wasi::{Wasi, WasiCtxBuilder}; |
|
||||
|
|
||||
fn main() -> Result<()> { |
|
||||
tracing_subscriber::FmtSubscriber::builder() |
|
||||
.with_env_filter(tracing_subscriber::EnvFilter::from_default_env()) |
|
||||
.with_ansi(true) |
|
||||
.init(); |
|
||||
|
|
||||
let store = Store::default(); |
|
||||
let mut linker = Linker::new(&store); |
|
||||
|
|
||||
// Create an instance of `Wasi` which contains a `WasiCtx`. Note that
|
|
||||
// `WasiCtx` provides a number of ways to configure what the target program
|
|
||||
// will have access to.
|
|
||||
let entry = VirtualDirEntry::File(Box::new(VecFileContents::with_content( |
|
||||
"world".as_bytes().to_owned(), |
|
||||
))); |
|
||||
let mut map = HashMap::new(); |
|
||||
map.insert("test.txt".to_string(), entry); |
|
||||
let dir = VirtualDirEntry::Directory(map); |
|
||||
let ctx = WasiCtxBuilder::new() |
|
||||
.inherit_stdout() |
|
||||
.inherit_stderr() |
|
||||
.preopened_virt(dir, ".") |
|
||||
.build()?; |
|
||||
let wasi = Wasi::new(&store, ctx); |
|
||||
wasi.add_to_linker(&mut linker)?; |
|
||||
|
|
||||
// Instantiate our module with the imports we've created, and run it.
|
|
||||
let module = Module::from_file(store.engine(), "target/wasm32-wasi/debug/wasi-fs.wasm")?; |
|
||||
linker.module("", &module)?; |
|
||||
linker.get_default("")?.get0::<()>()?()?; |
|
||||
|
|
||||
Ok(()) |
|
||||
} |
|
@ -1 +0,0 @@ |
|||||
world |
|
@ -1,10 +0,0 @@ |
|||||
[package] |
|
||||
name = "example-wasi-fs-wasm" |
|
||||
version = "0.0.0" |
|
||||
authors = ["The Wasmtime Project Developers"] |
|
||||
edition = "2018" |
|
||||
publish = false |
|
||||
|
|
||||
[[bin]] |
|
||||
path = "wasi-fs.rs" |
|
||||
name = "wasi-fs" |
|
@ -1,4 +0,0 @@ |
|||||
fn main() { |
|
||||
let contents = std::fs::read_to_string("test.txt").unwrap(); |
|
||||
println!("Hello, {}!", contents); |
|
||||
} |
|
Loading…
Reference in new issue