Skip to content

Commit 9263616

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 8a858a7 commit 9263616

File tree

2 files changed

+110
-0
lines changed

2 files changed

+110
-0
lines changed

uefi/src/mem/aligned_buffer.rs

+105
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,105 @@
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+
#[derive(Debug)]
12+
pub struct AlignedBuffer {
13+
ptr: *mut u8,
14+
layout: Layout,
15+
}
16+
17+
impl AlignedBuffer {
18+
/// Allocate a new memory region with the requested len and alignment.
19+
pub fn from_size_align(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+
25+
/// Get a pointer to the aligned memory region managed by this instance.
26+
#[must_use]
27+
pub const fn ptr(&self) -> *const u8 {
28+
self.ptr.cast_const()
29+
}
30+
31+
/// Get a mutable pointer to the aligned memory region managed by this instance.
32+
#[must_use]
33+
pub fn ptr_mut(&mut self) -> *mut u8 {
34+
self.ptr
35+
}
36+
37+
/// Get the size of the aligned memory region managed by this instance.
38+
#[must_use]
39+
pub const fn size(&self) -> usize {
40+
self.layout.size()
41+
}
42+
43+
/// Fill the aligned memory region with data from the given buffer.
44+
///
45+
/// The length of `src` must be the same as `self`.
46+
pub fn copy_from_slice(&mut self, src: &[u8]) {
47+
assert_eq!(self.size(), src.len());
48+
unsafe {
49+
self.ptr.copy_from(src.as_ptr(), src.len());
50+
}
51+
}
52+
53+
/// Check the buffer's alignment against the `required_alignment`.
54+
pub fn check_alignment(&self, required_alignment: usize) -> Result<(), AlignmentError> {
55+
//TODO: use bfr.addr() when it's available
56+
if (self.ptr as usize) % required_alignment != 0 {
57+
return Err(AlignmentError); //TODO: use >is_aligned_to< when it's available
58+
}
59+
Ok(())
60+
}
61+
}
62+
63+
impl Drop for AlignedBuffer {
64+
fn drop(&mut self) {
65+
unsafe {
66+
dealloc(self.ptr, self.layout);
67+
}
68+
}
69+
}
70+
71+
/// The `AlignmentError` is returned if a user-provided buffer doesn't fulfill alignment requirements.
72+
#[derive(Clone, PartialEq, Eq, Debug)]
73+
pub struct AlignmentError;
74+
impl Error for AlignmentError {}
75+
impl fmt::Display for AlignmentError {
76+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
77+
f.write_str("Buffer alignment does not fulfill requirements.")
78+
}
79+
}
80+
81+
#[cfg(test)]
82+
mod tests {
83+
use super::AlignedBuffer;
84+
85+
#[test]
86+
fn test_invalid_arguments() {
87+
// invalid alignments, valid len
88+
for request_alignment in [0, 3, 5, 7, 9] {
89+
for request_len in [1, 32, 64, 128, 1024] {
90+
assert!(AlignedBuffer::alloc(request_len, request_alignment).is_err());
91+
}
92+
}
93+
}
94+
95+
#[test]
96+
fn test_allocation_alignment() {
97+
for request_alignment in [1, 2, 4, 8, 16, 32, 64, 128] {
98+
for request_len in [1 as usize, 32, 64, 128, 1024] {
99+
let buffer = AlignedBuffer::alloc(request_len, request_alignment).unwrap();
100+
assert_eq!(buffer.ptr() as usize % request_alignment, 0);
101+
assert_eq!(buffer.len(), request_len);
102+
}
103+
}
104+
}
105+
}

uefi/src/mem/mod.rs

+5
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,11 @@ pub(crate) mod util;
1414
#[cfg(feature = "alloc")]
1515
pub(crate) use util::*;
1616

17+
#[cfg(feature = "alloc")]
18+
mod aligned_buffer;
19+
#[cfg(feature = "alloc")]
20+
pub use aligned_buffer::{AlignedBuffer, AlignmentError};
21+
1722
/// Wrapper for memory allocated with UEFI's pool allocator. The memory is freed
1823
/// on drop.
1924
#[derive(Debug)]

0 commit comments

Comments
 (0)