build-full.rs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  1. use core::cell::RefCell;
  2. use rustsbi::RustSBI;
  3. use sbi_spec::{
  4. binary::{HartMask, Physical, SbiRet, SharedPtr},
  5. nacl::shmem_size::NATIVE,
  6. };
  7. #[derive(RustSBI)]
  8. struct FullyImplemented {
  9. console: DummyConsole,
  10. cppc: DummyCppc,
  11. hsm: DummyHsm,
  12. ipi: DummyIpi,
  13. nacl: DummyNacl,
  14. pmu: DummyPmu,
  15. reset: DummyReset,
  16. fence: DummyFence,
  17. sta: DummySta,
  18. susp: DummySusp,
  19. timer: DummyTimer,
  20. info: DummyEnvInfo,
  21. }
  22. #[derive(RustSBI)]
  23. struct AlternateName {
  24. dbcn: DummyConsole,
  25. cppc: DummyCppc,
  26. hsm: DummyHsm,
  27. ipi: DummyIpi,
  28. nacl: DummyNacl,
  29. pmu: DummyPmu,
  30. srst: DummyReset,
  31. rfnc: DummyFence,
  32. sta: DummySta,
  33. susp: DummySusp,
  34. time: DummyTimer,
  35. info: DummyEnvInfo,
  36. }
  37. #[derive(RustSBI)]
  38. struct TupleStruct(
  39. #[rustsbi(dbcn)] DummyConsole,
  40. #[rustsbi(cppc)] DummyCppc,
  41. #[rustsbi(hsm)] DummyHsm,
  42. #[rustsbi(ipi)] DummyIpi,
  43. #[rustsbi(nacl)] DummyNacl,
  44. #[rustsbi(pmu)] DummyPmu,
  45. #[rustsbi(srst)] DummyReset,
  46. #[rustsbi(rfnc)] DummyFence,
  47. #[rustsbi(sta)] DummySta,
  48. #[rustsbi(susp)] DummySusp,
  49. #[rustsbi(time)] DummyTimer,
  50. #[rustsbi(info)] DummyEnvInfo,
  51. );
  52. #[cfg(feature = "machine")]
  53. #[derive(RustSBI)]
  54. struct UnitStruct;
  55. #[test]
  56. fn rustsbi_impl_id() {
  57. let sbi = FullyImplemented {
  58. console: DummyConsole,
  59. cppc: DummyCppc,
  60. hsm: DummyHsm,
  61. ipi: DummyIpi,
  62. nacl: DummyNacl,
  63. pmu: DummyPmu,
  64. reset: DummyReset,
  65. fence: DummyFence,
  66. sta: DummySta,
  67. susp: DummySusp,
  68. timer: DummyTimer(RefCell::new(0)),
  69. info: DummyEnvInfo,
  70. };
  71. assert_eq!(sbi.handle_ecall(0x10, 0x1, [0; 6]).value, 4);
  72. let sbi = AlternateName {
  73. dbcn: DummyConsole,
  74. cppc: DummyCppc,
  75. hsm: DummyHsm,
  76. ipi: DummyIpi,
  77. nacl: DummyNacl,
  78. pmu: DummyPmu,
  79. srst: DummyReset,
  80. rfnc: DummyFence,
  81. sta: DummySta,
  82. susp: DummySusp,
  83. time: DummyTimer(RefCell::new(0)),
  84. info: DummyEnvInfo,
  85. };
  86. assert_eq!(sbi.handle_ecall(0x10, 0x1, [0; 6]).value, 4);
  87. let sbi = TupleStruct(
  88. DummyConsole,
  89. DummyCppc,
  90. DummyHsm,
  91. DummyIpi,
  92. DummyNacl,
  93. DummyPmu,
  94. DummyReset,
  95. DummyFence,
  96. DummySta,
  97. DummySusp,
  98. DummyTimer(RefCell::new(0)),
  99. DummyEnvInfo,
  100. );
  101. assert_eq!(sbi.handle_ecall(0x10, 0x1, [0; 6]).value, 4);
  102. }
  103. #[cfg(feature = "machine")]
  104. #[test]
  105. fn unit_struct() {
  106. let sbi = UnitStruct;
  107. assert_eq!(sbi.handle_ecall(0x10, 0x1, [0; 6]).value, 4);
  108. }
  109. #[test]
  110. fn generated_extensions() {
  111. let sbi = FullyImplemented {
  112. console: DummyConsole,
  113. cppc: DummyCppc,
  114. hsm: DummyHsm,
  115. ipi: DummyIpi,
  116. nacl: DummyNacl,
  117. pmu: DummyPmu,
  118. reset: DummyReset,
  119. fence: DummyFence,
  120. sta: DummySta,
  121. susp: DummySusp,
  122. timer: DummyTimer(RefCell::new(0)),
  123. info: DummyEnvInfo,
  124. };
  125. assert_eq!(
  126. sbi.handle_ecall(0x10, 0, [0; 6]),
  127. SbiRet::success(0x02000000)
  128. );
  129. assert_eq!(sbi.handle_ecall(0x10, 1, [0; 6]), SbiRet::success(4));
  130. assert!(sbi.handle_ecall(0x10, 2, [0; 6]).is_ok());
  131. // All SBI 2.0 extensions, including Base, are supported
  132. for eid in [
  133. 0x10, 0x54494d45, 0x735049, 0x52464e43, 0x48534d, 0x53525354, 0x504d55, 0x4442434e,
  134. 0x53555350, 0x4e41434c, 0x535441, 0x43505043,
  135. ] {
  136. assert_eq!(
  137. sbi.handle_ecall(0x10, 3, [eid, 0, 0, 0, 0, 0]),
  138. SbiRet::success(1)
  139. );
  140. }
  141. // All legacy extensions are not supported
  142. for eid in 0x00..=0x08 {
  143. assert_eq!(
  144. sbi.handle_ecall(0x10, 3, [eid, 0, 0, 0, 0, 0]),
  145. SbiRet::success(0)
  146. );
  147. }
  148. assert_eq!(sbi.handle_ecall(0x4442434e, 0, [0; 6]), SbiRet::success(1));
  149. assert_eq!(sbi.handle_ecall(0x4442434e, 1, [0; 6]), SbiRet::success(2));
  150. assert_eq!(sbi.handle_ecall(0x4442434e, 2, [0; 6]), SbiRet::success(3));
  151. assert_eq!(sbi.handle_ecall(0x43505043, 0, [0; 6]), SbiRet::success(4));
  152. assert_eq!(sbi.handle_ecall(0x43505043, 1, [0; 6]), SbiRet::success(5));
  153. assert_eq!(sbi.handle_ecall(0x43505043, 2, [0; 6]), SbiRet::success(6));
  154. assert_eq!(sbi.handle_ecall(0x43505043, 3, [0; 6]), SbiRet::success(7));
  155. assert_eq!(sbi.handle_ecall(0x48534d, 0, [0; 6]), SbiRet::success(8));
  156. assert_eq!(sbi.handle_ecall(0x48534d, 1, [0; 6]), SbiRet::success(9));
  157. assert_eq!(sbi.handle_ecall(0x48534d, 2, [0; 6]), SbiRet::success(10));
  158. assert_eq!(sbi.handle_ecall(0x48534d, 3, [0; 6]), SbiRet::success(11));
  159. assert_eq!(sbi.handle_ecall(0x735049, 0, [0; 6]), SbiRet::success(12));
  160. assert_eq!(sbi.handle_ecall(0x4E41434C, 0, [0; 6]), SbiRet::success(13));
  161. assert_eq!(sbi.handle_ecall(0x4E41434C, 1, [0; 6]), SbiRet::success(14));
  162. assert_eq!(sbi.handle_ecall(0x4E41434C, 2, [0; 6]), SbiRet::success(15));
  163. assert_eq!(sbi.handle_ecall(0x4E41434C, 3, [0; 6]), SbiRet::success(16));
  164. assert_eq!(sbi.handle_ecall(0x4E41434C, 4, [0; 6]), SbiRet::success(17));
  165. assert_eq!(sbi.handle_ecall(0x504D55, 0, [0; 6]), SbiRet::success(18));
  166. assert_eq!(sbi.handle_ecall(0x504D55, 1, [0; 6]), SbiRet::success(19));
  167. assert_eq!(sbi.handle_ecall(0x504D55, 2, [0; 6]), SbiRet::success(20));
  168. assert_eq!(sbi.handle_ecall(0x504D55, 3, [0; 6]), SbiRet::success(21));
  169. assert_eq!(sbi.handle_ecall(0x504D55, 4, [0; 6]), SbiRet::success(22));
  170. assert_eq!(sbi.handle_ecall(0x504D55, 5, [0; 6]), SbiRet::success(23));
  171. assert_eq!(sbi.handle_ecall(0x504D55, 6, [0; 6]), SbiRet::success(24));
  172. assert_eq!(sbi.handle_ecall(0x53525354, 0, [0; 6]), SbiRet::success(25));
  173. assert_eq!(sbi.handle_ecall(0x52464E43, 0, [0; 6]), SbiRet::success(26));
  174. assert_eq!(sbi.handle_ecall(0x52464E43, 1, [0; 6]), SbiRet::success(27));
  175. assert_eq!(sbi.handle_ecall(0x52464E43, 2, [0; 6]), SbiRet::success(28));
  176. assert_eq!(sbi.handle_ecall(0x52464E43, 3, [0; 6]), SbiRet::success(29));
  177. assert_eq!(sbi.handle_ecall(0x52464E43, 4, [0; 6]), SbiRet::success(30));
  178. assert_eq!(sbi.handle_ecall(0x52464E43, 5, [0; 6]), SbiRet::success(31));
  179. assert_eq!(sbi.handle_ecall(0x52464E43, 6, [0; 6]), SbiRet::success(32));
  180. assert_eq!(sbi.handle_ecall(0x535441, 0, [0; 6]), SbiRet::success(33));
  181. assert_eq!(sbi.handle_ecall(0x53555350, 0, [0; 6]), SbiRet::success(34));
  182. assert!(sbi.handle_ecall(0x54494D45, 0, [0; 6]).is_ok());
  183. assert_eq!(sbi.timer.0.take(), 35);
  184. assert_eq!(sbi.handle_ecall(0x10, 4, [0; 6]), SbiRet::success(36));
  185. assert_eq!(sbi.handle_ecall(0x10, 5, [0; 6]), SbiRet::success(37));
  186. assert_eq!(sbi.handle_ecall(0x10, 6, [0; 6]), SbiRet::success(38));
  187. }
  188. struct DummyConsole;
  189. impl rustsbi::Console for DummyConsole {
  190. fn write(&self, _: Physical<&[u8]>) -> SbiRet {
  191. SbiRet::success(1)
  192. }
  193. fn read(&self, _: Physical<&mut [u8]>) -> SbiRet {
  194. SbiRet::success(2)
  195. }
  196. fn write_byte(&self, _: u8) -> SbiRet {
  197. SbiRet::success(3)
  198. }
  199. }
  200. struct DummyCppc;
  201. impl rustsbi::Cppc for DummyCppc {
  202. fn probe(&self, _: u32) -> SbiRet {
  203. SbiRet::success(4)
  204. }
  205. fn read(&self, _: u32) -> SbiRet {
  206. SbiRet::success(5)
  207. }
  208. fn read_hi(&self, _: u32) -> SbiRet {
  209. SbiRet::success(6)
  210. }
  211. fn write(&self, _: u32, _: u64) -> SbiRet {
  212. SbiRet::success(7)
  213. }
  214. }
  215. struct DummyHsm;
  216. impl rustsbi::Hsm for DummyHsm {
  217. fn hart_start(&self, _: usize, _: usize, _: usize) -> SbiRet {
  218. SbiRet::success(8)
  219. }
  220. fn hart_stop(&self) -> SbiRet {
  221. SbiRet::success(9)
  222. }
  223. fn hart_get_status(&self, _: usize) -> SbiRet {
  224. SbiRet::success(10)
  225. }
  226. fn hart_suspend(&self, _: u32, _: usize, _: usize) -> SbiRet {
  227. SbiRet::success(11)
  228. }
  229. }
  230. struct DummyIpi;
  231. impl rustsbi::Ipi for DummyIpi {
  232. fn send_ipi(&self, _: HartMask) -> SbiRet {
  233. SbiRet::success(12)
  234. }
  235. }
  236. struct DummyNacl;
  237. impl rustsbi::Nacl for DummyNacl {
  238. fn probe_feature(&self, _: u32) -> SbiRet {
  239. SbiRet::success(13)
  240. }
  241. fn set_shmem(&self, _: SharedPtr<[u8; NATIVE]>, _: usize) -> SbiRet {
  242. SbiRet::success(14)
  243. }
  244. fn sync_csr(&self, _: usize) -> SbiRet {
  245. SbiRet::success(15)
  246. }
  247. fn sync_hfence(&self, _: usize) -> SbiRet {
  248. SbiRet::success(16)
  249. }
  250. fn sync_sret(&self) -> SbiRet {
  251. SbiRet::success(17)
  252. }
  253. }
  254. struct DummyPmu;
  255. impl rustsbi::Pmu for DummyPmu {
  256. fn num_counters(&self) -> usize {
  257. 18
  258. }
  259. fn counter_get_info(&self, _: usize) -> SbiRet {
  260. SbiRet::success(19)
  261. }
  262. fn counter_config_matching(&self, _: usize, _: usize, _: usize, _: usize, _: u64) -> SbiRet {
  263. SbiRet::success(20)
  264. }
  265. fn counter_start(&self, _: usize, _: usize, _: usize, _: u64) -> SbiRet {
  266. SbiRet::success(21)
  267. }
  268. fn counter_stop(&self, _: usize, _: usize, _: usize) -> SbiRet {
  269. SbiRet::success(22)
  270. }
  271. fn counter_fw_read(&self, _: usize) -> SbiRet {
  272. SbiRet::success(23)
  273. }
  274. fn counter_fw_read_hi(&self, _: usize) -> SbiRet {
  275. SbiRet::success(24)
  276. }
  277. }
  278. struct DummyReset;
  279. impl rustsbi::Reset for DummyReset {
  280. fn system_reset(&self, _: u32, _: u32) -> SbiRet {
  281. SbiRet::success(25)
  282. }
  283. }
  284. struct DummyFence;
  285. impl rustsbi::Fence for DummyFence {
  286. fn remote_fence_i(&self, _: HartMask) -> SbiRet {
  287. SbiRet::success(26)
  288. }
  289. fn remote_sfence_vma(&self, _: HartMask, _: usize, _: usize) -> SbiRet {
  290. SbiRet::success(27)
  291. }
  292. fn remote_sfence_vma_asid(&self, _: HartMask, _: usize, _: usize, _: usize) -> SbiRet {
  293. SbiRet::success(28)
  294. }
  295. fn remote_hfence_gvma_vmid(&self, _: HartMask, _: usize, _: usize, _: usize) -> SbiRet {
  296. SbiRet::success(29)
  297. }
  298. fn remote_hfence_gvma(&self, _: HartMask, _: usize, _: usize) -> SbiRet {
  299. SbiRet::success(30)
  300. }
  301. fn remote_hfence_vvma_asid(&self, _: HartMask, _: usize, _: usize, _: usize) -> SbiRet {
  302. SbiRet::success(31)
  303. }
  304. fn remote_hfence_vvma(&self, _: HartMask, _: usize, _: usize) -> SbiRet {
  305. SbiRet::success(32)
  306. }
  307. }
  308. struct DummySta;
  309. impl rustsbi::Sta for DummySta {
  310. fn set_shmem(&self, _: SharedPtr<[u8; 64]>, _: usize) -> SbiRet {
  311. SbiRet::success(33)
  312. }
  313. }
  314. struct DummySusp;
  315. impl rustsbi::Susp for DummySusp {
  316. fn system_suspend(&self, _: u32, _: usize, _: usize) -> SbiRet {
  317. SbiRet::success(34)
  318. }
  319. }
  320. struct DummyTimer(RefCell<u64>);
  321. impl rustsbi::Timer for DummyTimer {
  322. fn set_timer(&self, _: u64) {
  323. self.0.replace(35);
  324. }
  325. }
  326. struct DummyEnvInfo;
  327. impl rustsbi::EnvInfo for DummyEnvInfo {
  328. fn mvendorid(&self) -> usize {
  329. 36
  330. }
  331. fn marchid(&self) -> usize {
  332. 37
  333. }
  334. fn mimpid(&self) -> usize {
  335. 38
  336. }
  337. }