trap.rs 15 KB

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