mirror of https://github.com/m-labs/artiq
71 lines
2.1 KiB
Rust
71 lines
2.1 KiB
Rust
use core::{mem, ptr};
|
|
use core::cell::RefCell;
|
|
use log::{self, Log, LogLevel, LogMetadata, LogRecord, LogLevelFilter};
|
|
use log_buffer::LogBuffer;
|
|
use clock;
|
|
|
|
pub struct BufferLogger {
|
|
buffer: RefCell<LogBuffer<&'static mut [u8]>>
|
|
}
|
|
|
|
unsafe impl Sync for BufferLogger {}
|
|
|
|
static mut LOGGER: *const BufferLogger = ptr::null();
|
|
|
|
impl BufferLogger {
|
|
pub fn new(buffer: &'static mut [u8]) -> BufferLogger {
|
|
BufferLogger {
|
|
buffer: RefCell::new(LogBuffer::new(buffer))
|
|
}
|
|
}
|
|
|
|
pub fn register<F: FnOnce()>(&self, f: F) {
|
|
// log::set_logger_raw captures a pointer to ourselves, so we must prevent
|
|
// ourselves from being moved or dropped after that function is called (and
|
|
// before log::shutdown_logger_raw is called).
|
|
unsafe {
|
|
log::set_logger_raw(|max_log_level| {
|
|
max_log_level.set(LogLevelFilter::Trace);
|
|
self as *const Log
|
|
}).expect("global logger can only be initialized once");
|
|
LOGGER = self;
|
|
}
|
|
f();
|
|
log::shutdown_logger_raw().unwrap();
|
|
unsafe {
|
|
LOGGER = ptr::null();
|
|
}
|
|
}
|
|
|
|
pub fn with_instance<R, F: FnOnce(&BufferLogger) -> R>(f: F) -> R {
|
|
f(unsafe { mem::transmute::<*const BufferLogger, &BufferLogger>(LOGGER) })
|
|
}
|
|
|
|
pub fn clear(&self) {
|
|
self.buffer.borrow_mut().clear()
|
|
}
|
|
|
|
pub fn extract<R, F: FnOnce(&str) -> R>(&self, f: F) -> R {
|
|
f(self.buffer.borrow_mut().extract())
|
|
}
|
|
}
|
|
|
|
impl Log for BufferLogger {
|
|
fn enabled(&self, _metadata: &LogMetadata) -> bool {
|
|
true
|
|
}
|
|
|
|
fn log(&self, record: &LogRecord) {
|
|
if self.enabled(record.metadata()) {
|
|
use core::fmt::Write;
|
|
writeln!(self.buffer.borrow_mut(),
|
|
"[{:12}us] {:>5}({}): {}",
|
|
clock::get_us(), record.level(), record.target(), record.args()).unwrap();
|
|
if record.level() <= LogLevel::Info {
|
|
println!("[{:12}us] {:>5}({}): {}",
|
|
clock::get_us(), record.level(), record.target(), record.args());
|
|
}
|
|
}
|
|
}
|
|
}
|