Skip to content

Commit ac852f3

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 172a86b commit ac852f3

File tree

2 files changed

+103
-0
lines changed

2 files changed

+103
-0
lines changed

uefi/src/helpers/aligned_buffer.rs

+97
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,97 @@
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.cast_const()
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+
/// Check the buffer's alignment against the `required_alignment`.
47+
pub fn check_alignment(&self, required_alignment: usize) -> Result<(), AlignmentError> {
48+
//TODO: use bfr.addr() when it's available
49+
if (self.ptr as usize) % required_alignment != 0 {
50+
return Err(AlignmentError); //TODO: use >is_aligned_to< when it's available
51+
}
52+
Ok(())
53+
}
54+
}
55+
impl Drop for AlignedBuffer {
56+
fn drop(&mut self) {
57+
unsafe {
58+
dealloc(self.ptr, self.layout);
59+
}
60+
}
61+
}
62+
63+
/// The `AlignmentError` is returned if a user-provided buffer doesn't fulfill alignment requirements.
64+
#[derive(Clone, PartialEq, Eq, Debug)]
65+
pub struct AlignmentError;
66+
impl Error for AlignmentError {}
67+
impl fmt::Display for AlignmentError {
68+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
69+
f.write_str("invalid parameters to Layout::from_size_align")
70+
}
71+
}
72+
73+
#[cfg(test)]
74+
mod tests {
75+
use super::AlignedBuffer;
76+
77+
#[test]
78+
fn test_invalid_arguments() {
79+
// invalid alignments, valid len
80+
for request_alignment in [0, 3, 5, 7, 9] {
81+
for request_len in [1, 32, 64, 128, 1024] {
82+
assert!(AlignedBuffer::alloc(request_len, request_alignment).is_err());
83+
}
84+
}
85+
}
86+
87+
#[test]
88+
fn test_allocation_alignment() {
89+
for request_alignment in [1, 2, 4, 8, 16, 32, 64, 128] {
90+
for request_len in [1 as usize, 32, 64, 128, 1024] {
91+
let buffer = AlignedBuffer::alloc(request_len, request_alignment).unwrap();
92+
assert_eq!(buffer.ptr() as usize % request_alignment, 0);
93+
assert_eq!(buffer.len(), request_len);
94+
}
95+
}
96+
}
97+
}

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)