//! Low-level allocation and OOM-handling utilities. mod arc; mod boxed; mod try_collect; mod try_new; mod vec; pub use boxed::{ BoxedSliceFromFallibleIterError, TooFewItemsOrOom, boxed_slice_write_iter, new_boxed_slice_from_fallible_iter, new_boxed_slice_from_iter, new_boxed_slice_from_iter_with_len, new_uninit_boxed_slice, }; pub use try_collect::{TryCollect, TryExtend, TryFromIterator}; pub use try_new::{TryNew, try_new}; pub use vec::Vec; use crate::error::OutOfMemory; use core::{alloc::Layout, ptr::NonNull}; /// Try to allocate a block of memory that fits the given layout, or return an /// `OutOfMemory` error. /// /// # Safety /// /// Same as `alloc::alloc::alloc`: layout must have non-zero size. #[inline] unsafe fn try_alloc(layout: Layout) -> Result, OutOfMemory> { // Safety: same as our safety conditions. debug_assert!(layout.size() > 0); let ptr = unsafe { std_alloc::alloc::alloc(layout) }; if let Some(ptr) = NonNull::new(ptr) { Ok(ptr) } else { Err(OutOfMemory::new(layout.size())) } } /// Tries to reallocate a block of memory, returning `OutOfMemory` on failure. /// /// Analogue of [`alloc::alloc::realloc`]. /// /// # Safety /// /// Same as `alloc::alloc::realloc`: `ptr` must be allocated with `layout`, /// `layout` must be nonzero in size, and `new_size` must be nonzero and valid. #[inline] unsafe fn try_realloc( ptr: *mut u8, layout: Layout, new_size: usize, ) -> Result, OutOfMemory> { // Safety: same as our safety conditions. debug_assert!(layout.size() > 0); debug_assert!(new_size > 0); let ptr = unsafe { std_alloc::alloc::realloc(ptr, layout, new_size) }; if let Some(ptr) = NonNull::new(ptr) { Ok(ptr) } else { Err(OutOfMemory::new(new_size)) } } /// An extension trait for ignoring `OutOfMemory` errors. /// /// Use this to unwrap a `Result` into its inner `T` or /// otherwise panic, leveraging the type system to be sure that you aren't ever /// accidentally unwrapping non-`OutOfMemory` errors. pub trait PanicOnOom { /// The non-`OutOfMemory` result of calling `panic_on_oom`. type Result; /// Panic on `OutOfMemory` errors, returning the non-`OutOfMemory` result. fn panic_on_oom(self) -> Self::Result; } impl PanicOnOom for Result { type Result = T; #[track_caller] fn panic_on_oom(self) -> Self::Result { match self { Ok(x) => x, Err(oom) => panic!("unhandled out-of-memory error: {oom}"), } } }