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-05-26 14:07:11 -07:00
|
|
|
use std::io::{BufRead, Read, Seek, SeekFrom, Write};
|
2022-09-15 01:17:05 -07:00
|
|
|
use std::os::unix::io::{AsRawFd, FromRawFd, OwnedFd, RawFd};
|
2023-05-26 14:07:11 -07:00
|
|
|
use std::{io, mem};
|
2022-09-15 01:17:05 -07:00
|
|
|
|
2022-09-25 16:35:28 -07:00
|
|
|
use libc::{c_char, c_uint, mode_t, EEXIST, ENOENT, O_CLOEXEC, O_PATH};
|
2022-09-15 01:17:05 -07:00
|
|
|
|
2023-05-24 19:11:56 -07:00
|
|
|
use crate::{bfmt_cstr, errno, error, xopen};
|
2022-09-15 01:17:05 -07:00
|
|
|
|
2022-09-21 03:09:46 +02:00
|
|
|
pub mod unsafe_impl {
|
2022-09-15 01:17:05 -07:00
|
|
|
use std::ffi::CStr;
|
|
|
|
|
|
|
|
use libc::c_char;
|
|
|
|
|
2022-09-15 16:56:22 -07:00
|
|
|
use crate::slice_from_ptr_mut;
|
|
|
|
|
2022-10-31 16:35:33 -07:00
|
|
|
pub unsafe fn readlink(path: *const c_char, buf: *mut u8, bufsz: usize) -> isize {
|
2022-09-15 01:17:05 -07:00
|
|
|
let r = libc::readlink(path, buf.cast(), bufsz - 1);
|
|
|
|
if r >= 0 {
|
|
|
|
*buf.offset(r) = b'\0';
|
|
|
|
}
|
2023-05-30 22:23:11 -07:00
|
|
|
r
|
2022-09-15 01:17:05 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[no_mangle]
|
|
|
|
unsafe extern "C" fn canonical_path(path: *const c_char, buf: *mut u8, bufsz: usize) -> isize {
|
2022-09-21 03:09:46 +02:00
|
|
|
super::realpath(CStr::from_ptr(path), slice_from_ptr_mut(buf, bufsz))
|
2022-09-15 01:17:05 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-25 16:35:28 -07:00
|
|
|
pub fn __open_fd_impl(path: &CStr, flags: i32, mode: mode_t) -> Option<OwnedFd> {
|
|
|
|
unsafe {
|
|
|
|
let fd = libc::open(path.as_ptr(), flags, mode as c_uint);
|
|
|
|
if fd >= 0 {
|
|
|
|
Some(OwnedFd::from_raw_fd(fd))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-21 03:09:46 +02:00
|
|
|
pub fn __xopen_fd_impl(path: &CStr, flags: i32, mode: mode_t) -> Option<OwnedFd> {
|
2023-05-30 22:23:11 -07:00
|
|
|
unsafe {
|
|
|
|
let fd = xopen(path.as_ptr(), flags, mode);
|
|
|
|
if fd >= 0 {
|
|
|
|
Some(OwnedFd::from_raw_fd(fd))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
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
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-09-15 01:17:05 -07:00
|
|
|
#[macro_export]
|
2022-09-21 03:09:46 +02:00
|
|
|
macro_rules! xopen_fd {
|
2022-09-15 01:17:05 -07:00
|
|
|
($path:expr, $flags:expr) => {
|
2023-05-02 16:49:43 -07:00
|
|
|
$crate::__xopen_fd_impl($path, $flags, 0)
|
2022-09-15 01:17:05 -07:00
|
|
|
};
|
|
|
|
($path:expr, $flags:expr, $mode:expr) => {
|
2023-05-02 16:49:43 -07:00
|
|
|
$crate::__xopen_fd_impl($path, $flags, $mode)
|
2022-09-15 01:17:05 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn readlink(path: &CStr, data: &mut [u8]) -> isize {
|
2022-10-31 16:35:33 -07:00
|
|
|
unsafe { unsafe_impl::readlink(path.as_ptr(), data.as_mut_ptr(), data.len()) }
|
2022-09-15 01:17:05 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fd_path(fd: RawFd, buf: &mut [u8]) -> isize {
|
|
|
|
let mut fd_buf: [u8; 40] = [0; 40];
|
|
|
|
let fd_path = bfmt_cstr!(&mut fd_buf, "/proc/self/fd/{}", fd);
|
|
|
|
readlink(fd_path, buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Inspired by https://android.googlesource.com/platform/bionic/+/master/libc/bionic/realpath.cpp
|
2022-09-21 03:09:46 +02:00
|
|
|
pub fn realpath(path: &CStr, buf: &mut [u8]) -> isize {
|
2022-09-25 16:35:28 -07:00
|
|
|
if let Some(fd) = open_fd!(path, O_PATH | O_CLOEXEC) {
|
2022-09-15 01:17:05 -07:00
|
|
|
let mut st1: libc::stat;
|
|
|
|
let mut st2: libc::stat;
|
2022-10-15 20:18:20 -07:00
|
|
|
let mut skip_check = false;
|
2022-09-15 01:17:05 -07:00
|
|
|
unsafe {
|
2023-05-30 22:23:11 -07:00
|
|
|
st1 = mem::zeroed();
|
2022-09-15 01:17:05 -07:00
|
|
|
if libc::fstat(fd.as_raw_fd(), &mut st1) < 0 {
|
2022-10-15 20:18:20 -07:00
|
|
|
// This shall only fail on Linux < 3.6
|
|
|
|
skip_check = true;
|
2022-09-15 01:17:05 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
let len = fd_path(fd.as_raw_fd(), buf);
|
|
|
|
unsafe {
|
2023-05-30 22:23:11 -07:00
|
|
|
st2 = mem::zeroed();
|
2022-09-15 01:17:05 -07:00
|
|
|
if libc::stat(buf.as_ptr().cast(), &mut st2) < 0
|
2022-10-15 20:18:20 -07:00
|
|
|
|| (!skip_check && (st2.st_dev != st1.st_dev || st2.st_ino != st1.st_ino))
|
2022-09-15 01:17:05 -07:00
|
|
|
{
|
|
|
|
*errno() = ENOENT;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
2023-05-30 22:23:11 -07:00
|
|
|
len
|
2022-09-15 01:17:05 -07:00
|
|
|
} else {
|
|
|
|
*errno() = ENOENT;
|
|
|
|
-1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
fn strscpy(dst: *mut c_char, src: *const c_char, size: usize) -> usize;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[no_mangle]
|
2023-05-30 22:23:11 -07:00
|
|
|
pub unsafe extern "C" fn mkdirs(path: *const c_char, mode: mode_t) -> i32 {
|
|
|
|
let mut buf = [0_u8; 4096];
|
|
|
|
let ptr: *mut c_char = buf.as_mut_ptr().cast();
|
|
|
|
let len = strscpy(ptr, path, buf.len());
|
|
|
|
let mut curr = &mut buf[1..len];
|
|
|
|
while let Some(p) = curr.iter().position(|c| *c == b'/') {
|
|
|
|
curr[p] = b'\0';
|
2022-09-15 01:17:05 -07:00
|
|
|
if libc::mkdir(ptr, mode) < 0 && *errno() != EEXIST {
|
|
|
|
return -1;
|
|
|
|
}
|
2023-05-30 22:23:11 -07:00
|
|
|
curr[p] = b'/';
|
|
|
|
curr = &mut curr[(p + 1)..];
|
|
|
|
}
|
|
|
|
if libc::mkdir(ptr, mode) < 0 && *errno() != EEXIST {
|
|
|
|
return -1;
|
2022-09-15 01:17:05 -07:00
|
|
|
}
|
2023-05-30 22:23:11 -07:00
|
|
|
0
|
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<()>;
|
|
|
|
}
|
|
|
|
|
|
|
|
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(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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('=') {
|
|
|
|
return f(key, value);
|
|
|
|
}
|
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(())
|
|
|
|
}
|
|
|
|
}
|