4
0

main.rs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488
  1. #![no_std]
  2. #![no_main]
  3. #![feature(alloc_error_handler)]
  4. #![feature(global_asm)]
  5. #![feature(llvm_asm)]
  6. #[cfg(not(test))]
  7. use core::alloc::Layout;
  8. #[cfg(not(test))]
  9. use core::panic::PanicInfo;
  10. use k210_hal::{clock::Clocks, fpioa, pac, prelude::*};
  11. use linked_list_allocator::LockedHeap;
  12. use rustsbi::{enter_privileged, print, println};
  13. use riscv::register::{
  14. mcause::{self, Exception, Interrupt, Trap},
  15. medeleg, mepc, mhartid, mideleg, mie, mip, misa::{self, MXL},
  16. mstatus::{self, MPP},
  17. mtval,
  18. mtvec::{self, TrapMode},
  19. satp,
  20. };
  21. #[global_allocator]
  22. static ALLOCATOR: LockedHeap = LockedHeap::empty();
  23. #[cfg(not(test))]
  24. #[panic_handler]
  25. fn panic(info: &PanicInfo) -> ! {
  26. println!("[rustsbi] {}", info);
  27. loop {}
  28. }
  29. #[cfg(not(test))]
  30. #[alloc_error_handler]
  31. fn oom(layout: Layout) -> ! {
  32. println!("[rustsbi] out of memory for layout {:?}", layout);
  33. loop {}
  34. }
  35. fn mp_hook() -> bool {
  36. use riscv::asm::wfi;
  37. use k210_hal::clint::msip;
  38. let hartid = mhartid::read();
  39. if hartid == 0 {
  40. true
  41. } else {
  42. unsafe {
  43. // Clear IPI
  44. msip::clear_ipi(hartid);
  45. // Start listening for software interrupts
  46. mie::set_msoft();
  47. loop {
  48. wfi();
  49. if mip::read().msoft() {
  50. break;
  51. }
  52. }
  53. // Stop listening for software interrupts
  54. mie::clear_msoft();
  55. // Clear IPI
  56. msip::clear_ipi(hartid);
  57. }
  58. false
  59. }
  60. }
  61. #[export_name = "_start"]
  62. #[link_section = ".text.entry"] // this is stable
  63. fn main() -> ! {
  64. unsafe {
  65. llvm_asm!(
  66. "
  67. csrr a2, mhartid
  68. lui t0, %hi(_max_hart_id)
  69. add t0, t0, %lo(_max_hart_id)
  70. bgtu a2, t0, _start_abort
  71. la sp, _stack_start
  72. lui t0, %hi(_hart_stack_size)
  73. add t0, t0, %lo(_hart_stack_size)
  74. .ifdef __riscv_mul
  75. mul t0, a2, t0
  76. .else
  77. beqz a2, 2f // Jump if single-hart
  78. mv t1, a2
  79. mv t2, t0
  80. 1:
  81. add t0, t0, t2
  82. addi t1, t1, -1
  83. bnez t1, 1b
  84. 2:
  85. .endif
  86. sub sp, sp, t0
  87. csrw mscratch, zero
  88. j _start_success
  89. _start_abort:
  90. wfi
  91. j _start_abort
  92. _start_success:
  93. "
  94. )
  95. };
  96. if mp_hook() {
  97. extern "C" {
  98. static mut _ebss: u32;
  99. static mut _sbss: u32;
  100. static mut _edata: u32;
  101. static mut _sdata: u32;
  102. static _sidata: u32;
  103. }
  104. unsafe {
  105. r0::zero_bss(&mut _sbss, &mut _ebss);
  106. r0::init_data(&mut _sdata, &mut _edata, &_sidata);
  107. }
  108. }
  109. extern "C" {
  110. fn _start_trap();
  111. }
  112. unsafe {
  113. mtvec::write(_start_trap as usize, TrapMode::Direct);
  114. }
  115. if mhartid::read() == 0 {
  116. extern "C" {
  117. fn _sheap();
  118. fn _heap_size();
  119. }
  120. let sheap = &mut _sheap as *mut _ as usize;
  121. let heap_size = &_heap_size as *const _ as usize;
  122. unsafe {
  123. ALLOCATOR.lock().init(sheap, heap_size);
  124. }
  125. let p = pac::Peripherals::take().unwrap();
  126. let mut sysctl = p.SYSCTL.constrain();
  127. let fpioa = p.FPIOA.split(&mut sysctl.apb0);
  128. let clocks = Clocks::new();
  129. let _uarths_tx = fpioa.io5.into_function(fpioa::UARTHS_TX);
  130. let _uarths_rx = fpioa.io4.into_function(fpioa::UARTHS_RX);
  131. // Configure UART
  132. let serial = p.UARTHS.configure(115_200.bps(), &clocks);
  133. let (tx, rx) = serial.split();
  134. use rustsbi::legacy_stdio::init_legacy_stdio_embedded_hal_fuse;
  135. init_legacy_stdio_embedded_hal_fuse(tx, rx);
  136. struct Ipi;
  137. impl rustsbi::Ipi for Ipi {
  138. fn max_hart_id(&self) -> usize {
  139. 1
  140. }
  141. fn send_ipi_many(&mut self, hart_mask: rustsbi::HartMask) {
  142. use k210_hal::clint::msip;
  143. for i in 0..=1 {
  144. if hart_mask.has_bit(i) {
  145. msip::set_ipi(i);
  146. msip::clear_ipi(i);
  147. }
  148. }
  149. }
  150. }
  151. use rustsbi::init_ipi;
  152. init_ipi(Ipi);
  153. struct Timer;
  154. impl rustsbi::Timer for Timer {
  155. fn set_timer(&mut self, stime_value: u64) {
  156. // This function must clear the pending timer interrupt bit as well.
  157. use k210_hal::clint::mtimecmp;
  158. mtimecmp::write(mhartid::read(), stime_value);
  159. unsafe { mip::clear_mtimer() };
  160. }
  161. }
  162. use rustsbi::init_timer;
  163. init_timer(Timer);
  164. struct Reset;
  165. impl rustsbi::Reset for Reset {
  166. fn reset(&self) -> ! {
  167. println!("[rustsbi] todo: shutdown all harts on k210; program halt");
  168. loop {}
  169. }
  170. }
  171. use rustsbi::init_reset;
  172. init_reset(Reset);
  173. use k210_hal::plic::Priority;
  174. use k210_hal::pac::Interrupt;
  175. use k210_hal::gpiohs::Edge;
  176. unsafe {
  177. pac::PLIC::set_threshold(mhartid::read(), Priority::P0);
  178. }
  179. let gpiohs = p.GPIOHS.split();
  180. fpioa.io16.into_function(fpioa::GPIOHS0);
  181. let mut boot = gpiohs.gpiohs0.into_pull_up_input();
  182. boot.trigger_on_edge(Edge::RISING | Edge::FALLING);
  183. unsafe {
  184. pac::PLIC::set_priority(Interrupt::GPIOHS0, Priority::P1);
  185. pac::PLIC::unmask(mhartid::read(), Interrupt::GPIOHS0);
  186. }
  187. boot.clear_interrupt_pending_bits();
  188. }
  189. unsafe {
  190. mideleg::set_sext();
  191. mideleg::set_stimer();
  192. mideleg::set_ssoft();
  193. medeleg::set_instruction_misaligned();
  194. medeleg::set_breakpoint();
  195. medeleg::set_user_env_call();
  196. /* MMU Exception Delegation
  197. /* Page Faults are *Reserved* in 1.9.1 version */
  198. - medeleg::set_instruction_page_fault();
  199. - medeleg::set_load_page_fault();
  200. - medeleg::set_store_page_fault();
  201. /* Actually, in 1.9.1 they are merged into more general exceptions */
  202. + medeleg::set_instruction_fault();
  203. + medeleg::set_load_fault();
  204. + medeleg::set_store_fault(); */
  205. medeleg::set_instruction_fault();
  206. medeleg::set_load_fault();
  207. medeleg::set_store_fault();
  208. mie::set_mext();
  209. // 不打开mie::set_mtimer
  210. mie::set_msoft();
  211. }
  212. if mhartid::read() == 0 {
  213. println!("[rustsbi] Version 0.1.0");
  214. println!("{}", rustsbi::LOGO);
  215. println!("[rustsbi] Platform: K210");
  216. let isa = misa::read();
  217. if let Some(isa) = isa {
  218. let mxl_str = match isa.mxl() {
  219. MXL::XLEN32 => "RV32",
  220. MXL::XLEN64 => "RV64",
  221. MXL::XLEN128 => "RV128",
  222. };
  223. print!("[rustsbi] misa: {}", mxl_str);
  224. for ext in 'A'..='Z' {
  225. if isa.has_extension(ext) {
  226. print!("{}", ext);
  227. }
  228. }
  229. println!("");
  230. }
  231. println!("[rustsbi] mideleg: {:#x}", mideleg::read().bits());
  232. println!("[rustsbi] medeleg: {:#x}", medeleg::read().bits());
  233. println!("[rustsbi] Kernel entry: 0x80020000");
  234. }
  235. extern "C" {
  236. fn _s_mode_start();
  237. }
  238. unsafe {
  239. mepc::write(_s_mode_start as usize);
  240. mstatus::set_mpp(MPP::Supervisor);
  241. enter_privileged(mhartid::read(), 0x2333333366666666);
  242. }
  243. }
  244. global_asm!(
  245. "
  246. .section .text
  247. .globl _s_mode_start
  248. _s_mode_start:
  249. 1: auipc ra, %pcrel_hi(1f)
  250. ld ra, %pcrel_lo(1b)(ra)
  251. jr ra
  252. .align 3
  253. 1: .dword 0x80020000
  254. "
  255. );
  256. // todo: configurable target address
  257. global_asm!(
  258. "
  259. .equ REGBYTES, 8
  260. .macro STORE reg, offset
  261. sd \\reg, \\offset*REGBYTES(sp)
  262. .endm
  263. .macro LOAD reg, offset
  264. ld \\reg, \\offset*REGBYTES(sp)
  265. .endm
  266. .section .text
  267. .global _start_trap
  268. .p2align 2
  269. _start_trap:
  270. csrrw sp, mscratch, sp
  271. bnez sp, 1f
  272. /* from M level, load sp */
  273. csrrw sp, mscratch, zero
  274. 1:
  275. addi sp, sp, -16 * REGBYTES
  276. STORE ra, 0
  277. STORE t0, 1
  278. STORE t1, 2
  279. STORE t2, 3
  280. STORE t3, 4
  281. STORE t4, 5
  282. STORE t5, 6
  283. STORE t6, 7
  284. STORE a0, 8
  285. STORE a1, 9
  286. STORE a2, 10
  287. STORE a3, 11
  288. STORE a4, 12
  289. STORE a5, 13
  290. STORE a6, 14
  291. STORE a7, 15
  292. mv a0, sp
  293. call _start_trap_rust
  294. LOAD ra, 0
  295. LOAD t0, 1
  296. LOAD t1, 2
  297. LOAD t2, 3
  298. LOAD t3, 4
  299. LOAD t4, 5
  300. LOAD t5, 6
  301. LOAD t6, 7
  302. LOAD a0, 8
  303. LOAD a1, 9
  304. LOAD a2, 10
  305. LOAD a3, 11
  306. LOAD a4, 12
  307. LOAD a5, 13
  308. LOAD a6, 14
  309. LOAD a7, 15
  310. addi sp, sp, 16 * REGBYTES
  311. csrrw sp, mscratch, sp
  312. mret
  313. "
  314. );
  315. #[allow(unused)]
  316. struct TrapFrame {
  317. ra: usize,
  318. t0: usize,
  319. t1: usize,
  320. t2: usize,
  321. t3: usize,
  322. t4: usize,
  323. t5: usize,
  324. t6: usize,
  325. a0: usize,
  326. a1: usize,
  327. a2: usize,
  328. a3: usize,
  329. a4: usize,
  330. a5: usize,
  331. a6: usize,
  332. a7: usize,
  333. }
  334. #[export_name = "_start_trap_rust"]
  335. extern "C" fn start_trap_rust(trap_frame: &mut TrapFrame) {
  336. let cause = mcause::read().cause();
  337. match cause {
  338. Trap::Exception(Exception::SupervisorEnvCall) => {
  339. let params = [trap_frame.a0, trap_frame.a1, trap_frame.a2, trap_frame.a3];
  340. let ans = rustsbi::ecall(trap_frame.a7, trap_frame.a6, params);
  341. trap_frame.a0 = ans.error;
  342. trap_frame.a1 = ans.value;
  343. mepc::write(mepc::read().wrapping_add(4));
  344. }
  345. Trap::Interrupt(Interrupt::MachineSoft) => {
  346. unsafe {
  347. mip::set_ssoft();
  348. mie::clear_msoft();
  349. }
  350. }
  351. Trap::Interrupt(Interrupt::MachineTimer) => {
  352. unsafe {
  353. mip::set_stimer();
  354. mie::clear_mtimer();
  355. }
  356. }
  357. Trap::Interrupt(Interrupt::MachineExternal) => {
  358. // to make UARTHS interrupt soft delegation work; ref: pull request #1
  359. // PLIC target0(Always Hart0-M-Interrupt) acquire
  360. let irq_id = unsafe { (0x0c20_0004 as *const u32).read_volatile() };
  361. // read from UARTHS RXFIFO
  362. let ch: u8 = unsafe { (0x3800_0004 as *const u32).read_volatile() & 0xFF } as u8;
  363. // black magic @_@, soft delegation won't success without it!
  364. print!("{}", 0 as char);
  365. // PLIC complete
  366. unsafe { (0x0c20_0004 as *mut u32).write_volatile(irq_id); }
  367. // communicate with delegated interrupt with stval CSR
  368. unsafe { llvm_asm!("csrw stval, $0" :: "r"(ch as usize) :: "volatile"); }
  369. // soft delegate to S Mode soft interrupt
  370. unsafe { mip::set_ssoft(); }
  371. }
  372. Trap::Exception(Exception::IllegalInstruction) => {
  373. let vaddr = mepc::read();
  374. let ins = unsafe { get_vaddr_u32(vaddr) };
  375. if ins & 0xFFFFF07F == 0xC0102073 { // rdtime instruction
  376. // rdtime is actually a csrrw instruction
  377. let rd = ((ins >> 7) & 0b1_1111) as u8;
  378. let mtime = k210_hal::clint::mtime::read();
  379. let time_usize = mtime as usize;
  380. set_rd(trap_frame, rd, time_usize);
  381. mepc::write(mepc::read().wrapping_add(4)); // skip current instruction
  382. } else if ins & 0xFE007FFF == 0x12000073 { // sfence.vma instruction
  383. // sfence.vma: | 31..25 funct7=SFENCE.VMA(0001001) | 24..20 rs2/asid | 19..15 rs1/vaddr |
  384. // 14..12 funct3=PRIV(000) | 11..7 rd, =0 | 6..0 opcode=SYSTEM(1110011) |
  385. // sfence.vm(1.9): | 31..=20 SFENCE.VM(000100000100) | 19..15 rs1/vaddr |
  386. // 14..12 funct3=PRIV(000) | 11..7 rd, =0 | 6..0 opcode=SYSTEM(1110011) |
  387. // discard rs2 // let _rs2_asid = ((ins >> 20) & 0b1_1111) as u8;
  388. // let rs1_vaddr = ((ins >> 15) & 0b1_1111) as u8;
  389. // read paging mode from satp (sptbr)
  390. let satp_bits = satp::read().bits();
  391. // bit 63..20 is not readable and writeable on K210, so we cannot
  392. // decide paging type from the 'satp' register.
  393. // that also means that the asid function is not usable on this chip.
  394. // we have to fix it to be Sv39.
  395. let ppn = satp_bits & 0xFFF_FFFF_FFFF; // 43..0 PPN WARL
  396. // write to sptbr
  397. let sptbr_bits = ppn & 0x3F_FFFF_FFFF;
  398. unsafe { llvm_asm!("csrw 0x180, $0"::"r"(sptbr_bits)) }; // write to sptbr
  399. // enable paging (in v1.9.1, mstatus: | 28..24 VM[4:0] WARL | ... )
  400. let mut mstatus_bits: usize;
  401. unsafe { llvm_asm!("csrr $0, mstatus":"=r"(mstatus_bits)) };
  402. mstatus_bits &= !0x1F00_0000;
  403. mstatus_bits |= 9 << 24;
  404. unsafe { llvm_asm!("csrw mstatus, $0"::"r"(mstatus_bits)) };
  405. // emulate with sfence.vm (declared in privileged spec v1.9)
  406. unsafe { llvm_asm!(".word 0x10400073") }; // sfence.vm x0
  407. // ::"r"(rs1_vaddr)
  408. mepc::write(mepc::read().wrapping_add(4)); // skip current instruction
  409. } else {
  410. panic!("invalid instruction, mepc: {:016x?}, instruction: {:08x?}", mepc::read(), ins);
  411. }
  412. }
  413. cause => panic!(
  414. "Unhandled trap! mcause: {:?}, mepc: {:016x?}, mtval: {:016x?}",
  415. cause,
  416. mepc::read(),
  417. mtval::read()
  418. ),
  419. }
  420. }
  421. #[inline]
  422. unsafe fn get_vaddr_u32(vaddr: usize) -> u32 {
  423. // todo: comment
  424. get_vaddr_u16(vaddr) as u32 |
  425. ((get_vaddr_u16(vaddr.wrapping_add(2)) as u32) << 16)
  426. }
  427. #[inline]
  428. unsafe fn get_vaddr_u16(vaddr: usize) -> u16 {
  429. let mut ans: u16;
  430. llvm_asm!("
  431. li t0, (1 << 17)
  432. csrrs t0, mstatus, t0
  433. lhu $0, 0($1)
  434. csrw mstatus, t0
  435. "
  436. :"=r"(ans)
  437. :"r"(vaddr)
  438. :"t0", "t1");
  439. ans
  440. }
  441. #[inline]
  442. fn set_rd(trap_frame: &mut TrapFrame, rd: u8, value: usize) {
  443. match rd {
  444. 10 => trap_frame.a0 = value,
  445. 11 => trap_frame.a1 = value,
  446. 12 => trap_frame.a2 = value,
  447. 13 => trap_frame.a3 = value,
  448. 14 => trap_frame.a4 = value,
  449. 15 => trap_frame.a5 = value,
  450. 16 => trap_frame.a6 = value,
  451. 17 => trap_frame.a7 = value,
  452. 5 => trap_frame.t0 = value,
  453. 6 => trap_frame.t1 = value,
  454. 7 => trap_frame.t2 = value,
  455. 28 => trap_frame.t3 = value,
  456. 29 => trap_frame.t4 = value,
  457. 30 => trap_frame.t5 = value,
  458. 31 => trap_frame.t6 = value,
  459. _ => panic!("invalid target `rd`"),
  460. }
  461. }