main.rs 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. // A test kernel to test RustSBI function on all platforms
  2. #![feature(naked_functions, asm)]
  3. #![no_std]
  4. #![no_main]
  5. #[macro_use]
  6. mod console;
  7. mod sbi;
  8. use riscv::register::{sepc, stvec::{self, TrapMode}, scause::{self, Trap, Exception}};
  9. pub extern "C" fn rust_main(hartid: usize, dtb_pa: usize) -> ! {
  10. println!("<< Test-kernel: Hart id = {}, DTB physical address = {:#x}", hartid, dtb_pa);
  11. test_base_extension();
  12. test_sbi_ins_emulation();
  13. unsafe { stvec::write(start_trap as usize, TrapMode::Direct) };
  14. println!(">> Test-kernel: Trigger illegal exception");
  15. unsafe { asm!("csrw mcycle, x0") }; // mcycle cannot be written, this is always a 4-byte illegal instruction
  16. println!("<< Test-kernel: SBI test SUCCESS, shutdown");
  17. sbi::shutdown()
  18. }
  19. fn test_base_extension() {
  20. println!(">> Test-kernel: Testing base extension");
  21. let base_version = sbi::probe_extension(sbi::EXTENSION_BASE);
  22. if base_version == 0 {
  23. println!("!! Test-kernel: no base extension probed; SBI call returned value '0'");
  24. println!("!! Test-kernel: This SBI implementation may only have legacy extension implemented");
  25. println!("!! Test-kernel: SBI test FAILED due to no base extension found");
  26. sbi::shutdown()
  27. }
  28. println!("<< Test-kernel: Base extension version: {:x}", base_version);
  29. println!("<< Test-kernel: SBI specification version: {:x}", sbi::get_spec_version());
  30. println!("<< Test-kernel: SBI implementation Id: {:x}", sbi::get_sbi_impl_id());
  31. println!("<< Test-kernel: SBI implementation version: {:x}", sbi::get_sbi_impl_version());
  32. println!("<< Test-kernel: Device mvendorid: {:x}", sbi::get_mvendorid());
  33. println!("<< Test-kernel: Device marchid: {:x}", sbi::get_marchid());
  34. println!("<< Test-kernel: Device mimpid: {:x}", sbi::get_mimpid());
  35. }
  36. fn test_sbi_ins_emulation() {
  37. println!(">> Test-kernel: Testing SBI instruction emulation");
  38. let time = riscv::register::time::read64();
  39. println!("<< Test-kernel: Current time: {:x}", time);
  40. }
  41. pub extern "C" fn rust_trap_exception() {
  42. let cause = scause::read().cause();
  43. println!("<< Test-kernel: Value of scause: {:?}", cause);
  44. if cause != Trap::Exception(Exception::IllegalInstruction) {
  45. println!("!! Test-kernel: Wrong cause associated to illegal instruction");
  46. sbi::shutdown()
  47. }
  48. println!("<< Test-kernel: Illegal exception delegate success");
  49. sepc::write(sepc::read().wrapping_add(4));
  50. }
  51. use core::panic::PanicInfo;
  52. #[cfg_attr(not(test), panic_handler)]
  53. #[allow(unused)]
  54. fn panic(info: &PanicInfo) -> ! {
  55. println!("!! Test-kernel: {}", info);
  56. println!("!! Test-kernel: SBI test FAILED due to panic");
  57. sbi::shutdown()
  58. }
  59. const BOOT_STACK_SIZE: usize = 4096 * 4 * 8;
  60. static mut BOOT_STACK: [u8; BOOT_STACK_SIZE] = [0; BOOT_STACK_SIZE];
  61. #[naked]
  62. #[link_section = ".text.entry"]
  63. #[export_name = "_start"]
  64. unsafe extern "C" fn entry() -> ! {
  65. asm!("
  66. # 1. set sp
  67. # sp = bootstack + (hartid + 1) * 0x10000
  68. add t0, a0, 1
  69. slli t0, t0, 14
  70. 1: auipc sp, %pcrel_hi({boot_stack})
  71. addi sp, sp, %pcrel_lo(1b)
  72. add sp, sp, t0
  73. # 2. jump to rust_main (absolute address)
  74. 1: auipc t0, %pcrel_hi({rust_main})
  75. addi t0, t0, %pcrel_lo(1b)
  76. jr t0
  77. ",
  78. boot_stack = sym BOOT_STACK,
  79. rust_main = sym rust_main,
  80. options(noreturn))
  81. }
  82. #[cfg(target_pointer_width = "128")]
  83. macro_rules! define_store_load {
  84. () => {
  85. ".altmacro
  86. .macro STORE reg, offset
  87. sq \\reg, \\offset* {REGBYTES} (sp)
  88. .endm
  89. .macro LOAD reg, offset
  90. lq \\reg, \\offset* {REGBYTES} (sp)
  91. .endm"
  92. };
  93. }
  94. #[cfg(target_pointer_width = "64")]
  95. macro_rules! define_store_load {
  96. () => {
  97. ".altmacro
  98. .macro STORE reg, offset
  99. sd \\reg, \\offset* {REGBYTES} (sp)
  100. .endm
  101. .macro LOAD reg, offset
  102. ld \\reg, \\offset* {REGBYTES} (sp)
  103. .endm"
  104. };
  105. }
  106. #[cfg(target_pointer_width = "32")]
  107. macro_rules! define_store_load {
  108. () => {
  109. ".altmacro
  110. .macro STORE reg, offset
  111. sw \\reg, \\offset* {REGBYTES} (sp)
  112. .endm
  113. .macro LOAD reg, offset
  114. lw \\reg, \\offset* {REGBYTES} (sp)
  115. .endm"
  116. };
  117. }
  118. #[naked]
  119. #[link_section = ".text"]
  120. unsafe extern "C" fn start_trap() {
  121. asm!(concat!(define_store_load!(), "
  122. .p2align 2
  123. addi sp, sp, -16 * {REGBYTES}
  124. STORE ra, 0
  125. STORE t0, 1
  126. STORE t1, 2
  127. STORE t2, 3
  128. STORE t3, 4
  129. STORE t4, 5
  130. STORE t5, 6
  131. STORE t6, 7
  132. STORE a0, 8
  133. STORE a1, 9
  134. STORE a2, 10
  135. STORE a3, 11
  136. STORE a4, 12
  137. STORE a5, 13
  138. STORE a6, 14
  139. STORE a7, 15
  140. mv a0, sp
  141. call {rust_trap_exception}
  142. LOAD ra, 0
  143. LOAD t0, 1
  144. LOAD t1, 2
  145. LOAD t2, 3
  146. LOAD t3, 4
  147. LOAD t4, 5
  148. LOAD t5, 6
  149. LOAD t6, 7
  150. LOAD a0, 8
  151. LOAD a1, 9
  152. LOAD a2, 10
  153. LOAD a3, 11
  154. LOAD a4, 12
  155. LOAD a5, 13
  156. LOAD a6, 14
  157. LOAD a7, 15
  158. addi sp, sp, 16 * {REGBYTES}
  159. sret
  160. "),
  161. REGBYTES = const core::mem::size_of::<usize>(),
  162. rust_trap_exception = sym rust_trap_exception,
  163. options(noreturn))
  164. }