- Update system table crc32 - Fix unsound use of Box - Free exit data - Code improvements - Introduce OwnedTable - Update r-efi to latest version - Use extended_varargs_abi_support for install_multiple_protocol_interfaces and uninstall_multiple_protocol_interfaces - Fix comments - Stub out args implementation Signed-off-by: Ayush Singh <ayushdevel1325@gmail.com>
419 lines
15 KiB
Rust
419 lines
15 KiB
Rust
//! Contains most of the shared UEFI specific stuff. Some of this might be moved to `std::os::uefi`
|
|
//! if needed but no point in adding extra public API when there is not Std support for UEFI in the
|
|
//! first place
|
|
//!
|
|
//! Some Nomenclature
|
|
//! * Protocol:
|
|
//! - Protocols serve to enable communication between separately built modules, including drivers.
|
|
//! - Every protocol has a GUID associated with it. The GUID serves as the name for the protocol.
|
|
//! - Protocols are produced and consumed.
|
|
//! - 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)
|
|
|
|
use r_efi::efi::{self, Guid};
|
|
use r_efi::protocols::{device_path, device_path_to_text};
|
|
|
|
use crate::ffi::{OsStr, OsString};
|
|
use crate::io::{self, const_io_error};
|
|
use crate::mem::{size_of, MaybeUninit};
|
|
use crate::os::uefi::{self, env::boot_services, ffi::OsStrExt, ffi::OsStringExt};
|
|
use crate::ptr::NonNull;
|
|
use crate::slice;
|
|
use crate::sync::atomic::{AtomicPtr, Ordering};
|
|
use crate::sys_common::wstr::WStrUnits;
|
|
|
|
type BootInstallMultipleProtocolInterfaces =
|
|
unsafe extern "efiapi" fn(_: *mut r_efi::efi::Handle, _: ...) -> r_efi::efi::Status;
|
|
|
|
type BootUninstallMultipleProtocolInterfaces =
|
|
unsafe extern "efiapi" fn(_: r_efi::efi::Handle, _: ...) -> r_efi::efi::Status;
|
|
|
|
const BOOT_SERVICES_UNAVAILABLE: io::Error =
|
|
const_io_error!(io::ErrorKind::Other, "Boot Services are no longer available");
|
|
|
|
/// Locate Handles with a particular Protocol GUID
|
|
/// Implemented using `EFI_BOOT_SERVICES.LocateHandles()`
|
|
///
|
|
/// Returns an array of [Handles](r_efi::efi::Handle) that support a specified protocol.
|
|
pub(crate) fn locate_handles(mut guid: Guid) -> io::Result<Vec<NonNull<crate::ffi::c_void>>> {
|
|
fn inner(
|
|
guid: &mut Guid,
|
|
boot_services: NonNull<r_efi::efi::BootServices>,
|
|
buf_size: &mut usize,
|
|
buf: *mut r_efi::efi::Handle,
|
|
) -> io::Result<()> {
|
|
let r = unsafe {
|
|
((*boot_services.as_ptr()).locate_handle)(
|
|
r_efi::efi::BY_PROTOCOL,
|
|
guid,
|
|
crate::ptr::null_mut(),
|
|
buf_size,
|
|
buf,
|
|
)
|
|
};
|
|
|
|
if r.is_error() { Err(crate::io::Error::from_raw_os_error(r.as_usize())) } else { Ok(()) }
|
|
}
|
|
|
|
let boot_services = boot_services().ok_or(BOOT_SERVICES_UNAVAILABLE)?.cast();
|
|
let mut buf_len = 0usize;
|
|
|
|
// This should always fail since the size of buffer is 0. This call should update the buf_len
|
|
// variable with the required buffer length
|
|
match inner(&mut guid, boot_services, &mut buf_len, crate::ptr::null_mut()) {
|
|
Ok(()) => unreachable!(),
|
|
Err(e) => match e.kind() {
|
|
io::ErrorKind::FileTooLarge => {}
|
|
_ => return Err(e),
|
|
},
|
|
}
|
|
|
|
// The returned buf_len is in bytes
|
|
assert_eq!(buf_len % size_of::<r_efi::efi::Handle>(), 0);
|
|
let num_of_handles = buf_len / size_of::<r_efi::efi::Handle>();
|
|
let mut buf: Vec<r_efi::efi::Handle> = Vec::with_capacity(num_of_handles);
|
|
match inner(&mut guid, boot_services, &mut buf_len, buf.as_mut_ptr()) {
|
|
Ok(()) => {
|
|
// This is safe because the call will succeed only if buf_len >= required length.
|
|
// Also, on success, the `buf_len` is updated with the size of bufferv (in bytes) written
|
|
unsafe { buf.set_len(num_of_handles) };
|
|
Ok(buf.into_iter().filter_map(|x| NonNull::new(x)).collect())
|
|
}
|
|
Err(e) => Err(e),
|
|
}
|
|
}
|
|
|
|
/// Open Protocol on a handle.
|
|
/// Internally just a call to `EFI_BOOT_SERVICES.OpenProtocol()`.
|
|
///
|
|
/// Queries a handle to determine if it supports a specified protocol. If the protocol is
|
|
/// supported by the handle, it opens the protocol on behalf of the calling agent.
|
|
pub(crate) fn open_protocol<T>(
|
|
handle: NonNull<crate::ffi::c_void>,
|
|
mut protocol_guid: Guid,
|
|
) -> io::Result<NonNull<T>> {
|
|
let boot_services: NonNull<efi::BootServices> =
|
|
boot_services().ok_or(BOOT_SERVICES_UNAVAILABLE)?.cast();
|
|
let system_handle = uefi::env::image_handle();
|
|
let mut protocol: MaybeUninit<*mut T> = MaybeUninit::uninit();
|
|
|
|
let r = unsafe {
|
|
((*boot_services.as_ptr()).open_protocol)(
|
|
handle.as_ptr(),
|
|
&mut protocol_guid,
|
|
protocol.as_mut_ptr().cast(),
|
|
system_handle.as_ptr(),
|
|
crate::ptr::null_mut(),
|
|
r_efi::system::OPEN_PROTOCOL_GET_PROTOCOL,
|
|
)
|
|
};
|
|
|
|
if r.is_error() {
|
|
Err(crate::io::Error::from_raw_os_error(r.as_usize()))
|
|
} else {
|
|
NonNull::new(unsafe { protocol.assume_init() })
|
|
.ok_or(const_io_error!(io::ErrorKind::Other, "null protocol"))
|
|
}
|
|
}
|
|
|
|
pub(crate) fn create_event(
|
|
signal: u32,
|
|
tpl: efi::Tpl,
|
|
handler: Option<efi::EventNotify>,
|
|
context: *mut crate::ffi::c_void,
|
|
) -> io::Result<NonNull<crate::ffi::c_void>> {
|
|
let boot_services: NonNull<efi::BootServices> =
|
|
boot_services().ok_or(BOOT_SERVICES_UNAVAILABLE)?.cast();
|
|
let mut event: r_efi::efi::Event = crate::ptr::null_mut();
|
|
let r = unsafe {
|
|
let create_event = (*boot_services.as_ptr()).create_event;
|
|
(create_event)(signal, tpl, handler, context, &mut event)
|
|
};
|
|
if r.is_error() {
|
|
Err(crate::io::Error::from_raw_os_error(r.as_usize()))
|
|
} else {
|
|
NonNull::new(event).ok_or(const_io_error!(io::ErrorKind::Other, "null protocol"))
|
|
}
|
|
}
|
|
|
|
/// # SAFETY
|
|
/// - The supplied event must be valid
|
|
pub(crate) unsafe fn close_event(evt: NonNull<crate::ffi::c_void>) -> io::Result<()> {
|
|
let boot_services: NonNull<efi::BootServices> =
|
|
boot_services().ok_or(BOOT_SERVICES_UNAVAILABLE)?.cast();
|
|
let r = unsafe {
|
|
let close_event = (*boot_services.as_ptr()).close_event;
|
|
(close_event)(evt.as_ptr())
|
|
};
|
|
|
|
if r.is_error() { Err(crate::io::Error::from_raw_os_error(r.as_usize())) } else { Ok(()) }
|
|
}
|
|
|
|
/// Get the Protocol for current system handle.
|
|
/// Note: Some protocols need to be manually freed. It is the callers responsibility to do so.
|
|
pub(crate) fn image_handle_protocol<T>(protocol_guid: Guid) -> io::Result<NonNull<T>> {
|
|
let system_handle = uefi::env::try_image_handle().ok_or(io::const_io_error!(
|
|
io::ErrorKind::NotFound,
|
|
"Protocol not found in Image handle"
|
|
))?;
|
|
open_protocol(system_handle, protocol_guid)
|
|
}
|
|
|
|
pub(crate) fn device_path_to_text(path: NonNull<device_path::Protocol>) -> io::Result<OsString> {
|
|
fn path_to_text(
|
|
protocol: NonNull<device_path_to_text::Protocol>,
|
|
path: NonNull<device_path::Protocol>,
|
|
) -> io::Result<OsString> {
|
|
let path_ptr: *mut r_efi::efi::Char16 = unsafe {
|
|
((*protocol.as_ptr()).convert_device_path_to_text)(
|
|
path.as_ptr(),
|
|
// DisplayOnly
|
|
r_efi::efi::Boolean::FALSE,
|
|
// AllowShortcuts
|
|
r_efi::efi::Boolean::FALSE,
|
|
)
|
|
};
|
|
|
|
// SAFETY: `convert_device_path_to_text` returns a pointer to a null-terminated UTF-16
|
|
// string, and that string cannot be deallocated prior to dropping the `WStrUnits`, so
|
|
// it's safe for `WStrUnits` to use.
|
|
let path_len = unsafe {
|
|
WStrUnits::new(path_ptr)
|
|
.ok_or(io::const_io_error!(io::ErrorKind::InvalidData, "Invalid path"))?
|
|
.count()
|
|
};
|
|
|
|
let path = OsString::from_wide(unsafe { slice::from_raw_parts(path_ptr.cast(), path_len) });
|
|
|
|
if let Some(boot_services) = crate::os::uefi::env::boot_services() {
|
|
let boot_services: NonNull<r_efi::efi::BootServices> = boot_services.cast();
|
|
unsafe {
|
|
((*boot_services.as_ptr()).free_pool)(path_ptr.cast());
|
|
}
|
|
}
|
|
|
|
Ok(path)
|
|
}
|
|
|
|
static LAST_VALID_HANDLE: AtomicPtr<crate::ffi::c_void> =
|
|
AtomicPtr::new(crate::ptr::null_mut());
|
|
|
|
if let Some(handle) = NonNull::new(LAST_VALID_HANDLE.load(Ordering::Acquire)) {
|
|
if let Ok(protocol) = open_protocol::<device_path_to_text::Protocol>(
|
|
handle,
|
|
device_path_to_text::PROTOCOL_GUID,
|
|
) {
|
|
return path_to_text(protocol, path);
|
|
}
|
|
}
|
|
|
|
let device_path_to_text_handles = locate_handles(device_path_to_text::PROTOCOL_GUID)?;
|
|
for handle in device_path_to_text_handles {
|
|
if let Ok(protocol) = open_protocol::<device_path_to_text::Protocol>(
|
|
handle,
|
|
device_path_to_text::PROTOCOL_GUID,
|
|
) {
|
|
LAST_VALID_HANDLE.store(handle.as_ptr(), Ordering::Release);
|
|
return path_to_text(protocol, path);
|
|
}
|
|
}
|
|
|
|
Err(io::const_io_error!(io::ErrorKind::NotFound, "No device path to text protocol found"))
|
|
}
|
|
|
|
/// Get RuntimeServices
|
|
pub(crate) fn runtime_services() -> Option<NonNull<r_efi::efi::RuntimeServices>> {
|
|
let system_table: NonNull<r_efi::efi::SystemTable> =
|
|
crate::os::uefi::env::try_system_table()?.cast();
|
|
let runtime_services = unsafe { (*system_table.as_ptr()).runtime_services };
|
|
NonNull::new(runtime_services)
|
|
}
|
|
|
|
pub(crate) struct DevicePath(NonNull<r_efi::protocols::device_path::Protocol>);
|
|
|
|
impl DevicePath {
|
|
pub(crate) fn from_text(p: &OsStr) -> io::Result<Self> {
|
|
fn inner(
|
|
p: &OsStr,
|
|
protocol: NonNull<r_efi::protocols::device_path_from_text::Protocol>,
|
|
) -> io::Result<DevicePath> {
|
|
let path_vec = p.encode_wide().chain(Some(0)).collect::<Vec<u16>>();
|
|
if path_vec[..path_vec.len() - 1].contains(&0) {
|
|
return Err(const_io_error!(
|
|
io::ErrorKind::InvalidInput,
|
|
"strings passed to UEFI cannot contain NULs",
|
|
));
|
|
}
|
|
|
|
let path =
|
|
unsafe { ((*protocol.as_ptr()).convert_text_to_device_path)(path_vec.as_ptr()) };
|
|
|
|
NonNull::new(path).map(DevicePath).ok_or_else(|| {
|
|
const_io_error!(io::ErrorKind::InvalidFilename, "Invalid Device Path")
|
|
})
|
|
}
|
|
|
|
static LAST_VALID_HANDLE: AtomicPtr<crate::ffi::c_void> =
|
|
AtomicPtr::new(crate::ptr::null_mut());
|
|
|
|
if let Some(handle) = NonNull::new(LAST_VALID_HANDLE.load(Ordering::Acquire)) {
|
|
if let Ok(protocol) = open_protocol::<r_efi::protocols::device_path_from_text::Protocol>(
|
|
handle,
|
|
r_efi::protocols::device_path_from_text::PROTOCOL_GUID,
|
|
) {
|
|
return inner(p, protocol);
|
|
}
|
|
}
|
|
|
|
let handles = locate_handles(r_efi::protocols::device_path_from_text::PROTOCOL_GUID)?;
|
|
for handle in handles {
|
|
if let Ok(protocol) = open_protocol::<r_efi::protocols::device_path_from_text::Protocol>(
|
|
handle,
|
|
r_efi::protocols::device_path_from_text::PROTOCOL_GUID,
|
|
) {
|
|
LAST_VALID_HANDLE.store(handle.as_ptr(), Ordering::Release);
|
|
return inner(p, protocol);
|
|
}
|
|
}
|
|
|
|
io::Result::Err(const_io_error!(
|
|
io::ErrorKind::NotFound,
|
|
"DevicePathFromText Protocol not found"
|
|
))
|
|
}
|
|
|
|
pub(crate) fn as_ptr(&self) -> *mut r_efi::protocols::device_path::Protocol {
|
|
self.0.as_ptr()
|
|
}
|
|
}
|
|
|
|
impl Drop for DevicePath {
|
|
fn drop(&mut self) {
|
|
if let Some(bt) = boot_services() {
|
|
let bt: NonNull<r_efi::efi::BootServices> = bt.cast();
|
|
unsafe {
|
|
((*bt.as_ptr()).free_pool)(self.0.as_ptr() as *mut crate::ffi::c_void);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub(crate) struct OwnedProtocol<T> {
|
|
guid: r_efi::efi::Guid,
|
|
handle: NonNull<crate::ffi::c_void>,
|
|
protocol: *mut T,
|
|
}
|
|
|
|
impl<T> OwnedProtocol<T> {
|
|
// FIXME: Consider using unsafe trait for matching protocol with guid
|
|
pub(crate) unsafe fn create(protocol: T, mut guid: r_efi::efi::Guid) -> io::Result<Self> {
|
|
let bt: NonNull<r_efi::efi::BootServices> =
|
|
boot_services().ok_or(BOOT_SERVICES_UNAVAILABLE)?.cast();
|
|
let protocol: *mut T = Box::into_raw(Box::new(protocol));
|
|
let mut handle: r_efi::efi::Handle = crate::ptr::null_mut();
|
|
|
|
// FIXME: Move into r-efi once extended_varargs_abi_support is stablized
|
|
let func: BootInstallMultipleProtocolInterfaces =
|
|
unsafe { crate::mem::transmute((*bt.as_ptr()).install_multiple_protocol_interfaces) };
|
|
|
|
let r = unsafe {
|
|
func(
|
|
&mut handle,
|
|
&mut guid as *mut _ as *mut crate::ffi::c_void,
|
|
protocol as *mut crate::ffi::c_void,
|
|
crate::ptr::null_mut() as *mut crate::ffi::c_void,
|
|
)
|
|
};
|
|
|
|
if r.is_error() {
|
|
drop(unsafe { Box::from_raw(protocol) });
|
|
return Err(crate::io::Error::from_raw_os_error(r.as_usize()));
|
|
};
|
|
|
|
let handle = NonNull::new(handle)
|
|
.ok_or(io::const_io_error!(io::ErrorKind::Uncategorized, "found null handle"))?;
|
|
|
|
Ok(Self { guid, handle, protocol })
|
|
}
|
|
|
|
pub(crate) fn handle(&self) -> NonNull<crate::ffi::c_void> {
|
|
self.handle
|
|
}
|
|
}
|
|
|
|
impl<T> Drop for OwnedProtocol<T> {
|
|
fn drop(&mut self) {
|
|
// Do not deallocate a runtime protocol
|
|
if let Some(bt) = boot_services() {
|
|
let bt: NonNull<r_efi::efi::BootServices> = bt.cast();
|
|
// FIXME: Move into r-efi once extended_varargs_abi_support is stablized
|
|
let func: BootUninstallMultipleProtocolInterfaces = unsafe {
|
|
crate::mem::transmute((*bt.as_ptr()).uninstall_multiple_protocol_interfaces)
|
|
};
|
|
let status = unsafe {
|
|
func(
|
|
self.handle.as_ptr(),
|
|
&mut self.guid as *mut _ as *mut crate::ffi::c_void,
|
|
self.protocol as *mut crate::ffi::c_void,
|
|
crate::ptr::null_mut() as *mut crate::ffi::c_void,
|
|
)
|
|
};
|
|
|
|
// Leak the protocol in case uninstall fails
|
|
if status == r_efi::efi::Status::SUCCESS {
|
|
let _ = unsafe { Box::from_raw(self.protocol) };
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T> AsRef<T> for OwnedProtocol<T> {
|
|
fn as_ref(&self) -> &T {
|
|
unsafe { self.protocol.as_ref().unwrap() }
|
|
}
|
|
}
|
|
|
|
pub(crate) struct OwnedTable<T> {
|
|
layout: crate::alloc::Layout,
|
|
ptr: *mut T,
|
|
}
|
|
|
|
impl<T> OwnedTable<T> {
|
|
pub(crate) fn from_table_header(hdr: &r_efi::efi::TableHeader) -> Self {
|
|
let header_size = hdr.header_size as usize;
|
|
let layout = crate::alloc::Layout::from_size_align(header_size, 8).unwrap();
|
|
let ptr = unsafe { crate::alloc::alloc(layout) as *mut T };
|
|
Self { layout, ptr }
|
|
}
|
|
|
|
pub(crate) const fn as_ptr(&self) -> *const T {
|
|
self.ptr
|
|
}
|
|
|
|
pub(crate) const fn as_mut_ptr(&self) -> *mut T {
|
|
self.ptr
|
|
}
|
|
}
|
|
|
|
impl OwnedTable<r_efi::efi::SystemTable> {
|
|
pub(crate) fn from_table(tbl: *const r_efi::efi::SystemTable) -> Self {
|
|
let hdr = unsafe { (*tbl).hdr };
|
|
|
|
let owned_tbl = Self::from_table_header(&hdr);
|
|
unsafe {
|
|
crate::ptr::copy_nonoverlapping(
|
|
tbl as *const u8,
|
|
owned_tbl.as_mut_ptr() as *mut u8,
|
|
hdr.header_size as usize,
|
|
)
|
|
};
|
|
|
|
owned_tbl
|
|
}
|
|
}
|
|
|
|
impl<T> Drop for OwnedTable<T> {
|
|
fn drop(&mut self) {
|
|
unsafe { crate::alloc::dealloc(self.ptr as *mut u8, self.layout) };
|
|
}
|
|
}
|