Browse Source

feat: use compatible prop to choose serial driver

Signed-off-by: Woshiluo Luo <woshiluo.luo@outlook.com>
Woshiluo Luo 4 months ago
parent
commit
7f9b4d25ec
4 changed files with 79 additions and 34 deletions
  1. 13 13
      Cargo.lock
  2. 14 13
      prototyper/src/board.rs
  3. 0 2
      prototyper/src/dt.rs
  4. 52 6
      prototyper/src/main.rs

+ 13 - 13
Cargo.lock

@@ -28,9 +28,9 @@ checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
 
 [[package]]
 name = "critical-section"
-version = "1.1.3"
+version = "1.2.0"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f64009896348fc5af4222e9cf7d7d82a95a256c634ebcf61c53e4ea461422242"
+checksum = "790eea4361631c5e7d22598ecd5723ff611904e3344ce8720784c93e3d83d40b"
 
 [[package]]
 name = "dtb-walker"
@@ -95,9 +95,9 @@ checksum = "de96540e0ebde571dc55c73d60ef407c653844e6f9a1e2fdbd40c07b9252d812"
 
 [[package]]
 name = "proc-macro2"
-version = "1.0.87"
+version = "1.0.92"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "b3e4daa0dcf6feba26f985457cdf104d4b4256fc5a09547140f3631bb076b19a"
+checksum = "37d3544b3f2748c54e147655edb5025752e2303145b5aefb3c3ea2c78b973bb0"
 dependencies = [
  "unicode-ident",
 ]
@@ -235,9 +235,9 @@ checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49"
 
 [[package]]
 name = "serde"
-version = "1.0.210"
+version = "1.0.215"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "c8e3592472072e6e22e0a54d5904d9febf8508f65fb8552499a1abc7d1078c3a"
+checksum = "6513c1ad0b11a9376da888e3e0baa0077f1aed55c17f50e7b2397136129fb88f"
 dependencies = [
  "serde_derive",
 ]
@@ -245,16 +245,16 @@ dependencies = [
 [[package]]
 name = "serde-device-tree"
 version = "0.0.1"
-source = "git+https://github.com/rustsbi/serde-device-tree#1c65f313c99ce73cfbfc954b1a58bcbcd885658c"
+source = "git+https://github.com/rustsbi/serde-device-tree#1ca431a23b70f9209caa3f72986f892508c17a69"
 dependencies = [
  "serde",
 ]
 
 [[package]]
 name = "serde_derive"
-version = "1.0.210"
+version = "1.0.215"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "243902eda00fad750862fc144cea25caca5e20d615af0a81bee94ca738f1df1f"
+checksum = "ad1e866f866923f252f05c889987993144fb74e722403468a4ebd70c3cd756c0"
 dependencies = [
  "proc-macro2",
  "quote",
@@ -278,9 +278,9 @@ dependencies = [
 
 [[package]]
 name = "syn"
-version = "2.0.79"
+version = "2.0.89"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "89132cd0bf050864e1d38dc3bbc07a0eb8e7530af26344d3d2bbbef83499f590"
+checksum = "44d46482f1c1c87acd84dea20c1bf5ebff4c757009ed6bf19cfd36fb10e92c4e"
 dependencies = [
  "proc-macro2",
  "quote",
@@ -304,9 +304,9 @@ dependencies = [
 
 [[package]]
 name = "unicode-ident"
-version = "1.0.13"
+version = "1.0.14"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "e91b56cd4cadaeb79bbf1a5645f6b4f8dc5bde8834ad5894a8db35fda9efa1fe"
+checksum = "adb9e6ca4f869e1180728b7950e35922a7fc6397f7b641499e8f3ef06e50dc83"
 
 [[package]]
 name = "vcell"

+ 14 - 13
prototyper/src/board.rs

@@ -1,7 +1,7 @@
 use aclint::SifiveClint;
 use core::{
     ops::Range,
-    ptr::{null, null_mut},
+    ptr::null_mut,
     sync::atomic::{AtomicPtr, Ordering::Release},
 };
 use sifive_test_device::SifiveTestDevice;
@@ -16,7 +16,7 @@ use crate::sbi::Sbi;
 
 pub struct Device {
     pub memory_range: Option<Range<usize>>,
-    pub uart: Mutex<MachineConsole>,
+    pub uart: Option<Mutex<MachineConsole>>,
     pub sifive_test: AtomicPtr<SifiveTestDevice>,
     pub sifive_clint: AtomicPtr<SifiveClint>,
 }
@@ -29,10 +29,7 @@ pub struct Board<'a> {
 pub(crate) static mut BOARD: Board<'static> = Board {
     device: Device {
         memory_range: None,
-        #[cfg(feature = "nemu")]
-        uart: Mutex::new(MachineConsole::UartAxiLite(MmioUartAxiLite::new(0))),
-        #[cfg(not(feature = "nemu"))]
-        uart: Mutex::new(MachineConsole::Uart16550(null())),
+        uart: None,
         sifive_test: AtomicPtr::new(null_mut()),
         sifive_clint: AtomicPtr::new(null_mut()),
     },
@@ -48,6 +45,12 @@ pub(crate) static mut BOARD: Board<'static> = Board {
 /// Console Device: Uart16550
 #[doc(hidden)]
 #[allow(unused)]
+pub enum MachineConsoleType {
+    Uart16550,
+    UartAxiLite,
+}
+#[doc(hidden)]
+#[allow(unused)]
 pub enum MachineConsole {
     Uart16550(*const Uart16550<u8>),
     UartAxiLite(MmioUartAxiLite),
@@ -72,15 +75,13 @@ impl ConsoleDevice for MachineConsole {
     }
 }
 
-// TODO: select driver follow fdt
-
 #[doc(hidden)]
-pub(crate) fn console_dev_init(base: usize) {
-    let new_console = match *unsafe { BOARD.device.uart.lock() } {
-        MachineConsole::Uart16550(_) => MachineConsole::Uart16550(base as _),
-        MachineConsole::UartAxiLite(_) => MachineConsole::UartAxiLite(MmioUartAxiLite::new(base)),
+pub(crate) fn console_dev_init(console_type: MachineConsoleType, base: usize) {
+    let new_console = match console_type {
+        MachineConsoleType::Uart16550 => MachineConsole::Uart16550(base as _),
+        MachineConsoleType::UartAxiLite => MachineConsole::UartAxiLite(MmioUartAxiLite::new(base)),
     };
-    *unsafe { BOARD.device.uart.lock() } = new_console;
+    unsafe { BOARD.device.uart = Some(Mutex::new(new_console)) };
 }
 
 /// Ipi Device: Sifive Clint

+ 0 - 2
prototyper/src/dt.rs

@@ -48,8 +48,6 @@ pub struct Cpu<'a> {
 /// System-on-chip components.
 #[derive(Deserialize, Debug)]
 pub struct Soc<'a> {
-    /// Serial (UART) device nodes.
-    pub serial: Option<NodeSeq<'a>>,
     /// Test device nodes.
     pub test: Option<NodeSeq<'a>>,
     /// CLINT (Core Local Interruptor) nodes.

+ 52 - 6
prototyper/src/main.rs

@@ -20,7 +20,7 @@ use core::sync::atomic::{AtomicBool, Ordering};
 
 use sbi::extensions;
 
-use crate::board::BOARD;
+use crate::board::{MachineConsoleType, BOARD};
 use crate::riscv_spec::{current_hartid, menvcfg};
 use crate::sbi::console::SbiConsole;
 use crate::sbi::extensions::{hart_extension_probe, Extension};
@@ -53,14 +53,62 @@ extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
         let dtb = dtb.share();
 
         // TODO: should remove `fail:device_tree_deserialize`.
+        let root: serde_device_tree::buildin::Node = serde_device_tree::from_raw_mut(&dtb).unwrap();
         let tree =
             serde_device_tree::from_raw_mut(&dtb).unwrap_or_else(fail::device_tree_deserialize);
         // 2. Init device
         // TODO: The device base address should be find in a better way.
-        let console_base = tree.soc.serial.unwrap().iter().next().unwrap();
+        'console_finder: for console_path in tree.chosen.stdout_path.iter() {
+            if let Some(node) = root.find(console_path) {
+                let compatible = node
+                    .props()
+                    .map(|mut prop_iter| {
+                        prop_iter
+                            .find(|prop_item| prop_item.get_name() == "compatible")
+                            .map(|prop_item| {
+                                prop_item.deserialize::<serde_device_tree::buildin::StrSeq>()
+                            })
+                    })
+                    .map_or_else(|| None, |v| v);
+                let regs = node
+                    .props()
+                    .map(|mut prop_iter| {
+                        prop_iter
+                            .find(|prop_item| prop_item.get_name() == "reg")
+                            .map(|prop_item| {
+                                let reg =
+                                    prop_item.deserialize::<serde_device_tree::buildin::Reg>();
+                                if let Some(range) = reg.iter().next() {
+                                    return Some(range);
+                                }
+                                None
+                            })
+                            .map_or_else(|| None, |v| v)
+                    })
+                    .map_or_else(|| None, |v| v);
+                if compatible.is_some() && regs.is_some() {
+                    for device_id in compatible.unwrap().iter() {
+                        if device_id == "ns16550a" {
+                            board::console_dev_init(
+                                MachineConsoleType::Uart16550,
+                                regs.unwrap().0.start,
+                            );
+                            break 'console_finder;
+                        }
+                        if device_id == "xlnx,xps-uartlite-1.00.a" {
+                            board::console_dev_init(
+                                MachineConsoleType::UartAxiLite,
+                                regs.unwrap().0.start,
+                            );
+                            break 'console_finder;
+                        }
+                    }
+                }
+            }
+        }
+
         let clint_device = tree.soc.clint.unwrap().iter().next().unwrap();
         let cpu_num = tree.cpus.cpu.len();
-        let console_base_address = console_base.at();
         let ipi_base_address = clint_device.at();
 
         // Initialize reset device if present.
@@ -71,7 +119,6 @@ extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
         }
 
         // Initialize console and IPI devices.
-        board::console_dev_init(usize::from_str_radix(console_base_address, 16).unwrap());
         board::ipi_dev_init(usize::from_str_radix(ipi_base_address, 16).unwrap());
 
         // 3. Init the SBI implementation
@@ -89,7 +136,7 @@ extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
         unsafe {
             BOARD.device.memory_range = Some(memory_range);
             BOARD.sbi = Sbi {
-                console: Some(SbiConsole::new(&BOARD.device.uart)),
+                console: Some(SbiConsole::new(BOARD.device.uart.as_ref().unwrap())),
                 ipi: Some(SbiIpi::new(&BOARD.device.sifive_clint, cpu_num)),
                 hsm: Some(SbiHsm),
                 reset: Some(SbiReset::new(&BOARD.device.sifive_test)),
@@ -114,7 +161,6 @@ extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
             info!("Model: {}", model.iter().next().unwrap_or("<unspecified>"));
         }
         info!("Clint device: {}", ipi_base_address);
-        info!("Console deivce: {}", console_base_address);
         info!(
             "Chosen stdout item: {}",
             tree.chosen