Skip to content

Commit 92f5cb9

Browse files
committed
uefi: Add initial incomplete safe wrapper for DevicePathUtilities protocol
1 parent 48295b3 commit 92f5cb9

File tree

4 files changed

+211
-24
lines changed

4 files changed

+211
-24
lines changed

uefi/CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,8 @@
55
- Added conversions between `proto::network::IpAddress` and `core::net` types.
66
- Added conversions between `proto::network::MacAddress` and the `[u8; 6]` type that's more commonly used to represent MAC addresses.
77
- Added `proto::media::disk_info::DiskInfo`.
8+
- Added `proto::device_path::DevicePath::append_path()`.
9+
- Added `proto::device_path::DevicePath::append_node()`.
810

911
## Changed
1012
- **Breaking:** Removed `BootPolicyError` as `BootPolicy` construction is no
@@ -15,6 +17,8 @@
1517
`&self`.
1618
- **Breaking:** The `pxe::Mode` struct is now opaque. Use method calls to access
1719
mode data instead of direct field access.
20+
- **Breaking:** `PoolDevicePathNode` and `PoolDevicePath` moved from module
21+
`proto::device_path::text` to `proto::device_path`.
1822
- `boot::memory_map()` will never return `Status::BUFFER_TOO_SMALL` from now on,
1923
as this is considered a hard internal error where users can't do anything
2024
about it anyway. It will panic instead.

uefi/src/proto/device_path/mod.rs

+103
Original file line numberDiff line numberDiff line change
@@ -77,13 +77,15 @@
7777
7878
pub mod build;
7979
pub mod text;
80+
pub mod util;
8081

8182
mod device_path_gen;
8283
pub use device_path_gen::{
8384
acpi, bios_boot_spec, end, hardware, media, messaging, DevicePathNodeEnum,
8485
};
8586
pub use uefi_raw::protocol::device_path::{DeviceSubType, DeviceType};
8687

88+
use crate::mem::PoolAllocation;
8789
use crate::proto::{unsafe_protocol, ProtocolPointer};
8890
use core::ffi::c_void;
8991
use core::fmt::{self, Debug, Display, Formatter};
@@ -94,6 +96,7 @@ use ptr_meta::Pointee;
9496
use {
9597
crate::boot::{self, OpenProtocolAttributes, OpenProtocolParams, ScopedProtocol, SearchType},
9698
crate::proto::device_path::text::{AllowShortcuts, DevicePathToText, DisplayOnly},
99+
crate::proto::device_path::util::DevicePathUtilities,
97100
crate::{CString16, Identify},
98101
alloc::borrow::ToOwned,
99102
alloc::boxed::Box,
@@ -108,6 +111,30 @@ opaque_type! {
108111
pub struct FfiDevicePath;
109112
}
110113

114+
/// Device path allocated from UEFI pool memory.
115+
#[derive(Debug)]
116+
pub struct PoolDevicePath(pub(crate) PoolAllocation);
117+
118+
impl Deref for PoolDevicePath {
119+
type Target = DevicePath;
120+
121+
fn deref(&self) -> &Self::Target {
122+
unsafe { DevicePath::from_ffi_ptr(self.0.as_ptr().as_ptr().cast()) }
123+
}
124+
}
125+
126+
/// Device path node allocated from UEFI pool memory.
127+
#[derive(Debug)]
128+
pub struct PoolDevicePathNode(pub(crate) PoolAllocation);
129+
130+
impl Deref for PoolDevicePathNode {
131+
type Target = DevicePathNode;
132+
133+
fn deref(&self) -> &Self::Target {
134+
unsafe { DevicePathNode::from_ffi_ptr(self.0.as_ptr().as_ptr().cast()) }
135+
}
136+
}
137+
111138
/// Header that appears at the start of every [`DevicePathNode`].
112139
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
113140
#[repr(C, packed)]
@@ -499,6 +526,25 @@ impl DevicePath {
499526
})
500527
.map_err(|_| DevicePathToTextError::OutOfMemory)
501528
}
529+
530+
/// Allocates and returns a new [`DevicePath`] by copying this one and appending the given `right` path.
531+
#[cfg(feature = "alloc")]
532+
pub fn append_path(&self, right: &Self) -> Result<PoolDevicePath, DevicePathUtilitiesError> {
533+
open_utility_protocol()?
534+
.append_path(self, right)
535+
.map_err(|_| DevicePathUtilitiesError::OutOfMemory)
536+
}
537+
538+
/// Allocates and returns a new [`DevicePath`] by copying this one and appending the given `right` node.
539+
#[cfg(feature = "alloc")]
540+
pub fn append_node(
541+
&self,
542+
right: &DevicePathNode,
543+
) -> Result<PoolDevicePath, DevicePathUtilitiesError> {
544+
open_utility_protocol()?
545+
.append_node(self, right)
546+
.map_err(|_| DevicePathUtilitiesError::OutOfMemory)
547+
}
502548
}
503549

504550
impl Debug for DevicePath {
@@ -745,6 +791,63 @@ fn open_text_protocol() -> Result<ScopedProtocol<DevicePathToText>, DevicePathTo
745791
.map_err(DevicePathToTextError::CantOpenProtocol)
746792
}
747793

794+
/// Errors that may occur when working with the [`DevicePathUtilities`] protocol.
795+
///
796+
/// These errors are typically encountered during operations involving device
797+
/// paths, such as appending or manipulating path segments.
798+
#[derive(Debug)]
799+
pub enum DevicePathUtilitiesError {
800+
/// Can't locate a handle buffer with handles associated with the
801+
/// [`DevicePathUtilities`] protocol.
802+
CantLocateHandleBuffer(crate::Error),
803+
/// No handle supporting the [`DevicePathUtilities`] protocol was found.
804+
NoHandle,
805+
/// The handle supporting the [`DevicePathUtilities`] protocol exists but
806+
/// it could not be opened.
807+
CantOpenProtocol(crate::Error),
808+
/// Memory allocation failed during device path operations.
809+
OutOfMemory,
810+
}
811+
812+
impl Display for DevicePathUtilitiesError {
813+
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
814+
write!(f, "{self:?}")
815+
}
816+
}
817+
818+
impl core::error::Error for DevicePathUtilitiesError {
819+
fn source(&self) -> Option<&(dyn core::error::Error + 'static)> {
820+
match self {
821+
Self::CantLocateHandleBuffer(e) => Some(e),
822+
Self::CantOpenProtocol(e) => Some(e),
823+
_ => None,
824+
}
825+
}
826+
}
827+
828+
/// Helper function to open the [`DevicePathUtilities`] protocol using the boot
829+
/// services.
830+
#[cfg(feature = "alloc")]
831+
fn open_utility_protocol() -> Result<ScopedProtocol<DevicePathUtilities>, DevicePathUtilitiesError>
832+
{
833+
let &handle = boot::locate_handle_buffer(SearchType::ByProtocol(&DevicePathToText::GUID))
834+
.map_err(DevicePathUtilitiesError::CantLocateHandleBuffer)?
835+
.first()
836+
.ok_or(DevicePathUtilitiesError::NoHandle)?;
837+
838+
unsafe {
839+
boot::open_protocol::<DevicePathUtilities>(
840+
OpenProtocolParams {
841+
handle,
842+
agent: boot::image_handle(),
843+
controller: None,
844+
},
845+
OpenProtocolAttributes::GetProtocol,
846+
)
847+
}
848+
.map_err(DevicePathUtilitiesError::CantOpenProtocol)
849+
}
850+
748851
#[cfg(test)]
749852
mod tests {
750853
use super::*;

uefi/src/proto/device_path/text.rs

+2-24
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@ use core::ops::Deref;
1818
use core::ptr::NonNull;
1919
use uefi_raw::protocol::device_path::{DevicePathFromTextProtocol, DevicePathToTextProtocol};
2020

21+
use super::{PoolDevicePath, PoolDevicePathNode};
22+
2123
/// Parameter for [`DevicePathToText`] that alters the output format.
2224
///
2325
/// * `DisplayOnly(false)` produces parseable output.
@@ -70,30 +72,6 @@ impl Deref for PoolString {
7072
}
7173
}
7274

73-
/// Device path allocated from UEFI pool memory.
74-
#[derive(Debug)]
75-
pub struct PoolDevicePath(PoolAllocation);
76-
77-
impl Deref for PoolDevicePath {
78-
type Target = DevicePath;
79-
80-
fn deref(&self) -> &Self::Target {
81-
unsafe { DevicePath::from_ffi_ptr(self.0.as_ptr().as_ptr().cast()) }
82-
}
83-
}
84-
85-
/// Device path node allocated from UEFI pool memory.
86-
#[derive(Debug)]
87-
pub struct PoolDevicePathNode(PoolAllocation);
88-
89-
impl Deref for PoolDevicePathNode {
90-
type Target = DevicePathNode;
91-
92-
fn deref(&self) -> &Self::Target {
93-
unsafe { DevicePathNode::from_ffi_ptr(self.0.as_ptr().as_ptr().cast()) }
94-
}
95-
}
96-
9775
/// Protocol for converting a [`DevicePath`] or `DevicePathNode`] to a string.
9876
#[derive(Debug)]
9977
#[repr(transparent)]

uefi/src/proto/device_path/util.rs

+102
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
3+
//! Protocol with utility functions for working with device paths.
4+
5+
use super::{DevicePath, DevicePathNode, PoolDevicePath};
6+
use crate::mem::PoolAllocation;
7+
use core::ptr::NonNull;
8+
use uefi_macros::unsafe_protocol;
9+
use uefi_raw::protocol::device_path::DevicePathUtilitiesProtocol;
10+
use uefi_raw::Status;
11+
12+
/// Protocol with utility functions for working with device paths.
13+
#[derive(Debug)]
14+
#[repr(transparent)]
15+
#[unsafe_protocol(DevicePathUtilitiesProtocol::GUID)]
16+
pub struct DevicePathUtilities(DevicePathUtilitiesProtocol);
17+
18+
impl DevicePathUtilities {
19+
/// Retrieves the size of the specified device path in bytes, including the
20+
/// end-of-device-path node.
21+
///
22+
/// # Parameters
23+
/// - `device_path`: A reference to the [`DevicePath`] whose size is to be determined.
24+
///
25+
/// # Returns
26+
/// The size of the specified device path in bytes.
27+
#[must_use]
28+
pub fn get_size(&self, device_path: &DevicePath) -> usize {
29+
unsafe { (self.0.get_device_path_size)(device_path.as_ffi_ptr().cast()) }
30+
}
31+
32+
/// Creates a new device path by appending the second device path to the first.
33+
///
34+
/// # Parameters
35+
/// - `path0`: A reference to the base device path.
36+
/// - `path1`: A reference to the device path to append.
37+
///
38+
/// # Returns
39+
/// A [`PoolDevicePath`] instance containing the newly created device path,
40+
/// or an error if memory could not be allocated.
41+
pub fn append_path(
42+
&self,
43+
path0: &DevicePath,
44+
path1: &DevicePath,
45+
) -> crate::Result<PoolDevicePath> {
46+
unsafe {
47+
let ptr =
48+
(self.0.append_device_path)(path0.as_ffi_ptr().cast(), path1.as_ffi_ptr().cast());
49+
NonNull::new(ptr.cast_mut())
50+
.map(|p| PoolDevicePath(PoolAllocation::new(p.cast())))
51+
.ok_or(Status::OUT_OF_RESOURCES.into())
52+
}
53+
}
54+
55+
/// Creates a new device path by appending a device node to the base device path.
56+
///
57+
/// # Parameters
58+
/// - `basepath`: A reference to the base device path.
59+
/// - `node`: A reference to the device node to append.
60+
///
61+
/// # Returns
62+
/// A [`PoolDevicePath`] instance containing the newly created device path,
63+
/// or an error if memory could not be allocated.
64+
pub fn append_node(
65+
&self,
66+
basepath: &DevicePath,
67+
node: &DevicePathNode,
68+
) -> crate::Result<PoolDevicePath> {
69+
unsafe {
70+
let ptr =
71+
(self.0.append_device_node)(basepath.as_ffi_ptr().cast(), node.as_ffi_ptr().cast());
72+
NonNull::new(ptr.cast_mut())
73+
.map(|p| PoolDevicePath(PoolAllocation::new(p.cast())))
74+
.ok_or(Status::OUT_OF_RESOURCES.into())
75+
}
76+
}
77+
78+
/// Creates a new device path by appending the specified device path instance to the base path.
79+
///
80+
/// # Parameters
81+
/// - `basepath`: A reference to the base device path.
82+
/// - `instance`: A reference to the device path instance to append.
83+
///
84+
/// # Returns
85+
/// A [`PoolDevicePath`] instance containing the newly created device path,
86+
/// or an error if memory could not be allocated.
87+
pub fn append_instance(
88+
&self,
89+
basepath: &DevicePath,
90+
instance: &DevicePath,
91+
) -> crate::Result<PoolDevicePath> {
92+
unsafe {
93+
let ptr = (self.0.append_device_path_instance)(
94+
basepath.as_ffi_ptr().cast(),
95+
instance.as_ffi_ptr().cast(),
96+
);
97+
NonNull::new(ptr.cast_mut())
98+
.map(|p| PoolDevicePath(PoolAllocation::new(p.cast())))
99+
.ok_or(Status::OUT_OF_RESOURCES.into())
100+
}
101+
}
102+
}

0 commit comments

Comments
 (0)