Skip to content

Commit

Permalink
Merge #1902
Browse files Browse the repository at this point in the history
1902: Fix memory leak in `InstanceAllocator` on failure r=MarkMcCaskey a=MarkMcCaskey

This code needs to be cleaned up, just hacked it out really quickly.

The idea is to use a builder pattern to create a type-safe state machine for setting up things for the `InstanceAllocator`; this should allow us to remove `unsafe` in our "public" API because we can guarantee certain invariants because of the restricted nature of these types.

The names for the new types are long and don't really make sense but I didn't want to get blocked on that: suggestions welcome!

In combination with #1865 , this PR should fix the all the memory leaks seen by asan while running `make test-cranelift-jit` 🎉 🎉 🎉 

Co-authored-by: Mark McCaskey <[email protected]>
  • Loading branch information
bors[bot] and Mark McCaskey authored Dec 11, 2020
2 parents c1402c8 + 569bfd5 commit a9fb192
Show file tree
Hide file tree
Showing 6 changed files with 267 additions and 216 deletions.
17 changes: 7 additions & 10 deletions lib/engine/src/artifact.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,8 +12,8 @@ use wasmer_types::{
SignatureIndex, TableIndex,
};
use wasmer_vm::{
FunctionBodyPtr, InstanceHandle, MemoryStyle, ModuleInfo, TableStyle, VMSharedSignatureIndex,
VMTrampoline,
FunctionBodyPtr, InstanceAllocator, InstanceHandle, MemoryStyle, ModuleInfo, TableStyle,
VMSharedSignatureIndex, VMTrampoline,
};

/// An `Artifact` is the product that the `Engine`
Expand Down Expand Up @@ -95,7 +95,6 @@ pub trait Artifact: Send + Sync + Upcastable {
self.preinstantiate()?;

let module = self.module();
let (instance_ptr, offsets) = InstanceHandle::allocate_instance(&module);
let (imports, import_initializers) = {
let mut imports = resolve_imports(
&module,
Expand All @@ -114,15 +113,14 @@ pub trait Artifact: Send + Sync + Upcastable {
};

// Get pointers to where metadata about local memories should live in VM memory.
let memory_definition_locations =
InstanceHandle::memory_definition_locations(instance_ptr, &offsets);
// Get pointers to where metadata about local tables should live in VM memory.

let (allocator, memory_definition_locations, table_definition_locations) =
InstanceAllocator::new(&*module);
let finished_memories = tunables
.create_memories(&module, self.memory_styles(), &memory_definition_locations)
.map_err(InstantiationError::Link)?
.into_boxed_slice();
// Get pointers to where metadata about local tables should live in VM memory.
let table_definition_locations =
InstanceHandle::table_definition_locations(instance_ptr, &offsets);
let finished_tables = tunables
.create_tables(&module, self.table_styles(), &table_definition_locations)
.map_err(InstantiationError::Link)?
Expand All @@ -135,8 +133,7 @@ pub trait Artifact: Send + Sync + Upcastable {
self.register_frame_info();

let handle = InstanceHandle::new(
instance_ptr,
offsets,
allocator,
module,
self.finished_functions().clone(),
self.finished_function_call_trampolines().clone(),
Expand Down
18 changes: 9 additions & 9 deletions lib/vm/src/export.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
// Attributions: https://github.com/wasmerio/wasmer/blob/master/ATTRIBUTIONS.md

use crate::global::Global;
use crate::instance::InstanceAllocator;
use crate::instance::InstanceRef;
use crate::memory::{Memory, MemoryStyle};
use crate::table::{Table, TableStyle};
use crate::vmcontext::{VMFunctionBody, VMFunctionEnvironment, VMFunctionKind, VMTrampoline};
Expand Down Expand Up @@ -49,8 +49,8 @@ pub struct VMExportFunction {
pub call_trampoline: Option<VMTrampoline>,

/// A “reference” to the instance through the
/// `InstanceAllocator`. `None` if it is a host function.
pub instance_allocator: Option<InstanceAllocator>,
/// `InstanceRef`. `None` if it is a host function.
pub instance_allocator: Option<InstanceRef>,
}

/// # Safety
Expand All @@ -74,8 +74,8 @@ pub struct VMExportTable {
pub from: Arc<dyn Table>,

/// A “reference” to the instance through the
/// `InstanceAllocator`. `None` if it is a host function.
pub instance_allocator: Option<InstanceAllocator>,
/// `InstanceRef`. `None` if it is a host function.
pub instance_allocator: Option<InstanceRef>,
}

/// # Safety
Expand Down Expand Up @@ -120,8 +120,8 @@ pub struct VMExportMemory {
pub from: Arc<dyn Memory>,

/// A “reference” to the instance through the
/// `InstanceAllocator`. `None` if it is a host function.
pub instance_allocator: Option<InstanceAllocator>,
/// `InstanceRef`. `None` if it is a host function.
pub instance_allocator: Option<InstanceRef>,
}

/// # Safety
Expand Down Expand Up @@ -166,8 +166,8 @@ pub struct VMExportGlobal {
pub from: Arc<Global>,

/// A “reference” to the instance through the
/// `InstanceAllocator`. `None` if it is a host function.
pub instance_allocator: Option<InstanceAllocator>,
/// `InstanceRef`. `None` if it is a host function.
pub instance_allocator: Option<InstanceRef>,
}

/// # Safety
Expand Down
197 changes: 197 additions & 0 deletions lib/vm/src/instance/allocator.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,197 @@
use super::{Instance, InstanceRef};
use crate::vmcontext::{VMMemoryDefinition, VMTableDefinition};
use crate::{ModuleInfo, VMOffsets};
use std::alloc::{self, Layout};
use std::convert::TryFrom;
use std::mem;
use std::ptr::{self, NonNull};
use wasmer_types::entity::EntityRef;
use wasmer_types::{LocalMemoryIndex, LocalTableIndex};

/// This is an intermediate type that manages the raw allocation and
/// metadata when creating an [`Instance`].
///
/// This type will free the allocated memory if it's dropped before
/// being used.
///
/// The [`InstanceAllocator::instance_layout`] computes the correct
/// layout to represent the wanted [`Instance`].
///
/// Then we use this layout to allocate an empty `Instance` properly.
pub struct InstanceAllocator {
/// The buffer that will contain the [`Instance`] and dynamic fields.
instance_ptr: NonNull<Instance>,
/// The layout of the `instance_ptr` buffer.
instance_layout: Layout,
/// Information about the offsets into the `instance_ptr` buffer for
/// the dynamic fields.
offsets: VMOffsets,
/// Whether or not this type has transferred ownership of the
/// `instance_ptr` buffer. If it has not when being dropped,
/// the buffer should be freed.
consumed: bool,
}

impl Drop for InstanceAllocator {
fn drop(&mut self) {
if !self.consumed {
// If `consumed` has not been set, then we still have ownership
// over the buffer and must free it.
let instance_ptr = self.instance_ptr.as_ptr();
unsafe {
std::alloc::dealloc(instance_ptr as *mut u8, self.instance_layout);
}
}
}
}

impl InstanceAllocator {
/// Allocates instance data for use with [`InstanceHandle::new`].
///
/// Returns a wrapper type around the allocation and 2 vectors of
/// pointers into the allocated buffer. These lists of pointers
/// correspond to the location in memory for the local memories and
/// tables respectively. These pointers should be written to before
/// calling [`InstanceHandle::new`].
pub fn new(
module: &ModuleInfo,
) -> (
InstanceAllocator,
Vec<NonNull<VMMemoryDefinition>>,
Vec<NonNull<VMTableDefinition>>,
) {
let offsets = VMOffsets::new(mem::size_of::<usize>() as u8, module);
let instance_layout = Self::instance_layout(&offsets);

#[allow(clippy::cast_ptr_alignment)]
let instance_ptr = unsafe { alloc::alloc(instance_layout) as *mut Instance };

let instance_ptr = if let Some(ptr) = NonNull::new(instance_ptr) {
ptr
} else {
alloc::handle_alloc_error(instance_layout);
};

let allocator = Self {
instance_ptr,
instance_layout,
offsets,
consumed: false,
};

// # Safety
// Both of these calls are safe because we allocate the pointer
// above with the same `offsets` that these functions use.
// Thus there will be enough valid memory for both of them.
let memories = unsafe { allocator.memory_definition_locations() };
let tables = unsafe { allocator.table_definition_locations() };

(allocator, memories, tables)
}

/// Calculate the appropriate layout for the [`Instance`].
fn instance_layout(offsets: &VMOffsets) -> Layout {
let vmctx_size = usize::try_from(offsets.size_of_vmctx())
.expect("Failed to convert the size of `vmctx` to a `usize`");

let instance_vmctx_layout =
Layout::array::<u8>(vmctx_size).expect("Failed to create a layout for `VMContext`");

let (instance_layout, _offset) = Layout::new::<Instance>()
.extend(instance_vmctx_layout)
.expect("Failed to extend to `Instance` layout to include `VMContext`");

instance_layout.pad_to_align()
}

/// Get the locations of where the local [`VMMemoryDefinition`]s should be stored.
///
/// This function lets us create `Memory` objects on the host with backing
/// memory in the VM.
///
/// # Safety
/// - `instance_ptr` must point to enough memory that all of the
/// offsets in `offsets` point to valid locations in memory,
/// i.e. `instance_ptr` must have been allocated by
/// `Self::new`.
unsafe fn memory_definition_locations(&self) -> Vec<NonNull<VMMemoryDefinition>> {
let num_memories = self.offsets.num_local_memories;
let num_memories = usize::try_from(num_memories).unwrap();
let mut out = Vec::with_capacity(num_memories);

// We need to do some pointer arithmetic now. The unit is `u8`.
let ptr = self.instance_ptr.cast::<u8>().as_ptr();
let base_ptr = ptr.add(mem::size_of::<Instance>());

for i in 0..num_memories {
let mem_offset = self
.offsets
.vmctx_vmmemory_definition(LocalMemoryIndex::new(i));
let mem_offset = usize::try_from(mem_offset).unwrap();

let new_ptr = NonNull::new_unchecked(base_ptr.add(mem_offset));

out.push(new_ptr.cast());
}

out
}

/// Get the locations of where the [`VMTableDefinition`]s should be stored.
///
/// This function lets us create [`Table`] objects on the host with backing
/// memory in the VM.
///
/// # Safety
/// - `instance_ptr` must point to enough memory that all of the
/// offsets in `offsets` point to valid locations in memory,
/// i.e. `instance_ptr` must have been allocated by
/// `Self::new`.
unsafe fn table_definition_locations(&self) -> Vec<NonNull<VMTableDefinition>> {
let num_tables = self.offsets.num_local_tables;
let num_tables = usize::try_from(num_tables).unwrap();
let mut out = Vec::with_capacity(num_tables);

// We need to do some pointer arithmetic now. The unit is `u8`.
let ptr = self.instance_ptr.cast::<u8>().as_ptr();
let base_ptr = ptr.add(std::mem::size_of::<Instance>());

for i in 0..num_tables {
let table_offset = self
.offsets
.vmctx_vmtable_definition(LocalTableIndex::new(i));
let table_offset = usize::try_from(table_offset).unwrap();

let new_ptr = NonNull::new_unchecked(base_ptr.add(table_offset));

out.push(new_ptr.cast());
}
out
}

/// Finish preparing by writing the [`Instance`] into memory.
pub(crate) fn write_instance(mut self, instance: Instance) -> InstanceRef {
// prevent the old state's drop logic from being called as we
// transition into the new state.
self.consumed = true;

unsafe {
// `instance` is moved at `instance_ptr`. This pointer has
// been allocated by `Self::allocate_instance` (so by
// `InstanceRef::allocate_instance`.
ptr::write(self.instance_ptr.as_ptr(), instance);
// Now `instance_ptr` is correctly initialized!
}
let instance = self.instance_ptr;
let instance_layout = self.instance_layout;

// This is correct because of the invariants of `Self` and
// because we write `Instance` to the pointer in this function.
unsafe { InstanceRef::new(instance, instance_layout) }
}

/// Get the [`VMOffsets`] for the allocated buffer.
pub(crate) fn offsets(&self) -> &VMOffsets {
&self.offsets
}
}
Loading

0 comments on commit a9fb192

Please sign in to comment.