mmtk::util::alloc

Trait Allocator

source
pub trait Allocator<VM: VMBinding>: Downcast {
    // Required methods
    fn get_tls(&self) -> VMThread;
    fn get_space(&self) -> &'static dyn Space<VM>;
    fn get_context(&self) -> &AllocatorContext<VM>;
    fn does_thread_local_allocation(&self) -> bool;
    fn alloc(&mut self, size: usize, align: usize, offset: usize) -> Address;
    fn alloc_slow_once(
        &mut self,
        size: usize,
        align: usize,
        offset: usize,
    ) -> Address;

    // Provided methods
    fn get_thread_local_buffer_granularity(&self) -> usize { ... }
    fn alloc_slow(
        &mut self,
        size: usize,
        align: usize,
        offset: usize,
    ) -> Address { ... }
    fn alloc_slow_inline(
        &mut self,
        size: usize,
        align: usize,
        offset: usize,
    ) -> Address { ... }
    fn alloc_slow_once_traced(
        &mut self,
        size: usize,
        align: usize,
        offset: usize,
    ) -> Address { ... }
    fn alloc_slow_once_precise_stress(
        &mut self,
        size: usize,
        align: usize,
        offset: usize,
        need_poll: bool,
    ) -> Address { ... }
    fn on_mutator_destroy(&mut self) { ... }
}
Expand description

A trait which implements allocation routines. Every allocator needs to implements this trait.

Required Methods§

source

fn get_tls(&self) -> VMThread

Return the VMThread associated with this allocator instance.

source

fn get_space(&self) -> &'static dyn Space<VM>

Return the Space instance associated with this allocator instance.

source

fn get_context(&self) -> &AllocatorContext<VM>

Return the context for the allocator.

source

fn does_thread_local_allocation(&self) -> bool

Return if this allocator can do thread local allocation. If an allocator does not do thread local allocation, each allocation will go to slowpath and will have a check for GC polls.

source

fn alloc(&mut self, size: usize, align: usize, offset: usize) -> Address

An allocation attempt. The implementation of this function depends on the allocator used. If an allocator supports thread local allocations, then the allocation will be serviced from its TLAB, otherwise it will default to using the slowpath, i.e. alloc_slow.

Note that in the case where the VM is out of memory, we invoke Collection::out_of_memory to inform the binding and then return a null pointer back to it. We have no assumptions on whether the VM will continue executing or abort immediately.

An allocator needs to make sure the object reference for the returned address is in the same chunk as the returned address (so the side metadata and the SFT for an object reference is valid). See crate::util::alloc::object_ref_guard.

Arguments:

  • size: the allocation size in bytes.
  • align: the required alignment in bytes.
  • offset the required offset in bytes.
source

fn alloc_slow_once( &mut self, size: usize, align: usize, offset: usize, ) -> Address

Single slow path allocation attempt. This is called by alloc_slow_inline. The implementation of this function depends on the allocator used. Generally, if an allocator supports thread local allocations, it will try to allocate more TLAB space here. If it doesn’t, then (generally) the allocator simply allocates enough space for the current object.

Arguments:

  • size: the allocation size in bytes.
  • align: the required alignment in bytes.
  • offset the required offset in bytes.

Provided Methods§

source

fn get_thread_local_buffer_granularity(&self) -> usize

Return at which granularity the allocator acquires memory from the global space and use them as thread local buffer. For example, the BumpAllocator acquires memory at 32KB blocks. Depending on the actual size for the current object, they always acquire memory of N*32KB (N>=1). Thus the BumpAllocator returns 32KB for this method. Only allocators that do thread local allocation need to implement this method.

source

fn alloc_slow(&mut self, size: usize, align: usize, offset: usize) -> Address

Slowpath allocation attempt. This function is explicitly not inlined for performance considerations.

Arguments:

  • size: the allocation size in bytes.
  • align: the required alignment in bytes.
  • offset the required offset in bytes.
source

fn alloc_slow_inline( &mut self, size: usize, align: usize, offset: usize, ) -> Address

Slowpath allocation attempt. This function executes the actual slowpath allocation. A slowpath allocation in MMTk attempts to allocate the object using the per-allocator definition of alloc_slow_once. This function also accounts for increasing the allocation bytes in order to support stress testing. In case precise stress testing is being used, the alloc_slow_once_precise_stress function is used instead.

Note that in the case where the VM is out of memory, we invoke Collection::out_of_memory with a AllocationError::HeapOutOfMemory error to inform the binding and then return a null pointer back to it. We have no assumptions on whether the VM will continue executing or abort immediately on a AllocationError::HeapOutOfMemory error.

Arguments:

  • size: the allocation size in bytes.
  • align: the required alignment in bytes.
  • offset the required offset in bytes.
source

fn alloc_slow_once_traced( &mut self, size: usize, align: usize, offset: usize, ) -> Address

A wrapper method for alloc_slow_once to insert USDT tracepoints.

Arguments:

  • size: the allocation size in bytes.
  • align: the required alignment in bytes.
  • offset the required offset in bytes.
source

fn alloc_slow_once_precise_stress( &mut self, size: usize, align: usize, offset: usize, need_poll: bool, ) -> Address

Single slowpath allocation attempt for stress test. When the stress factor is set (e.g. to N), we would expect for every N bytes allocated, we will trigger a stress GC. However, for allocators that do thread local allocation, they may allocate from their thread local buffer which does not have a GC poll check, and they may even allocate with the JIT generated allocation fastpath which is unaware of stress test GC. For both cases, we are not able to guarantee a stress GC is triggered every N bytes. To solve this, when the stress factor is set, we will call this method instead of the normal alloc_slow_once(). We expect the implementation of this slow allocation will trick the fastpath so every allocation will fail in the fastpath, jump to the slow path and eventually call this method again for the actual allocation.

The actual implementation about how to trick the fastpath may vary. For example, our bump pointer allocator will set the thread local buffer limit to the buffer size instead of the buffer end address. In this case, every fastpath check (cursor + size < limit) will fail, and jump to this slowpath. In the slowpath, we still allocate from the thread local buffer, and recompute the limit (remaining buffer size).

If an allocator does not do thread local allocation (which returns false for does_thread_local_allocation()), it does not need to override this method. The default implementation will simply call allow_slow_once() and it will work fine for allocators that do not have thread local allocation.

Arguments:

  • size: the allocation size in bytes.
  • align: the required alignment in bytes.
  • offset the required offset in bytes.
  • need_poll: if this is true, the implementation must poll for a GC, rather than attempting to allocate from the local buffer.
source

fn on_mutator_destroy(&mut self)

The crate::plan::Mutator that includes this allocator is going to be destroyed. Some allocators may need to save/transfer its thread local data to the space.

Implementations§

source§

impl<VM> dyn Allocator<VM>
where VM: Any + 'static + VMBinding,

source

pub fn is<__T: Allocator<VM>>(&self) -> bool

Returns true if the trait object wraps an object of type __T.

source

pub fn downcast<__T: Allocator<VM>>( self: Box<Self>, ) -> Result<Box<__T>, Box<Self>>

Returns a boxed object from a boxed trait object if the underlying object is of type __T. Returns the original boxed trait if it isn’t.

source

pub fn downcast_rc<__T: Allocator<VM>>( self: Rc<Self>, ) -> Result<Rc<__T>, Rc<Self>>

Returns an Rc-ed object from an Rc-ed trait object if the underlying object is of type __T. Returns the original Rc-ed trait if it isn’t.

source

pub fn downcast_ref<__T: Allocator<VM>>(&self) -> Option<&__T>

Returns a reference to the object within the trait object if it is of type __T, or None if it isn’t.

source

pub fn downcast_mut<__T: Allocator<VM>>(&mut self) -> Option<&mut __T>

Returns a mutable reference to the object within the trait object if it is of type __T, or None if it isn’t.

Implementors§