diff --git a/uefi/CHANGELOG.md b/uefi/CHANGELOG.md index e13eda4e9..5b67f786c 100644 --- a/uefi/CHANGELOG.md +++ b/uefi/CHANGELOG.md @@ -5,6 +5,8 @@ - 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 `proto::device_path::DevicePath::append_path()`. +- Added `proto::device_path::DevicePath::append_node()`. ## Changed - **Breaking:** Removed `BootPolicyError` as `BootPolicy` construction is no @@ -15,6 +17,8 @@ `&self`. - **Breaking:** The `pxe::Mode` struct is now opaque. Use method calls to access mode data instead of direct field access. +- **Breaking:** `PoolDevicePathNode` and `PoolDevicePath` moved from module + `proto::device_path::text` to `proto::device_path`. - `boot::memory_map()` will never return `Status::BUFFER_TOO_SMALL` from now on, as this is considered a hard internal error where users can't do anything about it anyway. It will panic instead. diff --git a/uefi/src/proto/device_path/mod.rs b/uefi/src/proto/device_path/mod.rs index 448de2c68..697f8ca9c 100644 --- a/uefi/src/proto/device_path/mod.rs +++ b/uefi/src/proto/device_path/mod.rs @@ -77,6 +77,7 @@ pub mod build; pub mod text; +pub mod util; mod device_path_gen; pub use device_path_gen::{ @@ -84,6 +85,7 @@ pub use device_path_gen::{ }; pub use uefi_raw::protocol::device_path::{DeviceSubType, DeviceType}; +use crate::mem::PoolAllocation; use crate::proto::{unsafe_protocol, ProtocolPointer}; use core::ffi::c_void; use core::fmt::{self, Debug, Display, Formatter}; @@ -94,6 +96,7 @@ use ptr_meta::Pointee; use { crate::boot::{self, OpenProtocolAttributes, OpenProtocolParams, ScopedProtocol, SearchType}, crate::proto::device_path::text::{AllowShortcuts, DevicePathToText, DisplayOnly}, + crate::proto::device_path::util::DevicePathUtilities, crate::{CString16, Identify}, alloc::borrow::ToOwned, alloc::boxed::Box, @@ -108,6 +111,30 @@ opaque_type! { pub struct FfiDevicePath; } +/// Device path allocated from UEFI pool memory. +#[derive(Debug)] +pub struct PoolDevicePath(pub(crate) PoolAllocation); + +impl Deref for PoolDevicePath { + type Target = DevicePath; + + fn deref(&self) -> &Self::Target { + unsafe { DevicePath::from_ffi_ptr(self.0.as_ptr().as_ptr().cast()) } + } +} + +/// Device path node allocated from UEFI pool memory. +#[derive(Debug)] +pub struct PoolDevicePathNode(pub(crate) PoolAllocation); + +impl Deref for PoolDevicePathNode { + type Target = DevicePathNode; + + fn deref(&self) -> &Self::Target { + unsafe { DevicePathNode::from_ffi_ptr(self.0.as_ptr().as_ptr().cast()) } + } +} + /// Header that appears at the start of every [`DevicePathNode`]. #[derive(Clone, Copy, Debug, Eq, PartialEq)] #[repr(C, packed)] @@ -499,6 +526,25 @@ impl DevicePath { }) .map_err(|_| DevicePathToTextError::OutOfMemory) } + + /// Allocates and returns a new [`DevicePath`] by copying this one and appending the given `right` path. + #[cfg(feature = "alloc")] + pub fn append_path(&self, right: &Self) -> Result { + open_utility_protocol()? + .append_path(self, right) + .map_err(|_| DevicePathUtilitiesError::OutOfMemory) + } + + /// Allocates and returns a new [`DevicePath`] by copying this one and appending the given `right` node. + #[cfg(feature = "alloc")] + pub fn append_node( + &self, + right: &DevicePathNode, + ) -> Result { + open_utility_protocol()? + .append_node(self, right) + .map_err(|_| DevicePathUtilitiesError::OutOfMemory) + } } impl Debug for DevicePath { @@ -745,6 +791,63 @@ fn open_text_protocol() -> Result, DevicePathTo .map_err(DevicePathToTextError::CantOpenProtocol) } +/// Errors that may occur when working with the [`DevicePathUtilities`] protocol. +/// +/// These errors are typically encountered during operations involving device +/// paths, such as appending or manipulating path segments. +#[derive(Debug)] +pub enum DevicePathUtilitiesError { + /// Can't locate a handle buffer with handles associated with the + /// [`DevicePathUtilities`] protocol. + CantLocateHandleBuffer(crate::Error), + /// No handle supporting the [`DevicePathUtilities`] protocol was found. + NoHandle, + /// The handle supporting the [`DevicePathUtilities`] protocol exists but + /// it could not be opened. + CantOpenProtocol(crate::Error), + /// Memory allocation failed during device path operations. + OutOfMemory, +} + +impl Display for DevicePathUtilitiesError { + fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result { + write!(f, "{self:?}") + } +} + +impl core::error::Error for DevicePathUtilitiesError { + fn source(&self) -> Option<&(dyn core::error::Error + 'static)> { + match self { + Self::CantLocateHandleBuffer(e) => Some(e), + Self::CantOpenProtocol(e) => Some(e), + _ => None, + } + } +} + +/// Helper function to open the [`DevicePathUtilities`] protocol using the boot +/// services. +#[cfg(feature = "alloc")] +fn open_utility_protocol() -> Result, DevicePathUtilitiesError> +{ + let &handle = boot::locate_handle_buffer(SearchType::ByProtocol(&DevicePathToText::GUID)) + .map_err(DevicePathUtilitiesError::CantLocateHandleBuffer)? + .first() + .ok_or(DevicePathUtilitiesError::NoHandle)?; + + unsafe { + boot::open_protocol::( + OpenProtocolParams { + handle, + agent: boot::image_handle(), + controller: None, + }, + OpenProtocolAttributes::GetProtocol, + ) + } + .map_err(DevicePathUtilitiesError::CantOpenProtocol) +} + #[cfg(test)] mod tests { use super::*; diff --git a/uefi/src/proto/device_path/text.rs b/uefi/src/proto/device_path/text.rs index 910b74d5c..6ea22e1a8 100644 --- a/uefi/src/proto/device_path/text.rs +++ b/uefi/src/proto/device_path/text.rs @@ -18,6 +18,8 @@ use core::ops::Deref; use core::ptr::NonNull; use uefi_raw::protocol::device_path::{DevicePathFromTextProtocol, DevicePathToTextProtocol}; +use super::{PoolDevicePath, PoolDevicePathNode}; + /// Parameter for [`DevicePathToText`] that alters the output format. /// /// * `DisplayOnly(false)` produces parseable output. @@ -70,30 +72,6 @@ impl Deref for PoolString { } } -/// Device path allocated from UEFI pool memory. -#[derive(Debug)] -pub struct PoolDevicePath(PoolAllocation); - -impl Deref for PoolDevicePath { - type Target = DevicePath; - - fn deref(&self) -> &Self::Target { - unsafe { DevicePath::from_ffi_ptr(self.0.as_ptr().as_ptr().cast()) } - } -} - -/// Device path node allocated from UEFI pool memory. -#[derive(Debug)] -pub struct PoolDevicePathNode(PoolAllocation); - -impl Deref for PoolDevicePathNode { - type Target = DevicePathNode; - - fn deref(&self) -> &Self::Target { - unsafe { DevicePathNode::from_ffi_ptr(self.0.as_ptr().as_ptr().cast()) } - } -} - /// Protocol for converting a [`DevicePath`] or `DevicePathNode`] to a string. #[derive(Debug)] #[repr(transparent)] diff --git a/uefi/src/proto/device_path/util.rs b/uefi/src/proto/device_path/util.rs new file mode 100644 index 000000000..f1a5f62e1 --- /dev/null +++ b/uefi/src/proto/device_path/util.rs @@ -0,0 +1,102 @@ +// SPDX-License-Identifier: MIT OR Apache-2.0 + +//! Protocol with utility functions for working with device paths. + +use super::{DevicePath, DevicePathNode, PoolDevicePath}; +use crate::mem::PoolAllocation; +use core::ptr::NonNull; +use uefi_macros::unsafe_protocol; +use uefi_raw::protocol::device_path::DevicePathUtilitiesProtocol; +use uefi_raw::Status; + +/// Protocol with utility functions for working with device paths. +#[derive(Debug)] +#[repr(transparent)] +#[unsafe_protocol(DevicePathUtilitiesProtocol::GUID)] +pub struct DevicePathUtilities(DevicePathUtilitiesProtocol); + +impl DevicePathUtilities { + /// Retrieves the size of the specified device path in bytes, including the + /// end-of-device-path node. + /// + /// # Parameters + /// - `device_path`: A reference to the [`DevicePath`] whose size is to be determined. + /// + /// # Returns + /// The size of the specified device path in bytes. + #[must_use] + pub fn get_size(&self, device_path: &DevicePath) -> usize { + unsafe { (self.0.get_device_path_size)(device_path.as_ffi_ptr().cast()) } + } + + /// Creates a new device path by appending the second device path to the first. + /// + /// # Parameters + /// - `path0`: A reference to the base device path. + /// - `path1`: A reference to the device path to append. + /// + /// # Returns + /// A [`PoolDevicePath`] instance containing the newly created device path, + /// or an error if memory could not be allocated. + pub fn append_path( + &self, + path0: &DevicePath, + path1: &DevicePath, + ) -> crate::Result { + unsafe { + let ptr = + (self.0.append_device_path)(path0.as_ffi_ptr().cast(), path1.as_ffi_ptr().cast()); + NonNull::new(ptr.cast_mut()) + .map(|p| PoolDevicePath(PoolAllocation::new(p.cast()))) + .ok_or(Status::OUT_OF_RESOURCES.into()) + } + } + + /// Creates a new device path by appending a device node to the base device path. + /// + /// # Parameters + /// - `basepath`: A reference to the base device path. + /// - `node`: A reference to the device node to append. + /// + /// # Returns + /// A [`PoolDevicePath`] instance containing the newly created device path, + /// or an error if memory could not be allocated. + pub fn append_node( + &self, + basepath: &DevicePath, + node: &DevicePathNode, + ) -> crate::Result { + unsafe { + let ptr = + (self.0.append_device_node)(basepath.as_ffi_ptr().cast(), node.as_ffi_ptr().cast()); + NonNull::new(ptr.cast_mut()) + .map(|p| PoolDevicePath(PoolAllocation::new(p.cast()))) + .ok_or(Status::OUT_OF_RESOURCES.into()) + } + } + + /// Creates a new device path by appending the specified device path instance to the base path. + /// + /// # Parameters + /// - `basepath`: A reference to the base device path. + /// - `instance`: A reference to the device path instance to append. + /// + /// # Returns + /// A [`PoolDevicePath`] instance containing the newly created device path, + /// or an error if memory could not be allocated. + pub fn append_instance( + &self, + basepath: &DevicePath, + instance: &DevicePath, + ) -> crate::Result { + unsafe { + let ptr = (self.0.append_device_path_instance)( + basepath.as_ffi_ptr().cast(), + instance.as_ffi_ptr().cast(), + ); + NonNull::new(ptr.cast_mut()) + .map(|p| PoolDevicePath(PoolAllocation::new(p.cast()))) + .ok_or(Status::OUT_OF_RESOURCES.into()) + } + } +}