Parcourir la source

SBI: introducing sbi compatibility

Pablo Graubner il y a 2 ans
Parent
commit
061c0aace3
6 fichiers modifiés avec 112 ajouts et 23 suppressions
  1. 3 0
      riscv-rt/Cargo.toml
  2. 16 1
      riscv-rt/asm.S
  3. 8 0
      riscv-rt/assemble.sh
  4. 7 1
      riscv-rt/build.rs
  5. 2 4
      riscv-rt/examples/multi_core.rs
  6. 76 17
      riscv-rt/src/lib.rs

+ 3 - 0
riscv-rt/Cargo.toml

@@ -10,6 +10,9 @@ keywords = ["riscv", "runtime", "startup"]
 license = "ISC"
 edition = "2018"
 
+[features]
+sbi = []
+
 [dependencies]
 r0 = "1.0.0"
 riscv = "0.8"

+ 16 - 1
riscv-rt/asm.S

@@ -46,8 +46,14 @@ _abs_start:
     .cfi_startproc
     .cfi_undefined ra
 
+    #ifdef SBI
+    csrw sie, 0     // interrupt disable 
+    csrw sip, 0     // no pending interrupts
+    #else
     csrw mie, 0
     csrw mip, 0
+    #endif
+    
 
     li  x1, 0
     li  x2, 0
@@ -84,8 +90,13 @@ _abs_start:
     la gp, __global_pointer$
     .option pop
 
-    // Check hart id
+    #ifdef SBI
+    // there is no equivalent of mhartid in supervisor mode.
+    // instead, the hartid is passed as paramter by SBI
+    mv t2, a0   
+    #else 
     csrr t2, mhartid
+    #endif
     lui t0, %hi(_max_hart_id)
     add t0, t0, %lo(_max_hart_id)
     bgtu t2, t0, abort
@@ -165,7 +176,11 @@ default_start_trap:
     LOAD a7, 15*REGBYTES(sp)
 
     addi sp, sp, 16*REGBYTES
+    #ifdef SBI
+    sret
+    #else
     mret
+    #endif
 
 /* Make sure there is an abort when linking */
 .section .text.abort

+ 8 - 0
riscv-rt/assemble.sh

@@ -31,6 +31,14 @@ do
 
     riscv64-unknown-elf-gcc -ggdb3 -fdebug-prefix-map=$(pwd)=/riscv-rt -c -mabi=lp64${abi} -march=rv64${ext} asm.S -o bin/$crate.o
     riscv64-unknown-elf-ar crs bin/riscv64${ext}-unknown-none-elf.a bin/$crate.o
+
+    #sbi
+    riscv64-unknown-elf-gcc -DSBI -ggdb3 -fdebug-prefix-map=$(pwd)=/riscv-rt -c -mabi=ilp32${abi} -march=rv32${ext} asm.S -o bin/$crate.o
+    riscv64-unknown-elf-ar crs bin/riscv32${ext}-unknown-none-elf-sbi.a bin/$crate.o
+
+    riscv64-unknown-elf-gcc -DSBI -ggdb3 -fdebug-prefix-map=$(pwd)=/riscv-rt -c -mabi=lp64${abi} -march=rv64${ext} asm.S -o bin/$crate.o
+    riscv64-unknown-elf-ar crs bin/riscv64${ext}-unknown-none-elf-sbi.a bin/$crate.o
+
 done
 
 rm bin/$crate.o

+ 7 - 1
riscv-rt/build.rs

@@ -14,7 +14,13 @@ fn main() {
     if target.starts_with("riscv") {
         let mut target = Target::from_target_str(&target);
         target.retain_extensions("imfdc");
-        let archive = format!("bin/{}.a", target.to_string());
+        let archive:String;
+        if cfg!(feature = "sbi") {
+            println!("======== compiling riscv-rt for sbi");
+            archive = format!("bin/{}-sbi.a", target.to_string());
+        } else {
+            archive = format!("bin/{}.a", target.to_string());
+        }
 
         fs::copy(&archive, out_dir.join(format!("lib{}.a", name))).unwrap();
         println!("cargo:rerun-if-changed={}", archive);

+ 2 - 4
riscv-rt/examples/multi_core.rs

@@ -11,7 +11,7 @@ use riscv_rt::entry;
 
 #[export_name = "_mp_hook"]
 #[rustfmt::skip]
-pub extern "Rust" fn user_mp_hook() -> bool {
+pub extern "Rust" fn user_mp_hook(hartid: usize) -> bool {
     let hartid = mhartid::read();
     if hartid == 0 {
         true
@@ -42,9 +42,7 @@ pub extern "Rust" fn user_mp_hook() -> bool {
 }
 
 #[entry]
-fn main() -> ! {
-    let hartid = mhartid::read();
-
+fn main(hartid: usize) -> ! {
     if hartid == 0 {
         // Waking hart 1...
         let addr = 0x02000004;

+ 76 - 17
riscv-rt/src/lib.rs

@@ -21,6 +21,8 @@
 //!   filename can be use instead of `memory.x`.
 //!
 //! - A `_sheap` symbol at whose address you can locate a heap.
+//! 
+//! - Support for a runtime in supervisor mode, bootstrapped via [Supervisor Binary Interface (SBI)](https://github.com/riscv-non-isa/riscv-sbi-doc) 
 //!
 //! ``` text
 //! $ cargo new --bin app && cd $_
@@ -230,12 +232,13 @@
 //! This function is called from all the harts and must return true only for one hart,
 //! which will perform memory initialization. For other harts it must return false
 //! and implement wake-up in platform-dependent way (e.g. after waiting for a user interrupt).
+//! The parameter `hartid` specifies the hartid of the caller.
 //!
 //! This function can be redefined in the following way:
 //!
 //! ``` no_run
 //! #[export_name = "_mp_hook"]
-//! pub extern "Rust" fn mp_hook() -> bool {
+//! pub extern "Rust" fn mp_hook(hartid: usize) -> bool {
 //!    // ...
 //! }
 //! ```
@@ -245,7 +248,7 @@
 //! ### `ExceptionHandler`
 //!
 //! This function is called when exception is occured. The exception reason can be decoded from the
-//! `mcause` register.
+//! `mcause`/`scause` register.
 //!
 //! This function can be redefined in the following way:
 //!
@@ -300,7 +303,7 @@
 //! ### `DefaultHandler`
 //!
 //! This function is called when interrupt without defined interrupt handler is occured.
-//! The interrupt reason can be decoded from the `mcause` register.
+//! The interrupt reason can be decoded from the `mcause`/`scause` register.
 //!
 //! This function can be redefined in the following way:
 //!
@@ -319,12 +322,44 @@
 //! ```
 //!
 //! Default implementation of this function stucks in a busy-loop.
+//! 
+//! # Features
+//! 
+//! ## `sbi`
+//! 
+//! The SBI runtime feature (`sbi`) can be activated via [Cargo features](https://doc.rust-lang.org/cargo/reference/features.html).
+//! 
+//! For example: 
+//! ``` text
+//! [dependencies]
+//! riscv-rt = {features=["sbi"]}
+//! ``` 
+//! Using the SBI requires riscv-rt to be run in supervisor mode instead of machine code.
+//! Internally, riscv-rt uses different versions of precompiled static libraries 
+//! for (i) machine mode and (ii) sbi. If the `sbi` feature was activated, 
+//! the build script selects the sbi library. While most registers/instructions have variants for 
+//! both `mcause` and `scause`, the `mhartid` hardware thread register is not available in supervisor 
+//! mode. Instead, the hartid is passed as parameter by the calling SBI.
+//! 
+//! QEMU supports [OpenSBI](https://github.com/riscv-software-src/opensbi) as default firmware.
+//! ``` text
+//! APP_BINARY=$(find target -name app)
+//! sudo qemu-system-riscv64 -m 2G -nographic -machine virt -kernel $APP_BINARY
+//! ```
+//! It requires the memory layout to be non-overlapping, like
+//! ``` text
+//! MEMORY
+//! {
+//!   RAM : ORIGIN = 0x80200000, LENGTH = 0x8000000
+//!   FLASH : ORIGIN = 0x20000000, LENGTH = 16M
+//! }
+//! ```
 
 // NOTE: Adapted from cortex-m/src/lib.rs
 #![no_std]
 #![deny(missing_docs)]
 
-use riscv::register::mcause;
+use riscv::register::{scause,mcause,mhartid};
 pub use riscv_rt_macros::{entry, pre_init};
 
 #[export_name = "error: riscv-rt appears more than once in the dependency graph"]
@@ -361,10 +396,18 @@ pub unsafe extern "C" fn start_rust(a0: usize, a1: usize, a2: usize) -> ! {
 
         fn _setup_interrupts();
 
-        fn _mp_hook() -> bool;
+        fn _mp_hook(hartid: usize) -> bool;
+    }
+
+    // sbi passes hartid as first parameter (a0)
+    let hartid : usize;
+    if cfg!(feature = "sbi") {
+        hartid = a0;
+    } else {
+        hartid = mhartid::read();
     }
 
-    if _mp_hook() {
+    if _mp_hook(hartid) {
         __pre_init();
 
         r0::zero_bss(&mut _sbss, &mut _ebss);
@@ -403,7 +446,7 @@ pub struct TrapFrame {
 
 /// Trap entry point rust (_start_trap_rust)
 ///
-/// `mcause` is read to determine the cause of the trap. XLEN-1 bit indicates
+/// `scause`/`mcause` is read to determine the cause of the trap. XLEN-1 bit indicates
 /// if it's an interrupt or an exception. The result is examined and ExceptionHandler
 /// or one of the core interrupt handlers is called.
 #[link_section = ".trap.rust"]
@@ -415,11 +458,22 @@ pub extern "C" fn start_trap_rust(trap_frame: *const TrapFrame) {
     }
 
     unsafe {
-        let cause = mcause::read();
-        if cause.is_exception() {
+        let code : usize;
+        let is_exception: bool;
+
+        if cfg!(feature = "sbi") {
+            let cause = scause::read();
+            is_exception = cause.is_exception();
+            code = cause.code();
+        } else {
+            let cause = mcause::read();
+            is_exception = cause.is_exception();
+            code = cause.code();
+        }
+
+        if is_exception {
             ExceptionHandler(&*trap_frame)
         } else {
-            let code = cause.code();
             if code < __INTERRUPTS.len() {
                 let h = &__INTERRUPTS[code];
                 if h.reserved == 0 {
@@ -430,7 +484,7 @@ pub extern "C" fn start_trap_rust(trap_frame: *const TrapFrame) {
             } else {
                 DefaultHandler();
             }
-        }
+        }    
     }
 }
 
@@ -529,9 +583,8 @@ pub unsafe extern "Rust" fn default_pre_init() {}
 #[doc(hidden)]
 #[no_mangle]
 #[rustfmt::skip]
-pub extern "Rust" fn default_mp_hook() -> bool {
-    use riscv::register::mhartid;
-    match mhartid::read() {
+pub extern "Rust" fn default_mp_hook(hartid: usize) -> bool {
+    match hartid {
         0 => true,
         _ => loop {
             unsafe { riscv::asm::wfi() }
@@ -539,14 +592,20 @@ pub extern "Rust" fn default_mp_hook() -> bool {
     }
 }
 
-/// Default implementation of `_setup_interrupts` that sets `mtvec` to a trap handler address.
+/// Default implementation of `_setup_interrupts` that sets `mtvec`/`stvec` to a trap handler address.
 #[doc(hidden)]
 #[no_mangle]
 #[rustfmt::skip]
 pub unsafe extern "Rust" fn default_setup_interrupts() {
-    use riscv::register::mtvec::{self, TrapMode};
     extern "C" {
         fn _start_trap();
     }
-    mtvec::write(_start_trap as usize, TrapMode::Direct);
+
+    if cfg!(feature = "sbi") {
+        use riscv::register::stvec::{self, TrapMode};
+        stvec::write(_start_trap as usize, TrapMode::Direct);
+    } else {
+        use riscv::register::mtvec::{self, TrapMode};
+        mtvec::write(_start_trap as usize, TrapMode::Direct);        
+    }
 }