|
10 | 10 | //! - More information about protocols can be found [here](https://edk2-docs.gitbook.io/edk-ii-uefi-driver-writer-s-guide/3_foundation/36_protocols_and_handles)
|
11 | 11 |
|
12 | 12 | use r_efi::efi::{self, Guid};
|
| 13 | +use r_efi::protocols::{device_path, device_path_to_text}; |
13 | 14 |
|
| 15 | +use crate::ffi::OsString; |
| 16 | +use crate::io::{self, const_io_error}; |
14 | 17 | use crate::mem::{size_of, MaybeUninit};
|
15 |
| -use crate::os::uefi; |
| 18 | +use crate::os::uefi::{self, env::boot_services, ffi::OsStringExt}; |
16 | 19 | use crate::ptr::NonNull;
|
17 |
| -use crate::{ |
18 |
| - io::{self, const_io_error}, |
19 |
| - os::uefi::env::boot_services, |
20 |
| -}; |
| 20 | +use crate::slice; |
| 21 | +use crate::sync::atomic::{AtomicPtr, Ordering}; |
| 22 | +use crate::sys_common::wstr::WStrUnits; |
21 | 23 |
|
22 | 24 | const BOOT_SERVICES_UNAVAILABLE: io::Error =
|
23 | 25 | const_io_error!(io::ErrorKind::Other, "Boot Services are no longer available");
|
@@ -142,7 +144,72 @@ pub(crate) unsafe fn close_event(evt: NonNull<crate::ffi::c_void>) -> io::Result
|
142 | 144 |
|
143 | 145 | /// Get the Protocol for current system handle.
|
144 | 146 | /// Note: Some protocols need to be manually freed. It is the callers responsibility to do so.
|
145 |
| -pub(crate) fn image_handle_protocol<T>(protocol_guid: Guid) -> Option<NonNull<T>> { |
146 |
| - let system_handle = uefi::env::try_image_handle()?; |
147 |
| - open_protocol(system_handle, protocol_guid).ok() |
| 147 | +pub(crate) fn image_handle_protocol<T>(protocol_guid: Guid) -> io::Result<NonNull<T>> { |
| 148 | + let system_handle = uefi::env::try_image_handle().ok_or(io::const_io_error!( |
| 149 | + io::ErrorKind::NotFound, |
| 150 | + "Protocol not found in Image handle" |
| 151 | + ))?; |
| 152 | + open_protocol(system_handle, protocol_guid) |
| 153 | +} |
| 154 | + |
| 155 | +pub(crate) fn device_path_to_text(path: NonNull<device_path::Protocol>) -> io::Result<OsString> { |
| 156 | + fn path_to_text( |
| 157 | + protocol: NonNull<device_path_to_text::Protocol>, |
| 158 | + path: NonNull<device_path::Protocol>, |
| 159 | + ) -> io::Result<OsString> { |
| 160 | + let path_ptr: *mut r_efi::efi::Char16 = unsafe { |
| 161 | + ((*protocol.as_ptr()).convert_device_path_to_text)( |
| 162 | + path.as_ptr(), |
| 163 | + // DisplayOnly |
| 164 | + r_efi::efi::Boolean::FALSE, |
| 165 | + // AllowShortcuts |
| 166 | + r_efi::efi::Boolean::FALSE, |
| 167 | + ) |
| 168 | + }; |
| 169 | + |
| 170 | + // SAFETY: `convert_device_path_to_text` returns a pointer to a null-terminated UTF-16 |
| 171 | + // string, and that string cannot be deallocated prior to dropping the `WStrUnits`, so |
| 172 | + // it's safe for `WStrUnits` to use. |
| 173 | + let path_len = unsafe { |
| 174 | + WStrUnits::new(path_ptr) |
| 175 | + .ok_or(io::const_io_error!(io::ErrorKind::InvalidData, "Invalid path"))? |
| 176 | + .count() |
| 177 | + }; |
| 178 | + |
| 179 | + let path = OsString::from_wide(unsafe { slice::from_raw_parts(path_ptr.cast(), path_len) }); |
| 180 | + |
| 181 | + if let Some(boot_services) = crate::os::uefi::env::boot_services() { |
| 182 | + let boot_services: NonNull<r_efi::efi::BootServices> = boot_services.cast(); |
| 183 | + unsafe { |
| 184 | + ((*boot_services.as_ptr()).free_pool)(path_ptr.cast()); |
| 185 | + } |
| 186 | + } |
| 187 | + |
| 188 | + Ok(path) |
| 189 | + } |
| 190 | + |
| 191 | + static LAST_VALID_HANDLE: AtomicPtr<crate::ffi::c_void> = |
| 192 | + AtomicPtr::new(crate::ptr::null_mut()); |
| 193 | + |
| 194 | + if let Some(handle) = NonNull::new(LAST_VALID_HANDLE.load(Ordering::Acquire)) { |
| 195 | + if let Ok(protocol) = open_protocol::<device_path_to_text::Protocol>( |
| 196 | + handle, |
| 197 | + device_path_to_text::PROTOCOL_GUID, |
| 198 | + ) { |
| 199 | + return path_to_text(protocol, path); |
| 200 | + } |
| 201 | + } |
| 202 | + |
| 203 | + let device_path_to_text_handles = locate_handles(device_path_to_text::PROTOCOL_GUID)?; |
| 204 | + for handle in device_path_to_text_handles { |
| 205 | + if let Ok(protocol) = open_protocol::<device_path_to_text::Protocol>( |
| 206 | + handle, |
| 207 | + device_path_to_text::PROTOCOL_GUID, |
| 208 | + ) { |
| 209 | + LAST_VALID_HANDLE.store(handle.as_ptr(), Ordering::Release); |
| 210 | + return path_to_text(protocol, path); |
| 211 | + } |
| 212 | + } |
| 213 | + |
| 214 | + Err(io::const_io_error!(io::ErrorKind::NotFound, "No device path to text protocol found")) |
148 | 215 | }
|
0 commit comments