main.rs 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. #![feature(alloc_error_handler)]
  2. #![feature(naked_functions)]
  3. #![feature(fn_align)]
  4. #![no_std]
  5. #![no_main]
  6. #![allow(static_mut_refs)]
  7. extern crate alloc;
  8. #[macro_use]
  9. extern crate log;
  10. #[macro_use]
  11. mod macros;
  12. mod cfg;
  13. mod devicetree;
  14. mod fail;
  15. mod firmware;
  16. mod platform;
  17. mod riscv;
  18. mod sbi;
  19. use core::arch::{asm, naked_asm};
  20. use crate::platform::PLATFORM;
  21. use crate::riscv::csr::menvcfg;
  22. use crate::riscv::current_hartid;
  23. use crate::sbi::extensions::{
  24. Extension, PrivilegedVersion, hart_extension_probe, hart_privileged_version,
  25. privileged_version_detection,
  26. };
  27. use crate::sbi::hart_context::NextStage;
  28. use crate::sbi::heap::sbi_heap_init;
  29. use crate::sbi::hsm::local_remote_hsm;
  30. use crate::sbi::ipi;
  31. use crate::sbi::trap;
  32. use crate::sbi::trap_stack;
  33. pub const R_RISCV_RELATIVE: usize = 3;
  34. #[unsafe(no_mangle)]
  35. extern "C" fn rust_main(_hart_id: usize, opaque: usize, nonstandard_a2: usize) {
  36. // Track whether SBI is initialized and ready.
  37. let boot_hart_info = firmware::get_boot_hart(opaque, nonstandard_a2);
  38. // boot hart task entry.
  39. if boot_hart_info.is_boot_hart {
  40. // Initialize the sbi heap
  41. sbi_heap_init();
  42. // parse the device tree
  43. let fdt_address = boot_hart_info.fdt_address;
  44. unsafe {
  45. PLATFORM.init(fdt_address);
  46. PLATFORM.print_board_info();
  47. }
  48. firmware::set_pmp(unsafe { PLATFORM.info.memory_range.as_ref().unwrap() });
  49. firmware::log_pmp_cfg(unsafe { PLATFORM.info.memory_range.as_ref().unwrap() });
  50. // Get boot information and prepare for kernel entry.
  51. let boot_info = firmware::get_boot_info(nonstandard_a2);
  52. let (mpp, next_addr) = (boot_info.mpp, boot_info.next_address);
  53. // Log boot hart ID and PMP information
  54. let hart_id = current_hartid();
  55. info!("{:<30}: {}", "Boot HART ID", hart_id);
  56. // Detection Priv Version
  57. privileged_version_detection();
  58. let priv_version = hart_privileged_version(hart_id);
  59. info!("{:<30}: {:?}", "Boot HART Privileged Version", priv_version);
  60. // Start kernel.
  61. local_remote_hsm().start(NextStage {
  62. start_addr: next_addr,
  63. next_mode: mpp,
  64. opaque: fdt_address,
  65. });
  66. info!(
  67. "Redirecting hart {} to 0x{:0>16x} in {:?} mode.",
  68. current_hartid(),
  69. next_addr,
  70. mpp
  71. );
  72. } else {
  73. // Other harts task entry.
  74. trap_stack::prepare_for_trap();
  75. // Wait for boot hart to complete SBI initialization.
  76. while !unsafe { PLATFORM.ready() } {
  77. core::hint::spin_loop()
  78. }
  79. firmware::set_pmp(unsafe { PLATFORM.info.memory_range.as_ref().unwrap() });
  80. // Detection Priv Version
  81. privileged_version_detection();
  82. }
  83. // Clear all pending IPIs.
  84. ipi::clear_all();
  85. // Configure CSRs and trap handling.
  86. unsafe {
  87. // Delegate all interrupts and exceptions to supervisor mode.
  88. asm!("csrw mideleg, {}", in(reg) !0);
  89. asm!("csrw medeleg, {}", in(reg) !0);
  90. asm!("csrw mcounteren, {}", in(reg) !0);
  91. asm!("csrw scounteren, {}", in(reg) !0);
  92. use ::riscv::register::{medeleg, mtvec};
  93. // Keep supervisor environment calls and illegal instructions in M-mode.
  94. medeleg::clear_supervisor_env_call();
  95. medeleg::clear_illegal_instruction();
  96. if hart_privileged_version(current_hartid()) >= PrivilegedVersion::Version1_12 {
  97. // Configure environment features based on available extensions.
  98. if hart_extension_probe(current_hartid(), Extension::Sstc) {
  99. menvcfg::set_bits(
  100. menvcfg::STCE | menvcfg::CBIE_INVALIDATE | menvcfg::CBCFE | menvcfg::CBZE,
  101. );
  102. } else {
  103. menvcfg::set_bits(menvcfg::CBIE_INVALIDATE | menvcfg::CBCFE | menvcfg::CBZE);
  104. }
  105. }
  106. // Set up trap handling.
  107. mtvec::write(fast_trap::trap_entry as _, mtvec::TrapMode::Direct);
  108. }
  109. }
  110. #[naked]
  111. #[unsafe(link_section = ".text.entry")]
  112. #[unsafe(export_name = "_start")]
  113. unsafe extern "C" fn start() -> ! {
  114. unsafe {
  115. naked_asm!(
  116. ".option arch, +a",
  117. // 1. Turn off interrupt.
  118. " csrw mie, zero",
  119. // 2. Initialize programming language runtime.
  120. // only clear bss if hartid matches preferred boot hart id.
  121. " csrr t0, mhartid",
  122. " bne t0, zero, 4f",
  123. " call {relocation_update}",
  124. "1:",
  125. // 3. Hart 0 clear bss segment.
  126. " lla t0, sbi_bss_start
  127. lla t1, sbi_bss_end
  128. 2: bgeu t0, t1, 3f
  129. sd zero, 0(t0)
  130. addi t0, t0, 8
  131. j 2b",
  132. "3: ", // Hart 0 set bss ready signal.
  133. " lla t0, 6f
  134. li t1, 1
  135. amoadd.w t0, t1, 0(t0)
  136. j 5f",
  137. "4:", // Other harts are waiting for bss ready signal.
  138. " li t1, 1
  139. lla t0, 6f
  140. lw t0, 0(t0)
  141. bne t0, t1, 4b",
  142. "5:",
  143. // 4. Prepare stack for each hart.
  144. " call {locate_stack}",
  145. " call {main}",
  146. " csrw mscratch, sp",
  147. " j {hart_boot}",
  148. " .balign 4",
  149. "6:", // bss ready signal.
  150. " .word 0",
  151. relocation_update = sym relocation_update,
  152. locate_stack = sym trap_stack::locate,
  153. main = sym rust_main,
  154. hart_boot = sym trap::boot::boot,
  155. )
  156. }
  157. }
  158. // Handle relocations for position-independent code
  159. #[naked]
  160. unsafe extern "C" fn relocation_update() {
  161. unsafe {
  162. naked_asm!(
  163. // Get load offset.
  164. " li t0, {START_ADDRESS}",
  165. " lla t1, sbi_start",
  166. " sub t2, t1, t0",
  167. // Foreach rela.dyn and update relocation.
  168. " lla t0, __rel_dyn_start",
  169. " lla t1, __rel_dyn_end",
  170. " li t3, {R_RISCV_RELATIVE}",
  171. "1:",
  172. " ld t4, 8(t0)",
  173. " bne t4, t3, 2f",
  174. " ld t4, 0(t0)", // Get offset
  175. " ld t5, 16(t0)", // Get append
  176. " add t4, t4, t2", // Add load offset to offset add append
  177. " add t5, t5, t2",
  178. " sd t5, 0(t4)", // Update address
  179. " addi t0, t0, 24", // Get next rela item
  180. "2:",
  181. " blt t0, t1, 1b",
  182. // Return
  183. " ret",
  184. R_RISCV_RELATIVE = const R_RISCV_RELATIVE,
  185. START_ADDRESS = const cfg::SBI_LINK_START_ADDRESS,
  186. )
  187. }
  188. }