Files
Magisk/native/src/base/logging.rs

164 lines
3.9 KiB
Rust
Raw Normal View History

2023-05-05 23:57:34 -07:00
use std::fmt::{Arguments, Display};
2022-07-06 01:16:08 -07:00
use std::io::{stderr, stdout, Write};
2022-07-01 04:53:41 -07:00
use std::process::exit;
use crate::ffi::LogLevel;
2023-05-29 01:27:40 -07:00
use crate::fmt_to_buf;
2022-07-01 04:53:41 -07:00
2022-07-06 01:16:08 -07:00
// Ugly hack to avoid using enum
#[allow(non_snake_case, non_upper_case_globals)]
mod LogFlag {
2022-08-08 22:53:37 -07:00
pub const DisableError: u32 = 1 << 0;
pub const DisableWarn: u32 = 1 << 1;
pub const DisableInfo: u32 = 1 << 2;
pub const DisableDebug: u32 = 1 << 3;
pub const ExitOnError: u32 = 1 << 4;
2022-07-06 01:16:08 -07:00
}
2022-07-01 04:53:41 -07:00
// We don't need to care about thread safety, because all
// logger changes will only happen on the main thread.
pub static mut LOGGER: Logger = Logger {
2022-07-06 01:16:08 -07:00
write: |_, _| {},
flags: 0,
2022-07-01 04:53:41 -07:00
};
#[derive(Copy, Clone)]
pub struct Logger {
2022-07-06 01:16:08 -07:00
pub write: fn(level: LogLevel, msg: &[u8]),
pub flags: u32,
}
pub fn exit_on_error(b: bool) {
unsafe {
if b {
LOGGER.flags |= LogFlag::ExitOnError;
} else {
LOGGER.flags &= !LogFlag::ExitOnError;
}
}
}
impl LogLevel {
2023-05-30 22:23:11 -07:00
fn as_disable_flag(&self) -> u32 {
2022-07-06 01:16:08 -07:00
match *self {
LogLevel::Error => LogFlag::DisableError,
LogLevel::Warn => LogFlag::DisableWarn,
LogLevel::Info => LogFlag::DisableInfo,
LogLevel::Debug => LogFlag::DisableDebug,
2022-07-22 03:53:50 -07:00
_ => 0,
2022-07-06 01:16:08 -07:00
}
}
2022-07-01 04:53:41 -07:00
}
2022-07-06 01:16:08 -07:00
pub fn set_log_level_state(level: LogLevel, enabled: bool) {
2023-05-30 22:23:11 -07:00
let flag = level.as_disable_flag();
2022-07-06 01:16:08 -07:00
unsafe {
if enabled {
LOGGER.flags &= !flag
} else {
LOGGER.flags |= flag
}
}
}
2022-07-01 04:53:41 -07:00
2023-05-09 19:14:08 -07:00
pub fn log_with_rs(level: LogLevel, msg: &[u8]) {
2022-07-06 01:16:08 -07:00
let logger = unsafe { LOGGER };
2023-05-30 22:23:11 -07:00
if (logger.flags & level.as_disable_flag()) != 0 {
2022-07-06 01:16:08 -07:00
return;
}
2023-05-09 19:14:08 -07:00
(logger.write)(level, msg);
2022-07-06 01:16:08 -07:00
if level == LogLevel::Error && (logger.flags & LogFlag::ExitOnError) != 0 {
exit(1);
}
2022-07-01 04:53:41 -07:00
}
2022-07-06 01:16:08 -07:00
pub fn log_impl(level: LogLevel, args: Arguments) {
let logger = unsafe { LOGGER };
2023-05-30 22:23:11 -07:00
if (logger.flags & level.as_disable_flag()) != 0 {
2022-07-06 01:16:08 -07:00
return;
}
2023-05-29 01:27:40 -07:00
let mut buf: [u8; 4096] = [0; 4096];
let len = fmt_to_buf(&mut buf, args);
(logger.write)(level, &buf[..len]);
2022-07-06 01:16:08 -07:00
if level == LogLevel::Error && (logger.flags & LogFlag::ExitOnError) != 0 {
exit(1);
}
2022-07-01 04:53:41 -07:00
}
pub fn cmdline_logging() {
2022-08-19 02:21:52 -07:00
fn cmdline_write(level: LogLevel, msg: &[u8]) {
2022-07-06 01:16:08 -07:00
if level == LogLevel::Info {
stdout().write_all(msg).ok();
} else {
stderr().write_all(msg).ok();
}
}
2022-07-05 21:13:09 -07:00
2022-07-01 04:53:41 -07:00
let logger = Logger {
2022-08-19 02:21:52 -07:00
write: cmdline_write,
2022-07-06 01:16:08 -07:00
flags: LogFlag::ExitOnError,
2022-07-01 04:53:41 -07:00
};
unsafe {
LOGGER = logger;
}
}
2022-08-08 22:53:37 -07:00
#[macro_export]
macro_rules! perror {
($fmt:expr) => {
$crate::log_impl($crate::ffi::LogLevel::Error, format_args_nl!(
concat!($fmt, " failed with {}: {}"),
2023-05-30 22:23:11 -07:00
$crate::errno(),
$crate::error_str()
2022-08-08 22:53:37 -07:00
))
};
($fmt:expr, $($args:tt)*) => {
$crate::log_impl($crate::ffi::LogLevel::Error, format_args_nl!(
concat!($fmt, " failed with {}: {}"),
$($args)*,
2023-05-30 22:23:11 -07:00
$crate::errno(),
$crate::error_str()
2022-08-08 22:53:37 -07:00
))
};
}
2022-07-01 04:53:41 -07:00
#[macro_export]
macro_rules! error {
2022-08-08 22:53:37 -07:00
($($args:tt)+) => ($crate::log_impl($crate::ffi::LogLevel::Error, format_args_nl!($($args)+)))
2022-07-01 04:53:41 -07:00
}
#[macro_export]
macro_rules! warn {
2022-08-08 22:53:37 -07:00
($($args:tt)+) => ($crate::log_impl($crate::ffi::LogLevel::Warn, format_args_nl!($($args)+)))
2022-07-01 04:53:41 -07:00
}
#[macro_export]
macro_rules! info {
2022-08-08 22:53:37 -07:00
($($args:tt)+) => ($crate::log_impl($crate::ffi::LogLevel::Info, format_args_nl!($($args)+)))
2022-07-01 04:53:41 -07:00
}
#[cfg(debug_assertions)]
#[macro_export]
macro_rules! debug {
2022-08-08 22:53:37 -07:00
($($args:tt)+) => ($crate::log_impl($crate::ffi::LogLevel::Debug, format_args_nl!($($args)+)))
2022-07-01 04:53:41 -07:00
}
#[cfg(not(debug_assertions))]
#[macro_export]
macro_rules! debug {
2022-08-08 22:53:37 -07:00
($($args:tt)+) => {};
2022-07-01 04:53:41 -07:00
}
2023-05-05 23:57:34 -07:00
pub trait ResultExt {
fn log(self) -> Self;
2023-05-05 23:57:34 -07:00
}
impl<T, E: Display> ResultExt for Result<T, E> {
fn log(self) -> Self {
if let Err(e) = &self {
error!("{:#}", e);
2023-05-05 23:57:34 -07:00
}
self
}
}