//! Minimal startup / runtime for RISC-V CPU's //! //! # Minimum Supported Rust Version (MSRV) //! //! This crate is guaranteed to compile on stable Rust 1.60 and up. It *might* //! compile with older versions but that may change in any new patch release. //! //! # Features //! //! This crate provides //! //! - Before main initialization of the `.bss` and `.data` sections. //! //! - `#[entry]` to declare the entry point of the program //! - `#[pre_init]` to run code *before* `static` variables are initialized //! //! - A linker script that encodes the memory layout of a generic RISC-V //! microcontroller. This linker script is missing some information that must //! be supplied through a `memory.x` file (see example below). This file //! must be supplied using rustflags and listed *before* `link.x`. Arbitrary //! 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, that can be bootstrapped via [Supervisor Binary Interface (SBI)](https://github.com/riscv-non-isa/riscv-sbi-doc) //! //! ``` text //! $ cargo new --bin app && cd $_ //! //! $ # add this crate as a dependency //! $ edit Cargo.toml && cat $_ //! [dependencies] //! riscv-rt = "0.6.1" //! panic-halt = "0.2.0" //! //! $ # memory layout of the device //! $ edit memory.x && cat $_ //! MEMORY //! { //! RAM : ORIGIN = 0x80000000, LENGTH = 16K //! FLASH : ORIGIN = 0x20000000, LENGTH = 16M //! } //! //! REGION_ALIAS("REGION_TEXT", FLASH); //! REGION_ALIAS("REGION_RODATA", FLASH); //! REGION_ALIAS("REGION_DATA", RAM); //! REGION_ALIAS("REGION_BSS", RAM); //! REGION_ALIAS("REGION_HEAP", RAM); //! REGION_ALIAS("REGION_STACK", RAM); //! //! $ edit src/main.rs && cat $_ //! ``` //! //! ``` ignore,no_run //! #![no_std] //! #![no_main] //! //! extern crate panic_halt; //! //! use riscv_rt::entry; //! //! // use `main` as the entry point of this application //! // `main` is not allowed to return //! #[entry] //! fn main() -> ! { //! // do something here //! loop { } //! } //! ``` //! //! ``` text //! $ mkdir .cargo && edit .cargo/config && cat $_ //! [target.riscv32imac-unknown-none-elf] //! rustflags = [ //! "-C", "link-arg=-Tmemory.x", //! "-C", "link-arg=-Tlink.x", //! ] //! //! [build] //! target = "riscv32imac-unknown-none-elf" //! $ edit build.rs && cat $_ //! ``` //! //! ``` ignore,no_run //! use std::env; //! use std::fs; //! use std::path::PathBuf; //! //! fn main() { //! let out_dir = PathBuf::from(env::var("OUT_DIR").unwrap()); //! //! // Put the linker script somewhere the linker can find it. //! fs::write(out_dir.join("memory.x"), include_bytes!("memory.x")).unwrap(); //! println!("cargo:rustc-link-search={}", out_dir.display()); //! println!("cargo:rerun-if-changed=memory.x"); //! //! println!("cargo:rerun-if-changed=build.rs"); //! } //! ``` //! //! ``` text //! $ cargo build //! //! $ riscv32-unknown-elf-objdump -Cd $(find target -name app) | head //! //! Disassembly of section .text: //! //! 20000000 <_start>: //! 20000000: 800011b7 lui gp,0x80001 //! 20000004: 80018193 addi gp,gp,-2048 # 80000800 <_stack_start+0xffffc800> //! 20000008: 80004137 lui sp,0x80004 //! ``` //! //! # Symbol interfaces //! //! This crate makes heavy use of symbols, linker sections and linker scripts to //! provide most of its functionality. Below are described the main symbol //! interfaces. //! //! ## `memory.x` //! //! This file supplies the information about the device to the linker. //! //! ### `MEMORY` //! //! The main information that this file must provide is the memory layout of //! the device in the form of the `MEMORY` command. The command is documented //! [here][2], but at a minimum you'll want to create at least one memory region. //! //! [2]: https://sourceware.org/binutils/docs/ld/MEMORY.html //! //! To support different relocation models (RAM-only, FLASH+RAM) multiple regions are used: //! //! - `REGION_TEXT` - for `.init`, `.trap` and `.text` sections //! - `REGION_RODATA` - for `.rodata` section and storing initial values for `.data` section //! - `REGION_DATA` - for `.data` section //! - `REGION_BSS` - for `.bss` section //! - `REGION_HEAP` - for the heap area //! - `REGION_STACK` - for hart stacks //! //! Specific aliases for these regions must be defined in `memory.x` file (see example below). //! //! ### `_stext` //! //! This symbol provides the loading address of `.text` section. This value can be changed //! to override the loading address of the firmware (for example, in case of bootloader present). //! //! If omitted this symbol value will default to `ORIGIN(REGION_TEXT)`. //! //! ### `_stack_start` //! //! This symbol provides the address at which the call stack will be allocated. //! The call stack grows downwards so this address is usually set to the highest //! valid RAM address plus one (this *is* an invalid address but the processor //! will decrement the stack pointer *before* using its value as an address). //! //! In case of multiple harts present, this address defines the initial stack pointer for hart 0. //! Stack pointer for hart `N` is calculated as `_stack_start - N * _hart_stack_size`. //! //! If omitted this symbol value will default to `ORIGIN(REGION_STACK) + LENGTH(REGION_STACK)`. //! //! #### Example //! //! Allocating the call stack on a different RAM region. //! //! ``` text //! MEMORY //! { //! L2_LIM : ORIGIN = 0x08000000, LENGTH = 1M //! RAM : ORIGIN = 0x80000000, LENGTH = 16K //! FLASH : ORIGIN = 0x20000000, LENGTH = 16M //! } //! //! REGION_ALIAS("REGION_TEXT", FLASH); //! REGION_ALIAS("REGION_RODATA", FLASH); //! REGION_ALIAS("REGION_DATA", RAM); //! REGION_ALIAS("REGION_BSS", RAM); //! REGION_ALIAS("REGION_HEAP", RAM); //! REGION_ALIAS("REGION_STACK", L2_LIM); //! //! _stack_start = ORIGIN(L2_LIM) + LENGTH(L2_LIM); //! ``` //! //! ### `_max_hart_id` //! //! This symbol defines the maximum hart id supported. All harts with id //! greater than `_max_hart_id` will be redirected to `abort()`. //! //! This symbol is supposed to be redefined in platform support crates for //! multi-core targets. //! //! If omitted this symbol value will default to 0 (single core). //! //! ### `_hart_stack_size` //! //! This symbol defines stack area size for *one* hart. //! //! If omitted this symbol value will default to 2K. //! //! ### `_heap_size` //! //! This symbol provides the size of a heap region. The default value is 0. You can set `_heap_size` //! to a non-zero value if you are planning to use heap allocations. //! //! ### `_sheap` //! //! This symbol is located in RAM right after the `.bss` and `.data` sections. //! You can use the address of this symbol as the start address of a heap //! region. This symbol is 4 byte aligned so that address will be a multiple of 4. //! //! #### Example //! //! ``` no_run //! extern crate some_allocator; //! //! extern "C" { //! static _sheap: u8; //! static _heap_size: u8; //! } //! //! fn main() { //! unsafe { //! let heap_bottom = &_sheap as *const u8 as usize; //! let heap_size = &_heap_size as *const u8 as usize; //! some_allocator::initialize(heap_bottom, heap_size); //! } //! } //! ``` //! //! ### `_mp_hook` //! //! 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(hartid: usize) -> bool { //! // ... //! } //! ``` //! //! Default implementation of this function wakes hart 0 and busy-loops all the other harts. //! //! //! ### Core exception handlers //! //! This functions are called when corresponding exception occurs. //! You can define an exception handler with one of the following names: //! * `InstructionMisaligned` //! * `InstructionFault` //! * `IllegalInstruction` //! * `Breakpoint` //! * `LoadMisaligned` //! * `LoadFault` //! * `StoreMisaligned` //! * `StoreFault` //! * `UserEnvCall` //! * `SupervisorEnvCall` //! * `MachineEnvCall` //! * `InstructionPageFault` //! * `LoadPageFault` //! * `StorePageFault` //! //! For example: //! ``` no_run //! #[export_name = "MachineEnvCall"] //! fn custom_menv_call_handler(trap_frame: &riscv_rt::TrapFrame) { //! // ... //! } //! ``` //! or //! ``` no_run //! #[no_mangle] //! fn MachineEnvCall(trap_frame: &riscv_rt::TrapFrame) -> ! { //! // ... //! } //! ``` //! //! If exception handler is not explicitly defined, `ExceptionHandler` is called. //! //! ### `ExceptionHandler` //! //! This function is called when exception without defined exception handler is occured. //! The exception reason can be decoded from the //! `mcause`/`scause` register. //! //! This function can be redefined in the following way: //! //! ``` no_run //! #[export_name = "ExceptionHandler"] //! fn custom_exception_handler(trap_frame: &riscv_rt::TrapFrame) -> ! { //! // ... //! } //! ``` //! or //! ``` no_run //! #[no_mangle] //! fn ExceptionHandler(trap_frame: &riscv_rt::TrapFrame) -> ! { //! // ... //! } //! ``` //! //! Default implementation of this function stucks in a busy-loop. //! //! //! ### Core interrupt handlers //! //! This functions are called when corresponding interrupt is occured. //! You can define an interrupt handler with one of the following names: //! * `UserSoft` //! * `SupervisorSoft` //! * `MachineSoft` //! * `UserTimer` //! * `SupervisorTimer` //! * `MachineTimer` //! * `UserExternal` //! * `SupervisorExternal` //! * `MachineExternal` //! //! For example: //! ``` no_run //! #[export_name = "MachineTimer"] //! fn custom_timer_handler() { //! // ... //! } //! ``` //! or //! ``` no_run //! #[no_mangle] //! fn MachineTimer() { //! // ... //! } //! ``` //! //! If interrupt handler is not explicitly defined, `DefaultHandler` is called. //! //! ### `DefaultHandler` //! //! This function is called when interrupt without defined interrupt handler is occured. //! The interrupt reason can be decoded from the `mcause`/`scause` register. //! //! This function can be redefined in the following way: //! //! ``` no_run //! #[export_name = "DefaultHandler"] //! fn custom_interrupt_handler() { //! // ... //! } //! ``` //! or //! ``` no_run //! #[no_mangle] //! fn DefaultHandler() { //! // ... //! } //! ``` //! //! Default implementation of this function stucks in a busy-loop. //! //! # Features //! //! ## `single-hart` //! //! This feature saves a little code size if there is only one hart on the target. //! //! ## `s-mode` //! //! The supervisor mode feature (`s-mode`) can be activated via [Cargo features](https://doc.rust-lang.org/cargo/reference/features.html). //! //! For example: //! ``` text //! [dependencies] //! riscv-rt = {features=["s-mode"]} //! ``` //! Internally, riscv-rt uses different versions of precompiled static libraries //! for (i) machine mode and (ii) supervisor mode. If the `s-mode` feature was activated, //! the build script selects the s-mode 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 a bootstrapping firmware (i.e., SBI). //! //! Use case: QEMU supports [OpenSBI](https://github.com/riscv-software-src/opensbi) as default firmware. //! Using the SBI requires riscv-rt to be run in supervisor mode instead of machine mode. //! ``` 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)] #[cfg(riscv)] mod asm; use core::sync::atomic::{compiler_fence, Ordering}; #[cfg(feature = "s-mode")] use riscv::register::{scause as xcause, stvec as xtvec, stvec::TrapMode as xTrapMode}; #[cfg(not(feature = "s-mode"))] use riscv::register::{mcause as xcause, mtvec as xtvec, mtvec::TrapMode as xTrapMode}; #[cfg(all(not(feature = "single-hart"), not(feature = "s-mode")))] use riscv::register::mhartid; #[cfg(all(feature = "s-mode", any(riscvf, riscvd)))] use riscv::register::sstatus as xstatus; #[cfg(all(not(feature = "s-mode"), any(riscvf, riscvd)))] use riscv::register::mstatus as xstatus; pub use riscv_rt_macros::{entry, pre_init}; /// We export this static with an informative name so that if an application attempts to link /// two copies of riscv-rt together, linking will fail. We also declare a links key in /// Cargo.toml which is the more modern way to solve the same problem, but we have to keep /// __ONCE__ around to prevent linking with versions before the links key was added. #[export_name = "error: riscv-rt appears more than once in the dependency graph"] #[doc(hidden)] pub static __ONCE__: () = (); /// Rust entry point (_start_rust) /// /// Zeros bss section, initializes data section and calls main. This function never returns. /// /// # Safety /// /// This function must be called only from assembly `_start` function. /// Do **NOT** call this function directly. #[link_section = ".init.rust"] #[export_name = "_start_rust"] pub unsafe extern "C" fn start_rust(a0: usize, a1: usize, a2: usize) -> ! { #[rustfmt::skip] extern "Rust" { // This symbol will be provided by the user via `#[entry]` fn main(a0: usize, a1: usize, a2: usize) -> !; // This symbol will be provided by the user via `#[pre_init]` fn __pre_init(); fn _setup_interrupts(); fn _mp_hook(hartid: usize) -> bool; } #[cfg(not(feature = "single-hart"))] let run_init = { // sbi passes hartid as first parameter (a0) #[cfg(feature = "s-mode")] let hartid = a0; #[cfg(not(feature = "s-mode"))] let hartid = mhartid::read(); _mp_hook(hartid) }; #[cfg(feature = "single-hart")] let run_init = true; if run_init { __pre_init(); // Initialize RAM // 1. Copy over .data from flash to RAM // 2. Zero out .bss #[cfg(target_arch = "riscv32")] core::arch::asm!( " // Copy over .data la {start},_sdata la {end},_edata la {input},_sidata bgeu {start},{end},2f 1: lw {a},0({input}) addi {input},{input},4 sw {a},0({start}) addi {start},{start},4 bltu {start},{end},1b 2: li {a},0 li {input},0 // Zero out .bss la {start},_sbss la {end},_ebss bgeu {start},{end},3f 2: sw zero,0({start}) addi {start},{start},4 bltu {start},{end},2b 3: li {start},0 li {end},0 ", start = out(reg) _, end = out(reg) _, input = out(reg) _, a = out(reg) _, ); #[cfg(target_arch = "riscv64")] core::arch::asm!( " // Copy over .data la {start},_sdata la {end},_edata la {input},_sidata bgeu {start},{end},2f 1: // .data Main Loop ld {a},0({input}) addi {input},{input},8 sd {a},0({start}) addi {start},{start},8 bltu {start},{end},1b 2: // .data zero registers li {a},0 li {input},0 la {start},_sbss la {end},_ebss bgeu {start},{end},4f 3: // .bss main loop sd zero,0({start}) addi {start},{start},8 bltu {start},{end},3b 4: // .bss zero registers // Zero out used registers li {start},0 li {end},0 ", start = out(reg) _, end = out(reg) _, input = out(reg) _, a = out(reg) _, ); compiler_fence(Ordering::SeqCst); } #[cfg(any(riscvf, riscvd))] { xstatus::set_fs(xstatus::FS::Initial); // Enable fpu in xstatus core::arch::asm!("fscsr x0"); // Zero out fcsr register csrrw x0, fcsr, x0 // Zero out floating point registers #[cfg(all(target_arch = "riscv32", riscvd))] riscv_rt_macros::loop_asm!("fcvt.d.w f{}, x0", 32); #[cfg(all(target_arch = "riscv64", riscvd))] riscv_rt_macros::loop_asm!("fmv.d.x f{}, x0", 32); #[cfg(not(riscvd))] riscv_rt_macros::loop_asm!("fmv.w.x f{}, x0", 32); } _setup_interrupts(); main(a0, a1, a2); } /// Registers saved in trap handler #[allow(missing_docs)] #[repr(C)] #[derive(Debug)] pub struct TrapFrame { pub ra: usize, pub t0: usize, pub t1: usize, pub t2: usize, pub t3: usize, pub t4: usize, pub t5: usize, pub t6: usize, pub a0: usize, pub a1: usize, pub a2: usize, pub a3: usize, pub a4: usize, pub a5: usize, pub a6: usize, pub a7: usize, } /// Trap entry point rust (_start_trap_rust) /// /// `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. /// /// # Safety /// /// This function must be called only from assembly `_start_trap` function. /// Do **NOT** call this function directly. #[link_section = ".trap.rust"] #[export_name = "_start_trap_rust"] pub unsafe extern "C" fn start_trap_rust(trap_frame: *const TrapFrame) { extern "C" { fn ExceptionHandler(trap_frame: &TrapFrame); fn DefaultHandler(); } let cause = xcause::read(); let code = cause.code(); if cause.is_exception() { let trap_frame = &*trap_frame; if code < __EXCEPTIONS.len() { let h = &__EXCEPTIONS[code]; if let Some(handler) = h { handler(trap_frame); } else { ExceptionHandler(trap_frame); } } else { ExceptionHandler(trap_frame); } ExceptionHandler(trap_frame) } else if code < __INTERRUPTS.len() { let h = &__INTERRUPTS[code]; if let Some(handler) = h { handler(); } else { DefaultHandler(); } } else { DefaultHandler(); } } #[doc(hidden)] #[no_mangle] #[allow(unused_variables, non_snake_case)] pub fn DefaultExceptionHandler(trap_frame: &TrapFrame) -> ! { loop { // Prevent this from turning into a UDF instruction // see rust-lang/rust#28728 for details continue; } } #[doc(hidden)] #[no_mangle] #[allow(non_snake_case)] pub fn DefaultInterruptHandler() { loop { // Prevent this from turning into a UDF instruction // see rust-lang/rust#28728 for details continue; } } extern "C" { fn InstructionMisaligned(trap_frame: &TrapFrame); fn InstructionFault(trap_frame: &TrapFrame); fn IllegalInstruction(trap_frame: &TrapFrame); fn Breakpoint(trap_frame: &TrapFrame); fn LoadMisaligned(trap_frame: &TrapFrame); fn LoadFault(trap_frame: &TrapFrame); fn StoreMisaligned(trap_frame: &TrapFrame); fn StoreFault(trap_frame: &TrapFrame); fn UserEnvCall(trap_frame: &TrapFrame); fn SupervisorEnvCall(trap_frame: &TrapFrame); fn MachineEnvCall(trap_frame: &TrapFrame); fn InstructionPageFault(trap_frame: &TrapFrame); fn LoadPageFault(trap_frame: &TrapFrame); fn StorePageFault(trap_frame: &TrapFrame); } #[doc(hidden)] #[no_mangle] pub static __EXCEPTIONS: [Option; 16] = [ Some(InstructionMisaligned), Some(InstructionFault), Some(IllegalInstruction), Some(Breakpoint), Some(LoadMisaligned), Some(LoadFault), Some(StoreMisaligned), Some(StoreFault), Some(UserEnvCall), Some(SupervisorEnvCall), None, Some(MachineEnvCall), Some(InstructionPageFault), Some(LoadPageFault), None, Some(StorePageFault), ]; extern "C" { fn SupervisorSoft(); fn MachineSoft(); fn SupervisorTimer(); fn MachineTimer(); fn SupervisorExternal(); fn MachineExternal(); } #[doc(hidden)] #[no_mangle] pub static __INTERRUPTS: [Option; 12] = [ None, Some(SupervisorSoft), None, Some(MachineSoft), None, Some(SupervisorTimer), None, Some(MachineTimer), None, Some(SupervisorExternal), None, Some(MachineExternal), ]; /// Default implementation of `_pre_init` does nothing. /// Users can override this function with the [`#[pre_init]`] macro. #[doc(hidden)] #[no_mangle] #[rustfmt::skip] pub extern "Rust" fn default_pre_init() {} /// Default implementation of `_mp_hook` wakes hart 0 and busy-loops all the other harts. #[doc(hidden)] #[no_mangle] #[rustfmt::skip] #[cfg(not(feature = "single-hart"))] pub extern "Rust" fn default_mp_hook(hartid: usize) -> bool { match hartid { 0 => true, _ => loop { unsafe { riscv::asm::wfi() } }, } } /// Default implementation of `_setup_interrupts` sets `mtvec`/`stvec` to the address of `_start_trap`. #[doc(hidden)] #[no_mangle] #[rustfmt::skip] pub unsafe extern "Rust" fn default_setup_interrupts() { extern "C" { fn _start_trap(); } xtvec::write(_start_trap as usize, xTrapMode::Direct); }