agave-scheduling-utils 3.1.13

Common utilities for building Agave scheduler implementations
Documentation
use {
    agave_scheduler_bindings::{SharableTransactionBatchRegion, SharableTransactionRegion},
    agave_transaction_view::transaction_data::TransactionData,
    core::ptr::NonNull,
    rts_alloc::Allocator,
};

pub struct TransactionPtr {
    ptr: NonNull<u8>,
    len: usize,
}

impl TransactionData for TransactionPtr {
    fn data(&self) -> &[u8] {
        unsafe { core::slice::from_raw_parts(self.ptr.as_ptr(), self.len) }
    }
}

impl TransactionPtr {
    /// # Safety
    /// - `sharable_transaction_region` must reference a valid offset and length
    ///   within the `allocator`.
    pub unsafe fn from_sharable_transaction_region(
        sharable_transaction_region: &SharableTransactionRegion,
        allocator: &Allocator,
    ) -> Self {
        let ptr = allocator.ptr_from_offset(sharable_transaction_region.offset);
        Self {
            ptr,
            len: sharable_transaction_region.length as usize,
        }
    }

    /// Translate the ptr type into a sharable region.
    ///
    /// # Safety
    /// - `allocator` must be the allocator owning the memory region pointed
    ///   to by `self`.
    pub unsafe fn to_sharable_transaction_region(
        &self,
        allocator: &Allocator,
    ) -> SharableTransactionRegion {
        // SAFETY: The `TransactionPtr` creation `Self::from_sharable_transaction_region`
        // is already conditioned on the offset being valid, if that safety constraint
        // was satisfied translation back to offset is safe.
        let offset = unsafe { allocator.offset(self.ptr) };
        SharableTransactionRegion {
            offset,
            length: self.len as u32,
        }
    }

    /// Frees the memory region pointed to in the `allocator`.
    /// This should only be called by the owner of the memory
    /// i.e. the external scheduler.
    ///
    /// # Safety
    /// - Data region pointed to by `TransactionPtr` belongs to the `allocator`.
    /// - Inner `ptr` must not have been previously freed.
    pub unsafe fn free(self, allocator: &Allocator) {
        unsafe { allocator.free(self.ptr) }
    }
}

/// A batch of transaction pointers that can be iterated over.
pub struct TransactionPtrBatch<'a> {
    ptr: NonNull<SharableTransactionRegion>,
    num_transactions: usize,
    allocator: &'a Allocator,
}

impl<'a> TransactionPtrBatch<'a> {
    /// # Safety
    /// - [`SharableTransactionBatchRegion`] must reference a valid offset and length
    ///   within the `allocator`.
    /// - ALL [`SharableTransactionRegion`]  within the batch must be valid.
    ///   See [`TransactionPtr::from_sharable_transaction_region`] for details.
    pub unsafe fn from_sharable_transaction_batch_region(
        sharable_transaction_batch_region: &SharableTransactionBatchRegion,
        allocator: &'a Allocator,
    ) -> Self {
        let ptr = allocator
            .ptr_from_offset(sharable_transaction_batch_region.transactions_offset)
            .cast();

        Self {
            ptr,
            num_transactions: usize::from(sharable_transaction_batch_region.num_transactions),
            allocator,
        }
    }

    /// Iterator returning [`TransactionPtr`] for each transaction in the batch.
    pub fn iter(&'a self) -> impl Iterator<Item = TransactionPtr> + 'a {
        (0..self.num_transactions)
            .map(|idx| unsafe { self.ptr.add(idx) })
            .map(|ptr| unsafe {
                TransactionPtr::from_sharable_transaction_region(ptr.as_ref(), self.allocator)
            })
    }

    /// Free all transactions in the batch, then free the batch itself.
    pub fn free(self) {
        for transaction_ptr in self.iter() {
            unsafe { transaction_ptr.free(self.allocator) }
        }

        unsafe { self.allocator.free(self.ptr.cast()) }
    }
}