Magisk/native/src/base/files.rs

143 lines
4.0 KiB
Rust
Raw Normal View History

2022-09-15 01:17:05 -07:00
use std::ffi::CStr;
use std::fs::File;
use std::io;
use std::io::BufRead;
2022-09-15 01:17:05 -07:00
use std::os::unix::io::{AsRawFd, FromRawFd, OwnedFd, RawFd};
use std::path::Path;
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
use crate::{bfmt_cstr, errno, xopen};
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';
}
return r;
}
#[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> {
2022-09-15 01:17:05 -07:00
let fd = xopen(path.as_ptr(), flags, mode);
if fd >= 0 {
unsafe { Some(OwnedFd::from_raw_fd(fd)) }
} else {
None
}
}
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 {
st1 = std::mem::zeroed();
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 {
st2 = std::mem::zeroed();
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;
}
}
return len;
} else {
*errno() = ENOENT;
-1
}
}
extern "C" {
fn strscpy(dst: *mut c_char, src: *const c_char, size: usize) -> usize;
}
#[no_mangle]
pub extern "C" fn mkdirs(path: *const c_char, mode: mode_t) -> i32 {
unsafe {
let mut buf = [0 as 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';
if libc::mkdir(ptr, mode) < 0 && *errno() != EEXIST {
return -1;
}
curr[p] = b'/';
curr = &mut curr[(p + 1)..];
}
if libc::mkdir(ptr, mode) < 0 && *errno() != EEXIST {
return -1;
}
0
}
}
pub fn read_lines<P: AsRef<Path>>(path: P) -> io::Result<io::Lines<io::BufReader<File>>> {
let file = File::open(path)?;
Ok(io::BufReader::new(file).lines())
}