console.rs 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. use core::{fmt, ptr::null, str::FromStr};
  2. use log::{Level, LevelFilter};
  3. use rustsbi::{Console, Physical, SbiRet};
  4. use spin::Mutex;
  5. use uart16550::Uart16550;
  6. pub struct ConsoleDevice<'a> {
  7. inner: &'a Mutex<MachineConsole>,
  8. }
  9. impl<'a> ConsoleDevice<'a> {
  10. pub fn new(inner: &'a Mutex<MachineConsole>) -> Self {
  11. Self { inner }
  12. }
  13. }
  14. #[doc(hidden)]
  15. pub(crate) static CONSOLE: Mutex<MachineConsole> = Mutex::new(MachineConsole::Uart16550(null()));
  16. pub fn init(base: usize) {
  17. *CONSOLE.lock() = MachineConsole::Uart16550(base as _);
  18. log_init();
  19. }
  20. impl<'a> Console for ConsoleDevice<'a> {
  21. #[inline]
  22. fn write(&self, bytes: Physical<&[u8]>) -> SbiRet {
  23. // TODO verify valid memory range for a `Physical` slice.
  24. let start = bytes.phys_addr_lo();
  25. let buf = unsafe { core::slice::from_raw_parts(start as *const u8, bytes.num_bytes()) };
  26. let console = self.inner.lock();
  27. match *console {
  28. MachineConsole::Uart16550(uart16550) => unsafe {
  29. (*uart16550).write(buf);
  30. },
  31. }
  32. drop(console);
  33. SbiRet::success(0)
  34. }
  35. #[inline]
  36. fn read(&self, bytes: Physical<&mut [u8]>) -> SbiRet {
  37. // TODO verify valid memory range for a `Physical` slice.
  38. let start = bytes.phys_addr_lo();
  39. let buf = unsafe { core::slice::from_raw_parts_mut(start as *mut u8, bytes.num_bytes()) };
  40. let console = self.inner.lock();
  41. let bytes_num: usize = match *console {
  42. MachineConsole::Uart16550(uart16550) => unsafe {
  43. (*uart16550).read(buf)
  44. },
  45. };
  46. drop(console);
  47. SbiRet::success(bytes_num)
  48. }
  49. #[inline]
  50. fn write_byte(&self, byte: u8) -> SbiRet {
  51. let console = self.inner.lock();
  52. match *console {
  53. MachineConsole::Uart16550(uart16550) => unsafe {
  54. (*uart16550).write(&[byte]);
  55. },
  56. }
  57. drop(console);
  58. SbiRet::success(0)
  59. }
  60. }
  61. #[doc(hidden)]
  62. pub enum MachineConsole {
  63. Uart16550(*const Uart16550<u8>),
  64. }
  65. impl fmt::Write for MachineConsole {
  66. #[inline]
  67. fn write_str(&mut self, s: &str) -> fmt::Result {
  68. let mut bytes = s.as_bytes();
  69. match self {
  70. Self::Uart16550(uart16550) => {
  71. while !bytes.is_empty() {
  72. let count = unsafe { &**uart16550 }.write(bytes);
  73. bytes = &bytes[count..];
  74. }
  75. }
  76. }
  77. Ok(())
  78. }
  79. }
  80. unsafe impl Send for MachineConsole {}
  81. unsafe impl Sync for MachineConsole {}
  82. pub fn log_init() {
  83. log::set_max_level(
  84. option_env!("RUST_LOG")
  85. .and_then(|s| LevelFilter::from_str(s).ok())
  86. .unwrap_or(LevelFilter::Info),
  87. );
  88. log::set_logger(&Logger).unwrap();
  89. }
  90. struct Logger;
  91. impl log::Log for Logger {
  92. #[inline]
  93. fn enabled(&self, _metadata: &log::Metadata) -> bool {
  94. true
  95. }
  96. #[inline]
  97. fn log(&self, record: &log::Record) {
  98. let color_code: u8 = match record.level() {
  99. Level::Error => 31,
  100. Level::Warn => 93,
  101. Level::Info => 34,
  102. Level::Debug => 32,
  103. Level::Trace => 90,
  104. };
  105. println!(
  106. "\x1b[{color_code}m[{:>5}] {}\x1b[0m",
  107. record.level(),
  108. record.args(),
  109. );
  110. }
  111. fn flush(&self) {}
  112. }
  113. // pub fn load_console_uart16550(uart16550: &Uart16550<u8>) {
  114. // let mut console = CONSOLE.lock();
  115. // *console = MachineConsole::Uart16550(uart16550);
  116. // drop(console);
  117. // }