1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
#![cfg_attr(target_os = "none", no_std)]
use core::fmt::Write;
use core::sync::atomic::{AtomicU32, Ordering};
use num_traits::ToPrimitive;
pub mod api;
mod cursor;
#[derive(Debug)]
pub enum LogError {
LoggerExists,
NoConnection,
}
struct XousLogger;
static XOUS_LOGGER: XousLogger = XousLogger {};
static XOUS_LOGGER_CONNECTION: AtomicU32 = AtomicU32::new(0);
impl XousLogger {
fn log_impl(&self, record: &log::Record) {
let mut log_record = api::LogRecord::default();
assert_eq!(core::mem::size_of::<api::LogRecord>(), 4096);
log_record.line = core::num::NonZeroU32::new(record.line().unwrap_or_default());
log_record.level = record.level() as u32;
let file = record.file().unwrap_or_default().as_bytes();
log_record.file_length = file.len().min(file.len()) as u32;
for (dest, src) in log_record.file.iter_mut().zip(file) {
*dest = *src;
}
let module = record.module_path().unwrap_or_default().as_bytes();
log_record.module_length = module.len().min(module.len()) as u32;
for (dest, src) in log_record.module.iter_mut().zip(module) {
*dest = *src;
}
let mut wrapper = cursor::BufferWrapper::new(&mut log_record.args);
write!(wrapper, "{}", record.args()).ok(); log_record.args_length = wrapper.len().min(log_record.args.len()) as u32;
let buf = unsafe {
xous::MemoryRange::new(
&log_record as *const api::LogRecord as usize,
core::mem::size_of::<api::LogRecord>(),
)
.unwrap()
};
xous::send_message(
XOUS_LOGGER_CONNECTION.load(Ordering::Relaxed),
xous::Message::new_lend(
crate::api::Opcode::LogRecord.to_usize().unwrap(),
buf,
None,
None,
),
)
.unwrap();
}
fn resume(&self) {
xous::send_message(
XOUS_LOGGER_CONNECTION.load(Ordering::Relaxed),
xous::Message::new_scalar(2000, 0, 0, 0, 0), )
.expect("couldn't send resume message to the logger implementation");
}
}
impl log::Log for XousLogger {
fn enabled(&self, _metadata: &log::Metadata) -> bool {
true
}
fn log(&self, record: &log::Record) {
XOUS_LOGGER.log_impl(record);
}
fn flush(&self) {}
}
pub fn init() -> Result<(), LogError> {
XOUS_LOGGER_CONNECTION.store(
xous::try_connect(xous::SID::from_bytes(b"xous-log-server ").unwrap())
.or(Err(LogError::NoConnection))?,
Ordering::Relaxed,
);
log::set_logger(&XOUS_LOGGER).map_err(|_| LogError::LoggerExists)?;
log::set_max_level(log::LevelFilter::Info);
Ok(())
}
pub fn init_wait() -> Result<(), ()> {
XOUS_LOGGER_CONNECTION.store(
xous::connect(xous::SID::from_bytes(b"xous-log-server ").unwrap()).or(Err(()))?,
Ordering::Relaxed,
);
log::set_logger(&XOUS_LOGGER).or(Err(()))?;
log::set_max_level(log::LevelFilter::Info);
Ok(())
}
pub fn resume() {
XOUS_LOGGER.resume();
}