David Craven пре 7 година
комит
2266921af5
7 измењених фајлова са 522 додато и 0 уклоњено
  1. 2 0
      riscv-rt/.gitignore
  2. 13 0
      riscv-rt/Cargo.toml
  3. 18 0
      riscv-rt/README.md
  4. 18 0
      riscv-rt/build.rs
  5. 83 0
      riscv-rt/link.x
  6. 45 0
      riscv-rt/src/lang_items.rs
  7. 343 0
      riscv-rt/src/lib.rs

+ 2 - 0
riscv-rt/.gitignore

@@ -0,0 +1,2 @@
+Cargo.lock
+target/

+ 13 - 0
riscv-rt/Cargo.toml

@@ -0,0 +1,13 @@
+[package]
+name = "riscv-rt"
+version = "0.1.0"
+repository = "https://github.com/dvc94ch/riscv-rt"
+authors = ["David Craven <david@craven.ch>"]
+categories = ["embedded", "no-std"]
+description = "Minimal runtime / startup for RISCV CPU's"
+keywords = ["riscv", "runtime", "startup"]
+license = "ISC"
+
+[dependencies]
+r0 = "0.2.1"
+riscv = "0.1.0"

+ 18 - 0
riscv-rt/README.md

@@ -0,0 +1,18 @@
+# `riscv-rt`
+
+> Minimal runtime / startup for RISCV CPU's.
+
+# License
+Copyright 2017 David Craven
+
+Permission to use, copy, modify, and/or distribute this software for any purpose
+with or without fee is hereby granted, provided that the above copyright notice
+and this permission notice appear in all copies.
+
+THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
+REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
+INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
+OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
+TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF
+THIS SOFTWARE.

+ 18 - 0
riscv-rt/build.rs

@@ -0,0 +1,18 @@
+// NOTE: Adapted from cortex-m/build.rs
+use std::env;
+use std::fs::File;
+use std::io::Write;
+use std::path::PathBuf;
+
+fn main() {
+    // Put the linker script somewhere the linker can find it
+    let out = &PathBuf::from(env::var_os("OUT_DIR").unwrap());
+    File::create(out.join("link.x"))
+        .unwrap()
+        .write_all(include_bytes!("link.x"))
+        .unwrap();
+    println!("cargo:rustc-link-search={}", out.display());
+
+    println!("cargo:rerun-if-changed=build.rs");
+    println!("cargo:rerun-if-changed=link.x");
+}

+ 83 - 0
riscv-rt/link.x

@@ -0,0 +1,83 @@
+/* NOTE: Adapted from cortex-m/link.x */
+INCLUDE memory.x
+
+PROVIDE(_stack_start = ORIGIN(RAM) + LENGTH(RAM));
+
+SECTIONS
+{
+  PROVIDE(_stext = ORIGIN(FLASH));
+
+  .text _stext : ALIGN(4)
+  {
+    /* Put reset handler first in .text section so it ends up as the entry */
+    /* point of the program. */
+    KEEP(*(.init));
+    KEEP(*(.init.rust));
+    KEEP(*(.trap));
+    KEEP(*(.trap.rust));
+
+    *(.text .text.*);
+  } > FLASH
+
+  .rodata : ALIGN(4)
+  {
+    *(.rodata .rodata.*);
+    . = ALIGN(4);
+  } > FLASH
+
+  .bss : ALIGN(4)
+  {
+    _sbss = .;
+    *(.bss .bss.*);
+    . = ALIGN(4);
+    _ebss = .;
+  } > RAM
+
+  .data : ALIGN(4)
+  {
+    _sidata = LOADADDR(.data);
+    _sdata = .;
+    /* Must be called __global_pointer$ for linker relaxations to work. */
+    PROVIDE(__global_pointer$ = . + 0x800);
+    *(.data .data.*);
+    . = ALIGN(4);
+    _edata = .;
+  } > RAM AT > FLASH /* LLD fails on AT > FLASH */
+
+  /* fake output .got section */
+  /* Dynamic relocations are unsupported. This section is only used to detect
+     relocatable code in the input files and raise an error if relocatable code
+     is found */
+  .got :
+  {
+    _sgot = .;
+    KEEP(*(.got .got.*));
+    _egot = .;
+  } > RAM AT > FLASH /* LLD fails on AT > FLASH */
+
+  /* The heap starts right after the .bss + .data section ends */
+  _sheap = _edata;
+
+  /* Due to an unfortunate combination of legacy concerns,
+     toolchain drawbacks, and insufficient attention to detail,
+     rustc has no choice but to mark .debug_gdb_scripts as allocatable.
+     We really do not want to upload it to our target, so we
+     remove the allocatable bit. Unfortunately, it appears
+     that the only way to do this in a linker script is
+     the extremely obscure "INFO" output section type specifier. */
+  /* a rustc hack will force the program to read the first byte of this section,
+     so we'll set the (fake) start address of this section to something we're
+     sure can be read at runtime: the start of the .text section */
+  /* LLD fails to parse _stext (INFO) */
+  .debug_gdb_scripts _stext (INFO) : {
+    KEEP(*(.debug_gdb_scripts))
+  }
+}
+
+/* Do not exceed this mark in the error messages below                | */
+ASSERT(_sgot == _egot, "
+.got section detected in the input files. Dynamic relocations are not
+supported. If you are linking to C code compiled using the `gcc` crate
+then modify your build script to compile the C code _without_ the
+-fPIC flag. See the documentation of the `gcc::Config.fpic` method for
+details.");

+ 45 - 0
riscv-rt/src/lang_items.rs

@@ -0,0 +1,45 @@
+// NOTE: Adapted from cortex-m/src/lang_items.rs
+
+use riscv::asm;
+
+/// Default panic handler
+#[linkage = "weak"]
+#[lang = "panic_fmt"]
+unsafe extern "C" fn panic_fmt(
+    _: ::core::fmt::Arguments, // fmt
+    _: &'static str, // file
+    _: u32, // line
+    _: u32, // col
+) -> ! {
+    asm::ebreak();
+    loop {}
+}
+
+/// Lang item required to make the normal `main` work in applications
+// This is how the `start` lang item works:
+// When `rustc` compiles a binary crate, it creates a `main` function that looks
+// like this:
+//
+// ```
+// #[export_name = "main"]
+// pub extern "C" fn rustc_main(argc: isize, argv: *const *const u8) -> isize {
+//     start(main)
+// }
+// ```
+//
+// Where `start` is this function and `main` is the binary crate's `main`
+// function.
+//
+// The final piece is that the entry point of our program, the reset handler,
+// has to call `rustc_main`. That's covered by the `reset_handler` function in
+// root of this crate.
+#[lang = "start"]
+extern "C" fn start(
+    main: fn(),
+    _argc: isize,
+    _argv: *const *const u8,
+) -> isize {
+    main();
+
+    0
+}

+ 343 - 0
riscv-rt/src/lib.rs

@@ -0,0 +1,343 @@
+//! Minimal startup / runtime for RISCV CPU's
+//!
+//! # Features
+//!
+//! This crate provides
+//!
+//! - Before main initialization of the `.bss` and `.data` sections.
+//!
+//! - Before main initialization of the FPU (for targets that have a FPU).
+//!
+//! - A `panic_fmt` implementation that just calls abort that you can opt into
+//!   through the "abort-on-panic" Cargo feature. If you don't use this feature
+//!   you'll have to provide the `panic_fmt` lang item yourself. Documentation
+//!   [here][1]
+//!
+//! [1]: https://doc.rust-lang.org/unstable-book/language-features/lang-items.html
+//!
+//! - A minimal `start` lang item to support the standard `fn main()`
+//!   interface. (The processor goes to sleep after returning from `main`)
+//!
+//! - 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).
+//!
+//! - A `_sheap` symbol at whose address you can locate a heap.
+//!
+//! ``` text
+//! $ cargo new --bin app && cd $_
+//!
+//! $ # add this crate as a dependency
+//! $ edit Cargo.toml && cat $_
+//! [dependencies.riscv-rt]
+//! version = "0.1.0"
+//!
+//! $ # tell Xargo which standard crates to build
+//! $ edit Xargo.toml && cat $_
+//! [dependencies.core]
+//! stage = 0
+//!
+//! [dependencies.compiler_builtins]
+//! features = ["mem"]
+//! stage = 1
+//!
+//! $ # memory layout of the device
+//! $ edit memory.x && cat $_
+//! MEMORY
+//! {
+//!   /* NOTE K = KiBi = 1024 bytes */
+//!   FLASH : ORIGIN = 0x08000000, LENGTH = 128K
+//!   RAM : ORIGIN = 0x20000000, LENGTH = 8K
+//! }
+//!
+//! $ edit src/main.rs && cat $_
+//! ```
+//!
+//! ``` ignore,no_run
+//! #![no_std]
+//!
+//! extern crate riscv_rt;
+//!
+//! fn main() {
+//!     // do something here
+//! }
+//! ```
+//!
+//! ``` text
+//! $ cargo install xargo
+//!
+//! $ xargo rustc --target riscv32-unknown-none -- \
+//!    -C link-arg=-Tlink.x -C linker=riscv32-unknown-elf-ld -Z linker-flavor=ld
+//!
+//! $ riscv32-unknown-elf-objdump -Cd $(find target -name app) | head
+//!
+//! Disassembly of section .text:
+//!
+//! 20400000 <_start>:
+//! 20400000:	800011b7    lui	gp,0x80001
+//! 20400004:	80018193    addi	gp,gp,-2048 # 80000800 <_stack_start+0xffffc800>
+//! 20400008:	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 two memory regions: one
+//! for Flash memory and another for RAM.
+//!
+//! [2]: https://sourceware.org/binutils/docs/ld/MEMORY.html
+//!
+//! The program instructions (the `.text` section) will be stored in the memory
+//! region named FLASH, and the program `static` variables (the sections `.bss`
+//! and `.data`) will be allocated in the memory region named RAM.
+//!
+//! ### `_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).
+//!
+//! If omitted this symbol value will default to `ORIGIN(RAM) + LENGTH(RAM)`.
+//!
+//! #### Example
+//!
+//! Allocating the call stack on a different RAM region.
+//!
+//! ```
+//! MEMORY
+//! {
+//!   /* call stack will go here */
+//!   CCRAM : ORIGIN = 0x10000000, LENGTH = 8K
+//!   FLASH : ORIGIN = 0x08000000, LENGTH = 256K
+//!   /* static variables will go here */
+//!   RAM : ORIGIN = 0x20000000, LENGTH = 40K
+//! }
+//!
+//! _stack_start = ORIGIN(CCRAM) + LENGTH(CCRAM);
+//! ```
+//!
+//! ### `_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
+//!
+//! ```
+//! extern crate some_allocator;
+//!
+//! // Size of the heap in bytes
+//! const SIZE: usize = 1024;
+//!
+//! extern "C" {
+//!     static mut _sheap: u8;
+//! }
+//!
+//! fn main() {
+//!     unsafe {
+//!         let start_address = &mut _sheap as *mut u8;
+//!         some_allocator::initialize(start_address, SIZE);
+//!     }
+//! }
+//! ```
+
+// NOTE: Adapted from cortex-m/src/lib.rs
+#![no_std]
+#![deny(missing_docs)]
+#![deny(warnings)]
+#![feature(asm)]
+#![feature(compiler_builtins_lib)]
+#![feature(const_fn)]
+#![feature(global_asm)]
+#![feature(lang_items)]
+#![feature(linkage)]
+#![feature(naked_functions)]
+#![feature(used)]
+
+extern crate compiler_builtins;
+extern crate riscv;
+extern crate r0;
+
+mod lang_items;
+
+use riscv::{asm, csr};
+pub use riscv::csr::{Trap, Interrupt, Exception};
+
+extern "C" {
+    // NOTE `rustc` forces this signature on us. See `src/lang_items.rs`
+    fn main(argc: isize, argv: *const *const u8) -> isize;
+
+    // Boundaries of the .bss section
+    static mut _ebss: u32;
+    static mut _sbss: u32;
+
+    // Boundaries of the .data section
+    static mut _edata: u32;
+    static mut _sdata: u32;
+
+    // Initial values of the .data section (stored in Flash)
+    static _sidata: u32;
+
+    // Address of _start_trap
+    static _start_trap: u32;
+}
+
+
+/// Entry point of all programs (_start).
+///
+/// It initializes DWARF call frame information, the stack pointer, the
+/// frame pointer (needed for closures to work in start_rust) and the global
+/// pointer. Then it calls _start_rust.
+// FIXME: .option push, .option norelax and .option pop directives aren't
+// supported by the llvm backend yet.
+global_asm!(r#"
+.section .init
+.globl _start
+_start:
+  .cfi_startproc
+  .cfi_undefined ra
+
+  // .option push
+  // .option norelax
+  lui gp, %hi(__global_pointer$)
+  addi gp, gp, %lo(__global_pointer$)
+  // .option pop
+
+  lui sp, %hi(_stack_start)
+  addi sp, sp, %lo(_stack_start)
+
+  add s0, sp, zero
+
+  jal zero, _start_rust
+
+  .cfi_endproc
+"#);
+
+
+/// Rust entry point (_start_rust)
+///
+/// Zeros bss section, initializes data section and calls main. This function
+/// never returns.
+#[naked]
+#[link_section = ".init.rust"]
+#[export_name = "_start_rust"]
+pub extern "C" fn start_rust() -> ! {
+    unsafe {
+        r0::zero_bss(&mut _sbss, &mut _ebss);
+        r0::init_data(&mut _sdata, &mut _edata, &_sidata);
+    }
+
+    // TODO: Enable FPU when available
+
+    // Set mtvec to _start_trap
+    unsafe {
+        // csr::mtvec.write(|w| w.bits(_start_trap));
+        asm!("csrrw zero, 0x305, $0"
+             :
+             : "r"(&_start_trap)
+             :
+             : "volatile");
+    }
+
+    // Neither `argc` or `argv` make sense in bare metal context so we
+    // just stub them
+    unsafe {
+        main(0, ::core::ptr::null());
+    }
+
+    // If `main` returns, then we go into "reactive" mode and simply attend
+    // interrupts as they occur.
+    loop {
+        asm::wfi();
+    }
+}
+
+
+/// Trap entry point (_start_trap)
+///
+/// Saves caller saved registers ra, t0..6, a0..7, calls _start_trap_rust,
+/// restores caller saved registers and then returns.
+global_asm!(r#"
+  .section .trap
+  .align 4
+  .global _start_trap
+
+_start_trap:
+  addi sp, sp, -16*4
+
+  sw ra, 1*4(sp)
+  sw t0, 2*4(sp)
+  sw t1, 3*4(sp)
+  sw t2, 4*4(sp)
+  sw t3, 5*4(sp)
+  sw t4, 6*4(sp)
+  sw t5, 7*4(sp)
+  sw t6, 8*4(sp)
+  sw a0, 9*4(sp)
+  sw a1, 10*4(sp)
+  sw a2, 11*4(sp)
+  sw a3, 12*4(sp)
+  sw a4, 13*4(sp)
+  sw a5, 14*4(sp)
+  sw a6, 15*4(sp)
+  sw a7, 16*4(sp)
+
+  jal ra, _start_trap_rust
+
+  lw ra, 1*4(sp)
+  lw t0, 2*4(sp)
+  lw t1, 3*4(sp)
+  lw t2, 4*4(sp)
+  lw t3, 5*4(sp)
+  lw t4, 6*4(sp)
+  lw t5, 7*4(sp)
+  lw t6, 8*4(sp)
+  lw a0, 9*4(sp)
+  lw a1, 10*4(sp)
+  lw a2, 11*4(sp)
+  lw a3, 12*4(sp)
+  lw a4, 13*4(sp)
+  lw a5, 14*4(sp)
+  lw a6, 15*4(sp)
+  lw a7, 16*4(sp)
+
+  addi sp, sp, 16*4
+  mret
+"#);
+
+
+/// Trap entry point rust (_start_trap_rust)
+///
+/// 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 converted to an element
+/// of the Interrupt or Exception enum and passed to handle_interrupt or
+/// handle_exception.
+#[link_section = ".trap.rust"]
+#[export_name = "_start_trap_rust"]
+pub extern "C" fn start_trap_rust() {
+    // dispatch trap to handler
+    trap_handler(csr::mcause.read().cause());
+    // mstatus, remain in M-mode after mret
+    csr::mstatus.set(|w| w.mpp(csr::MPP::Machine));
+}
+
+
+/// Default Trap Handler
+#[used]
+#[no_mangle]
+#[linkage = "weak"]
+fn trap_handler(_: Trap) {}