diff --git a/uefi/CHANGELOG.md b/uefi/CHANGELOG.md index e13eda4e9..dfb07a075 100644 --- a/uefi/CHANGELOG.md +++ b/uefi/CHANGELOG.md @@ -5,6 +5,7 @@ - Added conversions between `proto::network::IpAddress` and `core::net` types. - Added conversions between `proto::network::MacAddress` and the `[u8; 6]` type that's more commonly used to represent MAC addresses. - Added `proto::media::disk_info::DiskInfo`. +- Added `mem::AlignedBuffer`. ## Changed - **Breaking:** Removed `BootPolicyError` as `BootPolicy` construction is no diff --git a/uefi/src/mem/aligned_buffer.rs b/uefi/src/mem/aligned_buffer.rs new file mode 100644 index 000000000..4f9209366 --- /dev/null +++ b/uefi/src/mem/aligned_buffer.rs @@ -0,0 +1,123 @@ +// SPDX-License-Identifier: MIT OR Apache-2.0 + +use alloc::alloc::{alloc, dealloc, Layout, LayoutError}; +use core::error::Error; +use core::fmt; +use core::ptr::NonNull; + +/// Helper class to maintain the lifetime of a memory region allocated with a non-standard alignment. +/// Facilitates RAII to properly deallocate when lifetime of the object ends. +/// +/// Note: This uses the global Rust allocator under the hood. +#[derive(Debug)] +pub struct AlignedBuffer { + ptr: NonNull, + layout: Layout, +} + +impl AlignedBuffer { + /// Allocate a new memory region with the requested len and alignment. + /// + /// # Panics + /// This method panics when the allocation fails (e.g. due to an out of memory situation). + pub fn from_size_align(len: usize, alignment: usize) -> Result { + let layout = Layout::from_size_align(len, alignment)?; + Ok(Self::from_layout(layout)) + } + + /// Allocate a new memory region with the requested layout. + /// + /// # Panics + /// This method panics when the allocation fails (e.g. due to an out of memory situation). + #[must_use] + pub fn from_layout(layout: Layout) -> Self { + let ptr = unsafe { alloc(layout) }; + let ptr = NonNull::new(ptr).expect("Allocation failed"); + Self { ptr, layout } + } + + // TODO: Add non-panicking method variants as soon as alloc::AllocError was stabilized (#32838). + // - try_from_layout(layout: Layout) -> Result; + + /// Get a pointer to the aligned memory region managed by this instance. + #[must_use] + pub const fn ptr(&self) -> *const u8 { + self.ptr.as_ptr().cast_const() + } + + /// Get a mutable pointer to the aligned memory region managed by this instance. + #[must_use] + pub fn ptr_mut(&mut self) -> *mut u8 { + self.ptr.as_ptr() + } + + /// Get the size of the aligned memory region managed by this instance. + #[must_use] + pub const fn size(&self) -> usize { + self.layout.size() + } + + /// Fill the aligned memory region with data from the given buffer. + /// + /// The length of `src` must be the same as `self`. + pub fn copy_from_slice(&mut self, src: &[u8]) { + assert_eq!(self.size(), src.len()); + unsafe { + self.ptr_mut().copy_from(src.as_ptr(), src.len()); + } + } + + /// Check the buffer's alignment against the `required_alignment`. + pub fn check_alignment(&self, required_alignment: usize) -> Result<(), AlignmentError> { + //TODO: use bfr.addr() when it's available + if (self.ptr() as usize) % required_alignment != 0 { + return Err(AlignmentError); //TODO: use >is_aligned_to< when it's available + } + Ok(()) + } +} + +impl Drop for AlignedBuffer { + fn drop(&mut self) { + unsafe { + dealloc(self.ptr_mut(), self.layout); + } + } +} + +/// The `AlignmentError` is returned if a user-provided buffer doesn't fulfill alignment requirements. +#[derive(Clone, PartialEq, Eq, Debug)] +pub struct AlignmentError; +impl Error for AlignmentError {} +impl fmt::Display for AlignmentError { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + f.write_str("Buffer alignment does not fulfill requirements.") + } +} + +#[cfg(test)] +mod tests { + use super::AlignedBuffer; + + #[test] + fn test_invalid_arguments() { + // invalid alignments, valid len + for request_alignment in [0, 3, 5, 7, 9] { + for request_len in [1, 32, 64, 128, 1024] { + assert!(AlignedBuffer::from_size_align(request_len, request_alignment).is_err()); + } + } + } + + #[test] + fn test_allocation_alignment() { + for request_alignment in [1, 2, 4, 8, 16, 32, 64, 128] { + for request_len in [1 as usize, 32, 64, 128, 1024] { + let buffer = + AlignedBuffer::from_size_align(request_len, request_alignment).unwrap(); + assert_eq!(buffer.ptr() as usize % request_alignment, 0); + assert_eq!(buffer.size(), request_len); + } + } + } +} diff --git a/uefi/src/mem/mod.rs b/uefi/src/mem/mod.rs index 9bd4f7f77..122a2eb7f 100644 --- a/uefi/src/mem/mod.rs +++ b/uefi/src/mem/mod.rs @@ -14,6 +14,11 @@ pub(crate) mod util; #[cfg(feature = "alloc")] pub(crate) use util::*; +#[cfg(feature = "alloc")] +mod aligned_buffer; +#[cfg(feature = "alloc")] +pub use aligned_buffer::{AlignedBuffer, AlignmentError}; + /// Wrapper for memory allocated with UEFI's pool allocator. The memory is freed /// on drop. #[derive(Debug)]