Rearchitect logging

This commit is contained in:
topjohnwu 2022-07-06 01:16:08 -07:00
parent 2e52875b50
commit 70fd03d5fc
14 changed files with 172 additions and 154 deletions

View File

@ -11,23 +11,15 @@
using namespace std; using namespace std;
void nop_log(const char *, va_list) {}
log_callback log_cb = {
.d = nop_log,
.i = nop_log,
.w = nop_log,
.e = nop_log,
};
static bool EXIT_ON_ERROR = false;
static int fmt_and_log_with_rs(LogLevel level, const char *fmt, va_list ap) { static int fmt_and_log_with_rs(LogLevel level, const char *fmt, va_list ap) {
char buf[4096]; char buf[4096];
int ret = vsnprintf(buf, sizeof(buf), fmt, ap); int ret = vsnprintf(buf, sizeof(buf), fmt, ap);
log_with_rs(level, buf); log_with_rs(level, rust::Str(buf, ret));
return ret; return ret;
} }
int (*cpp_logger)(LogLevel level, const char *fmt, va_list ap) = fmt_and_log_with_rs;
// Used to override external C library logging // Used to override external C library logging
extern "C" int magisk_log_print(int prio, const char *tag, const char *fmt, ...) { extern "C" int magisk_log_print(int prio, const char *tag, const char *fmt, ...) {
LogLevel level; LogLevel level;
@ -55,48 +47,29 @@ extern "C" int magisk_log_print(int prio, const char *tag, const char *fmt, ...)
snprintf(fmt_buf + len, sizeof(fmt_buf) - len, ": %s", fmt); snprintf(fmt_buf + len, sizeof(fmt_buf) - len, ": %s", fmt);
va_list argv; va_list argv;
va_start(argv, fmt); va_start(argv, fmt);
int ret = fmt_and_log_with_rs(level, fmt_buf, argv); int ret = cpp_logger(level, fmt_buf, argv);
va_end(argv); va_end(argv);
return ret; return ret;
} }
#define rlog(prio) [](auto fmt, auto ap) { fmt_and_log_with_rs(LogLevel::prio, fmt, ap); } #define LOG_BODY(level) { \
void forward_logging_to_rs() {
log_cb.d = rlog(Debug);
log_cb.i = rlog(Info);
log_cb.w = rlog(Warn);
log_cb.e = rlog(Error);
}
void cmdline_logging() {
rust::cmdline_logging();
forward_logging_to_rs();
exit_on_error(true);
}
void exit_on_error(bool b) {
rust::exit_on_error(b);
EXIT_ON_ERROR = b;
}
#define LOG_BODY(prio) { \
va_list argv; \ va_list argv; \
va_start(argv, fmt); \ va_start(argv, fmt); \
log_cb.prio(fmt, argv); \ cpp_logger(LogLevel::level, fmt, argv); \
va_end(argv); \ va_end(argv); \
} }
// LTO will optimize out the NOP function // LTO will optimize out the NOP function
#if MAGISK_DEBUG #if MAGISK_DEBUG
void LOGD(const char *fmt, ...) { LOG_BODY(d) } void LOGD(const char *fmt, ...) { LOG_BODY(Debug) }
#else #else
void LOGD(const char *fmt, ...) {} void LOGD(const char *fmt, ...) {}
#endif #endif
void LOGI(const char *fmt, ...) { LOG_BODY(i) } void LOGI(const char *fmt, ...) { LOG_BODY(Info) }
void LOGW(const char *fmt, ...) { LOG_BODY(w) } void LOGW(const char *fmt, ...) { LOG_BODY(Warn) }
void LOGE(const char *fmt, ...) { LOG_BODY(e); if (EXIT_ON_ERROR) exit(EXIT_FAILURE); } void LOGE(const char *fmt, ...) { LOG_BODY(Error) }
// Export raw symbol to fortify compat // Export raw symbol to fortify compat
extern "C" void __vloge(const char* fmt, va_list ap) { extern "C" void __vloge(const char* fmt, va_list ap) {
log_cb.e(fmt, ap); cpp_logger(LogLevel::Error, fmt, ap);
} }

View File

@ -3,23 +3,12 @@
#include <cerrno> #include <cerrno>
#include <cstdarg> #include <cstdarg>
struct log_callback { #include <base-rs.hpp>
void (*d)(const char* fmt, va_list ap);
void (*i)(const char* fmt, va_list ap);
void (*w)(const char* fmt, va_list ap);
void (*e)(const char* fmt, va_list ap);
};
extern log_callback log_cb; extern int (*cpp_logger)(LogLevel level, const char *fmt, va_list ap);
void LOGD(const char *fmt, ...) __printflike(1, 2); void LOGD(const char *fmt, ...) __printflike(1, 2);
void LOGI(const char *fmt, ...) __printflike(1, 2); void LOGI(const char *fmt, ...) __printflike(1, 2);
void LOGW(const char *fmt, ...) __printflike(1, 2); void LOGW(const char *fmt, ...) __printflike(1, 2);
void LOGE(const char *fmt, ...) __printflike(1, 2); void LOGE(const char *fmt, ...) __printflike(1, 2);
#define PLOGE(fmt, args...) LOGE(fmt " failed with %d: %s\n", ##args, errno, std::strerror(errno)) #define PLOGE(fmt, args...) LOGE(fmt " failed with %d: %s\n", ##args, errno, std::strerror(errno))
void nop_log(const char *, va_list);
void forward_logging_to_rs();
void cmdline_logging();
void exit_on_error(bool b);

View File

@ -20,7 +20,6 @@ void unlock_blocks();
void reboot(); void reboot();
void start_log_daemon(); void start_log_daemon();
void setup_logfile(bool reset); void setup_logfile(bool reset);
void magisk_logging();
std::string read_certificate(int fd, int version = -1); std::string read_certificate(int fd, int version = -1);
// Module stuffs // Module stuffs

View File

@ -149,18 +149,6 @@ void magisk_log_write(int prio, const char *msg, int len) {
} }
} }
void magisk_logging() {
rust::magisk_logging();
forward_logging_to_rs();
exit_on_error(false);
}
void android_logging() {
rust::android_logging();
forward_logging_to_rs();
exit_on_error(false);
}
void start_log_daemon() { void start_log_daemon() {
int fds[2]; int fds[2];
if (pipe2(fds, O_CLOEXEC) == 0) { if (pipe2(fds, O_CLOEXEC) == 0) {

View File

@ -81,7 +81,6 @@ void exec_task(std::function<void()> &&task);
// Logging // Logging
extern std::atomic<int> logd_fd; extern std::atomic<int> logd_fd;
void android_logging();
extern "C" void magisk_log_write(int prio, const char *msg, int len); extern "C" void magisk_log_write(int prio, const char *msg, int len);
// Daemon handlers // Daemon handlers

View File

@ -121,6 +121,15 @@ static FILE *kmsg;
extern "C" void klog_write(const char *msg, int len) { extern "C" void klog_write(const char *msg, int len) {
fprintf(kmsg, "%.*s", len, msg); fprintf(kmsg, "%.*s", len, msg);
} }
static int klog_with_rs(LogLevel level, const char *fmt, va_list ap) {
char buf[4096];
strlcpy(buf, "magiskinit: ", sizeof(buf));
int len = vsnprintf(buf + 12, sizeof(buf) - 12, fmt, ap) + 12;
log_with_rs(level, rust::Str(buf, len));
return len;
}
void setup_klog() { void setup_klog() {
// Shut down first 3 fds // Shut down first 3 fds
int fd; int fd;
@ -148,8 +157,7 @@ void setup_klog() {
kmsg = fdopen(fd, "w"); kmsg = fdopen(fd, "w");
setbuf(kmsg, nullptr); setbuf(kmsg, nullptr);
rust::setup_klog(); rust::setup_klog();
forward_logging_to_rs(); cpp_logger = klog_with_rs;
exit_on_error(false);
// Disable kmsg rate limiting // Disable kmsg rate limiting
if (FILE *rate = fopen("/proc/sys/kernel/printk_devkmsg", "w")) { if (FILE *rate = fopen("/proc/sys/kernel/printk_devkmsg", "w")) {

View File

@ -337,8 +337,7 @@ int resetprop_main(int argc, char *argv[]) {
++argv; ++argv;
} }
if (!verbose) set_log_level_state(LogLevel::Debug, verbose);
log_cb.d = nop_log;
switch (argc) { switch (argc) {
case 0: case 0:

View File

@ -6,8 +6,7 @@ using namespace std;
void sepolicy::magisk_rules() { void sepolicy::magisk_rules() {
// Temp suppress warnings // Temp suppress warnings
auto bak = log_cb.w; set_log_level_state(LogLevel::Warn, false);
log_cb.w = nop_log;
// This indicates API 26+ // This indicates API 26+
bool new_rules = exists("untrusted_app_25"); bool new_rules = exists("untrusted_app_25");
@ -196,5 +195,5 @@ void sepolicy::magisk_rules() {
impl->strip_dontaudit(); impl->strip_dontaudit();
#endif #endif
log_cb.w = bak; set_log_level_state(LogLevel::Warn, true);
} }

View File

@ -17,12 +17,6 @@ using namespace std;
void *self_handle = nullptr; void *self_handle = nullptr;
static void zygisk_logging() {
rust::zygisk_logging();
forward_logging_to_rs();
exit_on_error(false);
}
// Make sure /proc/self/environ is sanitized // Make sure /proc/self/environ is sanitized
// Filter env and reset MM_ENV_END // Filter env and reset MM_ENV_END
static void sanitize_environ() { static void sanitize_environ() {

View File

@ -8,6 +8,7 @@ mod misc;
#[cxx::bridge] #[cxx::bridge]
pub mod ffi { pub mod ffi {
#[derive(Copy, Clone)]
pub enum LogLevel { pub enum LogLevel {
Error, Error,
Warn, Warn,
@ -17,13 +18,8 @@ pub mod ffi {
extern "Rust" { extern "Rust" {
fn log_with_rs(level: LogLevel, msg: &str); fn log_with_rs(level: LogLevel, msg: &str);
}
}
#[cxx::bridge(namespace = "rust")]
pub mod ffi2 {
extern "Rust" {
fn cmdline_logging();
fn exit_on_error(b: bool); fn exit_on_error(b: bool);
fn set_log_level_state(level: LogLevel, enabled: bool);
fn cmdline_logging();
} }
} }

View File

@ -1,64 +1,113 @@
use std::fmt::Arguments; use std::fmt::Arguments;
use std::io::{stderr, stdout, Write};
use std::process::exit; use std::process::exit;
use crate::ffi::LogLevel; use crate::ffi::LogLevel;
// Ugly hack to avoid using enum
#[allow(non_snake_case, non_upper_case_globals)]
mod LogFlag {
pub const DisableError: u32 = 0x1;
pub const DisableWarn: u32 = 0x2;
pub const DisableInfo: u32 = 0x4;
pub const DisableDebug: u32 = 0x8;
pub const ExitOnError: u32 = 0x10;
}
// We don't need to care about thread safety, because all // We don't need to care about thread safety, because all
// logger changes will only happen on the main thread. // logger changes will only happen on the main thread.
pub static mut LOGGER: Logger = Logger { pub static mut LOGGER: Logger = Logger {
d: nop_log, fmt: |_, _| {},
i: nop_log, write: |_, _| {},
w: nop_log, flags: 0,
e: nop_log,
}; };
static mut EXIT_ON_ERROR: bool = false;
#[derive(Copy, Clone)] #[derive(Copy, Clone)]
pub struct Logger { pub struct Logger {
pub d: fn(args: Arguments), pub fmt: fn(level: LogLevel, args: Arguments),
pub i: fn(args: Arguments), pub write: fn(level: LogLevel, msg: &[u8]),
pub w: fn(args: Arguments), pub flags: u32,
pub e: fn(args: Arguments),
}
pub fn nop_log(_: Arguments) {}
pub fn log_with_rs(level: LogLevel, msg: &str) {
log_impl(level, format_args!("{}", msg));
} }
pub fn exit_on_error(b: bool) { pub fn exit_on_error(b: bool) {
unsafe { EXIT_ON_ERROR = b; } unsafe {
if b {
LOGGER.flags |= LogFlag::ExitOnError;
} else {
LOGGER.flags &= !LogFlag::ExitOnError;
}
}
} }
pub fn cmdline_logging() { impl LogLevel {
fn print(args: Arguments) { print!("{}", args); } fn to_disable_flag(&self) -> u32 {
fn eprint(args: Arguments) { eprint!("{}", args); } match *self {
LogLevel::Error => LogFlag::DisableError,
LogLevel::Warn => LogFlag::DisableWarn,
LogLevel::Info => LogFlag::DisableInfo,
LogLevel::Debug => LogFlag::DisableDebug,
_ => 0
}
}
}
let logger = Logger { pub fn set_log_level_state(level: LogLevel, enabled: bool) {
d: eprint, let flag = level.to_disable_flag();
i: print,
w: eprint,
e: eprint,
};
unsafe { unsafe {
LOGGER = logger; if enabled {
EXIT_ON_ERROR = true; LOGGER.flags &= !flag
} else {
LOGGER.flags |= flag
}
}
}
pub fn log_with_rs(level: LogLevel, msg: &str) {
let logger = unsafe { LOGGER };
if (logger.flags & level.to_disable_flag()) != 0 {
return;
}
(logger.write)(level, msg.as_bytes());
if level == LogLevel::Error && (logger.flags & LogFlag::ExitOnError) != 0 {
exit(1);
} }
} }
pub fn log_impl(level: LogLevel, args: Arguments) { pub fn log_impl(level: LogLevel, args: Arguments) {
let logger = unsafe { LOGGER }; let logger = unsafe { LOGGER };
let aoe = unsafe { EXIT_ON_ERROR }; if (logger.flags & level.to_disable_flag()) != 0 {
match level { return;
LogLevel::Error => {
(logger.e)(args);
if aoe { exit(1); }
} }
LogLevel::Warn => (logger.w)(args), (logger.fmt)(level, args);
LogLevel::Info => (logger.i)(args), if level == LogLevel::Error && (logger.flags & LogFlag::ExitOnError) != 0 {
LogLevel::Debug => (logger.d)(args), exit(1);
_ => () }
}
pub fn cmdline_logging() {
fn print(level: LogLevel, args: Arguments) {
if level == LogLevel::Info {
print!("{}", args);
} else {
eprint!("{}", args);
}
}
fn write(level: LogLevel, msg: &[u8]) {
if level == LogLevel::Info {
stdout().write_all(msg).ok();
} else {
stderr().write_all(msg).ok();
}
}
let logger = Logger {
fmt: print,
write,
flags: LogFlag::ExitOnError,
};
unsafe {
LOGGER = logger;
} }
} }

View File

@ -7,12 +7,6 @@ mod logging;
pub mod ffi { pub mod ffi {
extern "Rust" { extern "Rust" {
fn rust_test_entry(); fn rust_test_entry();
}
}
#[cxx::bridge(namespace = "rust")]
pub mod ffi2 {
extern "Rust" {
fn android_logging(); fn android_logging();
fn magisk_logging(); fn magisk_logging();
fn zygisk_logging(); fn zygisk_logging();

View File

@ -1,5 +1,6 @@
use std::fmt::Arguments; use std::fmt::Arguments;
use base::*; use base::*;
use base::ffi::LogLevel;
#[allow(dead_code, non_camel_case_types)] #[allow(dead_code, non_camel_case_types)]
#[repr(i32)] #[repr(i32)]
@ -21,20 +22,34 @@ extern "C" {
fn zygisk_log_write(prio: i32, msg: *const u8, len: i32); fn zygisk_log_write(prio: i32, msg: *const u8, len: i32);
} }
fn level_to_prio(level: LogLevel) -> i32 {
match level {
LogLevel::Error => ALogPriority::ANDROID_LOG_ERROR as i32,
LogLevel::Warn => ALogPriority::ANDROID_LOG_WARN as i32,
LogLevel::Info => ALogPriority::ANDROID_LOG_INFO as i32,
LogLevel::Debug => ALogPriority::ANDROID_LOG_DEBUG as i32,
_ => 0
}
}
pub fn android_logging() { pub fn android_logging() {
fn android_log_impl(prio: i32, args: Arguments) { fn android_log_fmt(level: LogLevel, args: Arguments) {
let mut buf: [u8; 4096] = [0; 4096]; let mut buf: [u8; 4096] = [0; 4096];
fmt_to_buf(&mut buf, args); fmt_to_buf(&mut buf, args);
unsafe { unsafe {
__android_log_write(prio, b"Magisk\0".as_ptr(), buf.as_ptr()); __android_log_write(level_to_prio(level), b"Magisk\0".as_ptr(), buf.as_ptr());
}
}
fn android_log_write(level: LogLevel, msg: &[u8]) {
unsafe {
__android_log_write(level_to_prio(level), b"Magisk\0".as_ptr(), msg.as_ptr());
} }
} }
let logger = Logger { let logger = Logger {
d: |args| { android_log_impl(ALogPriority::ANDROID_LOG_DEBUG as i32, args) }, fmt: android_log_fmt,
i: |args| { android_log_impl(ALogPriority::ANDROID_LOG_INFO as i32, args) }, write: android_log_write,
w: |args| { android_log_impl(ALogPriority::ANDROID_LOG_WARN as i32, args) }, flags: 0,
e: |args| { android_log_impl(ALogPriority::ANDROID_LOG_ERROR as i32, args) }
}; };
exit_on_error(false); exit_on_error(false);
unsafe { unsafe {
@ -43,20 +58,25 @@ pub fn android_logging() {
} }
pub fn magisk_logging() { pub fn magisk_logging() {
fn magisk_log_impl(prio: i32, args: Arguments) { fn magisk_fmt(level: LogLevel, args: Arguments) {
let mut buf: [u8; 4096] = [0; 4096]; let mut buf: [u8; 4096] = [0; 4096];
let len = fmt_to_buf(&mut buf, args); let len = fmt_to_buf(&mut buf, args);
unsafe { unsafe {
__android_log_write(prio, b"Magisk\0".as_ptr(), buf.as_ptr()); __android_log_write(level_to_prio(level), b"Magisk\0".as_ptr(), buf.as_ptr());
magisk_log_write(prio, buf.as_ptr(), len as i32); magisk_log_write(level_to_prio(level), buf.as_ptr(), len as i32);
}
}
fn magisk_write(level: LogLevel, msg: &[u8]) {
unsafe {
__android_log_write(level_to_prio(level), b"Magisk\0".as_ptr(), msg.as_ptr());
magisk_log_write(level_to_prio(level), msg.as_ptr(), msg.len() as i32);
} }
} }
let logger = Logger { let logger = Logger {
d: |args| { magisk_log_impl(ALogPriority::ANDROID_LOG_DEBUG as i32, args) }, fmt: magisk_fmt,
i: |args| { magisk_log_impl(ALogPriority::ANDROID_LOG_INFO as i32, args) }, write: magisk_write,
w: |args| { magisk_log_impl(ALogPriority::ANDROID_LOG_WARN as i32, args) }, flags: 0,
e: |args| { magisk_log_impl(ALogPriority::ANDROID_LOG_ERROR as i32, args) }
}; };
exit_on_error(false); exit_on_error(false);
unsafe { unsafe {
@ -65,20 +85,25 @@ pub fn magisk_logging() {
} }
pub fn zygisk_logging() { pub fn zygisk_logging() {
fn zygisk_log_impl(prio: i32, args: Arguments) { fn zygisk_fmt(level: LogLevel, args: Arguments) {
let mut buf: [u8; 4096] = [0; 4096]; let mut buf: [u8; 4096] = [0; 4096];
let len = fmt_to_buf(&mut buf, args); let len = fmt_to_buf(&mut buf, args);
unsafe { unsafe {
__android_log_write(prio, b"Magisk\0".as_ptr(), buf.as_ptr()); __android_log_write(level_to_prio(level), b"Magisk\0".as_ptr(), buf.as_ptr());
zygisk_log_write(prio, buf.as_ptr(), len as i32); zygisk_log_write(level_to_prio(level), buf.as_ptr(), len as i32);
}
}
fn zygisk_write(level: LogLevel, msg: &[u8]) {
unsafe {
__android_log_write(level_to_prio(level), b"Magisk\0".as_ptr(), msg.as_ptr());
zygisk_log_write(level_to_prio(level), msg.as_ptr(), msg.len() as i32);
} }
} }
let logger = Logger { let logger = Logger {
d: |args| { zygisk_log_impl(ALogPriority::ANDROID_LOG_DEBUG as i32, args) }, fmt: zygisk_fmt,
i: |args| { zygisk_log_impl(ALogPriority::ANDROID_LOG_INFO as i32, args) }, write: zygisk_write,
w: |args| { zygisk_log_impl(ALogPriority::ANDROID_LOG_WARN as i32, args) }, flags: 0,
e: |args| { zygisk_log_impl(ALogPriority::ANDROID_LOG_ERROR as i32, args) }
}; };
exit_on_error(false); exit_on_error(false);
unsafe { unsafe {

View File

@ -1,15 +1,16 @@
use std::fmt::Arguments; use std::fmt::Arguments;
use base::*; use base::*;
use base::ffi::LogLevel;
extern "C" { extern "C" {
fn klog_write(msg: *const u8, len: i32); fn klog_write(msg: *const u8, len: i32);
} }
pub fn setup_klog() { pub fn setup_klog() {
fn klog_impl(args: Arguments) {
const PREFIX: &[u8; 12] = b"magiskinit: "; const PREFIX: &[u8; 12] = b"magiskinit: ";
const PFX_LEN: usize = PREFIX.len(); const PFX_LEN: usize = PREFIX.len();
fn klog_fmt(_: LogLevel, args: Arguments) {
let mut buf: [u8; 4096] = [0; 4096]; let mut buf: [u8; 4096] = [0; 4096];
buf[..PFX_LEN].copy_from_slice(PREFIX); buf[..PFX_LEN].copy_from_slice(PREFIX);
let len = fmt_to_buf(&mut buf[PFX_LEN..], args) + PFX_LEN; let len = fmt_to_buf(&mut buf[PFX_LEN..], args) + PFX_LEN;
@ -18,11 +19,16 @@ pub fn setup_klog() {
} }
} }
fn klog_write_impl(_: LogLevel, msg: &[u8]) {
unsafe {
klog_write(msg.as_ptr(), msg.len() as i32);
}
}
let logger = Logger { let logger = Logger {
d: klog_impl, fmt: klog_fmt,
i: klog_impl, write: klog_write_impl,
w: klog_impl, flags: 0,
e: klog_impl,
}; };
exit_on_error(false); exit_on_error(false);
unsafe { unsafe {