Ver código fonte

Merge pull request #14 from guttatus/bootflow

refactor(prototyper): remove unnecessary hsm status acquisition
guttatus 6 meses atrás
pai
commit
b85ed38e17

+ 6 - 5
prototyper/src/clint.rs

@@ -8,8 +8,8 @@ use core::{
 };
 use rustsbi::SbiRet;
 
-use crate::riscv_spec::{stimecmp,current_hartid};
 use crate::hsm::remote_hsm;
+use crate::riscv_spec::{current_hartid, stimecmp};
 
 pub(crate) static SIFIVECLINT: AtomicPtr<SifiveClint> = AtomicPtr::new(null_mut());
 
@@ -69,7 +69,7 @@ impl<'a> ClintDevice<'a> {
 impl<'a> ClintDevice<'a> {
     #[inline]
     pub fn get_time(&self) -> usize {
-        unsafe { (*self.clint.load(Relaxed)).read_mtime() as u32 as usize }
+        unsafe { (*self.clint.load(Relaxed)).read_mtime() as usize }
     }
 
     #[inline]
@@ -85,7 +85,6 @@ impl<'a> rustsbi::Timer for ClintDevice<'a> {
             // TODO: 添加CPU拓展探测机制,补充无Sstc拓展时的定时器设置
             stimecmp::set(stime_value);
             riscv::register::mie::set_mtimer();
-            // (*self.clint.load(Relaxed)).write_mtimecmp(current_hart_id, stime_value);
         }
     }
 }
@@ -95,8 +94,10 @@ impl<'a> rustsbi::Ipi for ClintDevice<'a> {
     fn send_ipi(&self, hart_mask: rustsbi::HartMask) -> SbiRet {
         for hart_id in 0..=self.max_hart_id {
             if hart_mask.has_bit(hart_id) && remote_hsm(hart_id).unwrap().allow_ipi() {
-                unsafe {
-                    (*self.clint.load(Relaxed)).set_msip(hart_id);
+                if hart_id == current_hartid() {
+                    unsafe { riscv::register::mip::set_ssoft(); }
+                } else {
+                    unsafe { (*self.clint.load(Relaxed)).set_msip(hart_id); }
                 }
             }
         }

+ 21 - 6
prototyper/src/console.rs

@@ -60,6 +60,27 @@ impl<'a> Console for ConsoleDevice<'a> {
     }
 }
 
+impl<'a> ConsoleDevice<'a> {
+    #[inline]
+    pub fn putchar(&self, c: usize) -> usize {
+        print!("{}", c as u8 as char);
+        0
+    }
+
+    #[inline]
+    pub fn getchar(&self) -> usize {
+        let mut c = 0u8;
+        let console = self.inner.lock();
+        match *console {
+            MachineConsole::Uart16550(uart16550) => unsafe {
+                loop { if (*uart16550).read(core::slice::from_mut(&mut c)) == 1 { break }}
+            }
+        };
+        drop(console);
+        c as _
+    }
+}
+
 #[doc(hidden)]
 pub enum MachineConsole {
     Uart16550(*const Uart16550<u8>),
@@ -119,9 +140,3 @@ impl log::Log for Logger {
 
     fn flush(&self) {}
 }
-
-// pub fn load_console_uart16550(uart16550: &Uart16550<u8>) {
-//     let mut console = CONSOLE.lock();
-//     *console = MachineConsole::Uart16550(uart16550);
-//     drop(console);
-// }

+ 6 - 1
prototyper/src/hsm.rs

@@ -201,6 +201,9 @@ impl rustsbi::Hsm for Hsm {
     #[inline]
     fn hart_stop(&self) -> SbiRet {
         local_hsm().stop();
+        clint::clear_msip();
+        unsafe { riscv::register::mie::clear_msoft(); }
+        riscv::asm::wfi();
         SbiRet::success(0)
     }
 
@@ -216,7 +219,9 @@ impl rustsbi::Hsm for Hsm {
         use rustsbi::spec::hsm::suspend_type::{NON_RETENTIVE, RETENTIVE};
         if matches!(suspend_type, NON_RETENTIVE | RETENTIVE) {
             local_hsm().suspend();
-            riscv::asm::wfi() ;
+            clint::clear_msip();
+            unsafe { riscv::register::mie::set_msoft(); }
+            riscv::asm::wfi();
             local_hsm().resume();
             SbiRet::success(0)
         } else {

+ 2 - 1
prototyper/src/main.rs

@@ -184,11 +184,12 @@ unsafe extern "C" fn start() -> ! {
          // 3. Prepare stack for each hart
         "   call    {locate_stack}",
         "   call    {main}",
+        "   csrw    mscratch, sp",
         "   j       {hart_boot}",
         magic = const dynamic::MAGIC,
         locate_stack = sym trap_stack::locate,
         main         = sym rust_main,
-        hart_boot         = sym trap::trap_vec,
+        hart_boot         = sym trap::msoft,
         options(noreturn)
     )
 }

+ 74 - 108
prototyper/src/trap.rs

@@ -9,10 +9,9 @@ use rustsbi::RustSBI;
 
 use crate::board::SBI;
 use crate::clint::{self, SIFIVECLINT};
-use crate::console::{MachineConsole, CONSOLE};
 use crate::hsm::local_hsm;
 use crate::rfence::{local_rfence, RFenceType};
-use crate::riscv_spec::{CSR_TIME, CSR_TIMEH, current_hartid};
+use crate::riscv_spec::{current_hartid, CSR_TIME, CSR_TIMEH};
 
 const PAGE_SIZE: usize = 4096;
 // TODO: `TLB_FLUSH_LIMIT` is a platform-dependent parameter
@@ -98,7 +97,7 @@ unsafe extern "C" fn mtimer() {
 /// machine soft 中断代理
 ///
 #[naked]
-unsafe extern "C" fn msoft() -> ! {
+pub unsafe extern "C" fn msoft() -> ! {
     asm!(
         ".align 2",
         "csrrw  sp, mscratch, sp",
@@ -180,7 +179,7 @@ unsafe extern "C" fn msoft() -> ! {
 }
 
 pub extern "C" fn msoft_hanlder(ctx: &mut SupervisorContext) {
-    #[inline]
+    #[inline(always)]
     fn boot(ctx: &mut SupervisorContext, start_addr: usize, opaque: usize) {
         unsafe {
             sstatus::clear_sie();
@@ -205,9 +204,7 @@ pub extern "C" fn msoft_hanlder(ctx: &mut SupervisorContext) {
         }
         Err(rustsbi::spec::hsm::HART_STOP) => {
             clint::clear_msip();
-            unsafe {
-                mie::set_msoft();
-            }
+            unsafe { mie::set_msoft(); }
             riscv::asm::wfi();
         }
         // RFence
@@ -228,7 +225,10 @@ pub fn msoft_rfence_handler() {
             },
             RFenceType::SFenceVma => {
                 // If the flush size is greater than the maximum limit then simply flush all
-                if (ctx.start_addr == 0 && ctx.size == 0) || (ctx.size == usize::MAX) || (ctx.size > TLB_FLUSH_LIMIT) {
+                if (ctx.start_addr == 0 && ctx.size == 0)
+                    || (ctx.size == usize::MAX)
+                    || (ctx.size > TLB_FLUSH_LIMIT)
+                {
                     unsafe { asm!("sfence.vma"); }
                 } else {
                     for offset in (0..ctx.size).step_by(PAGE_SIZE) {
@@ -242,7 +242,10 @@ pub fn msoft_rfence_handler() {
             RFenceType::SFenceVmaAsid => {
                 let asid = ctx.asid;
                 // If the flush size is greater than the maximum limit then simply flush all
-                if (ctx.start_addr == 0 && ctx.size == 0) || (ctx.size == usize::MAX) || (ctx.size > TLB_FLUSH_LIMIT) {
+                if (ctx.start_addr == 0 && ctx.size == 0)
+                    || (ctx.size == usize::MAX)
+                    || (ctx.size > TLB_FLUSH_LIMIT)
+                {
                     unsafe { asm!("sfence.vma {}, {}", in(reg) 0, in(reg) asid); }
                 } else {
                     for offset in (0..ctx.size).step_by(PAGE_SIZE) {
@@ -280,7 +283,7 @@ pub extern "C" fn fast_handler(
     a7: usize,
 ) -> FastResult {
     #[inline]
-    fn boot(mut ctx: FastContext, start_addr: usize, opaque: usize) -> FastResult {
+    fn resume(mut ctx: FastContext, start_addr: usize, opaque: usize) -> FastResult {
         unsafe {
             sstatus::clear_sie();
             satp::write(0);
@@ -290,113 +293,76 @@ pub extern "C" fn fast_handler(
         ctx.regs().pc = start_addr;
         ctx.call(2)
     }
-    loop {
-        match local_hsm().start() {
-            Ok(next_stage) => {
-                clint::clear_msip();
-                unsafe {
-                    mstatus::set_mpie();
-                    mstatus::set_mpp(next_stage.next_mode);
-                    mie::set_msoft();
-                    mie::set_mtimer();
-                }
-                break boot(ctx, next_stage.start_addr, next_stage.opaque);
-            }
-            Err(rustsbi::spec::hsm::HART_STOP) => {
-                clint::clear_msip();
-                unsafe {
-                    mie::set_msoft();
-                }
-                riscv::asm::wfi();
-            }
-            _ => match mcause::read().cause() {
-                // SBI call
-                T::Exception(E::SupervisorEnvCall) => {
-                    use sbi_spec::{base, hsm, legacy};
-                    let mut ret = unsafe { SBI.assume_init_mut() }.handle_ecall(
-                        a7,
-                        a6,
-                        [ctx.a0(), a1, a2, a3, a4, a5],
-                    );
-                    if ret.is_ok() {
-                        match (a7, a6) {
-                            // 关闭
-                            (hsm::EID_HSM, hsm::HART_STOP) => continue,
-                            // 不可恢复挂起
-                            (hsm::EID_HSM, hsm::HART_SUSPEND)
-                                if matches!(ctx.a0() as u32, hsm::suspend_type::NON_RETENTIVE) =>
-                            {
-                                break boot(ctx, a1, a2);
-                            }
-                            // legacy console 探测
-                            (base::EID_BASE, base::PROBE_EXTENSION)
-                                if matches!(
-                                    ctx.a0(),
-                                    legacy::LEGACY_CONSOLE_PUTCHAR | legacy::LEGACY_CONSOLE_GETCHAR
-                                ) =>
-                            {
-                                ret.value = 1;
-                            }
-                            _ => {}
-                        }
-                    } else {
-                        match a7 {
-                            legacy::LEGACY_CONSOLE_PUTCHAR => {
-                                print!("{}", ctx.a0() as u8 as char);
-                                ret.error = 0;
-                                ret.value = a1;
-                            }
-                            legacy::LEGACY_CONSOLE_GETCHAR => {
-                                let mut c = 0u8;
-                                let uart = CONSOLE.lock();
-                                match *uart {
-                                    MachineConsole::Uart16550(uart16550) => unsafe {
-                                        loop {
-                                            if (*uart16550).read(core::slice::from_mut(&mut c)) == 1
-                                            {
-                                                ret.error = c as _;
-                                                ret.value = a1;
-                                                break;
-                                            }
-                                        }
-                                    },
-                                }
-                                drop(uart);
-                            }
-                            _ => {}
-                        }
+    match mcause::read().cause() {
+        // SBI call
+        T::Exception(E::SupervisorEnvCall) => {
+            use sbi_spec::{base, hsm, legacy};
+            let mut ret = unsafe { SBI.assume_init_ref() }.handle_ecall(
+                a7,
+                a6,
+                [ctx.a0(), a1, a2, a3, a4, a5],
+            );
+            if ret.is_ok() {
+                match (a7, a6) {
+                    // 不可恢复挂起
+                    (hsm::EID_HSM, hsm::HART_SUSPEND)
+                        if matches!(ctx.a0() as u32, hsm::suspend_type::NON_RETENTIVE) =>
+                    {
+                        return resume(ctx, a1, a2);
                     }
-                    ctx.regs().a = [ret.error, ret.value, a2, a3, a4, a5, a6, a7];
-                    mepc::write(mepc::read() + 4);
-                    break ctx.restore();
+                    // legacy console 探测
+                    (base::EID_BASE, base::PROBE_EXTENSION)
+                        if matches!(
+                            ctx.a0(),
+                            legacy::LEGACY_CONSOLE_PUTCHAR | legacy::LEGACY_CONSOLE_GETCHAR
+                        ) =>
+                    {
+                        ret.value = 1;
+                    }
+                    _ => {}
                 }
-                T::Exception(E::IllegalInstruction) => {
-                    if mstatus::read().mpp() == mstatus::MPP::Machine {
-                        panic!("Cannot handle illegal instruction exception from M-MODE");
+            } else {
+                match a7 {
+                    legacy::LEGACY_CONSOLE_PUTCHAR => {
+                        ret.error = unsafe { SBI.assume_init_ref() }.uart16550.as_ref().unwrap().putchar(ctx.a0());
+                        ret.value = a1;
                     }
-
-                    ctx.regs().a = [ctx.a0(), a1, a2, a3, a4, a5, a6, a7];
-                    if !illegal_instruction_handler(&mut ctx) {
-                        delegate();
+                    legacy::LEGACY_CONSOLE_GETCHAR => {
+                        ret.error = unsafe { SBI.assume_init_ref() }.uart16550.as_ref().unwrap().getchar();
+                        ret.value = a1;
                     }
-                    break ctx.restore();
+                    _ => {}
                 }
-                // 其他陷入
-                trap => {
-                    println!(
-                        "
+            }
+            ctx.regs().a = [ret.error, ret.value, a2, a3, a4, a5, a6, a7];
+            mepc::write(mepc::read() + 4);
+            ctx.restore()
+        }
+        T::Exception(E::IllegalInstruction) => {
+            if mstatus::read().mpp() == mstatus::MPP::Machine {
+                panic!("Cannot handle illegal instruction exception from M-MODE");
+            }
+
+            ctx.regs().a = [ctx.a0(), a1, a2, a3, a4, a5, a6, a7];
+            if !illegal_instruction_handler(&mut ctx) {
+                delegate();
+            }
+            ctx.restore()
+        }
+        // 其他陷入
+        trap => {
+            println!(
+                "
 -----------------------------
 > trap:    {trap:?}
 > mepc:    {:#018x}
 > mtval:   {:#018x}
 -----------------------------
             ",
-                        mepc::read(),
-                        mtval::read()
-                    );
-                    panic!("Stopped with unsupported trap")
-                }
-            },
+                mepc::read(),
+                mtval::read()
+            );
+            panic!("Stopped with unsupported trap")
         }
     }
 }
@@ -433,7 +399,7 @@ fn illegal_instruction_handler(ctx: &mut FastContext) -> bool {
                     "Unsupported CSR rd: {}",
                     csr.rd()
                 );
-                ctx.regs().a[(csr.rd() - 10) as usize] = unsafe { SBI.assume_init_mut() }
+                ctx.regs().a[(csr.rd() - 10) as usize] = unsafe { SBI.assume_init_ref() }
                     .clint
                     .as_ref()
                     .unwrap()
@@ -445,7 +411,7 @@ fn illegal_instruction_handler(ctx: &mut FastContext) -> bool {
                     "Unsupported CSR rd: {}",
                     csr.rd()
                 );
-                ctx.regs().a[(csr.rd() - 10) as usize] = unsafe { SBI.assume_init_mut() }
+                ctx.regs().a[(csr.rd() - 10) as usize] = unsafe { SBI.assume_init_ref() }
                     .clint
                     .as_ref()
                     .unwrap()