Skip to content

Commit a2497a9

Browse files
committed
Expand sys/os for UEFI
- Implement current_exe() - Cache device_path_to_text protocol Signed-off-by: Ayush Singh <[email protected]>
1 parent 04521fd commit a2497a9

File tree

2 files changed

+81
-10
lines changed

2 files changed

+81
-10
lines changed

library/std/src/sys/pal/uefi/helpers.rs

+75-8
Original file line numberDiff line numberDiff line change
@@ -10,14 +10,16 @@
1010
//! - 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)
1111
1212
use r_efi::efi::{self, Guid};
13+
use r_efi::protocols::{device_path, device_path_to_text};
1314

15+
use crate::ffi::OsString;
16+
use crate::io::{self, const_io_error};
1417
use crate::mem::{size_of, MaybeUninit};
15-
use crate::os::uefi;
18+
use crate::os::uefi::{self, env::boot_services, ffi::OsStringExt};
1619
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;
2123

2224
const BOOT_SERVICES_UNAVAILABLE: io::Error =
2325
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
142144

143145
/// Get the Protocol for current system handle.
144146
/// 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"))
148215
}

library/std/src/sys/pal/uefi/os.rs

+6-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
use super::{unsupported, RawOsError};
1+
use super::{helpers, unsupported, RawOsError};
22
use crate::error::Error as StdError;
33
use crate::ffi::{OsStr, OsString};
44
use crate::fmt;
@@ -7,6 +7,7 @@ use crate::marker::PhantomData;
77
use crate::os::uefi;
88
use crate::path::{self, PathBuf};
99
use crate::ptr::NonNull;
10+
use r_efi::efi::protocols::{device_path, loaded_image_device_path};
1011
use r_efi::efi::Status;
1112

1213
pub fn errno() -> RawOsError {
@@ -164,7 +165,10 @@ impl fmt::Display for JoinPathsError {
164165
impl StdError for JoinPathsError {}
165166

166167
pub fn current_exe() -> io::Result<PathBuf> {
167-
unsupported()
168+
let protocol = helpers::image_handle_protocol::<device_path::Protocol>(
169+
loaded_image_device_path::PROTOCOL_GUID,
170+
)?;
171+
helpers::device_path_to_text(protocol).map(PathBuf::from)
168172
}
169173

170174
pub struct Env(!);

0 commit comments

Comments
 (0)