This commit removes the need to use `HostRef<Engine>` in the Rust API. Usage is retained in the C API in one location, but otherwise `Engine` can always be used directly. This is the first step of progress on #708 for the `Engine` type. Changes here include: * `Engine` is now `Clone`, and is documented as being cheap. It's not intended that cloning an engine creates a deep copy. * `Engine` is now both `Send` and `Sync`, and asserted to be so. * Usage of `Engine` in APIs no longer requires or uses `HostRef`.
141 lines
3.7 KiB
Rust
141 lines
3.7 KiB
Rust
//! Translation of multi example
|
|
|
|
use anyhow::{ensure, format_err, Context as _, Result};
|
|
use std::rc::Rc;
|
|
use wasmtime::*;
|
|
|
|
struct Callback;
|
|
|
|
impl Callable for Callback {
|
|
fn call(&self, args: &[Val], results: &mut [Val]) -> Result<(), Trap> {
|
|
println!("Calling back...");
|
|
println!("> {} {}", args[0].unwrap_i32(), args[1].unwrap_i64());
|
|
|
|
results[0] = Val::I64(args[1].unwrap_i64() + 1);
|
|
results[1] = Val::I32(args[0].unwrap_i32() + 1);
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
const WAT: &str = r#"
|
|
(module
|
|
(func $f (import "" "f") (param i32 i64) (result i64 i32))
|
|
|
|
(func $g (export "g") (param i32 i64) (result i64 i32)
|
|
(call $f (local.get 0) (local.get 1))
|
|
)
|
|
|
|
(func $round_trip_many
|
|
(export "round_trip_many")
|
|
(param i64 i64 i64 i64 i64 i64 i64 i64 i64 i64)
|
|
(result i64 i64 i64 i64 i64 i64 i64 i64 i64 i64)
|
|
local.get 0
|
|
local.get 1
|
|
local.get 2
|
|
local.get 3
|
|
local.get 4
|
|
local.get 5
|
|
local.get 6
|
|
local.get 7
|
|
local.get 8
|
|
local.get 9)
|
|
)
|
|
"#;
|
|
|
|
fn main() -> Result<()> {
|
|
// Initialize.
|
|
println!("Initializing...");
|
|
let mut cfg = Config::new();
|
|
cfg.features(wasmtime_jit::Features {
|
|
multi_value: true,
|
|
..Default::default()
|
|
});
|
|
let engine = Engine::new(&cfg);
|
|
let store = HostRef::new(Store::new(&engine));
|
|
|
|
// Load binary.
|
|
println!("Loading binary...");
|
|
let binary = wat::parse_str(WAT)?;
|
|
|
|
// Compile.
|
|
println!("Compiling module...");
|
|
let module = HostRef::new(Module::new(&store, &binary).context("Error compiling module!")?);
|
|
|
|
// Create external print functions.
|
|
println!("Creating callback...");
|
|
let callback_type = FuncType::new(
|
|
Box::new([ValType::I32, ValType::I64]),
|
|
Box::new([ValType::I64, ValType::I32]),
|
|
);
|
|
let callback_func = HostRef::new(Func::new(&store, callback_type, Rc::new(Callback)));
|
|
|
|
// Instantiate.
|
|
println!("Instantiating module...");
|
|
let imports = vec![callback_func.into()];
|
|
let instance = Instance::new(&store, &module, imports.as_slice())
|
|
.context("Error instantiating module!")?;
|
|
|
|
// Extract exports.
|
|
println!("Extracting export...");
|
|
let exports = instance.exports();
|
|
ensure!(!exports.is_empty(), "Error accessing exports!");
|
|
let g = exports[0].func().context("> Error accessing export $g!")?;
|
|
let round_trip_many = exports[1]
|
|
.func()
|
|
.context("> Error accessing export $round_trip_many")?;
|
|
|
|
// Call `$g`.
|
|
println!("Calling export \"g\"...");
|
|
let args = vec![Val::I32(1), Val::I64(3)];
|
|
let results = g
|
|
.borrow()
|
|
.call(&args)
|
|
.map_err(|e| format_err!("> Error calling g! {:?}", e))?;
|
|
|
|
println!("Printing result...");
|
|
println!("> {} {}", results[0].unwrap_i64(), results[1].unwrap_i32());
|
|
|
|
debug_assert_eq!(results[0].unwrap_i64(), 4);
|
|
debug_assert_eq!(results[1].unwrap_i32(), 2);
|
|
|
|
// Call `$round_trip_many`.
|
|
println!("Calling export \"round_trip_many\"...");
|
|
let args = vec![
|
|
Val::I64(0),
|
|
Val::I64(1),
|
|
Val::I64(2),
|
|
Val::I64(3),
|
|
Val::I64(4),
|
|
Val::I64(5),
|
|
Val::I64(6),
|
|
Val::I64(7),
|
|
Val::I64(8),
|
|
Val::I64(9),
|
|
];
|
|
let results = round_trip_many
|
|
.borrow()
|
|
.call(&args)
|
|
.map_err(|e| format_err!("> Error calling round_trip_many! {:?}", e))?;
|
|
|
|
println!("Printing result...");
|
|
print!(">");
|
|
for r in results.iter() {
|
|
print!(" {}", r.unwrap_i64());
|
|
}
|
|
println!();
|
|
|
|
debug_assert_eq!(results.len(), 10);
|
|
debug_assert!(args
|
|
.iter()
|
|
.zip(results.iter())
|
|
.all(|(a, r)| a.i64() == r.i64()));
|
|
|
|
// Shut down.
|
|
println!("Shutting down...");
|
|
drop(store);
|
|
|
|
// All done.
|
|
println!("Done.");
|
|
Ok(())
|
|
}
|