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
111
112
113
114
115
116
117
use crate::serial::Serial;
use core::{
fmt::write,
format_args,
sync::atomic::{AtomicU16, Ordering},
};
use log::*;
const COM1_PORT: u16 = 0x3f8;
static LOGGER: Logger = Logger(AtomicU16::new(COM1_PORT));
struct Logger(AtomicU16);
impl log::Log for Logger {
fn enabled(&self, _m: &Metadata) -> bool {
true
}
fn log(&self, record: &Record) {
let mut serial = Serial::new(self.0.load(Ordering::Relaxed));
let _ = write(
&mut serial,
format_args!(
"{:>8}: {} ({}, {}:{})\n",
record.level(),
record.args(),
record.target(),
record.file().unwrap_or("<unknown>"),
record.line().unwrap_or(0),
),
);
}
fn flush(&self) {}
}
fn set_logger_base(base: u16) {
LOGGER.0.store(base, Ordering::Relaxed);
}
pub struct Builder {
base: u16,
filter: LevelFilter,
}
impl Builder {
pub fn new() -> Self {
Self {
base: COM1_PORT,
filter: LevelFilter::Info,
}
}
pub fn base(mut self, base: u16) -> Self {
self.base = base;
self
}
pub fn filter(mut self, filter: LevelFilter) -> Self {
self.filter = filter;
self
}
pub fn setup(self) {
Serial::new(self.base).init();
set_logger_base(self.base);
set_logger(&LOGGER).unwrap();
set_max_level(self.filter);
}
}
pub fn builder() -> Builder {
Builder::new()
}
pub fn init() {
builder().filter(LevelFilter::Info).setup();
}
pub fn init_with_filter(filter: LevelFilter) {
builder().filter(filter).setup();
}