2023-05-26 14:07:11 -07:00
|
|
|
use mem::MaybeUninit;
|
2023-05-04 18:49:33 -07:00
|
|
|
use std::cmp::min;
|
2022-09-15 01:17:05 -07:00
|
|
|
use std::ffi::CStr;
|
2023-06-09 02:00:37 -07:00
|
|
|
use std::fs::File;
|
2023-05-26 14:07:11 -07:00
|
|
|
use std::io::{BufRead, Read, Seek, SeekFrom, Write};
|
2023-06-09 02:00:37 -07:00
|
|
|
use std::ops::Deref;
|
2023-06-12 14:58:13 -07:00
|
|
|
use std::os::android::fs::MetadataExt;
|
2023-06-09 02:00:37 -07:00
|
|
|
use std::os::fd::{AsFd, BorrowedFd, IntoRawFd};
|
2023-06-12 14:58:13 -07:00
|
|
|
use std::os::unix::fs::FileTypeExt;
|
2022-09-15 01:17:05 -07:00
|
|
|
use std::os::unix::io::{AsRawFd, FromRawFd, OwnedFd, RawFd};
|
2023-06-12 01:07:43 -07:00
|
|
|
use std::{io, mem, ptr, slice};
|
2022-09-15 01:17:05 -07:00
|
|
|
|
2023-09-12 17:35:01 -07:00
|
|
|
use libc::{
|
|
|
|
c_char, c_uint, dirent, mode_t, EEXIST, ENOENT, F_OK, O_CLOEXEC, O_PATH, O_RDONLY, O_RDWR,
|
|
|
|
};
|
2022-09-15 01:17:05 -07:00
|
|
|
|
2023-09-12 17:35:01 -07:00
|
|
|
use crate::{
|
|
|
|
copy_cstr, cstr, errno, error, FlatData, FsPath, FsPathBuf, LibcReturn, Utf8CStr, Utf8CStrArr,
|
|
|
|
Utf8CStrBuf,
|
|
|
|
};
|
2022-09-15 01:17:05 -07:00
|
|
|
|
2023-06-12 05:59:50 -07:00
|
|
|
pub fn __open_fd_impl(path: &Utf8CStr, flags: i32, mode: mode_t) -> io::Result<OwnedFd> {
|
2022-09-25 16:35:28 -07:00
|
|
|
unsafe {
|
2023-06-12 01:07:43 -07:00
|
|
|
let fd = libc::open(path.as_ptr(), flags, mode as c_uint).check_os_err()?;
|
|
|
|
Ok(OwnedFd::from_raw_fd(fd))
|
2022-09-15 01:17:05 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-25 16:35:28 -07:00
|
|
|
#[macro_export]
|
|
|
|
macro_rules! open_fd {
|
|
|
|
($path:expr, $flags:expr) => {
|
2023-05-02 16:49:43 -07:00
|
|
|
$crate::__open_fd_impl($path, $flags, 0)
|
2022-09-25 16:35:28 -07:00
|
|
|
};
|
|
|
|
($path:expr, $flags:expr, $mode:expr) => {
|
2023-05-02 16:49:43 -07:00
|
|
|
$crate::__open_fd_impl($path, $flags, $mode)
|
2022-09-25 16:35:28 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-07-06 04:54:33 -07:00
|
|
|
pub(crate) unsafe fn readlink_unsafe(path: *const c_char, buf: *mut u8, bufsz: usize) -> isize {
|
2023-06-09 02:00:37 -07:00
|
|
|
let r = libc::readlink(path, buf.cast(), bufsz - 1);
|
|
|
|
if r >= 0 {
|
|
|
|
*buf.offset(r) = b'\0';
|
|
|
|
}
|
|
|
|
r
|
2022-09-15 01:17:05 -07:00
|
|
|
}
|
|
|
|
|
2023-09-12 17:35:01 -07:00
|
|
|
pub fn fd_path(fd: RawFd, buf: &mut dyn Utf8CStrBuf) -> io::Result<()> {
|
|
|
|
let mut arr = Utf8CStrArr::<40>::new();
|
|
|
|
let path = FsPathBuf::new(&mut arr).join("/proc/self/fd").join_fmt(fd);
|
|
|
|
path.read_link(buf)
|
2022-09-15 01:17:05 -07:00
|
|
|
}
|
2023-02-12 16:36:38 +08:00
|
|
|
|
2023-05-26 14:07:11 -07:00
|
|
|
pub trait ReadExt {
|
|
|
|
fn skip(&mut self, len: usize) -> io::Result<()>;
|
2023-06-20 00:19:40 -07:00
|
|
|
fn read_flat_data<F: FlatData>(&mut self, data: &mut F) -> io::Result<()>;
|
2023-05-26 14:07:11 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: Read> ReadExt for T {
|
|
|
|
fn skip(&mut self, mut len: usize) -> io::Result<()> {
|
|
|
|
let mut buf = MaybeUninit::<[u8; 4096]>::uninit();
|
|
|
|
let buf = unsafe { buf.assume_init_mut() };
|
|
|
|
while len > 0 {
|
|
|
|
let l = min(buf.len(), len);
|
|
|
|
self.read_exact(&mut buf[..l])?;
|
|
|
|
len -= l;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-20 00:19:40 -07:00
|
|
|
|
|
|
|
fn read_flat_data<F: FlatData>(&mut self, data: &mut F) -> io::Result<()> {
|
|
|
|
self.read_exact(data.as_raw_bytes_mut())
|
|
|
|
}
|
2023-05-26 14:07:11 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub trait ReadSeekExt {
|
|
|
|
fn skip(&mut self, len: usize) -> io::Result<()>;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: Read + Seek> ReadSeekExt for T {
|
|
|
|
fn skip(&mut self, len: usize) -> io::Result<()> {
|
|
|
|
if self.seek(SeekFrom::Current(len as i64)).is_err() {
|
|
|
|
// If the file is not actually seekable, fallback to read
|
|
|
|
ReadExt::skip(self, len)?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-24 19:11:56 -07:00
|
|
|
pub trait BufReadExt {
|
|
|
|
fn foreach_lines<F: FnMut(&mut String) -> bool>(&mut self, f: F);
|
|
|
|
fn foreach_props<F: FnMut(&str, &str) -> bool>(&mut self, f: F);
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: BufRead> BufReadExt for T {
|
|
|
|
fn foreach_lines<F: FnMut(&mut String) -> bool>(&mut self, mut f: F) {
|
|
|
|
let mut buf = String::new();
|
|
|
|
loop {
|
|
|
|
match self.read_line(&mut buf) {
|
|
|
|
Ok(0) => break,
|
|
|
|
Ok(_) => {
|
|
|
|
if !f(&mut buf) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
error!("{}", e);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
buf.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn foreach_props<F: FnMut(&str, &str) -> bool>(&mut self, mut f: F) {
|
|
|
|
self.foreach_lines(|line| {
|
|
|
|
let line = line.trim();
|
|
|
|
if line.starts_with('#') {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if let Some((key, value)) = line.split_once('=') {
|
2023-06-20 00:19:40 -07:00
|
|
|
return f(key.trim(), value.trim());
|
2023-05-24 19:11:56 -07:00
|
|
|
}
|
2023-05-30 22:23:11 -07:00
|
|
|
true
|
2023-05-24 19:11:56 -07:00
|
|
|
});
|
|
|
|
}
|
2023-02-12 16:36:38 +08:00
|
|
|
}
|
2023-05-04 18:49:33 -07:00
|
|
|
|
|
|
|
pub trait WriteExt {
|
|
|
|
fn write_zeros(&mut self, len: usize) -> io::Result<()>;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: Write> WriteExt for T {
|
|
|
|
fn write_zeros(&mut self, mut len: usize) -> io::Result<()> {
|
2023-05-30 22:23:11 -07:00
|
|
|
let buf = [0_u8; 4096];
|
2023-05-04 18:49:33 -07:00
|
|
|
while len > 0 {
|
|
|
|
let l = min(buf.len(), len);
|
2023-05-30 22:23:11 -07:00
|
|
|
self.write_all(&buf[..l])?;
|
2023-05-04 18:49:33 -07:00
|
|
|
len -= l;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
2023-06-09 02:00:37 -07:00
|
|
|
|
|
|
|
pub struct DirEntry<'a> {
|
2023-06-12 05:05:57 -07:00
|
|
|
dir: &'a Directory,
|
2023-06-09 02:00:37 -07:00
|
|
|
entry: &'a dirent,
|
2023-06-10 01:50:18 -07:00
|
|
|
d_name_len: usize,
|
2023-06-09 02:00:37 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
impl DirEntry<'_> {
|
|
|
|
pub fn d_name(&self) -> &CStr {
|
|
|
|
unsafe {
|
|
|
|
CStr::from_bytes_with_nul_unchecked(slice::from_raw_parts(
|
|
|
|
self.d_name.as_ptr().cast(),
|
2023-06-10 01:50:18 -07:00
|
|
|
self.d_name_len,
|
2023-06-09 02:00:37 -07:00
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-12 17:35:01 -07:00
|
|
|
pub fn path(&self, buf: &mut dyn Utf8CStrBuf) -> io::Result<()> {
|
|
|
|
self.dir.path(buf)?;
|
|
|
|
buf.append("/");
|
|
|
|
buf.append_lossy(self.d_name().to_bytes());
|
|
|
|
Ok(())
|
2023-06-09 02:00:37 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_dir(&self) -> bool {
|
|
|
|
self.d_type == libc::DT_DIR
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_file(&self) -> bool {
|
|
|
|
self.d_type == libc::DT_REG
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_lnk(&self) -> bool {
|
|
|
|
self.d_type == libc::DT_LNK
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn unlink(&self) -> io::Result<()> {
|
|
|
|
let flag = if self.is_dir() { libc::AT_REMOVEDIR } else { 0 };
|
|
|
|
unsafe {
|
2023-06-12 01:07:43 -07:00
|
|
|
libc::unlinkat(self.dir.as_raw_fd(), self.d_name.as_ptr(), flag).check_os_err()?;
|
2023-06-09 02:00:37 -07:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn open_as_dir(&self) -> io::Result<Directory> {
|
|
|
|
if !self.is_dir() {
|
|
|
|
return Err(io::Error::from(io::ErrorKind::NotADirectory));
|
|
|
|
}
|
|
|
|
unsafe {
|
|
|
|
let fd = libc::openat(
|
|
|
|
self.dir.as_raw_fd(),
|
|
|
|
self.d_name.as_ptr(),
|
|
|
|
O_RDONLY | O_CLOEXEC,
|
2023-06-12 01:07:43 -07:00
|
|
|
)
|
|
|
|
.check_os_err()?;
|
2023-06-09 02:00:37 -07:00
|
|
|
Directory::try_from(OwnedFd::from_raw_fd(fd))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn open_as_file(&self, flags: i32) -> io::Result<File> {
|
|
|
|
if self.is_dir() {
|
|
|
|
return Err(io::Error::from(io::ErrorKind::IsADirectory));
|
|
|
|
}
|
|
|
|
unsafe {
|
|
|
|
let fd = libc::openat(
|
|
|
|
self.dir.as_raw_fd(),
|
|
|
|
self.d_name.as_ptr(),
|
|
|
|
flags | O_CLOEXEC,
|
2023-06-12 01:07:43 -07:00
|
|
|
)
|
|
|
|
.check_os_err()?;
|
2023-06-09 02:00:37 -07:00
|
|
|
Ok(File::from_raw_fd(fd))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Deref for DirEntry<'_> {
|
|
|
|
type Target = dirent;
|
|
|
|
|
|
|
|
fn deref(&self) -> &dirent {
|
|
|
|
self.entry
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-12 05:05:57 -07:00
|
|
|
pub struct Directory {
|
2023-06-09 02:00:37 -07:00
|
|
|
dirp: *mut libc::DIR,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub enum WalkResult {
|
|
|
|
Continue,
|
|
|
|
Abort,
|
|
|
|
Skip,
|
|
|
|
}
|
|
|
|
|
2023-06-12 05:05:57 -07:00
|
|
|
impl Directory {
|
2023-06-12 05:59:50 -07:00
|
|
|
pub fn open(path: &Utf8CStr) -> io::Result<Directory> {
|
2023-06-12 01:07:43 -07:00
|
|
|
let dirp = unsafe { libc::opendir(path.as_ptr()) }.check_os_err()?;
|
2023-06-12 05:05:57 -07:00
|
|
|
Ok(Directory { dirp })
|
2023-06-09 02:00:37 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn read(&mut self) -> io::Result<Option<DirEntry<'_>>> {
|
|
|
|
*errno() = 0;
|
|
|
|
let e = unsafe { libc::readdir(self.dirp) };
|
|
|
|
if e.is_null() {
|
|
|
|
return if *errno() != 0 {
|
|
|
|
Err(io::Error::last_os_error())
|
|
|
|
} else {
|
|
|
|
Ok(None)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
// Skip both "." and ".."
|
|
|
|
unsafe {
|
|
|
|
let entry = &*e;
|
|
|
|
let d_name = CStr::from_ptr(entry.d_name.as_ptr());
|
|
|
|
return if d_name == cstr!(".") || d_name == cstr!("..") {
|
|
|
|
self.read()
|
|
|
|
} else {
|
|
|
|
let e = DirEntry {
|
|
|
|
dir: self,
|
|
|
|
entry,
|
2023-06-10 01:50:18 -07:00
|
|
|
d_name_len: d_name.to_bytes_with_nul().len(),
|
2023-06-09 02:00:37 -07:00
|
|
|
};
|
|
|
|
Ok(Some(e))
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn rewind(&mut self) {
|
|
|
|
unsafe { libc::rewinddir(self.dirp) }
|
|
|
|
}
|
|
|
|
|
2023-09-12 17:35:01 -07:00
|
|
|
pub fn path(&self, buf: &mut dyn Utf8CStrBuf) -> io::Result<()> {
|
2023-06-09 02:00:37 -07:00
|
|
|
fd_path(self.as_raw_fd(), buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn post_order_walk<F: FnMut(&DirEntry) -> io::Result<WalkResult>>(
|
|
|
|
&mut self,
|
|
|
|
mut f: F,
|
|
|
|
) -> io::Result<WalkResult> {
|
|
|
|
self.post_order_walk_impl(&mut f)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn pre_order_walk<F: FnMut(&DirEntry) -> io::Result<WalkResult>>(
|
|
|
|
&mut self,
|
|
|
|
mut f: F,
|
|
|
|
) -> io::Result<WalkResult> {
|
|
|
|
self.pre_order_walk_impl(&mut f)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn remove_all(&mut self) -> io::Result<()> {
|
|
|
|
self.post_order_walk(|e| {
|
|
|
|
e.unlink()?;
|
|
|
|
Ok(WalkResult::Continue)
|
|
|
|
})?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-12 05:05:57 -07:00
|
|
|
impl Directory {
|
2023-06-09 02:00:37 -07:00
|
|
|
fn post_order_walk_impl<F: FnMut(&DirEntry) -> io::Result<WalkResult>>(
|
|
|
|
&mut self,
|
|
|
|
f: &mut F,
|
|
|
|
) -> io::Result<WalkResult> {
|
|
|
|
use WalkResult::*;
|
|
|
|
loop {
|
|
|
|
match self.read()? {
|
|
|
|
None => return Ok(Continue),
|
|
|
|
Some(ref e) => {
|
|
|
|
if e.is_dir() {
|
|
|
|
let mut dir = e.open_as_dir()?;
|
|
|
|
if let Abort = dir.post_order_walk_impl(f)? {
|
|
|
|
return Ok(Abort);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
match f(e)? {
|
|
|
|
Abort => return Ok(Abort),
|
|
|
|
Skip => return Ok(Continue),
|
|
|
|
Continue => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn pre_order_walk_impl<F: FnMut(&DirEntry) -> io::Result<WalkResult>>(
|
|
|
|
&mut self,
|
|
|
|
f: &mut F,
|
|
|
|
) -> io::Result<WalkResult> {
|
|
|
|
use WalkResult::*;
|
|
|
|
loop {
|
|
|
|
match self.read()? {
|
|
|
|
None => return Ok(Continue),
|
|
|
|
Some(ref e) => match f(e)? {
|
|
|
|
Abort => return Ok(Abort),
|
2023-06-14 18:44:53 +08:00
|
|
|
Skip => continue,
|
2023-06-09 02:00:37 -07:00
|
|
|
Continue => {
|
|
|
|
if e.is_dir() {
|
|
|
|
let mut dir = e.open_as_dir()?;
|
|
|
|
if let Abort = dir.pre_order_walk_impl(f)? {
|
|
|
|
return Ok(Abort);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-12 05:05:57 -07:00
|
|
|
impl TryFrom<OwnedFd> for Directory {
|
2023-06-09 02:00:37 -07:00
|
|
|
type Error = io::Error;
|
|
|
|
|
|
|
|
fn try_from(fd: OwnedFd) -> io::Result<Self> {
|
2023-06-12 01:07:43 -07:00
|
|
|
let dirp = unsafe { libc::fdopendir(fd.into_raw_fd()) }.check_os_err()?;
|
2023-06-12 05:05:57 -07:00
|
|
|
Ok(Directory { dirp })
|
2023-06-09 02:00:37 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-12 05:05:57 -07:00
|
|
|
impl AsRawFd for Directory {
|
2023-06-09 02:00:37 -07:00
|
|
|
fn as_raw_fd(&self) -> RawFd {
|
|
|
|
unsafe { libc::dirfd(self.dirp) }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-12 05:05:57 -07:00
|
|
|
impl AsFd for Directory {
|
|
|
|
fn as_fd(&self) -> BorrowedFd {
|
2023-06-09 02:00:37 -07:00
|
|
|
unsafe { BorrowedFd::borrow_raw(self.as_raw_fd()) }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-12 05:05:57 -07:00
|
|
|
impl Drop for Directory {
|
2023-06-09 02:00:37 -07:00
|
|
|
fn drop(&mut self) {
|
|
|
|
unsafe {
|
|
|
|
libc::closedir(self.dirp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-12 17:35:01 -07:00
|
|
|
impl FsPath {
|
|
|
|
pub fn open(&self, flags: i32) -> io::Result<File> {
|
|
|
|
Ok(File::from(open_fd!(self, flags)?))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create(&self, flags: i32, mode: mode_t) -> io::Result<File> {
|
|
|
|
Ok(File::from(open_fd!(self, flags, mode)?))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn exists(&self) -> bool {
|
|
|
|
unsafe { libc::access(self.as_ptr(), F_OK) == 0 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn rename_to<T: AsRef<Utf8CStr>>(&self, name: T) -> io::Result<()> {
|
|
|
|
unsafe { libc::rename(self.as_ptr(), name.as_ref().as_ptr()).as_os_err() }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn remove(&self) -> io::Result<()> {
|
|
|
|
unsafe { libc::remove(self.as_ptr()).as_os_err() }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn remove_all(&self) -> io::Result<()> {
|
|
|
|
let f = self.open(O_RDONLY | O_CLOEXEC)?;
|
2023-06-12 14:58:13 -07:00
|
|
|
let st = f.metadata()?;
|
|
|
|
if st.is_dir() {
|
|
|
|
let mut dir = Directory::try_from(OwnedFd::from(f))?;
|
2023-06-09 02:00:37 -07:00
|
|
|
dir.remove_all()?;
|
|
|
|
}
|
2023-09-12 17:35:01 -07:00
|
|
|
self.remove()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn read_link(&self, buf: &mut dyn Utf8CStrBuf) -> io::Result<()> {
|
|
|
|
buf.clear();
|
|
|
|
unsafe { readlink_unsafe(self.as_ptr(), buf.as_mut_ptr().cast(), buf.capacity()) }
|
|
|
|
.as_os_err()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn mkdirs(&self, mode: mode_t) -> io::Result<()> {
|
|
|
|
let mut buf = [0_u8; 4096];
|
|
|
|
let len = copy_cstr(&mut buf, self);
|
|
|
|
let buf = &mut buf[..len];
|
|
|
|
let mut off = 1;
|
|
|
|
unsafe {
|
|
|
|
while let Some(p) = buf[off..].iter().position(|c| *c == b'/') {
|
|
|
|
buf[off + p] = b'\0';
|
|
|
|
if libc::mkdir(buf.as_ptr().cast(), mode) < 0 && *errno() != EEXIST {
|
|
|
|
return Err(io::Error::last_os_error());
|
|
|
|
}
|
|
|
|
buf[off + p] = b'/';
|
|
|
|
off += p + 1;
|
|
|
|
}
|
|
|
|
if libc::mkdir(buf.as_ptr().cast(), mode) < 0 && *errno() != EEXIST {
|
|
|
|
return Err(io::Error::last_os_error());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*errno() = 0;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Inspired by https://android.googlesource.com/platform/bionic/+/master/libc/bionic/realpath.cpp
|
|
|
|
pub fn realpath(&self, buf: &mut dyn Utf8CStrBuf) -> io::Result<()> {
|
|
|
|
let fd = open_fd!(self, O_PATH | O_CLOEXEC)?;
|
|
|
|
let mut st1: libc::stat;
|
|
|
|
let mut st2: libc::stat;
|
|
|
|
let mut skip_check = false;
|
|
|
|
unsafe {
|
|
|
|
st1 = mem::zeroed();
|
|
|
|
if libc::fstat(fd.as_raw_fd(), &mut st1) < 0 {
|
|
|
|
// This will only fail on Linux < 3.6
|
|
|
|
skip_check = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fd_path(fd.as_raw_fd(), buf)?;
|
|
|
|
unsafe {
|
|
|
|
st2 = mem::zeroed();
|
|
|
|
libc::stat(buf.as_ptr(), &mut st2).as_os_err()?;
|
|
|
|
if !skip_check && (st2.st_dev != st1.st_dev || st2.st_ino != st1.st_ino) {
|
|
|
|
*errno() = ENOENT;
|
|
|
|
return Err(io::Error::last_os_error());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
2023-06-12 01:07:43 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct MappedFile(&'static mut [u8]);
|
|
|
|
|
|
|
|
impl MappedFile {
|
2023-06-12 05:59:50 -07:00
|
|
|
pub fn open(path: &Utf8CStr) -> io::Result<MappedFile> {
|
2023-06-12 01:07:43 -07:00
|
|
|
Ok(MappedFile(map_file(path, false)?))
|
|
|
|
}
|
|
|
|
|
2023-06-12 05:59:50 -07:00
|
|
|
pub fn open_rw(path: &Utf8CStr) -> io::Result<MappedFile> {
|
2023-06-12 01:07:43 -07:00
|
|
|
Ok(MappedFile(map_file(path, true)?))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create(fd: BorrowedFd, sz: usize, rw: bool) -> io::Result<MappedFile> {
|
|
|
|
Ok(MappedFile(map_fd(fd, sz, rw)?))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsRef<[u8]> for MappedFile {
|
|
|
|
fn as_ref(&self) -> &[u8] {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsMut<[u8]> for MappedFile {
|
|
|
|
fn as_mut(&mut self) -> &mut [u8] {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for MappedFile {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
unsafe {
|
|
|
|
libc::munmap(self.0.as_mut_ptr().cast(), self.0.len());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We mark the returned slice static because it is valid until explicitly unmapped
|
2023-06-12 05:59:50 -07:00
|
|
|
pub(crate) fn map_file(path: &Utf8CStr, rw: bool) -> io::Result<&'static mut [u8]> {
|
2023-06-12 14:58:13 -07:00
|
|
|
extern "C" {
|
|
|
|
// Don't use the declaration from the libc crate as request should be u32 not i32
|
|
|
|
fn ioctl(fd: RawFd, request: u32, ...) -> i32;
|
|
|
|
}
|
|
|
|
|
2023-06-25 07:21:35 +08:00
|
|
|
#[cfg(target_pointer_width = "64")]
|
2023-06-12 14:58:13 -07:00
|
|
|
const BLKGETSIZE64: u32 = 0x80081272;
|
|
|
|
|
2023-06-25 07:21:35 +08:00
|
|
|
#[cfg(target_pointer_width = "32")]
|
|
|
|
const BLKGETSIZE64: u32 = 0x80041272;
|
|
|
|
|
2023-06-12 20:00:27 +08:00
|
|
|
let flag = if rw { O_RDWR } else { O_RDONLY };
|
2023-06-12 14:58:13 -07:00
|
|
|
let f = File::from(open_fd!(path, flag | O_CLOEXEC)?);
|
|
|
|
|
|
|
|
let st = f.metadata()?;
|
|
|
|
let sz = if st.file_type().is_block_device() {
|
|
|
|
let mut sz = 0_u64;
|
2023-09-12 17:35:01 -07:00
|
|
|
unsafe { ioctl(f.as_raw_fd(), BLKGETSIZE64, &mut sz) }.as_os_err()?;
|
2023-06-12 14:58:13 -07:00
|
|
|
sz
|
|
|
|
} else {
|
|
|
|
st.st_size()
|
|
|
|
};
|
|
|
|
|
|
|
|
map_fd(f.as_fd(), sz as usize, rw)
|
2023-06-12 01:07:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn map_fd(fd: BorrowedFd, sz: usize, rw: bool) -> io::Result<&'static mut [u8]> {
|
|
|
|
let flag = if rw {
|
|
|
|
libc::MAP_SHARED
|
|
|
|
} else {
|
|
|
|
libc::MAP_PRIVATE
|
|
|
|
};
|
|
|
|
unsafe {
|
|
|
|
let ptr = libc::mmap(
|
|
|
|
ptr::null_mut(),
|
|
|
|
sz,
|
|
|
|
libc::PROT_READ | libc::PROT_WRITE,
|
|
|
|
flag,
|
|
|
|
fd.as_raw_fd(),
|
|
|
|
0,
|
|
|
|
);
|
|
|
|
if ptr == libc::MAP_FAILED {
|
2023-06-09 02:00:37 -07:00
|
|
|
return Err(io::Error::last_os_error());
|
|
|
|
}
|
2023-06-12 01:07:43 -07:00
|
|
|
Ok(slice::from_raw_parts_mut(ptr.cast(), sz))
|
2023-06-09 02:00:37 -07:00
|
|
|
}
|
|
|
|
}
|