4
0

lib.rs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774
  1. //! Minimal startup / runtime for RISC-V CPU's
  2. //!
  3. //! # Minimum Supported Rust Version (MSRV)
  4. //!
  5. //! This crate is guaranteed to compile on stable Rust 1.60 and up. It *might*
  6. //! compile with older versions but that may change in any new patch release.
  7. //!
  8. //! # Features
  9. //!
  10. //! This crate provides
  11. //!
  12. //! - Before main initialization of the `.bss` and `.data` sections.
  13. //!
  14. //! - `#[entry]` to declare the entry point of the program
  15. //! - `#[pre_init]` to run code *before* `static` variables are initialized
  16. //!
  17. //! - A linker script that encodes the memory layout of a generic RISC-V
  18. //! microcontroller. This linker script is missing some information that must
  19. //! be supplied through a `memory.x` file (see example below). This file
  20. //! must be supplied using rustflags and listed *before* `link.x`. Arbitrary
  21. //! filename can be use instead of `memory.x`.
  22. //!
  23. //! - A `_sheap` symbol at whose address you can locate a heap.
  24. //!
  25. //! - 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)
  26. //!
  27. //! ``` text
  28. //! $ cargo new --bin app && cd $_
  29. //!
  30. //! $ # add this crate as a dependency
  31. //! $ edit Cargo.toml && cat $_
  32. //! [dependencies]
  33. //! riscv-rt = "0.6.1"
  34. //! panic-halt = "0.2.0"
  35. //!
  36. //! $ # memory layout of the device
  37. //! $ edit memory.x && cat $_
  38. //! MEMORY
  39. //! {
  40. //! RAM : ORIGIN = 0x80000000, LENGTH = 16K
  41. //! FLASH : ORIGIN = 0x20000000, LENGTH = 16M
  42. //! }
  43. //!
  44. //! REGION_ALIAS("REGION_TEXT", FLASH);
  45. //! REGION_ALIAS("REGION_RODATA", FLASH);
  46. //! REGION_ALIAS("REGION_DATA", RAM);
  47. //! REGION_ALIAS("REGION_BSS", RAM);
  48. //! REGION_ALIAS("REGION_HEAP", RAM);
  49. //! REGION_ALIAS("REGION_STACK", RAM);
  50. //!
  51. //! $ edit src/main.rs && cat $_
  52. //! ```
  53. //!
  54. //! ``` ignore,no_run
  55. //! #![no_std]
  56. //! #![no_main]
  57. //!
  58. //! extern crate panic_halt;
  59. //!
  60. //! use riscv_rt::entry;
  61. //!
  62. //! // use `main` as the entry point of this application
  63. //! // `main` is not allowed to return
  64. //! #[entry]
  65. //! fn main() -> ! {
  66. //! // do something here
  67. //! loop { }
  68. //! }
  69. //! ```
  70. //!
  71. //! ``` text
  72. //! $ mkdir .cargo && edit .cargo/config && cat $_
  73. //! [target.riscv32imac-unknown-none-elf]
  74. //! rustflags = [
  75. //! "-C", "link-arg=-Tmemory.x",
  76. //! "-C", "link-arg=-Tlink.x",
  77. //! ]
  78. //!
  79. //! [build]
  80. //! target = "riscv32imac-unknown-none-elf"
  81. //! $ edit build.rs && cat $_
  82. //! ```
  83. //!
  84. //! ``` ignore,no_run
  85. //! use std::env;
  86. //! use std::fs;
  87. //! use std::path::PathBuf;
  88. //!
  89. //! fn main() {
  90. //! let out_dir = PathBuf::from(env::var("OUT_DIR").unwrap());
  91. //!
  92. //! // Put the linker script somewhere the linker can find it.
  93. //! fs::write(out_dir.join("memory.x"), include_bytes!("memory.x")).unwrap();
  94. //! println!("cargo:rustc-link-search={}", out_dir.display());
  95. //! println!("cargo:rerun-if-changed=memory.x");
  96. //!
  97. //! println!("cargo:rerun-if-changed=build.rs");
  98. //! }
  99. //! ```
  100. //!
  101. //! ``` text
  102. //! $ cargo build
  103. //!
  104. //! $ riscv32-unknown-elf-objdump -Cd $(find target -name app) | head
  105. //!
  106. //! Disassembly of section .text:
  107. //!
  108. //! 20000000 <_start>:
  109. //! 20000000: 800011b7 lui gp,0x80001
  110. //! 20000004: 80018193 addi gp,gp,-2048 # 80000800 <_stack_start+0xffffc800>
  111. //! 20000008: 80004137 lui sp,0x80004
  112. //! ```
  113. //!
  114. //! # Symbol interfaces
  115. //!
  116. //! This crate makes heavy use of symbols, linker sections and linker scripts to
  117. //! provide most of its functionality. Below are described the main symbol
  118. //! interfaces.
  119. //!
  120. //! ## `memory.x`
  121. //!
  122. //! This file supplies the information about the device to the linker.
  123. //!
  124. //! ### `MEMORY`
  125. //!
  126. //! The main information that this file must provide is the memory layout of
  127. //! the device in the form of the `MEMORY` command. The command is documented
  128. //! [here][2], but at a minimum you'll want to create at least one memory region.
  129. //!
  130. //! [2]: https://sourceware.org/binutils/docs/ld/MEMORY.html
  131. //!
  132. //! To support different relocation models (RAM-only, FLASH+RAM) multiple regions are used:
  133. //!
  134. //! - `REGION_TEXT` - for `.init`, `.trap` and `.text` sections
  135. //! - `REGION_RODATA` - for `.rodata` section and storing initial values for `.data` section
  136. //! - `REGION_DATA` - for `.data` section
  137. //! - `REGION_BSS` - for `.bss` section
  138. //! - `REGION_HEAP` - for the heap area
  139. //! - `REGION_STACK` - for hart stacks
  140. //!
  141. //! Specific aliases for these regions must be defined in `memory.x` file (see example below).
  142. //!
  143. //! ### `_stext`
  144. //!
  145. //! This symbol provides the loading address of `.text` section. This value can be changed
  146. //! to override the loading address of the firmware (for example, in case of bootloader present).
  147. //!
  148. //! If omitted this symbol value will default to `ORIGIN(REGION_TEXT)`.
  149. //!
  150. //! ### `_stack_start`
  151. //!
  152. //! This symbol provides the address at which the call stack will be allocated.
  153. //! The call stack grows downwards so this address is usually set to the highest
  154. //! valid RAM address plus one (this *is* an invalid address but the processor
  155. //! will decrement the stack pointer *before* using its value as an address).
  156. //!
  157. //! In case of multiple harts present, this address defines the initial stack pointer for hart 0.
  158. //! Stack pointer for hart `N` is calculated as `_stack_start - N * _hart_stack_size`.
  159. //!
  160. //! If omitted this symbol value will default to `ORIGIN(REGION_STACK) + LENGTH(REGION_STACK)`.
  161. //!
  162. //! #### Example
  163. //!
  164. //! Allocating the call stack on a different RAM region.
  165. //!
  166. //! ``` text
  167. //! MEMORY
  168. //! {
  169. //! L2_LIM : ORIGIN = 0x08000000, LENGTH = 1M
  170. //! RAM : ORIGIN = 0x80000000, LENGTH = 16K
  171. //! FLASH : ORIGIN = 0x20000000, LENGTH = 16M
  172. //! }
  173. //!
  174. //! REGION_ALIAS("REGION_TEXT", FLASH);
  175. //! REGION_ALIAS("REGION_RODATA", FLASH);
  176. //! REGION_ALIAS("REGION_DATA", RAM);
  177. //! REGION_ALIAS("REGION_BSS", RAM);
  178. //! REGION_ALIAS("REGION_HEAP", RAM);
  179. //! REGION_ALIAS("REGION_STACK", L2_LIM);
  180. //!
  181. //! _stack_start = ORIGIN(L2_LIM) + LENGTH(L2_LIM);
  182. //! ```
  183. //!
  184. //! ### `_max_hart_id`
  185. //!
  186. //! This symbol defines the maximum hart id supported. All harts with id
  187. //! greater than `_max_hart_id` will be redirected to `abort()`.
  188. //!
  189. //! This symbol is supposed to be redefined in platform support crates for
  190. //! multi-core targets.
  191. //!
  192. //! If omitted this symbol value will default to 0 (single core).
  193. //!
  194. //! ### `_hart_stack_size`
  195. //!
  196. //! This symbol defines stack area size for *one* hart.
  197. //!
  198. //! If omitted this symbol value will default to 2K.
  199. //!
  200. //! ### `_heap_size`
  201. //!
  202. //! This symbol provides the size of a heap region. The default value is 0. You can set `_heap_size`
  203. //! to a non-zero value if you are planning to use heap allocations.
  204. //!
  205. //! ### `_sheap`
  206. //!
  207. //! This symbol is located in RAM right after the `.bss` and `.data` sections.
  208. //! You can use the address of this symbol as the start address of a heap
  209. //! region. This symbol is 4 byte aligned so that address will be a multiple of 4.
  210. //!
  211. //! #### Example
  212. //!
  213. //! ``` no_run
  214. //! extern crate some_allocator;
  215. //!
  216. //! extern "C" {
  217. //! static _sheap: u8;
  218. //! static _heap_size: u8;
  219. //! }
  220. //!
  221. //! fn main() {
  222. //! unsafe {
  223. //! let heap_bottom = &_sheap as *const u8 as usize;
  224. //! let heap_size = &_heap_size as *const u8 as usize;
  225. //! some_allocator::initialize(heap_bottom, heap_size);
  226. //! }
  227. //! }
  228. //! ```
  229. //!
  230. //! ### `_mp_hook`
  231. //!
  232. //! This function is called from all the harts and must return true only for one hart,
  233. //! which will perform memory initialization. For other harts it must return false
  234. //! and implement wake-up in platform-dependent way (e.g. after waiting for a user interrupt).
  235. //! The parameter `hartid` specifies the hartid of the caller.
  236. //!
  237. //! This function can be redefined in the following way:
  238. //!
  239. //! ``` no_run
  240. //! #[export_name = "_mp_hook"]
  241. //! pub extern "Rust" fn mp_hook(hartid: usize) -> bool {
  242. //! // ...
  243. //! }
  244. //! ```
  245. //!
  246. //! Default implementation of this function wakes hart 0 and busy-loops all the other harts.
  247. //!
  248. //!
  249. //! ### Core exception handlers
  250. //!
  251. //! This functions are called when corresponding exception occurs.
  252. //! You can define an exception handler with one of the following names:
  253. //! * `InstructionMisaligned`
  254. //! * `InstructionFault`
  255. //! * `IllegalInstruction`
  256. //! * `Breakpoint`
  257. //! * `LoadMisaligned`
  258. //! * `LoadFault`
  259. //! * `StoreMisaligned`
  260. //! * `StoreFault`
  261. //! * `UserEnvCall`
  262. //! * `SupervisorEnvCall`
  263. //! * `MachineEnvCall`
  264. //! * `InstructionPageFault`
  265. //! * `LoadPageFault`
  266. //! * `StorePageFault`
  267. //!
  268. //! For example:
  269. //! ``` no_run
  270. //! #[export_name = "MachineEnvCall"]
  271. //! fn custom_menv_call_handler(trap_frame: &riscv_rt::TrapFrame) {
  272. //! // ...
  273. //! }
  274. //! ```
  275. //! or
  276. //! ``` no_run
  277. //! #[no_mangle]
  278. //! fn MachineEnvCall(trap_frame: &riscv_rt::TrapFrame) -> ! {
  279. //! // ...
  280. //! }
  281. //! ```
  282. //!
  283. //! If exception handler is not explicitly defined, `ExceptionHandler` is called.
  284. //!
  285. //! ### `ExceptionHandler`
  286. //!
  287. //! This function is called when exception without defined exception handler is occured.
  288. //! The exception reason can be decoded from the
  289. //! `mcause`/`scause` register.
  290. //!
  291. //! This function can be redefined in the following way:
  292. //!
  293. //! ``` no_run
  294. //! #[export_name = "ExceptionHandler"]
  295. //! fn custom_exception_handler(trap_frame: &riscv_rt::TrapFrame) -> ! {
  296. //! // ...
  297. //! }
  298. //! ```
  299. //! or
  300. //! ``` no_run
  301. //! #[no_mangle]
  302. //! fn ExceptionHandler(trap_frame: &riscv_rt::TrapFrame) -> ! {
  303. //! // ...
  304. //! }
  305. //! ```
  306. //!
  307. //! Default implementation of this function stucks in a busy-loop.
  308. //!
  309. //!
  310. //! ### Core interrupt handlers
  311. //!
  312. //! This functions are called when corresponding interrupt is occured.
  313. //! You can define an interrupt handler with one of the following names:
  314. //! * `UserSoft`
  315. //! * `SupervisorSoft`
  316. //! * `MachineSoft`
  317. //! * `UserTimer`
  318. //! * `SupervisorTimer`
  319. //! * `MachineTimer`
  320. //! * `UserExternal`
  321. //! * `SupervisorExternal`
  322. //! * `MachineExternal`
  323. //!
  324. //! For example:
  325. //! ``` no_run
  326. //! #[export_name = "MachineTimer"]
  327. //! fn custom_timer_handler() {
  328. //! // ...
  329. //! }
  330. //! ```
  331. //! or
  332. //! ``` no_run
  333. //! #[no_mangle]
  334. //! fn MachineTimer() {
  335. //! // ...
  336. //! }
  337. //! ```
  338. //!
  339. //! If interrupt handler is not explicitly defined, `DefaultHandler` is called.
  340. //!
  341. //! ### `DefaultHandler`
  342. //!
  343. //! This function is called when interrupt without defined interrupt handler is occured.
  344. //! The interrupt reason can be decoded from the `mcause`/`scause` register.
  345. //!
  346. //! This function can be redefined in the following way:
  347. //!
  348. //! ``` no_run
  349. //! #[export_name = "DefaultHandler"]
  350. //! fn custom_interrupt_handler() {
  351. //! // ...
  352. //! }
  353. //! ```
  354. //! or
  355. //! ``` no_run
  356. //! #[no_mangle]
  357. //! fn DefaultHandler() {
  358. //! // ...
  359. //! }
  360. //! ```
  361. //!
  362. //! Default implementation of this function stucks in a busy-loop.
  363. //!
  364. //! # Features
  365. //!
  366. //! ## `single-hart`
  367. //!
  368. //! This feature saves a little code size if there is only one hart on the target.
  369. //!
  370. //! ## `s-mode`
  371. //!
  372. //! The supervisor mode feature (`s-mode`) can be activated via [Cargo features](https://doc.rust-lang.org/cargo/reference/features.html).
  373. //!
  374. //! For example:
  375. //! ``` text
  376. //! [dependencies]
  377. //! riscv-rt = {features=["s-mode"]}
  378. //! ```
  379. //! Internally, riscv-rt uses different versions of precompiled static libraries
  380. //! for (i) machine mode and (ii) supervisor mode. If the `s-mode` feature was activated,
  381. //! the build script selects the s-mode library. While most registers/instructions have variants for
  382. //! both `mcause` and `scause`, the `mhartid` hardware thread register is not available in supervisor
  383. //! mode. Instead, the hartid is passed as parameter by a bootstrapping firmware (i.e., SBI).
  384. //!
  385. //! Use case: QEMU supports [OpenSBI](https://github.com/riscv-software-src/opensbi) as default firmware.
  386. //! Using the SBI requires riscv-rt to be run in supervisor mode instead of machine mode.
  387. //! ``` text
  388. //! APP_BINARY=$(find target -name app)
  389. //! sudo qemu-system-riscv64 -m 2G -nographic -machine virt -kernel $APP_BINARY
  390. //! ```
  391. //! It requires the memory layout to be non-overlapping, like
  392. //! ``` text
  393. //! MEMORY
  394. //! {
  395. //! RAM : ORIGIN = 0x80200000, LENGTH = 0x8000000
  396. //! FLASH : ORIGIN = 0x20000000, LENGTH = 16M
  397. //! }
  398. //! ```
  399. // NOTE: Adapted from cortex-m/src/lib.rs
  400. #![no_std]
  401. #![deny(missing_docs)]
  402. #[cfg(riscv)]
  403. mod asm;
  404. use core::sync::atomic::{compiler_fence, Ordering};
  405. #[cfg(feature = "s-mode")]
  406. use riscv::register::{scause as xcause, stvec as xtvec, stvec::TrapMode as xTrapMode};
  407. #[cfg(not(feature = "s-mode"))]
  408. use riscv::register::{mcause as xcause, mtvec as xtvec, mtvec::TrapMode as xTrapMode};
  409. #[cfg(all(not(feature = "single-hart"), not(feature = "s-mode")))]
  410. use riscv::register::mhartid;
  411. #[cfg(all(feature = "s-mode", any(riscvf, riscvd)))]
  412. use riscv::register::sstatus as xstatus;
  413. #[cfg(all(not(feature = "s-mode"), any(riscvf, riscvd)))]
  414. use riscv::register::mstatus as xstatus;
  415. pub use riscv_rt_macros::{entry, pre_init};
  416. /// We export this static with an informative name so that if an application attempts to link
  417. /// two copies of riscv-rt together, linking will fail. We also declare a links key in
  418. /// Cargo.toml which is the more modern way to solve the same problem, but we have to keep
  419. /// __ONCE__ around to prevent linking with versions before the links key was added.
  420. #[export_name = "error: riscv-rt appears more than once in the dependency graph"]
  421. #[doc(hidden)]
  422. pub static __ONCE__: () = ();
  423. /// Rust entry point (_start_rust)
  424. ///
  425. /// Zeros bss section, initializes data section and calls main. This function never returns.
  426. ///
  427. /// # Safety
  428. ///
  429. /// This function must be called only from assembly `_start` function.
  430. /// Do **NOT** call this function directly.
  431. #[link_section = ".init.rust"]
  432. #[export_name = "_start_rust"]
  433. pub unsafe extern "C" fn start_rust(a0: usize, a1: usize, a2: usize) -> ! {
  434. #[rustfmt::skip]
  435. extern "Rust" {
  436. // This symbol will be provided by the user via `#[entry]`
  437. fn main(a0: usize, a1: usize, a2: usize) -> !;
  438. // This symbol will be provided by the user via `#[pre_init]`
  439. fn __pre_init();
  440. fn _setup_interrupts();
  441. fn _mp_hook(hartid: usize) -> bool;
  442. }
  443. #[cfg(not(feature = "single-hart"))]
  444. let run_init = {
  445. // sbi passes hartid as first parameter (a0)
  446. #[cfg(feature = "s-mode")]
  447. let hartid = a0;
  448. #[cfg(not(feature = "s-mode"))]
  449. let hartid = mhartid::read();
  450. _mp_hook(hartid)
  451. };
  452. #[cfg(feature = "single-hart")]
  453. let run_init = true;
  454. if run_init {
  455. __pre_init();
  456. // Initialize RAM
  457. // 1. Copy over .data from flash to RAM
  458. // 2. Zero out .bss
  459. #[cfg(target_arch = "riscv32")]
  460. core::arch::asm!(
  461. "
  462. // Copy over .data
  463. la {start},_sdata
  464. la {end},_edata
  465. la {input},_sidata
  466. bgeu {start},{end},2f
  467. 1:
  468. lw {a},0({input})
  469. addi {input},{input},4
  470. sw {a},0({start})
  471. addi {start},{start},4
  472. bltu {start},{end},1b
  473. 2:
  474. li {a},0
  475. li {input},0
  476. // Zero out .bss
  477. la {start},_sbss
  478. la {end},_ebss
  479. bgeu {start},{end},3f
  480. 2:
  481. sw zero,0({start})
  482. addi {start},{start},4
  483. bltu {start},{end},2b
  484. 3:
  485. li {start},0
  486. li {end},0
  487. ",
  488. start = out(reg) _,
  489. end = out(reg) _,
  490. input = out(reg) _,
  491. a = out(reg) _,
  492. );
  493. #[cfg(target_arch = "riscv64")]
  494. core::arch::asm!(
  495. "
  496. // Copy over .data
  497. la {start},_sdata
  498. la {end},_edata
  499. la {input},_sidata
  500. bgeu {start},{end},2f
  501. 1: // .data Main Loop
  502. ld {a},0({input})
  503. addi {input},{input},8
  504. sd {a},0({start})
  505. addi {start},{start},8
  506. bltu {start},{end},1b
  507. 2: // .data zero registers
  508. li {a},0
  509. li {input},0
  510. la {start},_sbss
  511. la {end},_ebss
  512. bgeu {start},{end},4f
  513. 3: // .bss main loop
  514. sd zero,0({start})
  515. addi {start},{start},8
  516. bltu {start},{end},3b
  517. 4: // .bss zero registers
  518. // Zero out used registers
  519. li {start},0
  520. li {end},0
  521. ",
  522. start = out(reg) _,
  523. end = out(reg) _,
  524. input = out(reg) _,
  525. a = out(reg) _,
  526. );
  527. compiler_fence(Ordering::SeqCst);
  528. }
  529. #[cfg(any(riscvf, riscvd))]
  530. {
  531. xstatus::set_fs(xstatus::FS::Initial); // Enable fpu in xstatus
  532. core::arch::asm!("fscsr x0"); // Zero out fcsr register csrrw x0, fcsr, x0
  533. // Zero out floating point registers
  534. #[cfg(all(target_arch = "riscv32", riscvd))]
  535. riscv_rt_macros::loop_asm!("fcvt.d.w f{}, x0", 32);
  536. #[cfg(all(target_arch = "riscv64", riscvd))]
  537. riscv_rt_macros::loop_asm!("fmv.d.x f{}, x0", 32);
  538. #[cfg(not(riscvd))]
  539. riscv_rt_macros::loop_asm!("fmv.w.x f{}, x0", 32);
  540. }
  541. _setup_interrupts();
  542. main(a0, a1, a2);
  543. }
  544. /// Registers saved in trap handler
  545. #[allow(missing_docs)]
  546. #[repr(C)]
  547. #[derive(Debug)]
  548. pub struct TrapFrame {
  549. pub ra: usize,
  550. pub t0: usize,
  551. pub t1: usize,
  552. pub t2: usize,
  553. pub t3: usize,
  554. pub t4: usize,
  555. pub t5: usize,
  556. pub t6: usize,
  557. pub a0: usize,
  558. pub a1: usize,
  559. pub a2: usize,
  560. pub a3: usize,
  561. pub a4: usize,
  562. pub a5: usize,
  563. pub a6: usize,
  564. pub a7: usize,
  565. }
  566. /// Trap entry point rust (_start_trap_rust)
  567. ///
  568. /// `scause`/`mcause` is read to determine the cause of the trap. XLEN-1 bit indicates
  569. /// if it's an interrupt or an exception. The result is examined and ExceptionHandler
  570. /// or one of the core interrupt handlers is called.
  571. ///
  572. /// # Safety
  573. ///
  574. /// This function must be called only from assembly `_start_trap` function.
  575. /// Do **NOT** call this function directly.
  576. #[link_section = ".trap.rust"]
  577. #[export_name = "_start_trap_rust"]
  578. pub unsafe extern "C" fn start_trap_rust(trap_frame: *const TrapFrame) {
  579. extern "C" {
  580. fn ExceptionHandler(trap_frame: &TrapFrame);
  581. fn DefaultHandler();
  582. }
  583. let cause = xcause::read();
  584. let code = cause.code();
  585. if cause.is_exception() {
  586. let trap_frame = &*trap_frame;
  587. if code < __EXCEPTIONS.len() {
  588. let h = &__EXCEPTIONS[code];
  589. if let Some(handler) = h {
  590. handler(trap_frame);
  591. } else {
  592. ExceptionHandler(trap_frame);
  593. }
  594. } else {
  595. ExceptionHandler(trap_frame);
  596. }
  597. ExceptionHandler(trap_frame)
  598. } else if code < __INTERRUPTS.len() {
  599. let h = &__INTERRUPTS[code];
  600. if let Some(handler) = h {
  601. handler();
  602. } else {
  603. DefaultHandler();
  604. }
  605. } else {
  606. DefaultHandler();
  607. }
  608. }
  609. #[doc(hidden)]
  610. #[no_mangle]
  611. #[allow(unused_variables, non_snake_case)]
  612. pub fn DefaultExceptionHandler(trap_frame: &TrapFrame) -> ! {
  613. loop {
  614. // Prevent this from turning into a UDF instruction
  615. // see rust-lang/rust#28728 for details
  616. continue;
  617. }
  618. }
  619. #[doc(hidden)]
  620. #[no_mangle]
  621. #[allow(non_snake_case)]
  622. pub fn DefaultInterruptHandler() {
  623. loop {
  624. // Prevent this from turning into a UDF instruction
  625. // see rust-lang/rust#28728 for details
  626. continue;
  627. }
  628. }
  629. extern "C" {
  630. fn InstructionMisaligned(trap_frame: &TrapFrame);
  631. fn InstructionFault(trap_frame: &TrapFrame);
  632. fn IllegalInstruction(trap_frame: &TrapFrame);
  633. fn Breakpoint(trap_frame: &TrapFrame);
  634. fn LoadMisaligned(trap_frame: &TrapFrame);
  635. fn LoadFault(trap_frame: &TrapFrame);
  636. fn StoreMisaligned(trap_frame: &TrapFrame);
  637. fn StoreFault(trap_frame: &TrapFrame);
  638. fn UserEnvCall(trap_frame: &TrapFrame);
  639. fn SupervisorEnvCall(trap_frame: &TrapFrame);
  640. fn MachineEnvCall(trap_frame: &TrapFrame);
  641. fn InstructionPageFault(trap_frame: &TrapFrame);
  642. fn LoadPageFault(trap_frame: &TrapFrame);
  643. fn StorePageFault(trap_frame: &TrapFrame);
  644. }
  645. #[doc(hidden)]
  646. #[no_mangle]
  647. pub static __EXCEPTIONS: [Option<unsafe extern "C" fn(&TrapFrame)>; 16] = [
  648. Some(InstructionMisaligned),
  649. Some(InstructionFault),
  650. Some(IllegalInstruction),
  651. Some(Breakpoint),
  652. Some(LoadMisaligned),
  653. Some(LoadFault),
  654. Some(StoreMisaligned),
  655. Some(StoreFault),
  656. Some(UserEnvCall),
  657. Some(SupervisorEnvCall),
  658. None,
  659. Some(MachineEnvCall),
  660. Some(InstructionPageFault),
  661. Some(LoadPageFault),
  662. None,
  663. Some(StorePageFault),
  664. ];
  665. extern "C" {
  666. fn SupervisorSoft();
  667. fn MachineSoft();
  668. fn SupervisorTimer();
  669. fn MachineTimer();
  670. fn SupervisorExternal();
  671. fn MachineExternal();
  672. }
  673. #[doc(hidden)]
  674. #[no_mangle]
  675. pub static __INTERRUPTS: [Option<unsafe extern "C" fn()>; 12] = [
  676. None,
  677. Some(SupervisorSoft),
  678. None,
  679. Some(MachineSoft),
  680. None,
  681. Some(SupervisorTimer),
  682. None,
  683. Some(MachineTimer),
  684. None,
  685. Some(SupervisorExternal),
  686. None,
  687. Some(MachineExternal),
  688. ];
  689. /// Default implementation of `_pre_init` does nothing.
  690. /// Users can override this function with the [`#[pre_init]`] macro.
  691. #[doc(hidden)]
  692. #[no_mangle]
  693. #[rustfmt::skip]
  694. pub extern "Rust" fn default_pre_init() {}
  695. /// Default implementation of `_mp_hook` wakes hart 0 and busy-loops all the other harts.
  696. /// Users can override this function by defining their own `_mp_hook`.
  697. ///
  698. /// # Note
  699. ///
  700. /// If the `single-hart` feature is enabled, `_mp_hook` is not called.
  701. #[doc(hidden)]
  702. #[no_mangle]
  703. #[rustfmt::skip]
  704. pub extern "Rust" fn default_mp_hook(hartid: usize) -> bool {
  705. match hartid {
  706. 0 => true,
  707. _ => loop {
  708. unsafe { riscv::asm::wfi() }
  709. },
  710. }
  711. }
  712. /// Default implementation of `_setup_interrupts` sets `mtvec`/`stvec` to the address of `_start_trap`.
  713. #[doc(hidden)]
  714. #[no_mangle]
  715. #[rustfmt::skip]
  716. pub unsafe extern "Rust" fn default_setup_interrupts() {
  717. extern "C" {
  718. fn _start_trap();
  719. }
  720. xtvec::write(_start_trap as usize, xTrapMode::Direct);
  721. }