* Remove the `jit_function_registry` global state This commit removes on the final pieces of global state in wasmtime today, the `jit_function_registry` module. The purpose of this module is to help translate a native backtrace with native program counters into a wasm backtrace with module names, function names, and wasm module indices. To that end this module retained a global map of function ranges to this metadata information for each compiled function. It turns out that we already had a `NAMES` global in the `wasmtime` crate for symbolicating backtrace addresses, so this commit moves that global into its own file and restructures the internals to account for program counter ranges as well. The general set of changes here are: * Remove `jit_function_registry` * Remove `NAMES` * Create a new `frame_info` module which has a singleton global registering compiled module's frame information. * Update traps to use the `frame_info` module to symbolicate pcs, directly extracting a `FrameInfo` from the module. * Register and unregister information on a module level instead of on a per-function level (at least in terms of locking granluarity). This commit leaves the new `FRAME_INFO` global variable as the only remaining "critical" global variable in `wasmtime`, which only exists due to the API of `Trap` where it doesn't take in any extra context when capturing a stack trace through which we could hang off frame information. I'm thinking though that this is ok, and we can always tweak the API of `Trap` in the future if necessary if we truly need to accomodate this. * Remove a lazy_static dep * Add some comments and restructure
265 lines
8.7 KiB
Rust
265 lines
8.7 KiB
Rust
//! Define the `instantiate` function, which takes a byte array containing an
|
|
//! encoded wasm module and returns a live wasm instance. Also, define
|
|
//! `CompiledModule` to allow compiling and instantiating to be done as separate
|
|
//! steps.
|
|
|
|
use crate::compiler::Compiler;
|
|
use crate::imports::resolve_imports;
|
|
use crate::link::link_module;
|
|
use crate::resolver::Resolver;
|
|
use std::io::Write;
|
|
use std::rc::Rc;
|
|
use std::sync::Arc;
|
|
use thiserror::Error;
|
|
use wasmtime_debug::read_debuginfo;
|
|
use wasmtime_environ::entity::{BoxedSlice, PrimaryMap};
|
|
use wasmtime_environ::wasm::{DefinedFuncIndex, SignatureIndex};
|
|
use wasmtime_environ::{
|
|
CompileError, DataInitializer, DataInitializerLocation, Module, ModuleEnvironment,
|
|
};
|
|
use wasmtime_runtime::{
|
|
GdbJitImageRegistration, InstanceHandle, InstantiationError, TrapRegistration, VMFunctionBody,
|
|
VMSharedSignatureIndex,
|
|
};
|
|
|
|
/// An error condition while setting up a wasm instance, be it validation,
|
|
/// compilation, or instantiation.
|
|
#[derive(Error, Debug)]
|
|
pub enum SetupError {
|
|
/// The module did not pass validation.
|
|
#[error("Validation error: {0}")]
|
|
Validate(String),
|
|
|
|
/// A wasm translation error occured.
|
|
#[error("WebAssembly failed to compile")]
|
|
Compile(#[from] CompileError),
|
|
|
|
/// Some runtime resource was unavailable or insufficient, or the start function
|
|
/// trapped.
|
|
#[error("Instantiation failed during setup")]
|
|
Instantiate(#[from] InstantiationError),
|
|
|
|
/// Debug information generation error occured.
|
|
#[error("Debug information error")]
|
|
DebugInfo(#[from] anyhow::Error),
|
|
}
|
|
|
|
/// This is similar to `CompiledModule`, but references the data initializers
|
|
/// from the wasm buffer rather than holding its own copy.
|
|
struct RawCompiledModule<'data> {
|
|
module: Module,
|
|
finished_functions: BoxedSlice<DefinedFuncIndex, *mut [VMFunctionBody]>,
|
|
data_initializers: Box<[DataInitializer<'data>]>,
|
|
signatures: BoxedSlice<SignatureIndex, VMSharedSignatureIndex>,
|
|
dbg_jit_registration: Option<GdbJitImageRegistration>,
|
|
trap_registration: TrapRegistration,
|
|
}
|
|
|
|
impl<'data> RawCompiledModule<'data> {
|
|
/// Create a new `RawCompiledModule` by compiling the wasm module in `data` and instatiating it.
|
|
fn new(
|
|
compiler: &mut Compiler,
|
|
data: &'data [u8],
|
|
debug_info: bool,
|
|
) -> Result<Self, SetupError> {
|
|
let environ = ModuleEnvironment::new(compiler.frontend_config(), compiler.tunables());
|
|
|
|
let translation = environ
|
|
.translate(data)
|
|
.map_err(|error| SetupError::Compile(CompileError::Wasm(error)))?;
|
|
|
|
let debug_data = if debug_info {
|
|
Some(read_debuginfo(&data))
|
|
} else {
|
|
None
|
|
};
|
|
|
|
let (finished_functions, jt_offsets, relocations, dbg_image, trap_registration) = compiler
|
|
.compile(
|
|
&translation.module,
|
|
translation.module_translation.as_ref().unwrap(),
|
|
translation.function_body_inputs,
|
|
debug_data,
|
|
)?;
|
|
|
|
link_module(
|
|
&translation.module,
|
|
&finished_functions,
|
|
&jt_offsets,
|
|
relocations,
|
|
);
|
|
|
|
// Compute indices into the shared signature table.
|
|
let signatures = {
|
|
let signature_registry = compiler.signatures();
|
|
translation
|
|
.module
|
|
.signatures
|
|
.values()
|
|
.map(|sig| signature_registry.register(sig))
|
|
.collect::<PrimaryMap<_, _>>()
|
|
};
|
|
|
|
// Make all code compiled thus far executable.
|
|
compiler.publish_compiled_code();
|
|
|
|
let dbg_jit_registration = if let Some(img) = dbg_image {
|
|
let mut bytes = Vec::new();
|
|
bytes.write_all(&img).expect("all written");
|
|
let reg = GdbJitImageRegistration::register(bytes);
|
|
Some(reg)
|
|
} else {
|
|
None
|
|
};
|
|
|
|
Ok(Self {
|
|
module: translation.module,
|
|
finished_functions: finished_functions.into_boxed_slice(),
|
|
data_initializers: translation.data_initializers.into_boxed_slice(),
|
|
signatures: signatures.into_boxed_slice(),
|
|
dbg_jit_registration,
|
|
trap_registration,
|
|
})
|
|
}
|
|
}
|
|
|
|
/// A compiled wasm module, ready to be instantiated.
|
|
pub struct CompiledModule {
|
|
module: Arc<Module>,
|
|
finished_functions: BoxedSlice<DefinedFuncIndex, *mut [VMFunctionBody]>,
|
|
data_initializers: Box<[OwnedDataInitializer]>,
|
|
signatures: BoxedSlice<SignatureIndex, VMSharedSignatureIndex>,
|
|
dbg_jit_registration: Option<Rc<GdbJitImageRegistration>>,
|
|
trap_registration: TrapRegistration,
|
|
}
|
|
|
|
impl CompiledModule {
|
|
/// Compile a data buffer into a `CompiledModule`, which may then be instantiated.
|
|
pub fn new<'data>(
|
|
compiler: &mut Compiler,
|
|
data: &'data [u8],
|
|
debug_info: bool,
|
|
) -> Result<Self, SetupError> {
|
|
let raw = RawCompiledModule::<'data>::new(compiler, data, debug_info)?;
|
|
|
|
Ok(Self::from_parts(
|
|
raw.module,
|
|
raw.finished_functions,
|
|
raw.data_initializers
|
|
.iter()
|
|
.map(OwnedDataInitializer::new)
|
|
.collect::<Vec<_>>()
|
|
.into_boxed_slice(),
|
|
raw.signatures.clone(),
|
|
raw.dbg_jit_registration,
|
|
raw.trap_registration,
|
|
))
|
|
}
|
|
|
|
/// Construct a `CompiledModule` from component parts.
|
|
pub fn from_parts(
|
|
module: Module,
|
|
finished_functions: BoxedSlice<DefinedFuncIndex, *mut [VMFunctionBody]>,
|
|
data_initializers: Box<[OwnedDataInitializer]>,
|
|
signatures: BoxedSlice<SignatureIndex, VMSharedSignatureIndex>,
|
|
dbg_jit_registration: Option<GdbJitImageRegistration>,
|
|
trap_registration: TrapRegistration,
|
|
) -> Self {
|
|
Self {
|
|
module: Arc::new(module),
|
|
finished_functions,
|
|
data_initializers,
|
|
signatures,
|
|
dbg_jit_registration: dbg_jit_registration.map(Rc::new),
|
|
trap_registration,
|
|
}
|
|
}
|
|
|
|
/// Crate an `Instance` from this `CompiledModule`.
|
|
///
|
|
/// Note that if only one instance of this module is needed, it may be more
|
|
/// efficient to call the top-level `instantiate`, since that avoids copying
|
|
/// the data initializers.
|
|
///
|
|
/// # Unsafety
|
|
///
|
|
/// See `InstanceHandle::new`
|
|
pub unsafe fn instantiate(
|
|
&self,
|
|
resolver: &mut dyn Resolver,
|
|
) -> Result<InstanceHandle, InstantiationError> {
|
|
let data_initializers = self
|
|
.data_initializers
|
|
.iter()
|
|
.map(|init| DataInitializer {
|
|
location: init.location.clone(),
|
|
data: &*init.data,
|
|
})
|
|
.collect::<Vec<_>>();
|
|
let imports = resolve_imports(&self.module, resolver)?;
|
|
InstanceHandle::new(
|
|
Arc::clone(&self.module),
|
|
self.trap_registration.clone(),
|
|
self.finished_functions.clone(),
|
|
imports,
|
|
&data_initializers,
|
|
self.signatures.clone(),
|
|
self.dbg_jit_registration.as_ref().map(|r| Rc::clone(&r)),
|
|
Box::new(()),
|
|
)
|
|
}
|
|
|
|
/// Return a reference-counting pointer to a module.
|
|
pub fn module(&self) -> &Arc<Module> {
|
|
&self.module
|
|
}
|
|
|
|
/// Return a reference to a module.
|
|
pub fn module_ref(&self) -> &Module {
|
|
&self.module
|
|
}
|
|
|
|
/// Returns the map of all finished JIT functions compiled for this module
|
|
pub fn finished_functions(&self) -> &BoxedSlice<DefinedFuncIndex, *mut [VMFunctionBody]> {
|
|
&self.finished_functions
|
|
}
|
|
}
|
|
|
|
/// Similar to `DataInitializer`, but owns its own copy of the data rather
|
|
/// than holding a slice of the original module.
|
|
pub struct OwnedDataInitializer {
|
|
/// The location where the initialization is to be performed.
|
|
location: DataInitializerLocation,
|
|
|
|
/// The initialization data.
|
|
data: Box<[u8]>,
|
|
}
|
|
|
|
impl OwnedDataInitializer {
|
|
fn new(borrowed: &DataInitializer<'_>) -> Self {
|
|
Self {
|
|
location: borrowed.location.clone(),
|
|
data: borrowed.data.to_vec().into_boxed_slice(),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Create a new wasm instance by compiling the wasm module in `data` and instatiating it.
|
|
///
|
|
/// This is equivalent to createing a `CompiledModule` and calling `instantiate()` on it,
|
|
/// but avoids creating an intermediate copy of the data initializers.
|
|
///
|
|
/// # Unsafety
|
|
///
|
|
/// See `InstanceHandle::new`
|
|
#[allow(clippy::implicit_hasher)]
|
|
pub unsafe fn instantiate(
|
|
compiler: &mut Compiler,
|
|
data: &[u8],
|
|
resolver: &mut dyn Resolver,
|
|
debug_info: bool,
|
|
) -> Result<InstanceHandle, SetupError> {
|
|
let instance = CompiledModule::new(compiler, data, debug_info)?.instantiate(resolver)?;
|
|
Ok(instance)
|
|
}
|