Magisk/native/src/base/files.rs

224 lines
6.0 KiB
Rust
Raw Normal View History

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;
use crate::slice_from_ptr_mut;
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 {
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-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-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(())
}
}