Fix a "trampoline missing" panic with components (#4296)

One test case I wrote recently was to import a lowered function into a
wasm module and then immediately export it. This previously didn't work
because trampoline lookup would fail as the original
`VMCallerCheckedAnyfunc` function pointer points into the
`trampoline_obj` of a component which wasn't registered with the
`ModuleRegistry`. This plumbs through the necessary configuration to get
that all hooked up.
This commit is contained in:
Alex Crichton
2022-06-23 09:41:03 -05:00
committed by GitHub
parent 6778b4fce2
commit 445cc87a06
7 changed files with 204 additions and 28 deletions

View File

@@ -1,6 +1,7 @@
use crate::signatures::SignatureCollection; use crate::signatures::SignatureCollection;
use crate::{Engine, Module}; use crate::{Engine, Module};
use anyhow::{bail, Context, Result}; use anyhow::{bail, Context, Result};
use std::collections::HashMap;
use std::fs; use std::fs;
use std::ops::Range; use std::ops::Range;
use std::path::Path; use std::path::Path;
@@ -164,13 +165,33 @@ impl Component {
let code = trampoline_obj.publish()?; let code = trampoline_obj.publish()?;
let text = wasmtime_jit::subslice_range(code.text, code.mmap); let text = wasmtime_jit::subslice_range(code.text, code.mmap);
// This map is used to register all known tramplines in the
// `SignatureCollection` created below. This is later consulted during
// `ModuleRegistry::lookup_trampoline` if a trampoline needs to be
// located for a signature index where the original function pointer
// is that of the `trampolines` created above.
//
// This situation arises when a core wasm module imports a lowered
// function and then immediately exports it. Wasmtime will lookup an
// entry trampoline for the exported function which is actually a
// lowered host function, hence an entry in the `trampolines` variable
// above, and the type of that function will be stored in this
// `vmtrampolines` map since the type is guaranteed to have escaped
// from at least one of the modules we compiled prior.
let mut vmtrampolines = HashMap::new();
for (_, module) in static_modules.iter() {
for (idx, trampoline, _) in module.compiled_module().trampolines() {
vmtrampolines.insert(idx, trampoline);
}
}
// FIXME: for the same reason as above where each module is // FIXME: for the same reason as above where each module is
// re-registering everything this should only be registered once. This // re-registering everything this should only be registered once. This
// is benign for now but could do with refactorings later on. // is benign for now but could do with refactorings later on.
let signatures = SignatureCollection::new_for_module( let signatures = SignatureCollection::new_for_module(
engine.signatures(), engine.signatures(),
types.module_types(), types.module_types(),
[].into_iter(), vmtrampolines.into_iter(),
); );
Ok(Component { Ok(Component {
@@ -202,9 +223,13 @@ impl Component {
&self.inner.signatures &self.inner.signatures
} }
pub(crate) fn text(&self) -> &[u8] {
&self.inner.trampoline_obj.mmap()[self.inner.text.clone()]
}
pub(crate) fn trampoline_ptr(&self, index: LoweredIndex) -> NonNull<VMFunctionBody> { pub(crate) fn trampoline_ptr(&self, index: LoweredIndex) -> NonNull<VMFunctionBody> {
let info = &self.inner.trampolines[index]; let info = &self.inner.trampolines[index];
let text = &self.inner.trampoline_obj.mmap()[self.inner.text.clone()]; let text = self.text();
let trampoline = &text[info.start as usize..][..info.length as usize]; let trampoline = &text[info.start as usize..][..info.length as usize];
NonNull::new(trampoline.as_ptr() as *mut VMFunctionBody).unwrap() NonNull::new(trampoline.as_ptr() as *mut VMFunctionBody).unwrap()
} }

View File

@@ -210,6 +210,7 @@ impl<'a> Instantiator<'a> {
imports: &'a PrimaryMap<RuntimeImportIndex, RuntimeImport>, imports: &'a PrimaryMap<RuntimeImportIndex, RuntimeImport>,
) -> Instantiator<'a> { ) -> Instantiator<'a> {
let env_component = component.env_component(); let env_component = component.env_component();
store.modules_mut().register_component(component);
Instantiator { Instantiator {
component, component,
imports, imports,

View File

@@ -242,7 +242,7 @@ impl Instance {
// Register the module just before instantiation to ensure we keep the module // Register the module just before instantiation to ensure we keep the module
// properly referenced while in use by the store. // properly referenced while in use by the store.
store.modules_mut().register(module); store.modules_mut().register_module(module);
// The first thing we do is issue an instance allocation request // The first thing we do is issue an instance allocation request
// to the instance allocator. This, on success, will give us an // to the instance allocator. This, on success, will give us an

View File

@@ -1,5 +1,7 @@
//! Implements a registry of modules for a store. //! Implements a registry of modules for a store.
#[cfg(feature = "component-model")]
use crate::component::Component;
use crate::{Engine, Module}; use crate::{Engine, Module};
use std::{ use std::{
collections::BTreeMap, collections::BTreeMap,
@@ -24,12 +26,21 @@ lazy_static::lazy_static! {
#[derive(Default)] #[derive(Default)]
pub struct ModuleRegistry { pub struct ModuleRegistry {
// Keyed by the end address of the module's code in memory. // Keyed by the end address of the module's code in memory.
modules_with_code: BTreeMap<usize, Module>, //
// The value here is the start address and the module/component it
// corresponds to.
modules_with_code: BTreeMap<usize, (usize, ModuleOrComponent)>,
// Preserved for keeping data segments alive or similar // Preserved for keeping data segments alive or similar
modules_without_code: Vec<Module>, modules_without_code: Vec<Module>,
} }
enum ModuleOrComponent {
Module(Module),
#[cfg(feature = "component-model")]
Component(Component),
}
fn start(module: &Module) -> usize { fn start(module: &Module) -> usize {
assert!(!module.compiled_module().code().is_empty()); assert!(!module.compiled_module().code().is_empty());
module.compiled_module().code().as_ptr() as usize module.compiled_module().code().as_ptr() as usize
@@ -42,15 +53,23 @@ impl ModuleRegistry {
} }
fn module(&self, pc: usize) -> Option<&Module> { fn module(&self, pc: usize) -> Option<&Module> {
let (end, module) = self.modules_with_code.range(pc..).next()?; match self.module_or_component(pc)? {
if pc < start(module) || *end < pc { ModuleOrComponent::Module(m) => Some(m),
#[cfg(feature = "component-model")]
ModuleOrComponent::Component(_) => None,
}
}
fn module_or_component(&self, pc: usize) -> Option<&ModuleOrComponent> {
let (end, (start, module)) = self.modules_with_code.range(pc..).next()?;
if pc < *start || *end < pc {
return None; return None;
} }
Some(module) Some(module)
} }
/// Registers a new module with the registry. /// Registers a new module with the registry.
pub fn register(&mut self, module: &Module) { pub fn register_module(&mut self, module: &Module) {
let compiled_module = module.compiled_module(); let compiled_module = module.compiled_module();
// If there's not actually any functions in this module then we may // If there's not actually any functions in this module then we may
@@ -61,39 +80,70 @@ impl ModuleRegistry {
// modules and retain them. // modules and retain them.
if compiled_module.finished_functions().len() == 0 { if compiled_module.finished_functions().len() == 0 {
self.modules_without_code.push(module.clone()); self.modules_without_code.push(module.clone());
} else {
// The module code range is exclusive for end, so make it inclusive as it
// may be a valid PC value
let start_addr = start(module);
let end_addr = start_addr + compiled_module.code().len() - 1;
self.register(
start_addr,
end_addr,
ModuleOrComponent::Module(module.clone()),
);
}
}
#[cfg(feature = "component-model")]
pub fn register_component(&mut self, component: &Component) {
// If there's no text section associated with this component (e.g. no
// lowered functions) then there's nothing to register, otherwise it's
// registered along the same lines as modules above.
//
// Note that empty components don't need retaining here since it doesn't
// have data segments like empty modules.
let text = component.text();
if text.is_empty() {
return; return;
} }
let start = text.as_ptr() as usize;
self.register(
start,
start + text.len() - 1,
ModuleOrComponent::Component(component.clone()),
);
}
// The module code range is exclusive for end, so make it inclusive as it /// Registers a new module with the registry.
// may be a valid PC value fn register(&mut self, start_addr: usize, end_addr: usize, item: ModuleOrComponent) {
let start_addr = start(module);
let end_addr = start_addr + compiled_module.code().len() - 1;
// Ensure the module isn't already present in the registry // Ensure the module isn't already present in the registry
// This is expected when a module is instantiated multiple times in the // This is expected when a module is instantiated multiple times in the
// same store // same store
if let Some(m) = self.modules_with_code.get(&end_addr) { if let Some((other_start, _)) = self.modules_with_code.get(&end_addr) {
assert_eq!(start(m), start_addr); assert_eq!(*other_start, start_addr);
return; return;
} }
// Assert that this module's code doesn't collide with any other // Assert that this module's code doesn't collide with any other
// registered modules // registered modules
if let Some((_, prev)) = self.modules_with_code.range(end_addr..).next() { if let Some((_, (prev_start, _))) = self.modules_with_code.range(start_addr..).next() {
assert!(start(prev) > end_addr); assert!(*prev_start > end_addr);
} }
if let Some((prev_end, _)) = self.modules_with_code.range(..=start_addr).next_back() { if let Some((prev_end, _)) = self.modules_with_code.range(..=start_addr).next_back() {
assert!(*prev_end < start_addr); assert!(*prev_end < start_addr);
} }
let prev = self.modules_with_code.insert(end_addr, module.clone()); let prev = self.modules_with_code.insert(end_addr, (start_addr, item));
assert!(prev.is_none()); assert!(prev.is_none());
} }
/// Looks up a trampoline from an anyfunc. /// Looks up a trampoline from an anyfunc.
pub fn lookup_trampoline(&self, anyfunc: &VMCallerCheckedAnyfunc) -> Option<VMTrampoline> { pub fn lookup_trampoline(&self, anyfunc: &VMCallerCheckedAnyfunc) -> Option<VMTrampoline> {
let module = self.module(anyfunc.func_ptr.as_ptr() as usize)?; let signatures = match self.module_or_component(anyfunc.func_ptr.as_ptr() as usize)? {
module.signatures().trampoline(anyfunc.type_index) ModuleOrComponent::Module(m) => m.signatures(),
#[cfg(feature = "component-model")]
ModuleOrComponent::Component(c) => c.signatures(),
};
signatures.trampoline(anyfunc.type_index)
} }
} }

View File

@@ -19,7 +19,7 @@ use wasmtime_runtime::{VMSharedSignatureIndex, VMTrampoline};
pub struct SignatureCollection { pub struct SignatureCollection {
registry: Arc<RwLock<SignatureRegistryInner>>, registry: Arc<RwLock<SignatureRegistryInner>>,
signatures: PrimaryMap<SignatureIndex, VMSharedSignatureIndex>, signatures: PrimaryMap<SignatureIndex, VMSharedSignatureIndex>,
trampolines: HashMap<VMSharedSignatureIndex, (usize, VMTrampoline)>, trampolines: HashMap<VMSharedSignatureIndex, VMTrampoline>,
} }
impl SignatureCollection { impl SignatureCollection {
@@ -59,9 +59,7 @@ impl SignatureCollection {
/// Gets a trampoline for a registered signature. /// Gets a trampoline for a registered signature.
pub fn trampoline(&self, index: VMSharedSignatureIndex) -> Option<VMTrampoline> { pub fn trampoline(&self, index: VMSharedSignatureIndex) -> Option<VMTrampoline> {
self.trampolines self.trampolines.get(&index).copied()
.get(&index)
.map(|(_, trampoline)| *trampoline)
} }
} }
@@ -93,7 +91,7 @@ impl SignatureRegistryInner {
trampolines: impl Iterator<Item = (SignatureIndex, VMTrampoline)>, trampolines: impl Iterator<Item = (SignatureIndex, VMTrampoline)>,
) -> ( ) -> (
PrimaryMap<SignatureIndex, VMSharedSignatureIndex>, PrimaryMap<SignatureIndex, VMSharedSignatureIndex>,
HashMap<VMSharedSignatureIndex, (usize, VMTrampoline)>, HashMap<VMSharedSignatureIndex, VMTrampoline>,
) { ) {
let mut sigs = PrimaryMap::default(); let mut sigs = PrimaryMap::default();
let mut map = HashMap::default(); let mut map = HashMap::default();
@@ -104,7 +102,7 @@ impl SignatureRegistryInner {
} }
for (index, trampoline) in trampolines { for (index, trampoline) in trampolines {
map.insert(sigs[index], (1, trampoline)); map.insert(sigs[index], trampoline);
} }
(sigs, map) (sigs, map)
@@ -165,8 +163,8 @@ impl SignatureRegistryInner {
} else { } else {
// Otherwise, use the trampolines map, which has reference counts related // Otherwise, use the trampolines map, which has reference counts related
// to the stored index // to the stored index
for (index, (count, _)) in collection.trampolines.iter() { for (index, _) in collection.trampolines.iter() {
self.unregister_entry(*index, *count); self.unregister_entry(*index, 1);
} }
} }
} }

View File

@@ -3,7 +3,7 @@ use anyhow::Result;
use std::rc::Rc; use std::rc::Rc;
use std::sync::Arc; use std::sync::Arc;
use wasmtime::component::*; use wasmtime::component::*;
use wasmtime::{Store, Trap, TrapCode}; use wasmtime::{Store, StoreContextMut, Trap, TrapCode};
const CANON_32BIT_NAN: u32 = 0b01111111110000000000000000000000; const CANON_32BIT_NAN: u32 = 0b01111111110000000000000000000000;
const CANON_64BIT_NAN: u64 = 0b0111111111111000000000000000000000000000000000000000000000000000; const CANON_64BIT_NAN: u64 = 0b0111111111111000000000000000000000000000000000000000000000000000;
@@ -1858,3 +1858,58 @@ fn invalid_alignment() -> Result<()> {
Ok(()) Ok(())
} }
#[test]
fn drop_component_still_works() -> Result<()> {
let component = r#"
(component
(import "f" (func $f))
(core func $f_lower
(canon lower (func $f))
)
(core module $m
(import "" "" (func $f))
(func $f2
call $f
call $f
)
(export "f" (func $f2))
)
(core instance $i (instantiate $m
(with "" (instance
(export "" (func $f_lower))
))
))
(func (export "f")
(canon lift
(core func $i "f")
)
)
)
"#;
let (mut store, instance) = {
let engine = super::engine();
let component = Component::new(&engine, component)?;
let mut store = Store::new(&engine, 0);
let mut linker = Linker::new(&engine);
linker
.root()
.func_wrap("f", |mut store: StoreContextMut<'_, u32>| -> Result<()> {
*store.data_mut() += 1;
Ok(())
})?;
let instance = linker.instantiate(&mut store, &component)?;
(store, instance)
};
let f = instance.get_typed_func::<(), (), _>(&mut store, "f")?;
assert_eq!(*store.data(), 0);
f.call(&mut store, ())?;
assert_eq!(*store.data(), 2);
Ok(())
}

View File

@@ -614,3 +614,50 @@ fn bad_import_alignment() -> Result<()> {
assert!(trap.to_string().contains("pointer not aligned"), "{}", trap); assert!(trap.to_string().contains("pointer not aligned"), "{}", trap);
Ok(()) Ok(())
} }
#[test]
fn no_actual_wasm_code() -> Result<()> {
let component = r#"
(component
(import "f" (func $f))
(core func $f_lower
(canon lower (func $f))
)
(core module $m
(import "" "" (func $f))
(export "f" (func $f))
)
(core instance $i (instantiate $m
(with "" (instance
(export "" (func $f_lower))
))
))
(func (export "thunk")
(canon lift
(core func $i "f")
)
)
)
"#;
let engine = super::engine();
let component = Component::new(&engine, component)?;
let mut store = Store::new(&engine, 0);
let mut linker = Linker::new(&engine);
linker
.root()
.func_wrap("f", |mut store: StoreContextMut<'_, u32>| -> Result<()> {
*store.data_mut() += 1;
Ok(())
})?;
let instance = linker.instantiate(&mut store, &component)?;
let thunk = instance.get_typed_func::<(), (), _>(&mut store, "thunk")?;
assert_eq!(*store.data(), 0);
thunk.call(&mut store, ())?;
assert_eq!(*store.data(), 1);
Ok(())
}