Magisk/native/src/base/files.rs

519 lines
14 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-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;
use std::os::android::fs::MetadataExt;
2023-06-09 02:00:37 -07:00
use std::os::fd::{AsFd, BorrowedFd, IntoRawFd};
use std::os::unix::fs::FileTypeExt;
2022-09-15 01:17:05 -07:00
use std::os::unix::io::{AsRawFd, FromRawFd, OwnedFd, RawFd};
use std::{io, mem, ptr, slice};
2022-09-15 01:17:05 -07:00
use libc::{c_char, c_uint, dirent, mode_t, EEXIST, ENOENT, O_CLOEXEC, O_PATH, O_RDONLY, O_RDWR};
2022-09-15 01:17:05 -07:00
2023-06-12 05:59:50 -07:00
use crate::{bfmt_cstr, copy_cstr, cstr, errno, error, LibcReturn, Utf8CStr};
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 {
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-06-09 02:00:37 -07:00
pub unsafe fn readlink_unsafe(path: *const c_char, buf: *mut u8, bufsz: usize) -> isize {
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-06-12 05:59:50 -07:00
pub fn readlink(path: &Utf8CStr, data: &mut [u8]) -> io::Result<usize> {
let r =
unsafe { readlink_unsafe(path.as_ptr(), data.as_mut_ptr(), data.len()) }.check_os_err()?;
2023-06-09 02:00:37 -07:00
Ok(r as usize)
2022-09-15 01:17:05 -07:00
}
2023-06-09 02:00:37 -07:00
pub fn fd_path(fd: RawFd, buf: &mut [u8]) -> io::Result<usize> {
let mut fd_buf = [0_u8; 40];
2022-09-15 01:17:05 -07:00
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
2023-06-12 05:59:50 -07:00
pub fn realpath(path: &Utf8CStr, buf: &mut [u8]) -> io::Result<usize> {
2023-06-09 02:00:37 -07:00
let fd = open_fd!(path, 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 shall only fail on Linux < 3.6
skip_check = true;
2022-09-15 01:17:05 -07:00
}
2023-06-09 02:00:37 -07:00
}
let len = fd_path(fd.as_raw_fd(), buf)?;
unsafe {
st2 = mem::zeroed();
if libc::stat(buf.as_ptr().cast(), &mut st2) < 0
|| (!skip_check && (st2.st_dev != st1.st_dev || st2.st_ino != st1.st_ino))
{
*errno() = ENOENT;
return Err(io::Error::last_os_error());
2022-09-15 01:17:05 -07:00
}
}
2023-06-09 02:00:37 -07:00
Ok(len)
2022-09-15 01:17:05 -07:00
}
2023-06-12 05:59:50 -07:00
pub fn mkdirs(path: &Utf8CStr, mode: mode_t) -> io::Result<()> {
2023-05-30 22:23:11 -07:00
let mut buf = [0_u8; 4096];
2023-06-09 02:00:37 -07:00
let len = copy_cstr(&mut buf, path);
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());
2022-09-15 01:17:05 -07:00
}
}
2023-06-09 02:00:37 -07:00
*errno() = 0;
Ok(())
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(())
}
}
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
))
}
}
pub fn path(&self, buf: &mut [u8]) -> io::Result<usize> {
let mut len = self.dir.path(buf)?;
buf[len] = b'/';
len += 1;
len += copy_cstr(&mut buf[len..], self.d_name());
Ok(len)
}
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 {
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,
)
.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,
)
.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> {
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) }
}
pub fn path(&self, buf: &mut [u8]) -> io::Result<usize> {
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> {
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-06-12 05:59:50 -07:00
pub fn rm_rf(path: &Utf8CStr) -> io::Result<()> {
2023-06-09 02:00:37 -07:00
unsafe {
let f = File::from(open_fd!(path, O_RDONLY | O_CLOEXEC)?);
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()?;
}
libc::remove(path.as_ptr()).check_os_err()?;
}
Ok(())
}
pub struct MappedFile(&'static mut [u8]);
impl MappedFile {
2023-06-12 05:59:50 -07:00
pub fn open(path: &Utf8CStr) -> io::Result<MappedFile> {
Ok(MappedFile(map_file(path, false)?))
}
2023-06-12 05:59:50 -07:00
pub fn open_rw(path: &Utf8CStr) -> io::Result<MappedFile> {
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]> {
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;
}
const BLKGETSIZE64: u32 = 0x80081272;
2023-06-12 20:00:27 +08:00
let flag = if rw { O_RDWR } else { O_RDONLY };
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;
unsafe { ioctl(f.as_raw_fd(), BLKGETSIZE64, &mut sz) }.check_os_err()?;
sz
} else {
st.st_size()
};
map_fd(f.as_fd(), sz as usize, rw)
}
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());
}
Ok(slice::from_raw_parts_mut(ptr.cast(), sz))
2023-06-09 02:00:37 -07:00
}
}