blob: 094d1bfd31685b99f2f42b395812e65598051fbd [file] [log] [blame] [edit]
use libc::c_int;
cfg_select! {
not(
any(
all(target_os = "linux", not(target_env = "musl")),
target_os = "l4re",
target_os = "android",
target_os = "hurd",
)
) => {
use libc::{open as open64, openat as openat64};
}
_ => {
use libc::{open64, openat64};
}
}
use crate::ffi::CStr;
use crate::os::fd::{AsFd, BorrowedFd, IntoRawFd, OwnedFd, RawFd};
#[cfg(target_family = "unix")]
use crate::os::unix::io::{AsRawFd, FromRawFd};
#[cfg(target_os = "wasi")]
use crate::os::wasi::io::{AsRawFd, FromRawFd};
use crate::path::Path;
use crate::sys::fd::FileDesc;
use crate::sys::fs::OpenOptions;
use crate::sys::fs::unix::{File, debug_path_fd};
use crate::sys::helpers::run_path_with_cstr;
use crate::sys::{AsInner, FromInner, IntoInner, cvt_r};
use crate::{fmt, fs, io};
pub struct Dir(OwnedFd);
impl Dir {
pub fn open(path: &Path, opts: &OpenOptions) -> io::Result<Self> {
run_path_with_cstr(path, &|path| Self::open_with_c(path, opts))
}
pub fn open_file(&self, path: &Path, opts: &OpenOptions) -> io::Result<File> {
run_path_with_cstr(path.as_ref(), &|path| self.open_file_c(path, &opts))
}
pub fn open_with_c(path: &CStr, opts: &OpenOptions) -> io::Result<Self> {
let flags = libc::O_CLOEXEC
| libc::O_DIRECTORY
| opts.get_access_mode()?
| opts.get_creation_mode()?
| (opts.custom_flags as c_int & !libc::O_ACCMODE);
let fd = cvt_r(|| unsafe { open64(path.as_ptr(), flags, opts.mode as c_int) })?;
Ok(Self(unsafe { OwnedFd::from_raw_fd(fd) }))
}
fn open_file_c(&self, path: &CStr, opts: &OpenOptions) -> io::Result<File> {
let flags = libc::O_CLOEXEC
| opts.get_access_mode()?
| opts.get_creation_mode()?
| (opts.custom_flags as c_int & !libc::O_ACCMODE);
let fd = cvt_r(|| unsafe {
openat64(self.0.as_raw_fd(), path.as_ptr(), flags, opts.mode as c_int)
})?;
Ok(File(unsafe { FileDesc::from_raw_fd(fd) }))
}
}
impl fmt::Debug for Dir {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let fd = self.0.as_raw_fd();
let mut b = debug_path_fd(fd, f, "Dir");
b.finish()
}
}
#[unstable(feature = "dirfd", issue = "120426")]
impl AsRawFd for fs::Dir {
fn as_raw_fd(&self) -> RawFd {
self.as_inner().0.as_raw_fd()
}
}
#[unstable(feature = "dirfd", issue = "120426")]
impl IntoRawFd for fs::Dir {
fn into_raw_fd(self) -> RawFd {
self.into_inner().0.into_raw_fd()
}
}
#[unstable(feature = "dirfd", issue = "120426")]
impl FromRawFd for fs::Dir {
unsafe fn from_raw_fd(fd: RawFd) -> Self {
Self::from_inner(Dir(unsafe { FromRawFd::from_raw_fd(fd) }))
}
}
#[unstable(feature = "dirfd", issue = "120426")]
impl AsFd for fs::Dir {
fn as_fd(&self) -> BorrowedFd<'_> {
self.as_inner().0.as_fd()
}
}
#[unstable(feature = "dirfd", issue = "120426")]
impl From<fs::Dir> for OwnedFd {
fn from(value: fs::Dir) -> Self {
value.into_inner().0
}
}
#[unstable(feature = "dirfd", issue = "120426")]
impl From<OwnedFd> for fs::Dir {
fn from(value: OwnedFd) -> Self {
Self::from_inner(Dir(value))
}
}