main.rs 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. #![no_std]
  2. #![no_main]
  3. #![feature(naked_functions)]
  4. #![allow(static_mut_refs)]
  5. #[macro_use]
  6. extern crate rcore_console;
  7. use core::mem::MaybeUninit;
  8. use core::sync::{atomic::AtomicBool, atomic::AtomicU64, atomic::Ordering};
  9. use core::{arch::asm, ptr::null};
  10. use log::*;
  11. use sbi::SbiRet;
  12. use sbi_spec::binary::{HartMask, MaskError};
  13. use sbi_spec::hsm::hart_state;
  14. use sbi_testing::sbi;
  15. use serde::Deserialize;
  16. use serde_device_tree::{
  17. buildin::{Node, NodeSeq, Reg, StrSeq},
  18. Dtb, DtbPtr,
  19. };
  20. use uart16550::Uart16550;
  21. const RISCV_HEAD_FLAGS: u64 = 0;
  22. const RISCV_HEADER_VERSION: u32 = 0x2;
  23. const RISCV_IMAGE_MAGIC: u64 = 0x5643534952; /* Magic number, little endian, "RISCV" */
  24. const RISCV_IMAGE_MAGIC2: u32 = 0x05435352; /* Magic number 2, little endian, "RSC\x05" */
  25. /// boot header
  26. #[naked]
  27. #[no_mangle]
  28. #[link_section = ".head.text"]
  29. unsafe extern "C" fn _boot_header() -> ! {
  30. asm!(
  31. "j _start",
  32. ".word 0",
  33. ".balign 8",
  34. ".dword 0x200000",
  35. ".dword iend - istart",
  36. ".dword {RISCV_HEAD_FLAGS}",
  37. ".word {RISCV_HEADER_VERSION}",
  38. ".word 0",
  39. ".dword 0",
  40. ".dword {RISCV_IMAGE_MAGIC}",
  41. ".balign 4",
  42. ".word {RISCV_IMAGE_MAGIC2}",
  43. ".word 0",
  44. RISCV_HEAD_FLAGS = const RISCV_HEAD_FLAGS,
  45. RISCV_HEADER_VERSION = const RISCV_HEADER_VERSION,
  46. RISCV_IMAGE_MAGIC = const RISCV_IMAGE_MAGIC,
  47. RISCV_IMAGE_MAGIC2 = const RISCV_IMAGE_MAGIC2,
  48. options(noreturn)
  49. );
  50. }
  51. const STACK_SIZE: usize = 512 * 1024; // 512 KiB
  52. const MAX_HART_NUM: usize = 128;
  53. #[allow(dead_code)]
  54. #[derive(Copy, Clone)]
  55. struct HartStack([u8; STACK_SIZE]);
  56. impl HartStack {
  57. #[inline]
  58. pub const fn new() -> Self {
  59. HartStack([0; STACK_SIZE])
  60. }
  61. }
  62. #[link_section = ".bss.uninit"]
  63. static mut STACK: HartStack = HartStack::new();
  64. #[link_section = ".bss.uninit"]
  65. static mut HART_STACK: [HartStack; MAX_HART_NUM] = [HartStack::new(); MAX_HART_NUM];
  66. #[link_section = ".bss.uninit"]
  67. static mut IPI_SENT: [MaybeUninit<AtomicBool>; MAX_HART_NUM] =
  68. [const { MaybeUninit::uninit() }; MAX_HART_NUM];
  69. #[link_section = ".bss.uninit"]
  70. static mut SMP_COUNT: usize = 0;
  71. #[link_section = ".bss.uninit"]
  72. static mut BOOT_HART_ID: usize = 0;
  73. /// 内核入口。
  74. ///
  75. /// # Safety
  76. ///
  77. /// 裸函数。
  78. #[naked]
  79. #[no_mangle]
  80. #[link_section = ".text.entry"]
  81. unsafe extern "C" fn _start(hartid: usize, device_tree_paddr: usize) -> ! {
  82. asm!(
  83. // clear bss segment
  84. " la t0, sbss
  85. la t1, ebss
  86. 1: bgeu t0, t1, 2f
  87. sd zero, 0(t0)
  88. addi t0, t0, 8
  89. j 1b",
  90. "2:",
  91. " la sp, {stack} + {stack_size}",
  92. " j {main}",
  93. stack_size = const STACK_SIZE,
  94. stack = sym STACK,
  95. main = sym rust_main,
  96. options(noreturn),
  97. )
  98. }
  99. #[naked]
  100. #[no_mangle]
  101. unsafe extern "C" fn init_hart(hartid: usize, opaque: usize) {
  102. asm!(
  103. "add sp, a1, zero",
  104. "csrw sscratch, sp",
  105. "call {init_main}",
  106. init_main = sym init_main,
  107. options(noreturn),
  108. )
  109. }
  110. #[naked]
  111. #[no_mangle]
  112. unsafe extern "C" fn core_send_ipi(hartid: usize, opaque: usize) {
  113. asm!(
  114. "add sp, a1, zero",
  115. "csrw sscratch, sp",
  116. "call {send_ipi}",
  117. send_ipi = sym send_ipi,
  118. options(noreturn),
  119. )
  120. }
  121. extern "C" fn send_ipi(hartid: usize) -> ! {
  122. if unsafe { !(IPI_SENT[hartid].assume_init_mut().load(Ordering::Relaxed)) } {
  123. unsafe {
  124. IPI_SENT[hartid]
  125. .assume_init_mut()
  126. .swap(true, Ordering::AcqRel);
  127. };
  128. let mut mask = Some(HartMask::from_mask_base(0, 0));
  129. for i in 0..unsafe { SMP_COUNT } {
  130. if i == unsafe { BOOT_HART_ID } {
  131. continue;
  132. }
  133. if let Some(ref mut mask) = mask {
  134. match mask.insert(i) {
  135. Ok(_) => continue,
  136. Err(MaskError::InvalidBit) => {
  137. sbi::remote_sfence_vma(*mask, 0, 0);
  138. }
  139. Err(_) => unreachable!("Failed to construct mask"),
  140. }
  141. }
  142. mask = Some(HartMask::from_mask_base(0b1, i));
  143. }
  144. if let Some(mask) = mask {
  145. sbi::remote_sfence_vma(mask, 0, 0);
  146. }
  147. unsafe {
  148. WAIT_COUNT.fetch_sub(1, Ordering::AcqRel);
  149. while WAIT_COUNT.load(Ordering::Relaxed) != 0 {}
  150. }
  151. } else {
  152. unreachable!("resend {}", hartid);
  153. }
  154. sbi::hart_suspend(sbi::NonRetentive, core_send_ipi as _, unsafe {
  155. core::ptr::addr_of!(HART_STACK[hartid + 1]) as _
  156. });
  157. unreachable!()
  158. }
  159. extern "C" fn init_main(hartid: usize) -> ! {
  160. sbi::hart_suspend(sbi::NonRetentive, core_send_ipi as _, unsafe {
  161. core::ptr::addr_of!(HART_STACK[hartid + 1]) as _
  162. });
  163. unreachable!()
  164. }
  165. static mut WAIT_COUNT: AtomicU64 = AtomicU64::new(0);
  166. const SUSPENDED: SbiRet = SbiRet::success(hart_state::SUSPENDED);
  167. fn get_time() -> u64 {
  168. const CSR_TIME: u32 = 0xc01;
  169. let mut low_time: u64;
  170. unsafe {
  171. asm!("csrr {}, {CSR_TIME}", out(reg) low_time, CSR_TIME = const CSR_TIME);
  172. }
  173. low_time
  174. }
  175. extern "C" fn rust_main(hartid: usize, dtb_pa: usize) -> ! {
  176. #[derive(Deserialize)]
  177. struct Tree<'a> {
  178. cpus: Cpus<'a>,
  179. chosen: Chosen<'a>,
  180. }
  181. #[derive(Deserialize)]
  182. #[serde(rename_all = "kebab-case")]
  183. struct Cpus<'a> {
  184. timebase_frequency: u32,
  185. cpu: NodeSeq<'a>,
  186. }
  187. #[derive(Deserialize)]
  188. #[serde(rename_all = "kebab-case")]
  189. struct Chosen<'a> {
  190. stdout_path: StrSeq<'a>,
  191. }
  192. rcore_console::init_console(&Console);
  193. rcore_console::set_log_level(option_env!("LOG"));
  194. let dtb_ptr = DtbPtr::from_raw(dtb_pa as _).unwrap();
  195. let dtb = Dtb::from(dtb_ptr).share();
  196. let root: Node = serde_device_tree::from_raw_mut(&dtb).unwrap();
  197. let tree: Tree = root.deserialize();
  198. let stdout_path = tree.chosen.stdout_path.iter().next().unwrap();
  199. if let Some(node) = root.find(stdout_path) {
  200. let reg = node.get_prop("reg").unwrap().deserialize::<Reg>();
  201. let address = reg.iter().next().unwrap().0.start;
  202. unsafe { UART = Uart16550Map(address as _) };
  203. }
  204. let smp = tree.cpus.cpu.len();
  205. let frequency = tree.cpus.timebase_frequency;
  206. info!(
  207. r"
  208. ____ _ _ __ _
  209. | __ ) ___ _ __ ___| |__ | |/ /___ _ __ _ __ ___| |
  210. | _ \ / _ \ '_ \ / __| '_ \ | ' // _ \ '__| '_ \ / _ \ |
  211. | |_) | __/ | | | (__| | | | | . \ __/ | | | | | __/ |
  212. |____/ \___|_| |_|\___|_| |_| |_|\_\___|_| |_| |_|\___|_|
  213. ==========================================================
  214. | boot hart id | {hartid:20} |
  215. | smp | {smp:20} |
  216. | timebase frequency | {frequency:17} Hz |
  217. | dtb physical address | {dtb_pa:#20x} |
  218. ----------------------------------------------------------"
  219. );
  220. unsafe {
  221. SMP_COUNT = smp;
  222. BOOT_HART_ID = hartid;
  223. }
  224. for i in 0..smp {
  225. unsafe {
  226. IPI_SENT[i].write(AtomicBool::new(false));
  227. }
  228. if i != hartid {
  229. sbi::hart_start(i, init_hart as _, unsafe {
  230. core::ptr::addr_of!(HART_STACK[i + 1]) as _
  231. });
  232. while sbi::hart_get_status(i) != SUSPENDED {
  233. core::hint::spin_loop();
  234. }
  235. }
  236. }
  237. info!("Starting test");
  238. for i in 0..4 {
  239. info!("Test #{i} started");
  240. unsafe {
  241. for (i, ipi_sent) in IPI_SENT.iter_mut().enumerate().take(smp) {
  242. ipi_sent.assume_init_mut().swap(false, Ordering::AcqRel);
  243. if i != hartid {
  244. while sbi::hart_get_status(i) != SUSPENDED {}
  245. }
  246. }
  247. WAIT_COUNT.swap((smp - 1) as u64, Ordering::AcqRel);
  248. }
  249. debug!("send ipi!");
  250. let start_time = get_time();
  251. let mut mask = Some(HartMask::from_mask_base(0, 0));
  252. for i in 0..smp {
  253. if i == hartid {
  254. continue;
  255. }
  256. if let Some(ref mut mask) = mask {
  257. match mask.insert(i) {
  258. Ok(_) => continue,
  259. Err(MaskError::InvalidBit) => {
  260. sbi::send_ipi(*mask);
  261. }
  262. Err(_) => unreachable!("Failed to construct mask"),
  263. }
  264. }
  265. mask = Some(HartMask::from_mask_base(0b1, i));
  266. }
  267. if let Some(mask) = mask {
  268. sbi::send_ipi(mask);
  269. }
  270. while unsafe { WAIT_COUNT.load(Ordering::Acquire) } != 0 {}
  271. let end_time = get_time();
  272. println!("Test #{}: {}", i, end_time - start_time);
  273. }
  274. sbi::system_reset(sbi::Shutdown, sbi::NoReason);
  275. unreachable!()
  276. }
  277. #[cfg_attr(not(test), panic_handler)]
  278. fn panic(info: &core::panic::PanicInfo) -> ! {
  279. let (hart_id, pc): (usize, usize);
  280. unsafe { asm!("mv {}, tp", out(reg) hart_id) };
  281. unsafe { asm!("auipc {}, 0", out(reg) pc) };
  282. info!("[test-kernel-panic] hart {hart_id} {info}");
  283. info!("[test-kernel-panic] pc = {pc:#x}");
  284. info!("[test-kernel-panic] SBI test FAILED due to panic");
  285. sbi::system_reset(sbi::Shutdown, sbi::SystemFailure);
  286. loop {}
  287. }
  288. struct Console;
  289. static mut UART: Uart16550Map = Uart16550Map(null());
  290. pub struct Uart16550Map(*const Uart16550<u8>);
  291. unsafe impl Sync for Uart16550Map {}
  292. impl Uart16550Map {
  293. #[inline]
  294. pub fn get(&self) -> &Uart16550<u8> {
  295. unsafe { &*self.0 }
  296. }
  297. }
  298. impl rcore_console::Console for Console {
  299. #[inline]
  300. fn put_char(&self, c: u8) {
  301. unsafe { UART.get().write(core::slice::from_ref(&c)) };
  302. }
  303. #[inline]
  304. fn put_str(&self, s: &str) {
  305. unsafe { UART.get().write(s.as_bytes()) };
  306. }
  307. }