1 //! A C API for benchmarking Wasmtime's WebAssembly compilation, instantiation, 2 //! and execution. 3 //! 4 //! The API expects calls that match the following state machine: 5 //! 6 //! ```text 7 //! | 8 //! | 9 //! V 10 //! .---> wasm_bench_create 11 //! | | | 12 //! | | | 13 //! | | V 14 //! | | wasm_bench_compile 15 //! | | | | 16 //! | | | | .----. 17 //! | | | | | | 18 //! | | | V V | 19 //! | | | wasm_bench_instantiate <------. 20 //! | | | | | | 21 //! | | | | | | 22 //! | | | | | | 23 //! | | | .------' '-----> wasm_bench_execute 24 //! | | | | | 25 //! | | | | | 26 //! | V V V | 27 //! '------ wasm_bench_free <--------------------------' 28 //! | 29 //! | 30 //! V 31 //! ``` 32 //! 33 //! All API calls must happen on the same thread. 34 //! 35 //! Functions which return pointers use null as an error value. Function which 36 //! return `int` use `0` as OK and non-zero as an error value. 37 //! 38 //! # Example 39 //! 40 //! ``` 41 //! use std::ptr; 42 //! use wasmtime_bench_api::*; 43 //! 44 //! let working_dir = std::env::current_dir().unwrap().display().to_string(); 45 //! let stdout_path = "./stdout.log"; 46 //! let stderr_path = "./stderr.log"; 47 //! 48 //! // Functions to start/end timers for compilation. 49 //! // 50 //! // The `compilation_timer` pointer configured in the `WasmBenchConfig` is 51 //! // passed through. 52 //! extern "C" fn compilation_start(timer: *mut u8) { 53 //! // Start your compilation timer here. 54 //! } 55 //! extern "C" fn compilation_end(timer: *mut u8) { 56 //! // End your compilation timer here. 57 //! } 58 //! 59 //! // Similar for instantiation. 60 //! extern "C" fn instantiation_start(timer: *mut u8) { 61 //! // Start your instantiation timer here. 62 //! } 63 //! extern "C" fn instantiation_end(timer: *mut u8) { 64 //! // End your instantiation timer here. 65 //! } 66 //! 67 //! // Similar for execution. 68 //! extern "C" fn execution_start(timer: *mut u8) { 69 //! // Start your execution timer here. 70 //! } 71 //! extern "C" fn execution_end(timer: *mut u8) { 72 //! // End your execution timer here. 73 //! } 74 //! 75 //! let config = WasmBenchConfig { 76 //! working_dir_ptr: working_dir.as_ptr(), 77 //! working_dir_len: working_dir.len(), 78 //! stdout_path_ptr: stdout_path.as_ptr(), 79 //! stdout_path_len: stdout_path.len(), 80 //! stderr_path_ptr: stderr_path.as_ptr(), 81 //! stderr_path_len: stderr_path.len(), 82 //! stdin_path_ptr: ptr::null(), 83 //! stdin_path_len: 0, 84 //! compilation_timer: ptr::null_mut(), 85 //! compilation_start, 86 //! compilation_end, 87 //! instantiation_timer: ptr::null_mut(), 88 //! instantiation_start, 89 //! instantiation_end, 90 //! execution_timer: ptr::null_mut(), 91 //! execution_start, 92 //! execution_end, 93 //! execution_flags_ptr: ptr::null(), 94 //! execution_flags_len: 0, 95 //! }; 96 //! 97 //! let mut bench_api = ptr::null_mut(); 98 //! unsafe { 99 //! let code = wasm_bench_create(config, &mut bench_api); 100 //! assert_eq!(code, OK); 101 //! assert!(!bench_api.is_null()); 102 //! }; 103 //! 104 //! let wasm = wat::parse_bytes(br#" 105 //! (module 106 //! (func $bench_start (import "bench" "start")) 107 //! (func $bench_end (import "bench" "end")) 108 //! (func $start (export "_start") 109 //! call $bench_start 110 //! i32.const 1 111 //! i32.const 2 112 //! i32.add 113 //! drop 114 //! call $bench_end 115 //! ) 116 //! ) 117 //! "#).unwrap(); 118 //! 119 //! // This will call the `compilation_{start,end}` timing functions on success. 120 //! let code = unsafe { wasm_bench_compile(bench_api, wasm.as_ptr(), wasm.len()) }; 121 //! assert_eq!(code, OK); 122 //! 123 //! // This will call the `instantiation_{start,end}` timing functions on success. 124 //! let code = unsafe { wasm_bench_instantiate(bench_api) }; 125 //! assert_eq!(code, OK); 126 //! 127 //! // This will call the `execution_{start,end}` timing functions on success. 128 //! let code = unsafe { wasm_bench_execute(bench_api) }; 129 //! assert_eq!(code, OK); 130 //! 131 //! unsafe { 132 //! wasm_bench_free(bench_api); 133 //! } 134 //! ``` 135 136 mod unsafe_send_sync; 137 138 use crate::unsafe_send_sync::UnsafeSendSync; 139 use anyhow::{Context, Result}; 140 use clap::Parser; 141 use std::os::raw::{c_int, c_void}; 142 use std::slice; 143 use std::{env, path::PathBuf}; 144 use target_lexicon::Triple; 145 use wasi_common::{sync::WasiCtxBuilder, I32Exit, WasiCtx}; 146 use wasmtime::{Engine, Instance, Linker, Module, Store}; 147 use wasmtime_cli_flags::CommonOptions; 148 149 pub type ExitCode = c_int; 150 pub const OK: ExitCode = 0; 151 pub const ERR: ExitCode = -1; 152 153 // Randomize the location of heap objects to avoid accidental locality being an 154 // uncontrolled variable that obscures performance evaluation in our 155 // experiments. 156 #[cfg(feature = "shuffling-allocator")] 157 #[global_allocator] 158 static ALLOC: shuffling_allocator::ShufflingAllocator<std::alloc::System> = 159 shuffling_allocator::wrap!(&std::alloc::System); 160 161 /// Configuration options for the benchmark. 162 #[repr(C)] 163 pub struct WasmBenchConfig { 164 /// The working directory where benchmarks should be executed. 165 pub working_dir_ptr: *const u8, 166 pub working_dir_len: usize, 167 168 /// The file path that should be created and used as `stdout`. 169 pub stdout_path_ptr: *const u8, 170 pub stdout_path_len: usize, 171 172 /// The file path that should be created and used as `stderr`. 173 pub stderr_path_ptr: *const u8, 174 pub stderr_path_len: usize, 175 176 /// The (optional) file path that should be opened and used as `stdin`. If 177 /// not provided, then the WASI context will not have a `stdin` initialized. 178 pub stdin_path_ptr: *const u8, 179 pub stdin_path_len: usize, 180 181 /// The functions to start and stop performance timers/counters during Wasm 182 /// compilation. 183 pub compilation_timer: *mut u8, 184 pub compilation_start: extern "C" fn(*mut u8), 185 pub compilation_end: extern "C" fn(*mut u8), 186 187 /// The functions to start and stop performance timers/counters during Wasm 188 /// instantiation. 189 pub instantiation_timer: *mut u8, 190 pub instantiation_start: extern "C" fn(*mut u8), 191 pub instantiation_end: extern "C" fn(*mut u8), 192 193 /// The functions to start and stop performance timers/counters during Wasm 194 /// execution. 195 pub execution_timer: *mut u8, 196 pub execution_start: extern "C" fn(*mut u8), 197 pub execution_end: extern "C" fn(*mut u8), 198 199 /// The (optional) flags to use when running Wasmtime. These correspond to 200 /// the flags used when running Wasmtime from the command line. 201 pub execution_flags_ptr: *const u8, 202 pub execution_flags_len: usize, 203 } 204 205 impl WasmBenchConfig { 206 fn working_dir(&self) -> Result<PathBuf> { 207 let working_dir = 208 unsafe { std::slice::from_raw_parts(self.working_dir_ptr, self.working_dir_len) }; 209 let working_dir = std::str::from_utf8(working_dir) 210 .context("given working directory is not valid UTF-8")?; 211 Ok(working_dir.into()) 212 } 213 214 fn stdout_path(&self) -> Result<PathBuf> { 215 let stdout_path = 216 unsafe { std::slice::from_raw_parts(self.stdout_path_ptr, self.stdout_path_len) }; 217 let stdout_path = 218 std::str::from_utf8(stdout_path).context("given stdout path is not valid UTF-8")?; 219 Ok(stdout_path.into()) 220 } 221 222 fn stderr_path(&self) -> Result<PathBuf> { 223 let stderr_path = 224 unsafe { std::slice::from_raw_parts(self.stderr_path_ptr, self.stderr_path_len) }; 225 let stderr_path = 226 std::str::from_utf8(stderr_path).context("given stderr path is not valid UTF-8")?; 227 Ok(stderr_path.into()) 228 } 229 230 fn stdin_path(&self) -> Result<Option<PathBuf>> { 231 if self.stdin_path_ptr.is_null() { 232 return Ok(None); 233 } 234 235 let stdin_path = 236 unsafe { std::slice::from_raw_parts(self.stdin_path_ptr, self.stdin_path_len) }; 237 let stdin_path = 238 std::str::from_utf8(stdin_path).context("given stdin path is not valid UTF-8")?; 239 Ok(Some(stdin_path.into())) 240 } 241 242 fn execution_flags(&self) -> Result<CommonOptions> { 243 let flags = if self.execution_flags_ptr.is_null() { 244 "" 245 } else { 246 let execution_flags = unsafe { 247 std::slice::from_raw_parts(self.execution_flags_ptr, self.execution_flags_len) 248 }; 249 std::str::from_utf8(execution_flags) 250 .context("given execution flags string is not valid UTF-8")? 251 }; 252 let options = CommonOptions::try_parse_from( 253 ["wasmtime"] 254 .into_iter() 255 .chain(flags.split(' ').filter(|s| !s.is_empty())), 256 ) 257 .context("failed to parse options")?; 258 Ok(options) 259 } 260 } 261 262 /// Exposes a C-compatible way of creating the engine from the bytes of a single 263 /// Wasm module. 264 /// 265 /// On success, the `out_bench_ptr` is initialized to a pointer to a structure 266 /// that contains the engine's initialized state, and `0` is returned. On 267 /// failure, a non-zero status code is returned and `out_bench_ptr` is left 268 /// untouched. 269 #[no_mangle] 270 pub extern "C" fn wasm_bench_create( 271 config: WasmBenchConfig, 272 out_bench_ptr: *mut *mut c_void, 273 ) -> ExitCode { 274 let result = (|| -> Result<_> { 275 let working_dir = config.working_dir()?; 276 let working_dir = 277 cap_std::fs::Dir::open_ambient_dir(&working_dir, cap_std::ambient_authority()) 278 .with_context(|| { 279 format!( 280 "failed to preopen the working directory: {}", 281 working_dir.display(), 282 ) 283 })?; 284 285 let stdout_path = config.stdout_path()?; 286 let stderr_path = config.stderr_path()?; 287 let stdin_path = config.stdin_path()?; 288 let options = config.execution_flags()?; 289 290 let state = Box::new(BenchState::new( 291 options, 292 config.compilation_timer, 293 config.compilation_start, 294 config.compilation_end, 295 config.instantiation_timer, 296 config.instantiation_start, 297 config.instantiation_end, 298 config.execution_timer, 299 config.execution_start, 300 config.execution_end, 301 move || { 302 let mut cx = WasiCtxBuilder::new(); 303 304 let stdout = std::fs::File::create(&stdout_path) 305 .with_context(|| format!("failed to create {}", stdout_path.display()))?; 306 let stdout = cap_std::fs::File::from_std(stdout); 307 let stdout = wasi_common::sync::file::File::from_cap_std(stdout); 308 cx.stdout(Box::new(stdout)); 309 310 let stderr = std::fs::File::create(&stderr_path) 311 .with_context(|| format!("failed to create {}", stderr_path.display()))?; 312 let stderr = cap_std::fs::File::from_std(stderr); 313 let stderr = wasi_common::sync::file::File::from_cap_std(stderr); 314 cx.stderr(Box::new(stderr)); 315 316 if let Some(stdin_path) = &stdin_path { 317 let stdin = std::fs::File::open(stdin_path) 318 .with_context(|| format!("failed to open {}", stdin_path.display()))?; 319 let stdin = cap_std::fs::File::from_std(stdin); 320 let stdin = wasi_common::sync::file::File::from_cap_std(stdin); 321 cx.stdin(Box::new(stdin)); 322 } 323 324 // Allow access to the working directory so that the benchmark can read 325 // its input workload(s). 326 cx.preopened_dir(working_dir.try_clone()?, ".")?; 327 328 // Pass this env var along so that the benchmark program can use smaller 329 // input workload(s) if it has them and that has been requested. 330 if let Ok(val) = env::var("WASM_BENCH_USE_SMALL_WORKLOAD") { 331 cx.env("WASM_BENCH_USE_SMALL_WORKLOAD", &val)?; 332 } 333 334 Ok(cx.build()) 335 }, 336 )?); 337 Ok(Box::into_raw(state) as _) 338 })(); 339 340 if let Ok(bench_ptr) = result { 341 unsafe { 342 assert!(!out_bench_ptr.is_null()); 343 *out_bench_ptr = bench_ptr; 344 } 345 } 346 347 to_exit_code(result.map(|_| ())) 348 } 349 350 /// Free the engine state allocated by this library. 351 #[no_mangle] 352 pub extern "C" fn wasm_bench_free(state: *mut c_void) { 353 assert!(!state.is_null()); 354 unsafe { 355 drop(Box::from_raw(state as *mut BenchState)); 356 } 357 } 358 359 /// Compile the Wasm benchmark module. 360 #[no_mangle] 361 pub extern "C" fn wasm_bench_compile( 362 state: *mut c_void, 363 wasm_bytes: *const u8, 364 wasm_bytes_length: usize, 365 ) -> ExitCode { 366 let state = unsafe { (state as *mut BenchState).as_mut().unwrap() }; 367 let wasm_bytes = unsafe { slice::from_raw_parts(wasm_bytes, wasm_bytes_length) }; 368 let result = state.compile(wasm_bytes).context("failed to compile"); 369 to_exit_code(result) 370 } 371 372 /// Instantiate the Wasm benchmark module. 373 #[no_mangle] 374 pub extern "C" fn wasm_bench_instantiate(state: *mut c_void) -> ExitCode { 375 let state = unsafe { (state as *mut BenchState).as_mut().unwrap() }; 376 let result = state.instantiate().context("failed to instantiate"); 377 to_exit_code(result) 378 } 379 380 /// Execute the Wasm benchmark module. 381 #[no_mangle] 382 pub extern "C" fn wasm_bench_execute(state: *mut c_void) -> ExitCode { 383 let state = unsafe { (state as *mut BenchState).as_mut().unwrap() }; 384 let result = state.execute().context("failed to execute"); 385 to_exit_code(result) 386 } 387 388 /// Helper function for converting a Rust result to a C error code. 389 /// 390 /// This will print an error indicating some information regarding the failure. 391 fn to_exit_code<T>(result: impl Into<Result<T>>) -> ExitCode { 392 match result.into() { 393 Ok(_) => OK, 394 Err(error) => { 395 eprintln!("{:?}", error); 396 ERR 397 } 398 } 399 } 400 401 /// This structure contains the actual Rust implementation of the state required 402 /// to manage the Wasmtime engine between calls. 403 struct BenchState { 404 linker: Linker<HostState>, 405 compilation_timer: *mut u8, 406 compilation_start: extern "C" fn(*mut u8), 407 compilation_end: extern "C" fn(*mut u8), 408 instantiation_timer: *mut u8, 409 instantiation_start: extern "C" fn(*mut u8), 410 instantiation_end: extern "C" fn(*mut u8), 411 make_wasi_cx: Box<dyn FnMut() -> Result<WasiCtx>>, 412 module: Option<Module>, 413 store_and_instance: Option<(Store<HostState>, Instance)>, 414 epoch_interruption: bool, 415 fuel: Option<u64>, 416 } 417 418 struct HostState { 419 wasi: WasiCtx, 420 #[cfg(feature = "wasi-nn")] 421 wasi_nn: wasmtime_wasi_nn::WasiNnCtx, 422 } 423 424 impl BenchState { 425 fn new( 426 mut options: CommonOptions, 427 compilation_timer: *mut u8, 428 compilation_start: extern "C" fn(*mut u8), 429 compilation_end: extern "C" fn(*mut u8), 430 instantiation_timer: *mut u8, 431 instantiation_start: extern "C" fn(*mut u8), 432 instantiation_end: extern "C" fn(*mut u8), 433 execution_timer: *mut u8, 434 execution_start: extern "C" fn(*mut u8), 435 execution_end: extern "C" fn(*mut u8), 436 make_wasi_cx: impl FnMut() -> Result<WasiCtx> + 'static, 437 ) -> Result<Self> { 438 let mut config = options.config(Some(&Triple::host().to_string()))?; 439 // NB: always disable the compilation cache. 440 config.disable_cache(); 441 let engine = Engine::new(&config)?; 442 let mut linker = Linker::<HostState>::new(&engine); 443 444 // Define the benchmarking start/end functions. 445 let execution_timer = unsafe { 446 // Safe because this bench API's contract requires that its methods 447 // are only ever called from a single thread. 448 UnsafeSendSync::new(execution_timer) 449 }; 450 linker.func_wrap("bench", "start", move || { 451 execution_start(*execution_timer.get()); 452 Ok(()) 453 })?; 454 linker.func_wrap("bench", "end", move || { 455 execution_end(*execution_timer.get()); 456 Ok(()) 457 })?; 458 459 let epoch_interruption = options.wasm.epoch_interruption.unwrap_or(false); 460 let fuel = options.wasm.fuel; 461 462 if options.wasi.common != Some(false) { 463 wasi_common::sync::add_to_linker(&mut linker, |cx| &mut cx.wasi)?; 464 } 465 466 #[cfg(feature = "wasi-nn")] 467 if options.wasi.nn == Some(true) { 468 wasmtime_wasi_nn::witx::add_to_linker(&mut linker, |cx| &mut cx.wasi_nn)?; 469 } 470 471 Ok(Self { 472 linker, 473 compilation_timer, 474 compilation_start, 475 compilation_end, 476 instantiation_timer, 477 instantiation_start, 478 instantiation_end, 479 make_wasi_cx: Box::new(make_wasi_cx) as _, 480 module: None, 481 store_and_instance: None, 482 epoch_interruption, 483 fuel, 484 }) 485 } 486 487 fn compile(&mut self, bytes: &[u8]) -> Result<()> { 488 assert!( 489 self.module.is_none(), 490 "create a new engine to repeat compilation" 491 ); 492 493 (self.compilation_start)(self.compilation_timer); 494 let module = Module::from_binary(self.linker.engine(), bytes)?; 495 (self.compilation_end)(self.compilation_timer); 496 497 self.module = Some(module); 498 Ok(()) 499 } 500 501 fn instantiate(&mut self) -> Result<()> { 502 let module = self 503 .module 504 .as_ref() 505 .expect("compile the module before instantiating it"); 506 507 let host = HostState { 508 wasi: (self.make_wasi_cx)().context("failed to create a WASI context")?, 509 #[cfg(feature = "wasi-nn")] 510 wasi_nn: { 511 let (backends, registry) = wasmtime_wasi_nn::preload(&[])?; 512 wasmtime_wasi_nn::WasiNnCtx::new(backends, registry) 513 }, 514 }; 515 516 // NB: Start measuring instantiation time *after* we've created the WASI 517 // context, since that needs to do file I/O to setup 518 // stdin/stdout/stderr. 519 (self.instantiation_start)(self.instantiation_timer); 520 let mut store = Store::new(self.linker.engine(), host); 521 if self.epoch_interruption { 522 store.set_epoch_deadline(1); 523 } 524 if let Some(fuel) = self.fuel { 525 store.set_fuel(fuel).unwrap(); 526 } 527 528 let instance = self.linker.instantiate(&mut store, &module)?; 529 (self.instantiation_end)(self.instantiation_timer); 530 531 self.store_and_instance = Some((store, instance)); 532 Ok(()) 533 } 534 535 fn execute(&mut self) -> Result<()> { 536 let (mut store, instance) = self 537 .store_and_instance 538 .take() 539 .expect("instantiate the module before executing it"); 540 541 let start_func = instance.get_typed_func::<(), ()>(&mut store, "_start")?; 542 match start_func.call(&mut store, ()) { 543 Ok(_) => Ok(()), 544 Err(trap) => { 545 // Since _start will likely return by using the system `exit` call, we must 546 // check the trap code to see if it actually represents a successful exit. 547 if let Some(exit) = trap.downcast_ref::<I32Exit>() { 548 if exit.0 == 0 { 549 return Ok(()); 550 } 551 } 552 553 Err(trap) 554 } 555 } 556 } 557 } 558