Jelajahi Sumber

Add documents and prepare for 0.2.0-alpha.3 version

luojia65 3 tahun lalu
induk
melakukan
f4ec485d1b
2 mengubah file dengan 94 tambahan dan 2 penghapusan
  1. 2 2
      rustsbi/Cargo.toml
  2. 92 0
      rustsbi/src/lib.rs

+ 2 - 2
rustsbi/Cargo.toml

@@ -1,7 +1,7 @@
 [package]
 name = "rustsbi"
 description = "Minimal RISC-V's SBI implementation library in Rust"
-version = "0.2.0-alpha.2"
+version = "0.2.0-alpha.3"
 authors = ["luojia65 <me@luojia.cc>"]
 repository = "https://github.com/luojia65/rustsbi"
 documentation = "https://docs.rs/rustsbi"
@@ -17,7 +17,7 @@ edition = "2018"
 default-target = "riscv64imac-unknown-none-elf"
 targets = [
     "riscv32i-unknown-none-elf", "riscv32imc-unknown-none-elf", "riscv32imac-unknown-none-elf",
-    "riscv64imac-unknown-none-elf", "riscv64gc-unknown-none-elf",
+    "riscv64imac-unknown-none-elf",
 ]
 
 [dependencies]

+ 92 - 0
rustsbi/src/lib.rs

@@ -1,5 +1,97 @@
 //! A minimal RISC-V's SBI implementation in Rust.
 //!
+//! # How to use RustSBI 
+//!
+//! SBI features include boot sequence and a kernel environment. To bootstrap your kernel,
+//! place kernel into RustSBI implementation defined address, then RustSBI will prepare an
+//! environment and jump to this address.
+//!
+//! ## Make SBI environment calls
+//!
+//! To use the kernel environment, you either use SBI calls or emulated instructions.
+//! SBI calls are similar to operating systems' `syscall`s. RISC-V SBI defined many SBI modules,
+//! and in each module there are different functions, you should pick a function before calling.
+//! Then, you should prepare some parameters, whose definition are not the same among functions.
+//! 
+//! Now you have a module number, a function number, and a few SBI call parameters.
+//! You invoke a special `ecall` instruction on supervisor level, and it will trap into machine level
+//! SBI implementation. It will handle your `ecall`, similar to your kernel handling system calls 
+//! from user level. 
+//!
+//! SBI functions return two values other than one. First value will be an error number,
+//! it will tell if SBI call have succeeded, or which error have occurred. 
+//! Second value is the real return value, its meaning is different according to which function you calls.
+//!
+//! ## Call SBI in different programming languages
+//!
+//! Making SBI calls are similar to making system calls. 
+//! 
+//! Module number is required to put on register `a7`, function number on `a6`. 
+//! Parameters should be placed from `a0` to `a5`, first into `a0`, second into `a1`, etc.
+//! Unused parameters can be set to any value or leave untouched.
+//!
+//! After registers are ready, invoke an instruction called `ecall`. 
+//! Then, the return value is placed into `a0` and `a1` registers.
+//! The error value could be read from `a0`, and return value is placed into `a1`.
+//!
+//! In Rust, here is an example to call SBI functions using inline assembly:
+//!
+//! ```no_run
+//! #[inline(always)]
+//! fn sbi_call(extension: usize, function: usize, arg0: usize, arg1: usize) -> SbiRet {
+//!     let (error, value);
+//!     match () {
+//!         #[cfg(any(target_arch = "riscv32", target_arch = "riscv64"))]
+//!         () => unsafe { asm!(
+//!             "ecall", 
+//!             in("a0") arg0, in("a1") arg1,
+//!             in("a6") function, in("a7") extension,
+//!             lateout("a0") error, lateout("a1") value,
+//!         ) },
+//!         #[cfg(not(any(target_arch = "riscv32", target_arch = "riscv64")))]
+//!         () => {
+//!             drop((extension, function, arg0, arg1));
+//!             unimplemented!("not RISC-V instruction set architecture")
+//!         }
+//!     };
+//!     SbiRet { error, value }
+//! }
+//!
+//! #[inline]
+//! pub fn get_spec_version() -> usize {
+//!     sbi_call(EXTENSION_BASE, FUNCTION_BASE_GET_SPEC_VERSION, 0, 0).value
+//! }
+//! ```
+//!
+//! Complex SBI functions may fail. In this example we only take the value, but in complete designs 
+//! we should handle the `error` value returned from SbiRet.
+//!
+//! You may use other languages to call SBI environment. In C programming language, we can call like this:
+//!
+//! ```text
+//! #define SBI_CALL(module, funct, arg0, arg1, arg2, arg3) ({ \
+//!     register uintptr_t a0 asm ("a0") = (uintptr_t)(arg0); \
+//!     register uintptr_t a1 asm ("a1") = (uintptr_t)(arg1); \
+//!     register uintptr_t a2 asm ("a2") = (uintptr_t)(arg2); \
+//!     register uintptr_t a3 asm ("a3") = (uintptr_t)(arg3); \
+//!     register uintptr_t a7 asm ("a6") = (uintptr_t)(funct); \
+//!     register uintptr_t a7 asm ("a7") = (uintptr_t)(module); \
+//!     asm volatile ("ecall" \
+//!         : "+r" (a0), "+r" (a1) \
+//!         : "r" (a1), "r" (a2), "r" (a3), "r" (a6), "r" (a7) \
+//!         : "memory") \
+//!     {a0, a1}; \
+//! })
+//! 
+//! #define SBI_CALL_0(module, funct) SBI_CALL(module, funct, 0, 0, 0, 0)
+//!
+//! static inline unsigned long get_spec_version() {
+//!     SBI_CALL_0(EXTENSION_BASE, FUNCTION_BASE_GET_SPEC_VERSION).value
+//! }
+//! ```
+//!
+//! # Notes for RustSBI developers
+//!
 //! This library adapts to embedded Rust's `embedded-hal` crate to provide basical SBI features. 
 //! When building for own platform, implement traits in this library and pass them to the functions
 //! begin with `init`. After that, you may call `rustsbi::ecall` in your own exception handler