diff --git a/src/ld_so/access.rs b/src/ld_so/access.rs index c29f7f3417fe902a033ab7f8f4c68099ce0df0e2..1d083fe8936b900f839e03a6c646d82f07c607b4 100644 --- a/src/ld_so/access.rs +++ b/src/ld_so/access.rs @@ -1,67 +1,14 @@ -// Wrapper over the access syscall that doesn't touch errno variable, -// Do not use outside of ld_so - #[cfg(target_os = "redox")] use crate::header::unistd::{F_OK, R_OK, W_OK, X_OK}; use crate::{ c_str::{CStr, CString}, - platform::types::*, + error::Errno, + platform::{types::*, Pal, Sys}, }; -pub fn accessible(path: &str, mode: c_int) -> c_int { +pub fn accessible(path: &str, mode: c_int) -> Result<(), Errno> { let path_c = CString::new(path.as_bytes()).unwrap(); /*.map_err(|err| { Error::Malformed(format!("invalid path '{}': {}", path, err)) })?;*/ - unsafe { access(path_c.as_ptr(), mode) } -} - -#[cfg(target_os = "linux")] -unsafe fn access(path: *const c_char, mode: c_int) -> c_int { - let path = CStr::from_ptr(path); - syscall!(ACCESS, (path).as_ptr(), mode) as c_int -} - -// Wrapper over the systemcall, Do not use outside of ld_so -#[cfg(target_os = "redox")] -unsafe fn access(path: *const c_char, mode: c_int) -> c_int { - use core::str; - let path = match str::from_utf8(CStr::from_ptr(path).to_bytes()) { - Ok(ok) => ok, - Err(_) => return -1, - }; - let fd = match crate::platform::sys::path::open(path, syscall::O_CLOEXEC) { - Ok(fd) => fd, - _ => return -1, - }; - if mode == F_OK { - return 0; - } - let mut stat = syscall::Stat::default(); - if syscall::fstat(fd, &mut stat).is_err() { - return -1; - } - let _ = syscall::close(fd); - let uid = match syscall::getuid() { - Ok(uid) => uid, - Err(_) => return -1, - }; - let gid = match syscall::getgid() { - Ok(gid) => gid, - Err(_) => return -1, - }; - - let perms = if stat.st_uid as usize == uid { - stat.st_mode >> (3 * 2 & 0o7) - } else if stat.st_gid as usize == gid { - stat.st_mode >> (3 * 1 & 0o7) - } else { - stat.st_mode & 0o7 - }; - if (mode & R_OK == R_OK && perms & 0o4 != 0o4) - || (mode & W_OK == W_OK && perms & 0o2 != 0o2) - || (mode & X_OK == X_OK && perms & 0o1 != 0o1) - { - return -1; - } - 0 + unsafe { Sys::access(CStr::from_ptr(path_c.as_ptr()), mode) } } diff --git a/src/ld_so/linker.rs b/src/ld_so/linker.rs index 983715cbfdb2fa8aa5bf2376294d56202fc6d2a0..b6a90dd0833c688856e40e2b3c5be2166aa1063a 100644 --- a/src/ld_so/linker.rs +++ b/src/ld_so/linker.rs @@ -266,7 +266,7 @@ impl Linker { parent_runpath: &Option<String>, ) -> Result<String> { let mut full_path = name.to_string(); - if accessible(&full_path, F_OK) == 0 { + if accessible(&full_path, F_OK).is_ok() { return Ok(full_path); } else { let mut search_paths = Vec::new(); @@ -280,7 +280,7 @@ impl Linker { for part in search_paths.iter() { full_path = format!("{}/{}", part, name); trace!("trying path {}", full_path); - if accessible(&full_path, F_OK) == 0 { + if accessible(&full_path, F_OK).is_ok() { return Ok(full_path); } } diff --git a/src/ld_so/start.rs b/src/ld_so/start.rs index d0cf2c942fc439b36a0b58741240c57e6779c73d..0d0b0265d7b3135f566fb8ced2323ccc6b7a4793 100644 --- a/src/ld_so/start.rs +++ b/src/ld_so/start.rs @@ -113,7 +113,7 @@ fn resolve_path_name( name_or_path: &str, envs: &BTreeMap<String, String>, ) -> Option<(String, String)> { - if accessible(name_or_path, unistd::F_OK) == 0 { + if accessible(name_or_path, unistd::F_OK).is_ok() { return Some(( name_or_path.to_string(), name_or_path @@ -135,7 +135,7 @@ fn resolve_path_name( } else { format!("{}/{}", part, name_or_path) }; - if accessible(&path, unistd::F_OK) == 0 { + if accessible(&path, unistd::F_OK).is_ok() { return Some((path.to_string(), name_or_path.to_string())); } } diff --git a/src/platform/redox/mod.rs b/src/platform/redox/mod.rs index bda1862753c7c14feb7d19b2908e0079dedd8da0..e186b5de90d6a3a8ba3eddb607a64531a0594eed 100644 --- a/src/platform/redox/mod.rs +++ b/src/platform/redox/mod.rs @@ -78,7 +78,7 @@ pub struct Sys; impl Pal for Sys { fn access(path: CStr, mode: c_int) -> Result<()> { - let fd = File::open(path, fcntl::O_PATH | fcntl::O_CLOEXEC)?; + let fd = FdGuard::new(Sys::open(path, fcntl::O_PATH | fcntl::O_CLOEXEC, 0)? as usize); if mode == F_OK { return Ok(());