Magisk/native/src/core/daemon.rs

95 lines
2.4 KiB
Rust
Raw Normal View History

2023-05-09 18:54:38 -07:00
use std::cell::RefCell;
use std::fs::File;
2023-06-10 01:40:45 -07:00
use std::io;
2023-05-09 18:54:38 -07:00
use std::sync::{Mutex, OnceLock};
use crate::get_prop;
2023-09-27 15:21:24 -07:00
use base::{cstr, Directory, ResultExt, Utf8CStr, Utf8CStrBuf, Utf8CStrBufRef, WalkResult};
2023-06-10 01:40:45 -07:00
use crate::logging::{magisk_logging, zygisk_logging};
2023-05-09 18:54:38 -07:00
// Global magiskd singleton
pub static MAGISKD: OnceLock<MagiskD> = OnceLock::new();
#[derive(Default)]
pub struct MagiskD {
pub logd: Mutex<RefCell<Option<File>>>,
is_emulator: bool,
2023-05-09 18:54:38 -07:00
}
impl MagiskD {
pub fn is_emulator(&self) -> bool {
self.is_emulator
}
}
mod cxx_extern {
use base::libc::c_char;
extern "C" {
pub fn get_magisk_tmp() -> *const c_char;
}
}
pub fn get_magisk_tmp() -> &'static Utf8CStr {
unsafe { Utf8CStr::from_ptr(cxx_extern::get_magisk_tmp()).unwrap_unchecked() }
}
2023-05-09 18:54:38 -07:00
pub fn daemon_entry() {
let mut qemu = get_prop(cstr!("ro.kernel.qemu"), false);
if qemu.is_empty() {
qemu = get_prop(cstr!("ro.boot.qemu"), false);
}
let is_emulator = qemu == "1";
let magiskd = MagiskD {
logd: Default::default(),
is_emulator,
};
2023-05-09 18:54:38 -07:00
magiskd.start_log_daemon();
MAGISKD.set(magiskd).ok();
magisk_logging();
}
pub fn zygisk_entry() {
let magiskd = MagiskD::default();
MAGISKD.set(magiskd).ok();
zygisk_logging();
}
pub fn get_magiskd() -> &'static MagiskD {
unsafe { MAGISKD.get().unwrap_unchecked() }
2023-05-09 18:54:38 -07:00
}
2023-06-10 01:40:45 -07:00
pub fn find_apk_path(pkg: &[u8], data: &mut [u8]) -> usize {
use WalkResult::*;
fn inner(pkg: &[u8], buf: &mut dyn Utf8CStrBuf) -> io::Result<usize> {
2023-06-14 18:44:53 +08:00
let pkg = match Utf8CStr::from_bytes(pkg) {
Ok(pkg) => pkg,
Err(e) => return Err(io::Error::new(io::ErrorKind::Other, e)),
};
2023-06-10 01:40:45 -07:00
Directory::open(cstr!("/data/app"))?.pre_order_walk(|e| {
if !e.is_dir() {
return Ok(Skip);
}
let d_name = e.d_name().to_bytes();
2023-06-14 18:44:53 +08:00
if d_name.starts_with(pkg.as_bytes()) && d_name[pkg.len()] == b'-' {
2023-06-10 01:40:45 -07:00
// Found the APK path, we can abort now
e.path(buf)?;
2023-06-10 01:40:45 -07:00
return Ok(Abort);
}
if d_name.starts_with(b"~~") {
return Ok(Continue);
}
Ok(Skip)
})?;
if !buf.is_empty() {
2023-09-27 15:21:24 -07:00
buf.push_str("/base.apk");
2023-06-10 01:40:45 -07:00
}
Ok(buf.len())
2023-06-10 01:40:45 -07:00
}
2023-09-27 15:21:24 -07:00
inner(pkg, &mut Utf8CStrBufRef::from(data))
.log()
.unwrap_or(0)
2023-06-10 01:40:45 -07:00
}