Bladeren bron

feat(prototyper): add payload support

feat(prototyper): hack for nemu

Signed-off-by: Woshiluo Luo <woshiluo.luo@outlook.com>
Woshiluo Luo 5 maanden geleden
bovenliggende
commit
48785be8b4
5 gewijzigde bestanden met toevoegingen van 156 en 13 verwijderingen
  1. 11 0
      Makefile.toml
  2. 44 1
      prototyper/build.rs
  3. 6 4
      prototyper/src/fail.rs
  4. 60 8
      prototyper/src/main.rs
  5. 35 0
      prototyper/src/payload.rs

+ 11 - 0
Makefile.toml

@@ -5,6 +5,17 @@ default_to_workspace = false
 command = "cargo"
 args = ["clean"]
 
+[tasks.prototyper-nemu-build]
+command = "cargo"
+args = ["build", "-prustsbi-prototyper", "--release", "--features=nemu,fw_payload"]
+
+[tasks.prototyper-nemu]
+command = "rust-objcopy"
+args = ["--binary-architecture=riscv64", "target/riscv64imac-unknown-none-elf/release/rustsbi-prototyper",
+        "--output-target=binary", "target/riscv64imac-unknown-none-elf/release/rustsbi-prototyper.bin"
+]
+dependencies = ["prototyper-nemu-build"]
+
 [tasks.prototyper-build]
 command = "cargo"
 args = ["build", "-prustsbi-prototyper", "--release"]

+ 44 - 1
prototyper/build.rs

@@ -6,11 +6,54 @@ fn main() {
 
     std::fs::write(ld, LINKER_SCRIPT).unwrap();
 
-    println!("cargo:rerun-if-env-changed=RUST_LOG");
+    println!("cargo:rerun-if-env-changed=RUST_LOG,PROTOTYPER_FDT,PROTOTYPER_IMAGE");
     println!("cargo:rustc-link-arg=-T{}", ld.display());
     println!("cargo:rustc-link-search={}", out.display());
 }
 
+#[cfg(feature = "fw_payload")]
+const LINKER_SCRIPT: &[u8] = b"OUTPUT_ARCH(riscv)
+ENTRY(_start) 
+SECTIONS {
+    . = 0x80000000;
+    .text : ALIGN(8) { 
+        *(.text.entry)
+        *(.text .text.*)
+    }
+    .rodata : ALIGN(8) { 
+        srodata = .;
+        *(.rodata .rodata.*)
+        *(.srodata .srodata.*)
+        . = ALIGN(8);  
+        erodata = .;
+    } 
+    .data : ALIGN(8) { 
+        sdata = .;
+        *(.data .data.*)
+        *(.sdata .sdata.*)
+        . = ALIGN(8); 
+        edata = .;
+    }
+    sidata = LOADADDR(.data);
+    .bss (NOLOAD) : ALIGN(8) {  
+        *(.bss.uninit)
+        sbss = .;
+        *(.bss .bss.*)
+        *(.sbss .sbss.*)
+        ebss = .;
+    } 
+    /DISCARD/ : {
+        *(.eh_frame)
+    }
+    .text 0x80100000 : ALIGN(8) {
+        *(.fw_fdt)
+    }
+    .text 0x80200000 : ALIGN(8) {
+        *(.fw_payload)
+    }
+}";
+
+#[cfg(not(feature = "fw_payload"))]
 const LINKER_SCRIPT: &[u8] = b"OUTPUT_ARCH(riscv)
 ENTRY(_start) 
 SECTIONS {

+ 6 - 4
prototyper/src/fail.rs

@@ -1,11 +1,11 @@
 use riscv::register::mstatus;
 use serde_device_tree::Dtb;
 
+use crate::dt::{self, ParseDeviceTreeError, Tree};
 use crate::sbi::reset;
-use crate::{
-    dt::{self, ParseDeviceTreeError, Tree},
-    dynamic,
-};
+
+#[cfg(not(feature = "fw_payload"))]
+use crate::dynamic;
 
 #[cold]
 pub fn device_tree_format(err: dt::ParseDeviceTreeError) -> Dtb {
@@ -22,6 +22,7 @@ pub fn device_tree_deserialize<'a>(err: serde_device_tree::error::Error) -> Tree
 }
 
 #[cold]
+#[cfg(not(feature = "fw_payload"))]
 pub fn invalid_dynamic_data(err: dynamic::DynamicError) -> (mstatus::MPP, usize) {
     error!("Invalid data in dynamic information:");
     if err.invalid_mpp {
@@ -44,6 +45,7 @@ pub fn invalid_dynamic_data(err: dynamic::DynamicError) -> (mstatus::MPP, usize)
 }
 
 #[cold]
+#[cfg(not(feature = "fw_payload"))]
 pub fn no_dynamic_info_available(err: dynamic::DynamicReadError) -> dynamic::DynamicInfo {
     if let Some(bad_paddr) = err.bad_paddr {
         error!(

+ 60 - 8
prototyper/src/main.rs

@@ -10,8 +10,11 @@ mod macros;
 
 mod board;
 mod dt;
+#[cfg(not(feature = "fw_payload"))]
 mod dynamic;
 mod fail;
+#[cfg(feature = "fw_payload")]
+mod payload;
 mod riscv_spec;
 mod sbi;
 
@@ -34,37 +37,53 @@ use crate::sbi::SBI;
 #[no_mangle]
 extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
     // parse dynamic information
+    #[cfg(not(feature = "fw_payload"))]
     let info = dynamic::read_paddr(nonstandard_a2).unwrap_or_else(fail::no_dynamic_info_available);
     static GENESIS: AtomicBool = AtomicBool::new(true);
     static SBI_READY: AtomicBool = AtomicBool::new(false);
 
+    #[cfg(not(feature = "fw_payload"))]
     let is_boot_hart = if info.boot_hart == usize::MAX {
         GENESIS.swap(false, Ordering::AcqRel)
     } else {
         current_hartid() == info.boot_hart
     };
+    #[cfg(feature = "fw_payload")]
+    let is_boot_hart = true;
+
+    #[cfg(feature = "fw_payload")]
+    let fdt_address = payload::get_fdt_address();
+    #[cfg(not(feature = "fw_payload"))]
+    let fdt_address = opaque;
 
     if is_boot_hart {
+        #[cfg(feature = "fw_payload")]
+        let (mpp, next_addr) = (mstatus::MPP::Supervisor, payload::get_image_address());
+        #[cfg(not(feature = "fw_payload"))]
         let (mpp, next_addr) =
             dynamic::mpp_next_addr(&info).unwrap_or_else(fail::invalid_dynamic_data);
 
-        // parse the device tree
-
         // 1. Init FDT
-        let dtb = dt::parse_device_tree(opaque).unwrap_or_else(fail::device_tree_format);
+        // parse the device tree
+        let dtb = dt::parse_device_tree(fdt_address).unwrap_or_else(fail::device_tree_format);
         let dtb = dtb.share();
         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 reset_device = tree.soc.test.unwrap().iter().next().unwrap();
         let console_base = tree.soc.serial.unwrap().iter().next().unwrap();
         let clint_device = tree.soc.clint.unwrap().iter().next().unwrap();
-        let reset_base_address = reset_device.at();
         let console_base_address = console_base.at();
         let ipi_base_address = clint_device.at();
-        board::reset_dev_init(usize::from_str_radix(reset_base_address, 16).unwrap());
+
+        // Set reset device if found it
+        if let Some(test) = tree.soc.test {
+            let reset_device = test.iter().next().unwrap();
+            let reset_base_address = reset_device.at();
+            board::reset_dev_init(usize::from_str_radix(reset_base_address, 16).unwrap());
+        }
+
         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());
         // Assume sstc is enabled only if all hart has sstc ext
@@ -82,6 +101,8 @@ extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
                 isa.iter().find(|&x| x == "sstc").is_some()
             })
             .all(|x| x);
+        #[cfg(feature = "nemu")]
+        let sstc_support = true;
         // 3. Init SBI
         unsafe {
             SBI_IMPL = MaybeUninit::new(SBI {
@@ -105,7 +126,6 @@ extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
         info!("Support sstc: {sstc_support}");
         info!("Clint device: {}", ipi_base_address);
         info!("Console deivce: {}", console_base_address);
-        info!("Reset device: {}", reset_base_address);
         info!(
             "Chosen stdout item: {}",
             tree.chosen
@@ -131,7 +151,7 @@ extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
         local_remote_hsm().start(NextStage {
             start_addr: next_addr,
             next_mode: mpp,
-            opaque,
+            opaque: fdt_address,
         });
 
         info!(
@@ -179,6 +199,7 @@ extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
 #[naked]
 #[link_section = ".text.entry"]
 #[export_name = "_start"]
+#[cfg(not(feature = "fw_payload"))]
 unsafe extern "C" fn start() -> ! {
     core::arch::asm!(
         // 1. Turn off interrupt
@@ -216,6 +237,37 @@ unsafe extern "C" fn start() -> ! {
     )
 }
 
+#[naked]
+#[link_section = ".text.entry"]
+#[export_name = "_start"]
+#[cfg(feature = "fw_payload")]
+unsafe extern "C" fn start() -> ! {
+    core::arch::asm!(
+        // 1. Turn off interrupt
+        "   csrw    mie, zero",
+        // 2. Initialize programming langauge runtime
+        // only clear bss if hartid matches preferred boot hart id
+        "   csrr    t0, mhartid",
+        // 3. clear bss segment
+        "   la      t0, sbss
+            la      t1, ebss
+        1:  bgeu    t0, t1, 2f
+            sd      zero, 0(t0)
+            addi    t0, t0, 8
+            j       1b",
+        "2:",
+         // 4. Prepare stack for each hart
+        "   call    {locate_stack}",
+        "   call    {main}",
+        "   csrw    mscratch, sp",
+        "   j       {hart_boot}",
+        locate_stack = sym trap_stack::locate,
+        main         = sym rust_main,
+        hart_boot    = sym trap::msoft,
+        options(noreturn)
+    )
+}
+
 #[panic_handler]
 fn panic(info: &core::panic::PanicInfo) -> ! {
     use riscv::register::*;

+ 35 - 0
prototyper/src/payload.rs

@@ -0,0 +1,35 @@
+use core::arch::asm;
+
+#[naked]
+#[link_section = ".fw_fdt"]
+pub unsafe extern "C" fn raw_fdt() {
+    asm!(
+        concat!(".incbin \"", env!("PROTOTYPER_FDT"), "\""),
+        options(noreturn)
+    );
+}
+
+#[naked]
+#[link_section = ".fw_payload"]
+pub unsafe extern "C" fn fw_payload_image() {
+    asm!(
+        concat!(".incbin \"", env!("PROTOTYPER_IMAGE"), "\""),
+        options(noreturn)
+    );
+}
+
+pub fn get_fdt_address() -> usize {
+    let addr: usize;
+    unsafe {
+        asm!("la {}, {fdt}", out(reg) addr, fdt = sym raw_fdt, options(nomem));
+    }
+    addr
+}
+
+pub fn get_image_address() -> usize {
+    let addr: usize;
+    unsafe {
+        asm!("la {}, {image}", out(reg) addr, image = sym fw_payload_image, options(nomem));
+    }
+    addr
+}