diff options
Diffstat (limited to 'alacritty/src/macos/proc.rs')
-rw-r--r-- | alacritty/src/macos/proc.rs | 160 |
1 files changed, 160 insertions, 0 deletions
diff --git a/alacritty/src/macos/proc.rs b/alacritty/src/macos/proc.rs new file mode 100644 index 00000000..be7e4891 --- /dev/null +++ b/alacritty/src/macos/proc.rs @@ -0,0 +1,160 @@ +use std::ffi::{CStr, CString, IntoStringError}; +use std::fmt::{self, Display, Formatter}; +use std::io; +use std::mem::{self, MaybeUninit}; +use std::os::raw::{c_int, c_void}; +use std::path::PathBuf; + +/// Error during working directory retrieval. +#[derive(Debug)] +pub enum Error { + Io(io::Error), + + /// Error converting into utf8 string. + IntoString(IntoStringError), + + /// Expected return size didn't match libproc's. + InvalidSize, +} + +impl std::error::Error for Error { + fn source(&self) -> Option<&(dyn std::error::Error + 'static)> { + match self { + Error::InvalidSize => None, + Error::Io(err) => err.source(), + Error::IntoString(err) => err.source(), + } + } +} + +impl Display for Error { + fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result { + match self { + Error::InvalidSize => write!(f, "Invalid proc_pidinfo return size"), + Error::Io(err) => write!(f, "Error getting current working directory: {}", err), + Error::IntoString(err) => { + write!(f, "Error when parsing current working directory: {}", err) + }, + } + } +} + +impl From<io::Error> for Error { + fn from(val: io::Error) -> Self { + Error::Io(val) + } +} + +impl From<IntoStringError> for Error { + fn from(val: IntoStringError) -> Self { + Error::IntoString(val) + } +} + +pub fn cwd(pid: c_int) -> Result<PathBuf, Error> { + let mut info = MaybeUninit::<sys::proc_vnodepathinfo>::uninit(); + let info_ptr = info.as_mut_ptr() as *mut c_void; + let size = mem::size_of::<sys::proc_vnodepathinfo>() as c_int; + + let c_str = unsafe { + let pidinfo_size = sys::proc_pidinfo(pid, sys::PROC_PIDVNODEPATHINFO, 0, info_ptr, size); + match pidinfo_size { + c if c < 0 => return Err(io::Error::last_os_error().into()), + s if s != size => return Err(Error::InvalidSize), + _ => CStr::from_ptr(info.assume_init().pvi_cdir.vip_path.as_ptr()), + } + }; + + Ok(CString::from(c_str).into_string().map(PathBuf::from)?) +} + +/// Bindings for libproc. +#[allow(non_camel_case_types)] +mod sys { + use std::os::raw::{c_char, c_int, c_longlong, c_void}; + + pub const PROC_PIDVNODEPATHINFO: c_int = 9; + + type gid_t = c_int; + type off_t = c_longlong; + type uid_t = c_int; + type fsid_t = fsid; + + #[repr(C)] + #[derive(Debug, Copy, Clone)] + pub struct fsid { + pub val: [i32; 2usize], + } + + #[repr(C)] + #[derive(Debug, Copy, Clone)] + pub struct vinfo_stat { + pub vst_dev: u32, + pub vst_mode: u16, + pub vst_nlink: u16, + pub vst_ino: u64, + pub vst_uid: uid_t, + pub vst_gid: gid_t, + pub vst_atime: i64, + pub vst_atimensec: i64, + pub vst_mtime: i64, + pub vst_mtimensec: i64, + pub vst_ctime: i64, + pub vst_ctimensec: i64, + pub vst_birthtime: i64, + pub vst_birthtimensec: i64, + pub vst_size: off_t, + pub vst_blocks: i64, + pub vst_blksize: i32, + pub vst_flags: u32, + pub vst_gen: u32, + pub vst_rdev: u32, + pub vst_qspare: [i64; 2usize], + } + + #[repr(C)] + #[derive(Debug, Copy, Clone)] + pub struct vnode_info { + pub vi_stat: vinfo_stat, + pub vi_type: c_int, + pub vi_pad: c_int, + pub vi_fsid: fsid_t, + } + + #[repr(C)] + #[derive(Copy, Clone)] + pub struct vnode_info_path { + pub vip_vi: vnode_info, + pub vip_path: [c_char; 1024usize], + } + + #[repr(C)] + #[derive(Copy, Clone)] + pub struct proc_vnodepathinfo { + pub pvi_cdir: vnode_info_path, + pub pvi_rdir: vnode_info_path, + } + + extern "C" { + pub fn proc_pidinfo( + pid: c_int, + flavor: c_int, + arg: u64, + buffer: *mut c_void, + buffersize: c_int, + ) -> c_int; + } +} + +#[cfg(test)] +mod tests { + use super::*; + + use std::env; + use std::process; + + #[test] + fn cwd_matches_current_dir() { + assert_eq!(cwd(process::id() as i32).ok(), env::current_dir().ok()); + } +} |