Skip to content

Commit cbd28fe

Browse files
committed
helpers: Add AlignedBuffer
AlignedBuffer is a helper class that manages the livetime of a memory region, allocated using a certain alignment. Like Box, it handles deallocation when the object isn't used anymore.
1 parent 60cf8a0 commit cbd28fe

File tree

2 files changed

+85
-0
lines changed

2 files changed

+85
-0
lines changed

uefi/src/helpers/aligned_buffer.rs

+79
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
3+
use alloc::alloc::{alloc, dealloc, Layout, LayoutError};
4+
use core::error::Error;
5+
use core::fmt;
6+
7+
/// Helper class to maintain the lifetime of a memory region allocated with a non-standard alignment.
8+
/// Facilitates RAII to properly deallocate when lifetime of the object ends.
9+
///
10+
/// Note: This uses the global Rust allocator under the hood.
11+
#[allow(clippy::len_without_is_empty)]
12+
#[derive(Debug)]
13+
pub struct AlignedBuffer {
14+
ptr: *mut u8,
15+
layout: Layout,
16+
}
17+
impl AlignedBuffer {
18+
/// Allocate a new memory region with the requested len and alignment.
19+
pub fn alloc(len: usize, alignment: usize) -> Result<Self, LayoutError> {
20+
let layout = Layout::from_size_align(len, alignment)?;
21+
let ptr = unsafe { alloc(layout) };
22+
Ok(Self { ptr, layout })
23+
}
24+
/// Get a pointer to the aligned memory region managed by this instance.
25+
#[must_use]
26+
pub const fn ptr(&self) -> *const u8 {
27+
self.ptr as *const u8
28+
}
29+
/// Get a mutable pointer to the aligned memory region managed by this instance.
30+
#[must_use]
31+
pub fn ptr_mut(&mut self) -> *mut u8 {
32+
self.ptr
33+
}
34+
/// Get the size of the aligned memory region managed by this instance.
35+
#[must_use]
36+
pub const fn len(&self) -> usize {
37+
self.layout.size()
38+
}
39+
/// Fill the aligned memory region with data from the given buffer.
40+
pub fn copy_from(&mut self, data: &[u8]) {
41+
let len = data.len().min(self.len());
42+
unsafe {
43+
self.ptr.copy_from(data.as_ptr(), len);
44+
}
45+
}
46+
}
47+
impl Drop for AlignedBuffer {
48+
fn drop(&mut self) {
49+
unsafe {
50+
dealloc(self.ptr, self.layout);
51+
}
52+
}
53+
}
54+
55+
/// The `AlignmentError` is returned if a user-provided buffer doesn't fulfill alignment requirements.
56+
#[derive(Clone, PartialEq, Eq, Debug)]
57+
pub struct AlignmentError;
58+
impl Error for AlignmentError {}
59+
impl fmt::Display for AlignmentError {
60+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
61+
f.write_str("invalid parameters to Layout::from_size_align")
62+
}
63+
}
64+
65+
#[cfg(test)]
66+
mod tests {
67+
use super::AlignedBuffer;
68+
69+
#[test]
70+
fn test_allocation_alignment() {
71+
for request_alignment in [1, 2, 4, 8, 16, 32, 64, 128] {
72+
for request_len in [1 as usize, 32, 64, 128, 1024] {
73+
let buffer = AlignedBuffer::alloc(request_len, request_alignment).unwrap();
74+
assert_eq!(buffer.ptr() as usize % request_alignment, 0);
75+
assert_eq!(buffer.len(), request_len);
76+
}
77+
}
78+
}
79+
}

uefi/src/helpers/mod.rs

+6
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,12 @@ use crate::Result;
2424
#[doc(hidden)]
2525
pub use println::_print;
2626

27+
#[cfg(feature = "alloc")]
28+
mod aligned_buffer;
29+
#[doc(hidden)]
30+
#[cfg(feature = "alloc")]
31+
pub use aligned_buffer::{AlignedBuffer, AlignmentError};
32+
2733
#[cfg(feature = "global_allocator")]
2834
mod global_allocator;
2935
#[cfg(feature = "logger")]

0 commit comments

Comments
 (0)