use r_efi::efi::Status; use r_efi::efi::protocols::{device_path, loaded_image_device_path}; use super::{RawOsError, helpers, unsupported_err}; use crate::ffi::{OsStr, OsString}; use crate::marker::PhantomData; use crate::os::uefi; use crate::path::{self, PathBuf}; use crate::ptr::NonNull; use crate::{fmt, io}; pub fn errno() -> RawOsError { 0 } pub fn error_string(errno: RawOsError) -> String { // Keep the List in Alphabetical Order // The Messages are taken from UEFI Specification Appendix D - Status Codes #[rustfmt::skip] let msg = match r_efi::efi::Status::from_usize(errno) { Status::ABORTED => "The operation was aborted.", Status::ACCESS_DENIED => "Access was denied.", Status::ALREADY_STARTED => "The protocol has already been started.", Status::BAD_BUFFER_SIZE => "The buffer was not the proper size for the request.", Status::BUFFER_TOO_SMALL => "The buffer is not large enough to hold the requested data. The required buffer size is returned in the appropriate parameter when this error occurs.", Status::COMPROMISED_DATA => "The security status of the data is unknown or compromised and the data must be updated or replaced to restore a valid security status.", Status::CONNECTION_FIN => "The receiving operation fails because the communication peer has closed the connection and there is no more data in the receive buffer of the instance.", Status::CONNECTION_REFUSED => "The receiving or transmission operation fails because this connection is refused.", Status::CONNECTION_RESET => "The connect fails because the connection is reset either by instance itself or the communication peer.", Status::CRC_ERROR => "A CRC error was detected.", Status::DEVICE_ERROR => "The physical device reported an error while attempting the operation.", Status::END_OF_FILE => "The end of the file was reached.", Status::END_OF_MEDIA => "Beginning or end of media was reached", Status::HOST_UNREACHABLE => "The remote host is not reachable.", Status::HTTP_ERROR => "A HTTP error occurred during the network operation.", Status::ICMP_ERROR => "An ICMP error occurred during the network operation.", Status::INCOMPATIBLE_VERSION => "The function encountered an internal version that was incompatible with a version requested by the caller.", Status::INVALID_LANGUAGE => "The language specified was invalid.", Status::INVALID_PARAMETER => "A parameter was incorrect.", Status::IP_ADDRESS_CONFLICT => "There is an address conflict address allocation", Status::LOAD_ERROR => "The image failed to load.", Status::MEDIA_CHANGED => "The medium in the device has changed since the last access.", Status::NETWORK_UNREACHABLE => "The network containing the remote host is not reachable.", Status::NO_MAPPING => "A mapping to a device does not exist.", Status::NO_MEDIA => "The device does not contain any medium to perform the operation.", Status::NO_RESPONSE => "The server was not found or did not respond to the request.", Status::NOT_FOUND => "The item was not found.", Status::NOT_READY => "There is no data pending upon return.", Status::NOT_STARTED => "The protocol has not been started.", Status::OUT_OF_RESOURCES => "A resource has run out.", Status::PROTOCOL_ERROR => "A protocol error occurred during the network operation.", Status::PROTOCOL_UNREACHABLE => "An ICMP protocol unreachable error is received.", Status::SECURITY_VIOLATION => "The function was not performed due to a security violation.", Status::TFTP_ERROR => "A TFTP error occurred during the network operation.", Status::TIMEOUT => "The timeout time expired.", Status::UNSUPPORTED => "The operation is not supported.", Status::VOLUME_FULL => "There is no more space on the file system.", Status::VOLUME_CORRUPTED => "An inconstancy was detected on the file system causing the operating to fail.", Status::WRITE_PROTECTED => "The device cannot be written to.", _ => return format!("Status: {errno}"), }; msg.to_owned() } pub fn getcwd() -> io::Result { match helpers::open_shell() { Some(shell) => { // SAFETY: path_ptr is managed by UEFI shell and should not be deallocated let path_ptr = unsafe { ((*shell.as_ptr()).get_cur_dir)(crate::ptr::null_mut()) }; helpers::os_string_from_raw(path_ptr) .map(PathBuf::from) .ok_or(io::const_error!(io::ErrorKind::InvalidData, "invalid path")) } None => { let mut t = current_exe()?; // SAFETY: This should never fail since the disk prefix will be present even for root // executables assert!(t.pop()); Ok(t) } } } pub fn chdir(p: &path::Path) -> io::Result<()> { let shell = helpers::open_shell().ok_or(unsupported_err())?; let mut p = helpers::os_string_to_raw(p.as_os_str()) .ok_or(io::const_error!(io::ErrorKind::InvalidData, "invalid path"))?; let r = unsafe { ((*shell.as_ptr()).set_cur_dir)(crate::ptr::null_mut(), p.as_mut_ptr()) }; if r.is_error() { Err(io::Error::from_raw_os_error(r.as_usize())) } else { Ok(()) } } pub struct SplitPaths<'a>(!, PhantomData<&'a ()>); pub fn split_paths(_unparsed: &OsStr) -> SplitPaths<'_> { panic!("unsupported") } impl<'a> Iterator for SplitPaths<'a> { type Item = PathBuf; fn next(&mut self) -> Option { self.0 } } #[derive(Debug)] pub struct JoinPathsError; pub fn join_paths(_paths: I) -> Result where I: Iterator, T: AsRef, { Err(JoinPathsError) } impl fmt::Display for JoinPathsError { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { "not supported on this platform yet".fmt(f) } } impl crate::error::Error for JoinPathsError {} pub fn current_exe() -> io::Result { let protocol = helpers::image_handle_protocol::( loaded_image_device_path::PROTOCOL_GUID, )?; helpers::device_path_to_text(protocol).map(PathBuf::from) } pub fn temp_dir() -> PathBuf { panic!("no filesystem on this platform") } pub fn home_dir() -> Option { None } pub fn exit(code: i32) -> ! { if let (Some(boot_services), Some(handle)) = (uefi::env::boot_services(), uefi::env::try_image_handle()) { let boot_services: NonNull = boot_services.cast(); let _ = unsafe { ((*boot_services.as_ptr()).exit)( handle.as_ptr(), Status::from_usize(code as usize), 0, crate::ptr::null_mut(), ) }; } crate::intrinsics::abort() } pub fn getpid() -> u32 { panic!("no pids on this platform") }