2023-05-09 18:54:38 -07:00
|
|
|
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};
|
|
|
|
|
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
|
|
|
|
2023-10-30 14:24:21 -07:00
|
|
|
use crate::get_prop;
|
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 {
|
2023-10-30 14:24:21 -07:00
|
|
|
pub logd: Mutex<Option<File>>,
|
2023-09-28 15:37:11 -07:00
|
|
|
is_emulator: bool,
|
2023-05-09 18:54:38 -07:00
|
|
|
}
|
|
|
|
|
2023-10-30 03:14:25 -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() {
|
2023-09-28 15:37:11 -07:00
|
|
|
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 {
|
2023-09-28 15:37:11 -07:00
|
|
|
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::*;
|
2023-09-12 17:35:01 -07:00
|
|
|
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
|
2023-09-12 17:35:01 -07:00
|
|
|
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)
|
|
|
|
})?;
|
2023-09-12 17:35:01 -07:00
|
|
|
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
|
|
|
}
|
2023-09-12 17:35:01 -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))
|
2023-09-12 17:35:01 -07:00
|
|
|
.log()
|
|
|
|
.unwrap_or(0)
|
2023-06-10 01:40:45 -07:00
|
|
|
}
|