Эх сурвалжийг харах

feat(arch): impl CPU data structure for riscv64.

Heinz 8 сар өмнө
parent
commit
56083a0e82
1 өөрчлөгдсөн 151 нэмэгдсэн , 0 устгасан
  1. 151 0
      arch/src/riscv64/init.rs

+ 151 - 0
arch/src/riscv64/init.rs

@@ -0,0 +1,151 @@
+use core::pin::Pin;
+use riscv::{asm::sfence_vma_all, register::{
+    mhartid, sscratch, sstatus, stvec::{self, Stvec}
+}};
+use sbi::PhysicalAddress;
+
+/// TODO:
+/// 解析设备树
+/// 
+
+use super::{config::smp::MAX_HART, enable_sse, setup_kernel_page_table, InterruptControl};
+
+/// RISC-V Hart
+pub struct CPU {
+    hart_id: usize,
+    interrupt: InterruptControl,
+}
+
+impl CPU {
+    pub fn new() -> Self {
+        let hart_id = read_hart_id();
+        Self {
+            hart_id: hart_id,
+            interrupt: InterruptControl::new(hart_id),
+        }
+    }
+
+    /// Load CPU specific configurations for the current Hart.
+    ///
+    /// # Safety
+    /// This function performs low-level hardware initialization and should
+    /// only be called once per Hart during its boot sequence.
+    pub unsafe fn init(self: Pin<&mut Self>) {
+        enable_sse();
+        let self_mut = self.get_unchecked_mut();
+
+        sscratch::write(self_mut.hart_id as usize);
+
+        // TODO: in somewhere
+        setup_trap_vector(__trap_handler_entry as usize);
+
+        // CLINT, 10_000 ms
+        self_mut.interrupt.setup_timer(10_000);
+
+        // Supervisor Mode Status Register (sstatus)
+        // SUM (Supervisor User Memory access): support S-mode access user memory
+        // MXR (Make Executable Readable)
+        // SIE (Supervisor Interrupt Enable): enable S-mode interrupt
+        let mut current_sstatus = sstatus::read();
+        current_sstatus.set_spp(sstatus::SPP::Supervisor);
+        current_sstatus.set_sum(true);
+        current_sstatus.set_mxr(true);
+        current_sstatus.set_sie(true);
+        sstatus::write(current_sstatus);
+
+        // setup kernel page table and flush tlb
+        setup_kernel_page_table();
+    }
+
+    /// Boot all other hart.
+    pub unsafe fn bootstrap_cpus(&self) {
+        extern "C" {
+        fn ap_boot_entry();
+        }
+        // TODO: 获取系统中的总 Hart 数量
+        let total_harts = MAX_HART;
+
+        let ap_entry_point = PhysicalAddress::new(ap_boot_entry as usize);
+
+        for i in 1..total_harts {
+            sbi::hsm::hart_start(i, ap_entry_point, 0)
+                .expect("Failed to start secondary hart via SBI");
+        }
+    }
+
+    pub fn cpuid(&self) -> usize {
+        self.hart_id
+    }
+}
+
+fn read_hart_id() -> usize {
+    mhartid::read()
+}
+
+#[macro_export]
+macro_rules! define_smp_bootstrap {
+    ($cpu_count:literal, $ap_entry:ident, $alloc_kstack:tt) => {
+        #[no_mangle]
+        static BOOT_SEMAPHORE: core::sync::atomic::AtomicU64 =
+            core::sync::atomic::AtomicU64::new(0);
+        #[no_mangle]
+        static BOOT_STACK: core::sync::atomic::AtomicU64 =
+            core::sync::atomic::AtomicU64::new(0);
+
+        #[no_mangle]
+        static CPU_COUNT: core::sync::atomic::AtomicU64 =
+            core::sync::atomic::AtomicU64::new(0);
+
+        core::arch::global_asm!(
+            r#"
+        .section .text.ap_boot
+        .globl ap_boot_entry
+
+        ap_boot_entry:
+            csrr a0, mhartid
+
+        1:
+            lw t0, AP_BOOT_STACK.addr
+            beqz t0, 1b
+            li t1, 0
+            sw t1, AP_BOOT_STACK.addr
+            mv sp, t0
+
+        2:
+            lw t0, AP_BOOT_SEMAPHORE.addr
+            beqz t0, 2b
+
+            li t1, 0
+            sw t1, AP_BOOT_SEMAPHORE.addr
+
+            li t0, 1
+            amoswap.w.aq rl t0, a0, ONLINE_HART_COUNT.addr
+
+            call $ap_entry
+            j .
+            "#,
+            BOOT_SEMAPHORE = sym BOOT_SEMAPHORE,
+            BOOT_STACK = sym BOOT_STACK,
+            CPU_COUNT = sym CPU_COUNT,
+            AP_ENTRY = sym $ap_entry,
+        );
+
+        pub unsafe fn wait_cpus_online() {
+            use core::sync::atomic::Ordering;
+            while CPU_COUNT.load(Ordering::Acquire) != $cpu_count - 1 {
+                if BOOT_STACK.load(Ordering::Acquire) == 0 {
+                    let stack_bottom = $alloc_kstack as u64;
+                    BOOT_STACK.store(stack_bottom, Ordering::Release);
+                }
+                $crate::pause();
+            }
+        }
+    };
+}
+
+fn setup_trap_vector(trap_entry_addr: usize) {
+    unsafe {
+        stvec::write(Stvec::from_bits(trap_entry_addr));
+    }
+    sfence_vma_all();
+}