instance.rs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808
  1. use crate::{
  2. spec::binary::SbiRet, Console, Cppc, Fence, HartMask, Hsm, Ipi, Pmu, Reset, Susp, Timer,
  3. IMPL_ID_RUSTSBI, RUSTSBI_VERSION, SBI_SPEC_MAJOR, SBI_SPEC_MINOR,
  4. };
  5. use core::convert::Infallible;
  6. #[cfg(feature = "machine")]
  7. use riscv::register::{marchid, mimpid, mvendorid};
  8. use spec::binary::Physical;
  9. /// RustSBI instance including standard extensions
  10. ///
  11. /// By now RustSBI supports to run instance based interface on systems has environment pointer width
  12. /// that is the same as supervisor pointer width.
  13. #[derive(Clone, Debug)]
  14. pub struct RustSBI<T, I, R, H, S, P, C, SU, CP> {
  15. timer: Option<T>,
  16. ipi: Option<I>,
  17. rfnc: Option<R>,
  18. hsm: Option<H>,
  19. srst: Option<S>,
  20. pmu: Option<P>,
  21. dbcn: Option<C>,
  22. susp: Option<SU>,
  23. cppc: Option<CP>,
  24. #[cfg(not(feature = "machine"))]
  25. info: MachineInfo,
  26. }
  27. /// Machine information for SBI environment
  28. ///
  29. /// This structure is useful to build an SBI environment when RustSBI is not run directly on RISC-V machine mode.
  30. #[cfg(not(feature = "machine"))]
  31. #[derive(Clone, Copy, Debug)]
  32. pub struct MachineInfo {
  33. /// Register `mvendorid` for supervisor environment
  34. pub mvendorid: usize,
  35. /// Register `marchid` for supervisor environment
  36. pub marchid: usize,
  37. /// Register `mimpid` for supervisor environment
  38. pub mimpid: usize,
  39. }
  40. impl<T: Timer, I: Ipi, R: Fence, H: Hsm, S: Reset, P: Pmu, C: Console, SU: Susp, CP: Cppc>
  41. RustSBI<T, I, R, H, S, P, C, SU, CP>
  42. {
  43. /// Create RustSBI instance on current machine environment for all the SBI extensions
  44. #[cfg(feature = "machine")]
  45. #[inline]
  46. pub const fn new_machine(
  47. timer: T,
  48. ipi: I,
  49. rfnc: R,
  50. hsm: H,
  51. srst: S,
  52. pmu: P,
  53. dbcn: C,
  54. susp: SU,
  55. cppc: CP,
  56. ) -> Self {
  57. Self {
  58. timer: Some(timer),
  59. ipi: Some(ipi),
  60. rfnc: Some(rfnc),
  61. hsm: Some(hsm),
  62. srst: Some(srst),
  63. pmu: Some(pmu),
  64. dbcn: Some(dbcn),
  65. susp: Some(susp),
  66. cppc: Some(cppc),
  67. }
  68. }
  69. /// Create RustSBI instance on given machine information for all the SBI extensions
  70. #[cfg(not(feature = "machine"))]
  71. #[allow(clippy::too_many_arguments)] // fixme: is it possible to have a better design here?
  72. #[inline]
  73. pub const fn with_machine_info(
  74. timer: T,
  75. ipi: I,
  76. rfnc: R,
  77. hsm: H,
  78. srst: S,
  79. pmu: P,
  80. dbcn: C,
  81. susp: SU,
  82. cppc: CP,
  83. info: MachineInfo,
  84. ) -> Self {
  85. Self {
  86. timer: Some(timer),
  87. ipi: Some(ipi),
  88. rfnc: Some(rfnc),
  89. hsm: Some(hsm),
  90. srst: Some(srst),
  91. pmu: Some(pmu),
  92. dbcn: Some(dbcn),
  93. susp: Some(susp),
  94. cppc: Some(cppc),
  95. info,
  96. }
  97. }
  98. /// Handle supervisor environment call with given parameters and return the `SbiRet` result.
  99. #[inline]
  100. pub fn handle_ecall(&mut self, extension: usize, function: usize, param: [usize; 6]) -> SbiRet {
  101. match extension {
  102. spec::rfnc::EID_RFNC => {
  103. let Some(rfnc) = &mut self.rfnc else {
  104. return SbiRet::not_supported();
  105. };
  106. let [param0, param1, param2, param3, param4] =
  107. [param[0], param[1], param[2], param[3], param[4]];
  108. let hart_mask = crate::HartMask::from_mask_base(param0, param1);
  109. match function {
  110. spec::rfnc::REMOTE_FENCE_I => rfnc.remote_fence_i(hart_mask),
  111. spec::rfnc::REMOTE_SFENCE_VMA => {
  112. rfnc.remote_sfence_vma(hart_mask, param2, param3)
  113. }
  114. spec::rfnc::REMOTE_SFENCE_VMA_ASID => {
  115. rfnc.remote_sfence_vma_asid(hart_mask, param2, param3, param4)
  116. }
  117. spec::rfnc::REMOTE_HFENCE_GVMA_VMID => {
  118. rfnc.remote_hfence_gvma_vmid(hart_mask, param2, param3, param4)
  119. }
  120. spec::rfnc::REMOTE_HFENCE_GVMA => {
  121. rfnc.remote_hfence_gvma(hart_mask, param2, param3)
  122. }
  123. spec::rfnc::REMOTE_HFENCE_VVMA_ASID => {
  124. rfnc.remote_hfence_vvma_asid(hart_mask, param2, param3, param4)
  125. }
  126. spec::rfnc::REMOTE_HFENCE_VVMA => {
  127. rfnc.remote_hfence_vvma(hart_mask, param2, param3)
  128. }
  129. _ => SbiRet::not_supported(),
  130. }
  131. }
  132. spec::time::EID_TIME => match () {
  133. #[cfg(target_pointer_width = "64")]
  134. () => {
  135. let Some(timer) = &mut self.timer else {
  136. return SbiRet::not_supported();
  137. };
  138. let [param0] = [param[0]];
  139. match function {
  140. spec::time::SET_TIMER => {
  141. timer.set_timer(param0 as _);
  142. SbiRet::success(0)
  143. }
  144. _ => SbiRet::not_supported(),
  145. }
  146. }
  147. #[cfg(target_pointer_width = "32")]
  148. () => {
  149. let Some(timer) = &mut self.timer else {
  150. return SbiRet::not_supported();
  151. };
  152. let [param0, param1] = [param[0], param[1]];
  153. match function {
  154. spec::time::SET_TIMER => {
  155. timer.set_timer(concat_u32(param1, param0));
  156. SbiRet::success(0)
  157. }
  158. _ => SbiRet::not_supported(),
  159. }
  160. }
  161. },
  162. spec::spi::EID_SPI => {
  163. let Some(ipi) = &mut self.ipi else {
  164. return SbiRet::not_supported();
  165. };
  166. let [param0, param1] = [param[0], param[1]];
  167. match function {
  168. spec::spi::SEND_IPI => ipi.send_ipi(HartMask::from_mask_base(param0, param1)),
  169. _ => SbiRet::not_supported(),
  170. }
  171. }
  172. spec::base::EID_BASE => {
  173. let [param0] = [param[0]];
  174. let value = match function {
  175. spec::base::GET_SBI_SPEC_VERSION => (SBI_SPEC_MAJOR << 24) | (SBI_SPEC_MINOR),
  176. spec::base::GET_SBI_IMPL_ID => IMPL_ID_RUSTSBI,
  177. spec::base::GET_SBI_IMPL_VERSION => RUSTSBI_VERSION,
  178. spec::base::PROBE_EXTENSION => {
  179. // only provides probes to standard extensions. If you have customized extensions to be probed,
  180. // run it even before this `handle_ecall` function.
  181. self.probe_extension(param0)
  182. }
  183. spec::base::GET_MVENDORID => match () {
  184. #[cfg(feature = "machine")]
  185. () => mvendorid::read().map(|r| r.bits()).unwrap_or(0),
  186. #[cfg(not(feature = "machine"))]
  187. () => self.info.mvendorid,
  188. },
  189. spec::base::GET_MARCHID => match () {
  190. #[cfg(feature = "machine")]
  191. () => marchid::read().map(|r| r.bits()).unwrap_or(0),
  192. #[cfg(not(feature = "machine"))]
  193. () => self.info.marchid,
  194. },
  195. spec::base::GET_MIMPID => match () {
  196. #[cfg(feature = "machine")]
  197. () => mimpid::read().map(|r| r.bits()).unwrap_or(0),
  198. #[cfg(not(feature = "machine"))]
  199. () => self.info.mimpid,
  200. },
  201. _ => return SbiRet::not_supported(),
  202. };
  203. SbiRet::success(value)
  204. }
  205. spec::hsm::EID_HSM => {
  206. let Some(hsm) = &mut self.hsm else {
  207. return SbiRet::not_supported();
  208. };
  209. let [param0, param1, param2] = [param[0], param[1], param[2]];
  210. match function {
  211. spec::hsm::HART_START => hsm.hart_start(param0, param1, param2),
  212. spec::hsm::HART_STOP => hsm.hart_stop(),
  213. spec::hsm::HART_GET_STATUS => hsm.hart_get_status(param0),
  214. spec::hsm::HART_SUSPEND => {
  215. if let Ok(suspend_type) = u32::try_from(param0) {
  216. hsm.hart_suspend(suspend_type, param1, param2)
  217. } else {
  218. SbiRet::invalid_param()
  219. }
  220. }
  221. _ => SbiRet::not_supported(),
  222. }
  223. }
  224. spec::srst::EID_SRST => {
  225. let Some(srst) = &mut self.srst else {
  226. return SbiRet::not_supported();
  227. };
  228. let [param0, param1] = [param[0], param[1]];
  229. match function {
  230. spec::srst::SYSTEM_RESET => {
  231. match (u32::try_from(param0), u32::try_from(param1)) {
  232. (Ok(reset_type), Ok(reset_reason)) => {
  233. srst.system_reset(reset_type, reset_reason)
  234. }
  235. (_, _) => SbiRet::invalid_param(),
  236. }
  237. }
  238. _ => SbiRet::not_supported(),
  239. }
  240. }
  241. spec::pmu::EID_PMU => match () {
  242. #[cfg(target_pointer_width = "64")]
  243. () => {
  244. let Some(pmu) = &mut self.pmu else {
  245. return SbiRet::not_supported();
  246. };
  247. let [param0, param1, param2, param3, param4] =
  248. [param[0], param[1], param[2], param[3], param[4]];
  249. match function {
  250. spec::pmu::PMU_NUM_COUNTERS => SbiRet::success(pmu.num_counters()),
  251. spec::pmu::PMU_COUNTER_GET_INFO => pmu.counter_get_info(param0),
  252. spec::pmu::PMU_COUNTER_CONFIG_MATCHING => {
  253. pmu.counter_config_matching(param0, param1, param2, param3, param4 as _)
  254. }
  255. spec::pmu::PMU_COUNTER_START => {
  256. pmu.counter_start(param0, param1, param2, param3 as _)
  257. }
  258. spec::pmu::PMU_COUNTER_STOP => pmu.counter_stop(param0, param1, param2),
  259. spec::pmu::PMU_COUNTER_FW_READ => pmu.counter_fw_read(param0),
  260. spec::pmu::PMU_COUNTER_FW_READ_HI => pmu.counter_fw_read_hi(param0),
  261. _ => SbiRet::not_supported(),
  262. }
  263. }
  264. #[cfg(target_pointer_width = "32")]
  265. () => {
  266. let Some(pmu) = &mut self.pmu else {
  267. return SbiRet::not_supported();
  268. };
  269. let [param0, param1, param2, param3, param4, param5] =
  270. [param[0], param[1], param[2], param[3], param[4], param[5]];
  271. match function {
  272. spec::pmu::PMU_NUM_COUNTERS => SbiRet::success(pmu.num_counters()),
  273. spec::pmu::PMU_COUNTER_GET_INFO => pmu.counter_get_info(param0),
  274. spec::pmu::PMU_COUNTER_CONFIG_MATCHING => pmu.counter_config_matching(
  275. param0,
  276. param1,
  277. param2,
  278. param3,
  279. concat_u32(param5, param4),
  280. ),
  281. spec::pmu::PMU_COUNTER_START => {
  282. pmu.counter_start(param0, param1, param2, concat_u32(param4, param3))
  283. }
  284. spec::pmu::PMU_COUNTER_STOP => pmu.counter_stop(param0, param1, param2),
  285. spec::pmu::PMU_COUNTER_FW_READ => pmu.counter_fw_read(param0),
  286. spec::pmu::PMU_COUNTER_FW_READ_HI => pmu.counter_fw_read_hi(param0),
  287. _ => SbiRet::not_supported(),
  288. }
  289. }
  290. },
  291. spec::dbcn::EID_DBCN => {
  292. let Some(dbcn) = &mut self.dbcn else {
  293. return SbiRet::not_supported();
  294. };
  295. let [param0, param1, param2] = [param[0], param[1], param[2]];
  296. match function {
  297. spec::dbcn::CONSOLE_WRITE => {
  298. let bytes = Physical::new(param0, param1, param2);
  299. dbcn.write(bytes)
  300. }
  301. spec::dbcn::CONSOLE_READ => {
  302. let bytes = Physical::new(param0, param1, param2);
  303. dbcn.read(bytes)
  304. }
  305. spec::dbcn::CONSOLE_WRITE_BYTE => dbcn.write_byte((param0 & 0xFF) as u8),
  306. _ => SbiRet::not_supported(),
  307. }
  308. }
  309. spec::susp::EID_SUSP => {
  310. let Some(susp) = &mut self.susp else {
  311. return SbiRet::not_supported();
  312. };
  313. let [param0, param1, param2] = [param[0], param[1], param[2]];
  314. match function {
  315. spec::susp::SUSPEND => match u32::try_from(param0) {
  316. Ok(sleep_type) => susp.system_suspend(sleep_type, param1, param2),
  317. _ => SbiRet::invalid_param(),
  318. },
  319. _ => SbiRet::not_supported(),
  320. }
  321. }
  322. spec::cppc::EID_CPPC => match () {
  323. #[cfg(target_pointer_width = "64")]
  324. () => {
  325. let Some(cppc) = &mut self.cppc else {
  326. return SbiRet::not_supported();
  327. };
  328. let [param0, param1] = [param[0], param[1]];
  329. match function {
  330. spec::cppc::PROBE => match u32::try_from(param0) {
  331. Ok(reg_id) => cppc.probe(reg_id),
  332. _ => SbiRet::invalid_param(),
  333. },
  334. spec::cppc::READ => match u32::try_from(param0) {
  335. Ok(reg_id) => cppc.read(reg_id),
  336. _ => SbiRet::invalid_param(),
  337. },
  338. spec::cppc::READ_HI => match u32::try_from(param0) {
  339. Ok(reg_id) => cppc.read_hi(reg_id),
  340. _ => SbiRet::invalid_param(),
  341. },
  342. spec::cppc::WRITE => match u32::try_from(param0) {
  343. Ok(reg_id) => cppc.write(reg_id, param1 as _),
  344. _ => SbiRet::invalid_param(),
  345. },
  346. _ => SbiRet::not_supported(),
  347. }
  348. }
  349. #[cfg(target_pointer_width = "32")]
  350. () => {
  351. let Some(cppc) = &mut self.cppc else {
  352. return SbiRet::not_supported();
  353. };
  354. let [param0, param1, param2] = [param[0], param[1], param[2]];
  355. match function {
  356. spec::cppc::PROBE => cppc.probe(param0 as _),
  357. spec::cppc::READ => cppc.read(param0 as _),
  358. spec::cppc::READ_HI => cppc.read_hi(param0 as _),
  359. spec::cppc::WRITE => cppc.write(param0 as _, concat_u32(param2, param1)),
  360. _ => SbiRet::not_supported(),
  361. }
  362. }
  363. },
  364. _ => SbiRet::not_supported(),
  365. }
  366. }
  367. #[inline]
  368. fn probe_extension(&self, extension: usize) -> usize {
  369. let ans = match extension {
  370. spec::base::EID_BASE => true,
  371. spec::time::EID_TIME => self.timer.is_some(),
  372. spec::spi::EID_SPI => self.ipi.is_some(),
  373. spec::rfnc::EID_RFNC => self.rfnc.is_some(),
  374. spec::srst::EID_SRST => self.srst.is_some(),
  375. spec::hsm::EID_HSM => self.hsm.is_some(),
  376. spec::pmu::EID_PMU => self.pmu.is_some(),
  377. spec::dbcn::EID_DBCN => self.dbcn.is_some(),
  378. spec::susp::EID_SUSP => self.susp.is_some(),
  379. spec::cppc::EID_CPPC => self.cppc.is_some(),
  380. _ => false,
  381. };
  382. if ans {
  383. spec::base::UNAVAILABLE_EXTENSION.wrapping_add(1)
  384. } else {
  385. spec::base::UNAVAILABLE_EXTENSION
  386. }
  387. }
  388. }
  389. #[cfg(target_pointer_width = "32")]
  390. #[inline]
  391. const fn concat_u32(h: usize, l: usize) -> u64 {
  392. ((h as u64) << 32) | (l as u64)
  393. }
  394. /// Structure to build a RustSBI instance
  395. pub struct Builder<T, I, R, H, S, P, C, SU, CP> {
  396. inner: RustSBI<T, I, R, H, S, P, C, SU, CP>,
  397. }
  398. impl
  399. Builder<
  400. Infallible,
  401. Infallible,
  402. Infallible,
  403. Infallible,
  404. Infallible,
  405. Infallible,
  406. Infallible,
  407. Infallible,
  408. Infallible,
  409. >
  410. {
  411. /// Create a new `Builder` from current machine environment
  412. #[inline]
  413. #[cfg(feature = "machine")]
  414. pub const fn new_machine() -> Builder<
  415. Infallible,
  416. Infallible,
  417. Infallible,
  418. Infallible,
  419. Infallible,
  420. Infallible,
  421. Infallible,
  422. Infallible,
  423. Infallible,
  424. > {
  425. Builder {
  426. inner: RustSBI {
  427. timer: None,
  428. ipi: None,
  429. rfnc: None,
  430. hsm: None,
  431. srst: None,
  432. pmu: None,
  433. dbcn: None,
  434. susp: None,
  435. cppc: None,
  436. },
  437. }
  438. }
  439. /// Create a new `Builder` from machine information
  440. #[inline]
  441. #[cfg(not(feature = "machine"))]
  442. pub const fn with_machine_info(
  443. info: MachineInfo,
  444. ) -> Builder<
  445. Infallible,
  446. Infallible,
  447. Infallible,
  448. Infallible,
  449. Infallible,
  450. Infallible,
  451. Infallible,
  452. Infallible,
  453. Infallible,
  454. > {
  455. Builder {
  456. inner: RustSBI {
  457. timer: None,
  458. ipi: None,
  459. rfnc: None,
  460. hsm: None,
  461. srst: None,
  462. pmu: None,
  463. dbcn: None,
  464. susp: None,
  465. cppc: None,
  466. info,
  467. },
  468. }
  469. }
  470. }
  471. // fixme: in future releases we may use type-changing struct update syntax like:
  472. // Builder { inner: RustSBI { timer: None, ..self.inner } }
  473. // https://github.com/rust-lang/rust/issues/86555
  474. // fixme: struct `Infallible` should be replaced to never type once it's stablized
  475. impl<T, I, R, H, S, P, C, SU, CP> Builder<T, I, R, H, S, P, C, SU, CP> {
  476. /// Add Timer programmer extension to RustSBI
  477. #[inline]
  478. pub fn with_timer<T2: Timer>(self, timer: T2) -> Builder<T2, I, R, H, S, P, C, SU, CP> {
  479. Builder {
  480. inner: RustSBI {
  481. timer: Some(timer),
  482. ipi: self.inner.ipi,
  483. rfnc: self.inner.rfnc,
  484. hsm: self.inner.hsm,
  485. srst: self.inner.srst,
  486. pmu: self.inner.pmu,
  487. dbcn: self.inner.dbcn,
  488. susp: self.inner.susp,
  489. cppc: self.inner.cppc,
  490. #[cfg(not(feature = "machine"))]
  491. info: self.inner.info,
  492. },
  493. }
  494. }
  495. /// Add Inter-processor Interrupt extension to RustSBI
  496. #[inline]
  497. pub fn with_ipi<I2: Ipi>(self, ipi: I2) -> Builder<T, I2, R, H, S, P, C, SU, CP> {
  498. Builder {
  499. inner: RustSBI {
  500. timer: self.inner.timer,
  501. ipi: Some(ipi),
  502. rfnc: self.inner.rfnc,
  503. hsm: self.inner.hsm,
  504. srst: self.inner.srst,
  505. pmu: self.inner.pmu,
  506. dbcn: self.inner.dbcn,
  507. susp: self.inner.susp,
  508. cppc: self.inner.cppc,
  509. #[cfg(not(feature = "machine"))]
  510. info: self.inner.info,
  511. },
  512. }
  513. }
  514. /// Add Remote Fence extension to RustSBI
  515. #[inline]
  516. pub fn with_fence<R2: Fence>(self, fence: R2) -> Builder<T, I, R2, H, S, P, C, SU, CP> {
  517. Builder {
  518. inner: RustSBI {
  519. timer: self.inner.timer,
  520. ipi: self.inner.ipi,
  521. rfnc: Some(fence),
  522. hsm: self.inner.hsm,
  523. srst: self.inner.srst,
  524. pmu: self.inner.pmu,
  525. dbcn: self.inner.dbcn,
  526. susp: self.inner.susp,
  527. cppc: self.inner.cppc,
  528. #[cfg(not(feature = "machine"))]
  529. info: self.inner.info,
  530. },
  531. }
  532. }
  533. /// Add Hart State Monitor extension to RustSBI
  534. #[inline]
  535. pub fn with_hsm<H2: Hsm>(self, hsm: H2) -> Builder<T, I, R, H2, S, P, C, SU, CP> {
  536. Builder {
  537. inner: RustSBI {
  538. timer: self.inner.timer,
  539. ipi: self.inner.ipi,
  540. rfnc: self.inner.rfnc,
  541. hsm: Some(hsm),
  542. srst: self.inner.srst,
  543. pmu: self.inner.pmu,
  544. dbcn: self.inner.dbcn,
  545. susp: self.inner.susp,
  546. cppc: self.inner.cppc,
  547. #[cfg(not(feature = "machine"))]
  548. info: self.inner.info,
  549. },
  550. }
  551. }
  552. /// Add System Reset extension to RustSBI
  553. #[inline]
  554. pub fn with_reset<S2: Reset>(self, reset: S2) -> Builder<T, I, R, H, S2, P, C, SU, CP> {
  555. Builder {
  556. inner: RustSBI {
  557. timer: self.inner.timer,
  558. ipi: self.inner.ipi,
  559. rfnc: self.inner.rfnc,
  560. hsm: self.inner.hsm,
  561. srst: Some(reset),
  562. pmu: self.inner.pmu,
  563. dbcn: self.inner.dbcn,
  564. susp: self.inner.susp,
  565. cppc: self.inner.cppc,
  566. #[cfg(not(feature = "machine"))]
  567. info: self.inner.info,
  568. },
  569. }
  570. }
  571. /// Add Performance Monitor Unit extension to RustSBI
  572. #[inline]
  573. pub fn with_pmu<P2: Pmu>(self, pmu: P2) -> Builder<T, I, R, H, S, P2, C, SU, CP> {
  574. Builder {
  575. inner: RustSBI {
  576. timer: self.inner.timer,
  577. ipi: self.inner.ipi,
  578. rfnc: self.inner.rfnc,
  579. hsm: self.inner.hsm,
  580. srst: self.inner.srst,
  581. pmu: Some(pmu),
  582. dbcn: self.inner.dbcn,
  583. susp: self.inner.susp,
  584. cppc: self.inner.cppc,
  585. #[cfg(not(feature = "machine"))]
  586. info: self.inner.info,
  587. },
  588. }
  589. }
  590. /// Add Debug Console extension to RustSBI
  591. #[inline]
  592. pub fn with_console<C2: Console>(self, console: C2) -> Builder<T, I, R, H, S, P, C2, SU, CP> {
  593. Builder {
  594. inner: RustSBI {
  595. timer: self.inner.timer,
  596. ipi: self.inner.ipi,
  597. rfnc: self.inner.rfnc,
  598. hsm: self.inner.hsm,
  599. srst: self.inner.srst,
  600. pmu: self.inner.pmu,
  601. dbcn: Some(console),
  602. susp: self.inner.susp,
  603. cppc: self.inner.cppc,
  604. #[cfg(not(feature = "machine"))]
  605. info: self.inner.info,
  606. },
  607. }
  608. }
  609. /// Add System Suspend extension to RustSBI
  610. #[inline]
  611. pub fn with_susp<SU2: Susp>(self, susp: SU2) -> Builder<T, I, R, H, S, P, C, SU2, CP> {
  612. Builder {
  613. inner: RustSBI {
  614. timer: self.inner.timer,
  615. ipi: self.inner.ipi,
  616. rfnc: self.inner.rfnc,
  617. hsm: self.inner.hsm,
  618. srst: self.inner.srst,
  619. pmu: self.inner.pmu,
  620. dbcn: self.inner.dbcn,
  621. susp: Some(susp),
  622. cppc: self.inner.cppc,
  623. #[cfg(not(feature = "machine"))]
  624. info: self.inner.info,
  625. },
  626. }
  627. }
  628. /// Add CPPC extension to RustSBI
  629. #[inline]
  630. pub fn with_cppc<CP2: Cppc>(self, cppc: CP2) -> Builder<T, I, R, H, S, P, C, SU, CP2> {
  631. Builder {
  632. inner: RustSBI {
  633. timer: self.inner.timer,
  634. ipi: self.inner.ipi,
  635. rfnc: self.inner.rfnc,
  636. hsm: self.inner.hsm,
  637. srst: self.inner.srst,
  638. pmu: self.inner.pmu,
  639. dbcn: self.inner.dbcn,
  640. susp: self.inner.susp,
  641. cppc: Some(cppc),
  642. #[cfg(not(feature = "machine"))]
  643. info: self.inner.info,
  644. },
  645. }
  646. }
  647. /// Build the target RustSBI instance
  648. #[inline]
  649. pub fn build(self) -> RustSBI<T, I, R, H, S, P, C, SU, CP> {
  650. self.inner
  651. }
  652. }
  653. // Placeholder for a structure that implements all RustSBI traits but is never accessed
  654. // fixme: Should be replaced to never type `!` once it's stablized
  655. // https://github.com/rust-lang/rust/issues/35121
  656. // fixme: should be replaced to impl SomeTrait for ! once never type is stablized
  657. impl Timer for Infallible {
  658. fn set_timer(&self, _: u64) {
  659. unreachable!()
  660. }
  661. }
  662. impl Ipi for Infallible {
  663. fn send_ipi(&self, _: HartMask) -> SbiRet {
  664. unreachable!()
  665. }
  666. }
  667. impl Fence for Infallible {
  668. fn remote_fence_i(&self, _: HartMask) -> SbiRet {
  669. unreachable!()
  670. }
  671. fn remote_sfence_vma(&self, _: HartMask, _: usize, _: usize) -> SbiRet {
  672. unreachable!()
  673. }
  674. fn remote_sfence_vma_asid(&self, _: HartMask, _: usize, _: usize, _: usize) -> SbiRet {
  675. unreachable!()
  676. }
  677. fn remote_hfence_gvma_vmid(&self, _: HartMask, _: usize, _: usize, _: usize) -> SbiRet {
  678. unreachable!()
  679. }
  680. fn remote_hfence_gvma(&self, _: HartMask, _: usize, _: usize) -> SbiRet {
  681. unreachable!()
  682. }
  683. fn remote_hfence_vvma_asid(&self, _: HartMask, _: usize, _: usize, _: usize) -> SbiRet {
  684. unreachable!()
  685. }
  686. fn remote_hfence_vvma(&self, _: HartMask, _: usize, _: usize) -> SbiRet {
  687. unreachable!()
  688. }
  689. }
  690. impl Hsm for Infallible {
  691. fn hart_start(&self, _: usize, _: usize, _: usize) -> SbiRet {
  692. unreachable!()
  693. }
  694. fn hart_stop(&self) -> SbiRet {
  695. unreachable!()
  696. }
  697. fn hart_get_status(&self, _: usize) -> SbiRet {
  698. unreachable!()
  699. }
  700. fn hart_suspend(&self, _: u32, _: usize, _: usize) -> SbiRet {
  701. unreachable!()
  702. }
  703. }
  704. impl Reset for Infallible {
  705. fn system_reset(&self, _: u32, _: u32) -> SbiRet {
  706. unreachable!()
  707. }
  708. }
  709. impl Pmu for Infallible {
  710. fn num_counters(&self) -> usize {
  711. unreachable!()
  712. }
  713. fn counter_get_info(&self, _: usize) -> SbiRet {
  714. unreachable!()
  715. }
  716. fn counter_config_matching(&self, _: usize, _: usize, _: usize, _: usize, _: u64) -> SbiRet {
  717. unreachable!()
  718. }
  719. fn counter_start(&self, _: usize, _: usize, _: usize, _: u64) -> SbiRet {
  720. unreachable!()
  721. }
  722. fn counter_stop(&self, _: usize, _: usize, _: usize) -> SbiRet {
  723. unreachable!()
  724. }
  725. fn counter_fw_read(&self, _: usize) -> SbiRet {
  726. unreachable!()
  727. }
  728. fn counter_fw_read_hi(&self, _: usize) -> SbiRet {
  729. unreachable!()
  730. }
  731. }
  732. impl Console for Infallible {
  733. fn write(&self, _: Physical<&[u8]>) -> SbiRet {
  734. unreachable!()
  735. }
  736. fn read(&self, _: Physical<&mut [u8]>) -> SbiRet {
  737. unreachable!()
  738. }
  739. fn write_byte(&self, _: u8) -> SbiRet {
  740. unreachable!()
  741. }
  742. }
  743. impl Susp for Infallible {
  744. fn system_suspend(&self, _: u32, _: usize, _: usize) -> SbiRet {
  745. unreachable!()
  746. }
  747. }
  748. impl Cppc for Infallible {
  749. fn probe(&self, _: u32) -> SbiRet {
  750. unreachable!()
  751. }
  752. fn read(&self, _: u32) -> SbiRet {
  753. unreachable!()
  754. }
  755. fn read_hi(&self, _: u32) -> SbiRet {
  756. unreachable!()
  757. }
  758. fn write(&self, _: u32, _: u64) -> SbiRet {
  759. unreachable!()
  760. }
  761. }