This commit removes Wasmtime's dependency on the `region` crate. The motivation for this came about when I was updating dependencies and saw that `region` had a new major version at 3.0.0 as opposed to our currently used 2.3 track. In reviewing the use cases of `region` within Wasmtime I found two trends in particular which motivated this commit: * Some unix-specific areas of `wasmtime_runtime` use `rustix::mm::mprotect` instead of `region::protect` already. This means that the usage of `region::protect` for changing virtual memory protections was already inconsistent. * Many uses of `region::protect` were already in unix-specific regions which could make use of `rustix`. Overall I opted to remove the dependency on the `region` crate to avoid chasing its versions over time. Unix-specific changes of protections were easily changed to `rustix::mm::mprotect`. There were two locations where a windows/unix split is now required and I subjectively ruled "that seems ok". Finally removing `region` also meant that the "what is the current page size" query needed to be inlined into `wasmtime_runtime`, which I have also subjectively ruled "that seems fine". Finally one final refactoring here was that the `unix.rs` and `linux.rs` split for the pooling allocator was merged. These two files already only differed in one function so I slapped a `cfg_if!` in there to help reduce the duplication.
234 lines
8.4 KiB
Rust
234 lines
8.4 KiB
Rust
//! Runtime library support for Wasmtime.
|
|
|
|
#![deny(missing_docs, trivial_numeric_casts, unused_extern_crates)]
|
|
#![warn(unused_import_braces)]
|
|
#![cfg_attr(feature = "clippy", plugin(clippy(conf_file = "../../clippy.toml")))]
|
|
#![cfg_attr(
|
|
feature = "cargo-clippy",
|
|
allow(clippy::new_without_default, clippy::new_without_default)
|
|
)]
|
|
#![cfg_attr(
|
|
feature = "cargo-clippy",
|
|
warn(
|
|
clippy::float_arithmetic,
|
|
clippy::mut_mut,
|
|
clippy::nonminimal_bool,
|
|
clippy::map_unwrap_or,
|
|
clippy::clippy::print_stdout,
|
|
clippy::unicode_not_nfc,
|
|
clippy::use_self
|
|
)
|
|
)]
|
|
#![cfg_attr(not(memory_init_cow), allow(unused_variables, unreachable_code))]
|
|
|
|
use anyhow::Error;
|
|
use std::sync::atomic::{AtomicU64, AtomicUsize, Ordering};
|
|
use std::sync::Arc;
|
|
use wasmtime_environ::DefinedFuncIndex;
|
|
use wasmtime_environ::DefinedMemoryIndex;
|
|
use wasmtime_environ::FunctionInfo;
|
|
use wasmtime_environ::SignatureIndex;
|
|
|
|
#[cfg(feature = "component-model")]
|
|
pub mod component;
|
|
mod export;
|
|
mod externref;
|
|
mod imports;
|
|
mod instance;
|
|
mod memory;
|
|
mod mmap;
|
|
mod mmap_vec;
|
|
mod table;
|
|
mod traphandlers;
|
|
mod vmcontext;
|
|
|
|
pub mod debug_builtins;
|
|
pub mod libcalls;
|
|
|
|
pub use wasmtime_jit_debug::gdb_jit_int::GdbJitImageRegistration;
|
|
|
|
pub use crate::export::*;
|
|
pub use crate::externref::*;
|
|
pub use crate::imports::Imports;
|
|
pub use crate::instance::{
|
|
allocate_single_memory_instance, InstanceAllocationRequest, InstanceAllocator, InstanceHandle,
|
|
InstantiationError, LinkError, OnDemandInstanceAllocator, StorePtr,
|
|
};
|
|
#[cfg(feature = "pooling-allocator")]
|
|
pub use crate::instance::{InstanceLimits, PoolingAllocationStrategy, PoolingInstanceAllocator};
|
|
pub use crate::memory::{
|
|
DefaultMemoryCreator, Memory, RuntimeLinearMemory, RuntimeMemoryCreator, SharedMemory,
|
|
};
|
|
pub use crate::mmap::Mmap;
|
|
pub use crate::mmap_vec::MmapVec;
|
|
pub use crate::table::{Table, TableElement};
|
|
pub use crate::traphandlers::{
|
|
catch_traps, init_traps, raise_lib_trap, raise_user_trap, resume_panic, tls_eager_initialize,
|
|
Backtrace, SignalHandler, TlsRestore, Trap, TrapReason,
|
|
};
|
|
pub use crate::vmcontext::{
|
|
VMCallerCheckedAnyfunc, VMContext, VMFunctionBody, VMFunctionImport, VMGlobalDefinition,
|
|
VMGlobalImport, VMInvokeArgument, VMMemoryDefinition, VMMemoryImport, VMOpaqueContext,
|
|
VMRuntimeLimits, VMSharedSignatureIndex, VMTableDefinition, VMTableImport, VMTrampoline,
|
|
ValRaw,
|
|
};
|
|
|
|
mod module_id;
|
|
pub use module_id::{CompiledModuleId, CompiledModuleIdAllocator};
|
|
|
|
#[cfg(memory_init_cow)]
|
|
mod cow;
|
|
#[cfg(memory_init_cow)]
|
|
pub use crate::cow::{MemoryImage, MemoryImageSlot, ModuleMemoryImages};
|
|
|
|
#[cfg(not(memory_init_cow))]
|
|
mod cow_disabled;
|
|
#[cfg(not(memory_init_cow))]
|
|
pub use crate::cow_disabled::{MemoryImage, MemoryImageSlot, ModuleMemoryImages};
|
|
|
|
/// Version number of this crate.
|
|
pub const VERSION: &str = env!("CARGO_PKG_VERSION");
|
|
|
|
/// Dynamic runtime functionality needed by this crate throughout the execution
|
|
/// of a wasm instance.
|
|
///
|
|
/// This trait is used to store a raw pointer trait object within each
|
|
/// `VMContext`. This raw pointer trait object points back to the
|
|
/// `wasmtime::Store` internally but is type-erased so this `wasmtime_runtime`
|
|
/// crate doesn't need the entire `wasmtime` crate to build.
|
|
///
|
|
/// Note that this is an extra-unsafe trait because no heed is paid to the
|
|
/// lifetime of this store or the Send/Sync-ness of this store. All of that must
|
|
/// be respected by embedders (e.g. the `wasmtime::Store` structure). The theory
|
|
/// is that `wasmtime::Store` handles all this correctly.
|
|
pub unsafe trait Store {
|
|
/// Returns the raw pointer in memory where this store's shared
|
|
/// `VMRuntimeLimits` structure is located.
|
|
///
|
|
/// Used to configure `VMContext` initialization and store the right pointer
|
|
/// in the `VMContext`.
|
|
fn vmruntime_limits(&self) -> *mut VMRuntimeLimits;
|
|
|
|
/// Returns a pointer to the global epoch counter.
|
|
///
|
|
/// Used to configure the `VMContext` on initialization.
|
|
fn epoch_ptr(&self) -> *const AtomicU64;
|
|
|
|
/// Returns the externref management structures necessary for this store.
|
|
///
|
|
/// The first element returned is the table in which externrefs are stored
|
|
/// throughout wasm execution, and the second element is how to look up
|
|
/// module information for gc requests.
|
|
fn externref_activations_table(
|
|
&mut self,
|
|
) -> (&mut VMExternRefActivationsTable, &dyn ModuleInfoLookup);
|
|
|
|
/// Callback invoked to allow the store's resource limiter to reject a
|
|
/// memory grow operation.
|
|
fn memory_growing(
|
|
&mut self,
|
|
current: usize,
|
|
desired: usize,
|
|
maximum: Option<usize>,
|
|
) -> Result<bool, Error>;
|
|
/// Callback invoked to notify the store's resource limiter that a memory
|
|
/// grow operation has failed.
|
|
fn memory_grow_failed(&mut self, error: &Error);
|
|
/// Callback invoked to allow the store's resource limiter to reject a
|
|
/// table grow operation.
|
|
fn table_growing(
|
|
&mut self,
|
|
current: u32,
|
|
desired: u32,
|
|
maximum: Option<u32>,
|
|
) -> Result<bool, Error>;
|
|
/// Callback invoked to notify the store's resource limiter that a table
|
|
/// grow operation has failed.
|
|
fn table_grow_failed(&mut self, error: &Error);
|
|
/// Callback invoked whenever fuel runs out by a wasm instance. If an error
|
|
/// is returned that's raised as a trap. Otherwise wasm execution will
|
|
/// continue as normal.
|
|
fn out_of_gas(&mut self) -> Result<(), Error>;
|
|
/// Callback invoked whenever an instance observes a new epoch
|
|
/// number. Cannot fail; cooperative epoch-based yielding is
|
|
/// completely semantically transparent. Returns the new deadline.
|
|
fn new_epoch(&mut self) -> Result<u64, Error>;
|
|
}
|
|
|
|
/// Functionality required by this crate for a particular module. This
|
|
/// is chiefly needed for lazy initialization of various bits of
|
|
/// instance state.
|
|
///
|
|
/// When an instance is created, it holds an Arc<dyn ModuleRuntimeInfo>
|
|
/// so that it can get to signatures, metadata on functions, memory and
|
|
/// funcref-table images, etc. All of these things are ordinarily known
|
|
/// by the higher-level layers of Wasmtime. Specifically, the main
|
|
/// implementation of this trait is provided by
|
|
/// `wasmtime::module::ModuleInner`. Since the runtime crate sits at
|
|
/// the bottom of the dependence DAG though, we don't know or care about
|
|
/// that; we just need some implementor of this trait for each
|
|
/// allocation request.
|
|
pub trait ModuleRuntimeInfo: Send + Sync + 'static {
|
|
/// The underlying Module.
|
|
fn module(&self) -> &Arc<wasmtime_environ::Module>;
|
|
|
|
/// The signatures.
|
|
fn signature(&self, index: SignatureIndex) -> VMSharedSignatureIndex;
|
|
|
|
/// The base address of where JIT functions are located.
|
|
fn image_base(&self) -> usize;
|
|
|
|
/// Descriptors about each compiled function, such as the offset from
|
|
/// `image_base`.
|
|
fn function_info(&self, func_index: DefinedFuncIndex) -> &FunctionInfo;
|
|
|
|
/// Returns the `MemoryImage` structure used for copy-on-write
|
|
/// initialization of the memory, if it's applicable.
|
|
fn memory_image(&self, memory: DefinedMemoryIndex)
|
|
-> anyhow::Result<Option<&Arc<MemoryImage>>>;
|
|
|
|
/// A unique ID for this particular module. This can be used to
|
|
/// allow for fastpaths to optimize a "re-instantiate the same
|
|
/// module again" case.
|
|
fn unique_id(&self) -> Option<CompiledModuleId>;
|
|
|
|
/// A slice pointing to all data that is referenced by this instance.
|
|
fn wasm_data(&self) -> &[u8];
|
|
|
|
/// Returns an array, indexed by `SignatureIndex` of all
|
|
/// `VMSharedSignatureIndex` entries corresponding to the `SignatureIndex`.
|
|
fn signature_ids(&self) -> &[VMSharedSignatureIndex];
|
|
}
|
|
|
|
/// Returns the host OS page size, in bytes.
|
|
pub fn page_size() -> usize {
|
|
static PAGE_SIZE: AtomicUsize = AtomicUsize::new(0);
|
|
|
|
return match PAGE_SIZE.load(Ordering::Relaxed) {
|
|
0 => {
|
|
let size = get_page_size();
|
|
assert!(size != 0);
|
|
PAGE_SIZE.store(size, Ordering::Relaxed);
|
|
size
|
|
}
|
|
n => n,
|
|
};
|
|
|
|
#[cfg(windows)]
|
|
fn get_page_size() -> usize {
|
|
use std::mem::MaybeUninit;
|
|
use windows_sys::Win32::System::SystemInformation::*;
|
|
|
|
unsafe {
|
|
let mut info = MaybeUninit::uninit();
|
|
GetSystemInfo(info.as_mut_ptr());
|
|
info.assume_init_ref().dwPageSize as usize
|
|
}
|
|
}
|
|
|
|
#[cfg(unix)]
|
|
fn get_page_size() -> usize {
|
|
unsafe { libc::sysconf(libc::_SC_PAGESIZE) as usize }
|
|
}
|
|
}
|