2023-06-29 16:44:44 -07:00
|
|
|
use std::fmt;
|
2024-03-05 01:48:35 -08:00
|
|
|
use std::fmt::Arguments;
|
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;
|
|
|
|
|
|
2024-03-05 01:48:35 -08:00
|
|
|
use num_derive::{FromPrimitive, ToPrimitive};
|
|
|
|
|
use num_traits::FromPrimitive;
|
|
|
|
|
|
2023-10-13 16:59:54 -07:00
|
|
|
use crate::ffi::LogLevelCxx;
|
2025-02-17 11:32:21 -08:00
|
|
|
use crate::{cstr_buf, Utf8CStr};
|
2023-06-29 16:44:44 -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
|
|
|
}
|
|
|
|
|
|
2023-10-13 16:59:54 -07:00
|
|
|
#[derive(Copy, Clone, FromPrimitive, ToPrimitive)]
|
|
|
|
|
#[repr(i32)]
|
|
|
|
|
pub enum LogLevel {
|
|
|
|
|
ErrorCxx = LogLevelCxx::ErrorCxx.repr,
|
|
|
|
|
Error = LogLevelCxx::Error.repr,
|
|
|
|
|
Warn = LogLevelCxx::Warn.repr,
|
|
|
|
|
Info = LogLevelCxx::Info.repr,
|
|
|
|
|
Debug = LogLevelCxx::Debug.repr,
|
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
};
|
|
|
|
|
|
2023-09-12 17:35:20 -07:00
|
|
|
type LogWriter = fn(level: LogLevel, msg: &Utf8CStr);
|
2024-03-05 01:48:35 -08:00
|
|
|
pub(crate) type Formatter<'a> = &'a mut dyn fmt::Write;
|
2023-09-12 17:35:01 -07:00
|
|
|
|
2022-07-01 04:53:41 -07:00
|
|
|
#[derive(Copy, Clone)]
|
|
|
|
|
pub struct Logger {
|
2023-09-12 17:35:01 -07:00
|
|
|
pub write: LogWriter,
|
2022-07-06 01:16:08 -07:00
|
|
|
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 {
|
2023-06-29 16:44:44 -07:00
|
|
|
LogLevel::Error | LogLevel::ErrorCxx => LogFlag::DisableError,
|
2022-07-06 01:16:08 -07:00
|
|
|
LogLevel::Warn => LogFlag::DisableWarn,
|
|
|
|
|
LogLevel::Info => LogFlag::DisableInfo,
|
|
|
|
|
LogLevel::Debug => LogFlag::DisableDebug,
|
|
|
|
|
}
|
|
|
|
|
}
|
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-09-12 17:35:01 -07:00
|
|
|
fn log_with_writer<F: FnOnce(LogWriter)>(level: LogLevel, f: F) {
|
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-06-29 16:44:44 -07:00
|
|
|
f(logger.write);
|
2023-10-13 16:59:54 -07:00
|
|
|
if matches!(level, LogLevel::ErrorCxx) && (logger.flags & LogFlag::ExitOnError) != 0 {
|
2025-03-04 08:21:55 +08:00
|
|
|
exit(-1);
|
2022-07-06 01:16:08 -07:00
|
|
|
}
|
2022-07-01 04:53:41 -07:00
|
|
|
}
|
|
|
|
|
|
2023-12-26 23:08:06 +08:00
|
|
|
pub fn log_from_cxx(level: LogLevelCxx, msg: &Utf8CStr) {
|
2023-10-13 16:59:54 -07:00
|
|
|
if let Some(level) = LogLevel::from_i32(level.repr) {
|
|
|
|
|
log_with_writer(level, |write| write(level, msg));
|
|
|
|
|
}
|
2023-06-29 16:44:44 -07:00
|
|
|
}
|
|
|
|
|
|
2023-09-12 17:35:01 -07:00
|
|
|
pub fn log_with_formatter<F: FnOnce(Formatter) -> fmt::Result>(level: LogLevel, f: F) {
|
|
|
|
|
log_with_writer(level, |write| {
|
2025-02-17 11:32:21 -08:00
|
|
|
let mut buf = cstr_buf::default();
|
2023-09-12 17:35:01 -07:00
|
|
|
f(&mut buf).ok();
|
2023-09-12 17:35:20 -07:00
|
|
|
write(level, &buf);
|
2023-06-29 16:44:44 -07:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn log_with_args(level: LogLevel, args: Arguments) {
|
|
|
|
|
log_with_formatter(level, |w| w.write_fmt(args));
|
2022-07-01 04:53:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn cmdline_logging() {
|
2023-09-12 17:35:20 -07:00
|
|
|
fn cmdline_write(level: LogLevel, msg: &Utf8CStr) {
|
2023-10-13 16:59:54 -07:00
|
|
|
if matches!(level, LogLevel::Info) {
|
2023-09-12 17:35:20 -07:00
|
|
|
stdout().write_all(msg.as_bytes()).ok();
|
2022-07-06 01:16:08 -07:00
|
|
|
} else {
|
2023-09-12 17:35:20 -07:00
|
|
|
stderr().write_all(msg.as_bytes()).ok();
|
2022-07-06 01:16:08 -07:00
|
|
|
}
|
|
|
|
|
}
|
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;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[macro_export]
|
|
|
|
|
macro_rules! error {
|
2023-06-29 16:44:44 -07:00
|
|
|
($($args:tt)+) => {
|
2023-10-13 16:59:54 -07:00
|
|
|
$crate::log_with_args($crate::LogLevel::Error, format_args_nl!($($args)+))
|
2023-06-29 16:44:44 -07:00
|
|
|
}
|
2022-07-01 04:53:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[macro_export]
|
|
|
|
|
macro_rules! warn {
|
2023-06-29 16:44:44 -07:00
|
|
|
($($args:tt)+) => {
|
2023-10-13 16:59:54 -07:00
|
|
|
$crate::log_with_args($crate::LogLevel::Warn, format_args_nl!($($args)+))
|
2023-06-29 16:44:44 -07:00
|
|
|
}
|
2022-07-01 04:53:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[macro_export]
|
|
|
|
|
macro_rules! info {
|
2023-06-29 16:44:44 -07:00
|
|
|
($($args:tt)+) => {
|
2023-10-13 16:59:54 -07:00
|
|
|
$crate::log_with_args($crate::LogLevel::Info, format_args_nl!($($args)+))
|
2023-06-29 16:44:44 -07:00
|
|
|
}
|
2022-07-01 04:53:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
|
#[macro_export]
|
|
|
|
|
macro_rules! debug {
|
2023-06-29 16:44:44 -07:00
|
|
|
($($args:tt)+) => {
|
2023-10-13 16:59:54 -07:00
|
|
|
$crate::log_with_args($crate::LogLevel::Debug, format_args_nl!($($args)+))
|
2023-06-29 16:44:44 -07:00
|
|
|
}
|
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
|
|
|
}
|