use super::procfs::set_permissions_through_proc_self_fd;
use crate::fs::{errors, open, OpenOptions, Permissions};
use posish::fs::{fchmod, Mode, OFlags, RawMode};
use std::{
fs, io,
os::unix::fs::{OpenOptionsExt, PermissionsExt},
path::Path,
sync::atomic::{AtomicBool, Ordering::Relaxed},
};
pub(crate) fn set_permissions_impl(
start: &fs::File,
path: &Path,
perm: Permissions,
) -> io::Result<()> {
static FCHMOD_PATH_BADF: AtomicBool = AtomicBool::new(false);
let std_perm = perm.into_std(start)?;
if !FCHMOD_PATH_BADF.load(Relaxed) {
let opath_result = open(
start,
path,
OpenOptions::new()
.read(true)
.custom_flags(OFlags::PATH.bits() as i32),
);
if let Ok(file) = opath_result {
match set_file_permissions(&file, std_perm.clone()) {
Ok(()) => return Ok(()),
Err(err) => match posish::io::Error::from_io_error(&err) {
Some(posish::io::Error::BADF) => FCHMOD_PATH_BADF.store(true, Relaxed),
_ => return Err(err),
},
}
}
}
match open(start, path, OpenOptions::new().read(true)) {
Ok(file) => return set_file_permissions(&file, std_perm),
Err(err) => match posish::io::Error::from_io_error(&err) {
Some(posish::io::Error::ACCES) => (),
_ => return Err(err),
},
}
match open(start, path, OpenOptions::new().write(true)) {
Ok(file) => return set_file_permissions(&file, std_perm),
Err(err) => match posish::io::Error::from_io_error(&err) {
Some(posish::io::Error::ACCES) | Some(posish::io::Error::ISDIR) => (),
_ => return Err(err),
},
}
set_permissions_through_proc_self_fd(start, path, std_perm)
}
fn set_file_permissions(file: &fs::File, perm: fs::Permissions) -> io::Result<()> {
let mode = Mode::from_bits(perm.mode() as RawMode).ok_or_else(errors::invalid_flags)?;
Ok(fchmod(file, mode)?)
}