trap.rs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501
  1. use aclint::SifiveClint as Clint;
  2. use core::arch::asm;
  3. use fast_trap::{trap_entry, FastContext, FastResult};
  4. use riscv::register::{
  5. mcause::{self, Exception as E, Trap as T},
  6. mepc, mie, mstatus, mtval, satp, sstatus,
  7. };
  8. use rustsbi::RustSBI;
  9. use crate::board::SBI;
  10. use crate::clint::{self, SIFIVECLINT};
  11. use crate::console::{MachineConsole, CONSOLE};
  12. use crate::hsm::local_hsm;
  13. use crate::rfence::{local_rfence, RFenceType};
  14. use crate::riscv_spec::{CSR_TIME, CSR_TIMEH, current_hartid};
  15. const PAGE_SIZE: usize = 4096;
  16. #[naked]
  17. pub(crate) unsafe extern "C" fn trap_vec() {
  18. asm!(
  19. ".align 2",
  20. ".option push",
  21. ".option norvc",
  22. "j {default}", // exception
  23. "j {default}", // supervisor software
  24. "j {default}", // reserved
  25. "j {msoft} ", // machine software
  26. "j {default}", // reserved
  27. "j {default}", // supervisor timer
  28. "j {default}", // reserved
  29. "j {mtimer}", // machine timer
  30. "j {default}", // reserved
  31. "j {default}", // supervisor external
  32. "j {default}", // reserved
  33. "j {default}", // machine external
  34. ".option pop",
  35. default = sym trap_entry,
  36. msoft = sym msoft,
  37. mtimer = sym mtimer,
  38. options(noreturn)
  39. )
  40. }
  41. /// machine timer 中断代理
  42. ///
  43. /// # Safety
  44. ///
  45. /// 裸函数。
  46. #[naked]
  47. unsafe extern "C" fn mtimer() {
  48. asm!(
  49. // 换栈:
  50. // sp : M sp
  51. // mscratch: S sp
  52. " csrrw sp, mscratch, sp",
  53. // 保护
  54. " addi sp, sp, -4*8
  55. sd ra, 0*8(sp)
  56. sd a0, 1*8(sp)
  57. sd a1, 2*8(sp)
  58. sd a2, 3*8(sp)
  59. ",
  60. // 清除 mtimecmp
  61. " la a0, {clint_ptr}
  62. ld a0, (a0)
  63. csrr a1, mhartid
  64. addi a2, zero, -1
  65. call {set_mtimecmp}
  66. ",
  67. // 设置 stip
  68. " li a0, {mip_stip}
  69. csrrs zero, mip, a0
  70. ",
  71. // 恢复
  72. " ld ra, 0*8(sp)
  73. ld a0, 1*8(sp)
  74. ld a1, 2*8(sp)
  75. ld a2, 3*8(sp)
  76. addi sp, sp, 4*8
  77. ",
  78. // 换栈:
  79. // sp : S sp
  80. // mscratch: M sp
  81. " csrrw sp, mscratch, sp",
  82. // 返回
  83. " mret",
  84. mip_stip = const 1 << 5,
  85. clint_ptr = sym SIFIVECLINT,
  86. // Clint::write_mtimecmp_naked(&self, hart_idx, val)
  87. set_mtimecmp = sym Clint::write_mtimecmp_naked,
  88. options(noreturn)
  89. )
  90. }
  91. /// machine soft 中断代理
  92. ///
  93. #[naked]
  94. unsafe extern "C" fn msoft() -> ! {
  95. asm!(
  96. ".align 2",
  97. "csrrw sp, mscratch, sp",
  98. "addi sp, sp, -32*8",
  99. "sd ra, 0*8(sp)
  100. sd gp, 2*8(sp)
  101. sd tp, 3*8(sp)
  102. sd t0, 4*8(sp)
  103. sd t1, 5*8(sp)
  104. sd t2, 6*8(sp)
  105. sd s0, 7*8(sp)
  106. sd s1, 8*8(sp)
  107. sd a0, 9*8(sp)
  108. sd a1, 10*8(sp)
  109. sd a2, 11*8(sp)
  110. sd a3, 12*8(sp)
  111. sd a4, 13*8(sp)
  112. sd a5, 14*8(sp)
  113. sd a6, 15*8(sp)
  114. sd a7, 16*8(sp)
  115. sd s2, 17*8(sp)
  116. sd s3, 18*8(sp)
  117. sd s4, 19*8(sp)
  118. sd s5, 20*8(sp)
  119. sd s6, 21*8(sp)
  120. sd s7, 22*8(sp)
  121. sd s8, 23*8(sp)
  122. sd s9, 24*8(sp)
  123. sd s10, 25*8(sp)
  124. sd s11, 26*8(sp)
  125. sd t3, 27*8(sp)
  126. sd t4, 28*8(sp)
  127. sd t5, 29*8(sp)
  128. sd t6, 30*8(sp)",
  129. "csrr t0, mepc
  130. sd t0, 31*8(sp)",
  131. "csrr t2, mscratch",
  132. "sd t2, 1*8(sp)",
  133. "mv a0, sp",
  134. "call {msoft_hanlder}",
  135. "ld t0, 31*8(sp)
  136. csrw mepc, t0",
  137. "ld ra, 0*8(sp)
  138. ld gp, 2*8(sp)
  139. ld tp, 3*8(sp)
  140. ld t0, 4*8(sp)
  141. ld t1, 5*8(sp)
  142. ld t2, 6*8(sp)
  143. ld s0, 7*8(sp)
  144. ld s1, 8*8(sp)
  145. ld a0, 9*8(sp)
  146. ld a1, 10*8(sp)
  147. ld a2, 11*8(sp)
  148. ld a3, 12*8(sp)
  149. ld a4, 13*8(sp)
  150. ld a5, 14*8(sp)
  151. ld a6, 15*8(sp)
  152. ld a7, 16*8(sp)
  153. ld s2, 17*8(sp)
  154. ld s3, 18*8(sp)
  155. ld s4, 19*8(sp)
  156. ld s5, 20*8(sp)
  157. ld s6, 21*8(sp)
  158. ld s7, 22*8(sp)
  159. ld s8, 23*8(sp)
  160. ld s9, 24*8(sp)
  161. ld s10, 25*8(sp)
  162. ld s11, 26*8(sp)
  163. ld t3, 27*8(sp)
  164. ld t4, 28*8(sp)
  165. ld t5, 29*8(sp)
  166. ld t6, 30*8(sp)",
  167. "addi sp, sp, 32*8",
  168. "csrrw sp, mscratch, sp",
  169. "mret",
  170. msoft_hanlder = sym msoft_hanlder,
  171. options(noreturn)
  172. );
  173. }
  174. pub extern "C" fn msoft_hanlder(ctx: &mut SupervisorContext) {
  175. #[inline]
  176. fn boot(ctx: &mut SupervisorContext, start_addr: usize, opaque: usize) {
  177. unsafe {
  178. sstatus::clear_sie();
  179. satp::write(0);
  180. }
  181. ctx.a0 = current_hartid();
  182. ctx.a1 = opaque;
  183. ctx.mepc = start_addr;
  184. }
  185. match local_hsm().start() {
  186. // HSM Start
  187. Ok(next_stage) => {
  188. clint::clear_msip();
  189. unsafe {
  190. mstatus::set_mpie();
  191. mstatus::set_mpp(next_stage.next_mode);
  192. mie::set_msoft();
  193. mie::set_mtimer();
  194. }
  195. boot(ctx, next_stage.start_addr, next_stage.opaque);
  196. }
  197. Err(rustsbi::spec::hsm::HART_STOP) => {
  198. clint::clear_msip();
  199. unsafe {
  200. mie::set_msoft();
  201. }
  202. riscv::asm::wfi();
  203. }
  204. // RFence
  205. _ => {
  206. msoft_rfence_handler();
  207. }
  208. }
  209. }
  210. pub fn msoft_rfence_handler() {
  211. let rfence_context = local_rfence().get();
  212. match rfence_context {
  213. Some(ctx) => match ctx.op {
  214. RFenceType::FenceI => unsafe {
  215. asm!("fence.i");
  216. local_rfence().clear();
  217. clint::clear_msip();
  218. },
  219. RFenceType::SFenceVma => {
  220. if (ctx.start_addr == 0 && ctx.size == 0) || (ctx.size == usize::MAX) {
  221. unsafe {
  222. asm!("sfence.vma");
  223. }
  224. } else {
  225. for offset in (0..ctx.size).step_by(PAGE_SIZE) {
  226. let addr = ctx.start_addr + offset;
  227. unsafe {
  228. asm!("sfence.vma {}", in(reg) addr);
  229. }
  230. }
  231. }
  232. local_rfence().clear();
  233. clint::clear_msip();
  234. }
  235. RFenceType::SFenceVmaAsid => {
  236. let asid = ctx.asid;
  237. if (ctx.start_addr == 0 && ctx.size == 0) || (ctx.size == usize::MAX) {
  238. unsafe {
  239. asm!("sfence.vma {}, {}", in(reg) 0, in(reg) asid);
  240. }
  241. } else {
  242. for offset in (0..ctx.size).step_by(PAGE_SIZE) {
  243. let addr = ctx.start_addr + offset;
  244. unsafe {
  245. asm!("sfence.vma {}, {}", in(reg) addr, in(reg) asid);
  246. }
  247. }
  248. }
  249. local_rfence().clear();
  250. clint::clear_msip();
  251. }
  252. rfencetype => {
  253. error!("Unsupported RFence Type: {:?}!", rfencetype);
  254. local_rfence().clear();
  255. clint::clear_msip();
  256. }
  257. },
  258. None => {
  259. clint::clear_msip();
  260. unsafe {
  261. riscv::register::mip::set_ssoft();
  262. }
  263. }
  264. }
  265. }
  266. /// Fast trap
  267. pub extern "C" fn fast_handler(
  268. mut ctx: FastContext,
  269. a1: usize,
  270. a2: usize,
  271. a3: usize,
  272. a4: usize,
  273. a5: usize,
  274. a6: usize,
  275. a7: usize,
  276. ) -> FastResult {
  277. #[inline]
  278. fn boot(mut ctx: FastContext, start_addr: usize, opaque: usize) -> FastResult {
  279. unsafe {
  280. sstatus::clear_sie();
  281. satp::write(0);
  282. }
  283. ctx.regs().a[0] = current_hartid();
  284. ctx.regs().a[1] = opaque;
  285. ctx.regs().pc = start_addr;
  286. ctx.call(2)
  287. }
  288. loop {
  289. match local_hsm().start() {
  290. Ok(next_stage) => {
  291. clint::clear_msip();
  292. unsafe {
  293. mstatus::set_mpie();
  294. mstatus::set_mpp(next_stage.next_mode);
  295. mie::set_msoft();
  296. mie::set_mtimer();
  297. }
  298. break boot(ctx, next_stage.start_addr, next_stage.opaque);
  299. }
  300. Err(rustsbi::spec::hsm::HART_STOP) => {
  301. clint::clear_msip();
  302. unsafe {
  303. mie::set_msoft();
  304. }
  305. riscv::asm::wfi();
  306. }
  307. _ => match mcause::read().cause() {
  308. // SBI call
  309. T::Exception(E::SupervisorEnvCall) => {
  310. use sbi_spec::{base, hsm, legacy};
  311. let mut ret = unsafe { SBI.assume_init_mut() }.handle_ecall(
  312. a7,
  313. a6,
  314. [ctx.a0(), a1, a2, a3, a4, a5],
  315. );
  316. if ret.is_ok() {
  317. match (a7, a6) {
  318. // 关闭
  319. (hsm::EID_HSM, hsm::HART_STOP) => continue,
  320. // 不可恢复挂起
  321. (hsm::EID_HSM, hsm::HART_SUSPEND)
  322. if matches!(ctx.a0() as u32, hsm::suspend_type::NON_RETENTIVE) =>
  323. {
  324. break boot(ctx, a1, a2);
  325. }
  326. // legacy console 探测
  327. (base::EID_BASE, base::PROBE_EXTENSION)
  328. if matches!(
  329. ctx.a0(),
  330. legacy::LEGACY_CONSOLE_PUTCHAR | legacy::LEGACY_CONSOLE_GETCHAR
  331. ) =>
  332. {
  333. ret.value = 1;
  334. }
  335. _ => {}
  336. }
  337. } else {
  338. match a7 {
  339. legacy::LEGACY_CONSOLE_PUTCHAR => {
  340. print!("{}", ctx.a0() as u8 as char);
  341. ret.error = 0;
  342. ret.value = a1;
  343. }
  344. legacy::LEGACY_CONSOLE_GETCHAR => {
  345. let mut c = 0u8;
  346. let uart = CONSOLE.lock();
  347. match *uart {
  348. MachineConsole::Uart16550(uart16550) => unsafe {
  349. loop {
  350. if (*uart16550).read(core::slice::from_mut(&mut c)) == 1
  351. {
  352. ret.error = c as _;
  353. ret.value = a1;
  354. break;
  355. }
  356. }
  357. },
  358. }
  359. drop(uart);
  360. }
  361. _ => {}
  362. }
  363. }
  364. ctx.regs().a = [ret.error, ret.value, a2, a3, a4, a5, a6, a7];
  365. mepc::write(mepc::read() + 4);
  366. break ctx.restore();
  367. }
  368. T::Exception(E::IllegalInstruction) => {
  369. if mstatus::read().mpp() == mstatus::MPP::Machine {
  370. panic!("Cannot handle illegal instruction exception from M-MODE");
  371. }
  372. ctx.regs().a = [ctx.a0(), a1, a2, a3, a4, a5, a6, a7];
  373. if !illegal_instruction_handler(&mut ctx) {
  374. delegate();
  375. }
  376. break ctx.restore();
  377. }
  378. // 其他陷入
  379. trap => {
  380. println!(
  381. "
  382. -----------------------------
  383. > trap: {trap:?}
  384. > mepc: {:#018x}
  385. > mtval: {:#018x}
  386. -----------------------------
  387. ",
  388. mepc::read(),
  389. mtval::read()
  390. );
  391. panic!("Stopped with unsupported trap")
  392. }
  393. },
  394. }
  395. }
  396. }
  397. #[inline]
  398. fn delegate() {
  399. use riscv::register::{mcause, mepc, mtval, scause, sepc, sstatus, stval, stvec};
  400. unsafe {
  401. // TODO: 当支持中断嵌套时,需要从ctx里获取mpec。当前ctx.reg().pc与mepc不一致
  402. sepc::write(mepc::read());
  403. scause::write(mcause::read().bits());
  404. stval::write(mtval::read());
  405. sstatus::clear_sie();
  406. if mstatus::read().mpp() == mstatus::MPP::Supervisor {
  407. sstatus::set_spp(sstatus::SPP::Supervisor);
  408. } else {
  409. sstatus::set_spp(sstatus::SPP::User);
  410. }
  411. mepc::write(stvec::read().address());
  412. }
  413. }
  414. #[inline]
  415. fn illegal_instruction_handler(ctx: &mut FastContext) -> bool {
  416. use riscv::register::{mepc, mtval};
  417. use riscv_decode::{decode, Instruction};
  418. let inst = decode(mtval::read() as u32);
  419. match inst {
  420. Ok(Instruction::Csrrs(csr)) => match csr.csr() {
  421. CSR_TIME => {
  422. assert!(
  423. 10 <= csr.rd() && csr.rd() <= 17,
  424. "Unsupported CSR rd: {}",
  425. csr.rd()
  426. );
  427. ctx.regs().a[(csr.rd() - 10) as usize] = unsafe { SBI.assume_init_mut() }
  428. .clint
  429. .as_ref()
  430. .unwrap()
  431. .get_time();
  432. }
  433. CSR_TIMEH => {
  434. assert!(
  435. 10 <= csr.rd() && csr.rd() <= 17,
  436. "Unsupported CSR rd: {}",
  437. csr.rd()
  438. );
  439. ctx.regs().a[(csr.rd() - 10) as usize] = unsafe { SBI.assume_init_mut() }
  440. .clint
  441. .as_ref()
  442. .unwrap()
  443. .get_timeh();
  444. }
  445. _ => return false,
  446. },
  447. _ => return false,
  448. }
  449. mepc::write(mepc::read() + 4);
  450. true
  451. }
  452. #[derive(Debug)]
  453. #[repr(C)]
  454. pub struct SupervisorContext {
  455. pub ra: usize, // 0
  456. pub sp: usize,
  457. pub gp: usize,
  458. pub tp: usize,
  459. pub t0: usize,
  460. pub t1: usize,
  461. pub t2: usize,
  462. pub s0: usize,
  463. pub s1: usize,
  464. pub a0: usize,
  465. pub a1: usize,
  466. pub a2: usize,
  467. pub a3: usize,
  468. pub a4: usize,
  469. pub a5: usize,
  470. pub a6: usize,
  471. pub a7: usize,
  472. pub s2: usize,
  473. pub s3: usize,
  474. pub s4: usize,
  475. pub s5: usize,
  476. pub s6: usize,
  477. pub s7: usize,
  478. pub s8: usize,
  479. pub s9: usize,
  480. pub s10: usize,
  481. pub s11: usize,
  482. pub t3: usize,
  483. pub t4: usize,
  484. pub t5: usize,
  485. pub t6: usize, // 30
  486. pub mepc: usize, // 31
  487. }