console.rs 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. use alloc::boxed::Box;
  2. use core::fmt::{self, Write};
  3. use rustsbi::{Console, Physical, SbiRet};
  4. use spin::Mutex;
  5. use crate::platform::PLATFORM;
  6. /// A trait that must be implemented by console devices to provide basic I/O functionality.
  7. pub trait ConsoleDevice {
  8. /// Reads bytes from the console into the provided buffer.
  9. ///
  10. /// # Returns
  11. /// The number of bytes that were successfully read.
  12. fn read(&self, buf: &mut [u8]) -> usize;
  13. /// Writes bytes from the provided buffer to the console.
  14. ///
  15. /// # Returns
  16. /// The number of bytes that were successfully written.
  17. fn write(&self, buf: &[u8]) -> usize;
  18. }
  19. /// An implementation of the SBI console interface that wraps a console device.
  20. ///
  21. /// This provides a safe interface for interacting with console hardware through the
  22. /// SBI specification.
  23. pub struct SbiConsole {
  24. inner: Mutex<Box<dyn ConsoleDevice>>,
  25. }
  26. impl SbiConsole {
  27. /// Creates a new SBI console that wraps the provided locked console device.
  28. ///
  29. /// # Arguments
  30. /// * `inner` - A mutex containing the console device implementation
  31. #[inline]
  32. pub fn new(inner: Mutex<Box<dyn ConsoleDevice>>) -> Self {
  33. Self { inner }
  34. }
  35. /// Writes a single character to the console.
  36. ///
  37. /// # Arguments
  38. /// * `c` - The character to write, as a usize
  39. ///
  40. /// # Returns
  41. /// Always returns 0 to indicate success
  42. #[inline]
  43. pub fn putchar(&mut self, c: usize) -> usize {
  44. self.write_char(c as u8 as char).unwrap();
  45. 0
  46. }
  47. /// Reads a single character from the console.
  48. ///
  49. /// This method will block until a character is available to be read.
  50. ///
  51. /// # Returns
  52. /// The read character as a usize
  53. #[inline]
  54. pub fn getchar(&self) -> usize {
  55. let mut c = 0u8;
  56. let console = self.inner.lock();
  57. // Block until we successfully read 1 byte
  58. while console.read(core::slice::from_mut(&mut c)) != 1 {
  59. core::hint::spin_loop();
  60. }
  61. c as usize
  62. }
  63. }
  64. impl Console for SbiConsole {
  65. /// Write a physical memory buffer to the console.
  66. #[inline]
  67. fn write(&self, bytes: Physical<&[u8]>) -> SbiRet {
  68. // TODO: verify valid memory range for a `Physical` slice.
  69. let start = bytes.phys_addr_lo();
  70. let buf = unsafe { core::slice::from_raw_parts(start as *const u8, bytes.num_bytes()) };
  71. let bytes_written = self.inner.lock().write(buf);
  72. SbiRet::success(bytes_written)
  73. }
  74. /// Read from console into a physical memory buffer.
  75. #[inline]
  76. fn read(&self, bytes: Physical<&mut [u8]>) -> SbiRet {
  77. // TODO: verify valid memory range for a `Physical` slice.
  78. let start = bytes.phys_addr_lo();
  79. let buf = unsafe { core::slice::from_raw_parts_mut(start as *mut u8, bytes.num_bytes()) };
  80. let bytes_read = self.inner.lock().read(buf);
  81. SbiRet::success(bytes_read)
  82. }
  83. /// Write a single byte to the console.
  84. #[inline]
  85. fn write_byte(&self, byte: u8) -> SbiRet {
  86. self.inner.lock().write(&[byte]);
  87. SbiRet::success(0)
  88. }
  89. }
  90. impl fmt::Write for SbiConsole {
  91. /// Implement Write trait for string formatting.
  92. #[inline]
  93. fn write_str(&mut self, s: &str) -> fmt::Result {
  94. let mut bytes = s.as_bytes();
  95. let console = self.inner.lock();
  96. // Write all bytes in chunks
  97. while !bytes.is_empty() {
  98. let count = console.write(bytes);
  99. bytes = &bytes[count..];
  100. }
  101. Ok(())
  102. }
  103. }
  104. /// Global function to write a character to the console.
  105. #[inline]
  106. pub fn putchar(c: usize) -> usize {
  107. unsafe { PLATFORM.sbi.console.as_mut().unwrap().putchar(c) }
  108. }
  109. /// Global function to read a character from the console.
  110. #[inline]
  111. pub fn getchar() -> usize {
  112. unsafe { PLATFORM.sbi.console.as_mut().unwrap().getchar() }
  113. }