non-usize-sbiret-emulator.rs 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325
  1. use XReg::*;
  2. /// This example illustrates how we use non-usize SBI register values on emulators.
  3. use core::ops::ControlFlow;
  4. use sbi_spec::binary::SbiRet;
  5. /// We take RISC-V RV128I as an example, as by now (2025 CE) there are nearly no common host
  6. /// platforms that supports 128-bit pointer width. It will illustrate how we write an emulator
  7. /// with pointer width different than the host platform.
  8. ///
  9. /// This emulator starts at S-mode, allowing the emulated supervisor software to call via
  10. /// the Supervisor Binary Interface (SBI) by using the `ecall` instruction.
  11. pub struct SimpleRv128IHart {
  12. xregs: [u128; 32],
  13. pc: u128,
  14. inst_memory: InstMemory<0x2000_0000, { 0x4000 / 4 }>,
  15. }
  16. // Run the emulator.
  17. fn main() {
  18. let mut memory = InstMemory::new_unimp();
  19. // Call SBI probe_extension to probe the BASE extension which always exists
  20. memory.li(0x0, A0, 0x10);
  21. memory.li(0x4, A6, 3);
  22. memory.li(0x8, A7, 0x10);
  23. memory.ecall(0xC);
  24. // Call SBI system_reset to shutdown emulation
  25. memory.li(0x10, A0, 0x0);
  26. memory.li(0x14, A1, 0x0);
  27. memory.li(0x18, A6, 0x0);
  28. memory.li(0x1C, A7, 0x53525354);
  29. memory.ecall(0x24);
  30. let mut hart = SimpleRv128IHart::new(memory);
  31. let emulation_result = loop {
  32. match hart.stepi() {
  33. Ok(()) => {}
  34. Err(Exception::SupervisorEcall) => match machine_firmware_handle_ecall(&mut hart) {
  35. ControlFlow::Break(value) => break value,
  36. ControlFlow::Continue(()) => continue,
  37. },
  38. Err(e) => {
  39. println!("Emulation failed for unexpected exception: {:?}", e);
  40. return;
  41. }
  42. }
  43. };
  44. println!("Emulation success! Result: {}", emulation_result);
  45. }
  46. /* -- Implementations of SimpleRv128Platform -- */
  47. pub struct InstMemory<const BASE: usize, const N_INSNS: usize> {
  48. inner: [u32; N_INSNS],
  49. }
  50. const OPCODE_OP_IMM: u32 = 0b001_0011;
  51. const OPCODE_LUI: u32 = 0b011_0111;
  52. const FUNCT3_OP_ADD_SUB: u32 = 0b000;
  53. impl<const BASE: usize, const N_INSNS: usize> InstMemory<BASE, N_INSNS> {
  54. pub fn new_unimp() -> Self {
  55. Self {
  56. inner: [0; N_INSNS],
  57. }
  58. }
  59. pub fn addi(&mut self, offset: usize, rd: XReg, rs: XReg, simm12: impl Into<Simm12>) {
  60. let funct3 = FUNCT3_OP_ADD_SUB;
  61. let opcode = OPCODE_OP_IMM;
  62. let word = (u32::from(simm12.into().0) << 20)
  63. | ((rs as u32) << 15)
  64. | (funct3 << 12)
  65. | ((rd as u32) << 7)
  66. | opcode;
  67. self.inner[offset / 4] = word;
  68. }
  69. pub fn lui(&mut self, offset: usize, rd: XReg, simm20: impl Into<Simm20>) {
  70. let opcode = OPCODE_LUI;
  71. let word = (u32::from(simm20.into().0) << 12) | ((rd as u32) << 7) | opcode;
  72. self.inner[offset / 4] = word;
  73. }
  74. pub fn li(&mut self, offset: usize, rd: XReg, imm: u32) {
  75. let simm20 = (imm >> 12) & 0xFFFFF;
  76. let simm12 = imm & 0xFFF;
  77. if simm20 != 0 {
  78. self.lui(offset, rd, simm20);
  79. self.addi(offset + 0x4, rd, rd, simm12);
  80. } else {
  81. self.addi(offset, rd, XReg::Zero, simm12);
  82. }
  83. }
  84. pub fn ecall(&mut self, offset: usize) {
  85. let word = 0b000000000000_00000_000_00000_1110011;
  86. self.inner[offset / 4] = word;
  87. }
  88. pub fn get(&mut self, ptr: u128) -> Option<u32> {
  89. if ptr % 4 != 0 || ptr >= (BASE + 4 * N_INSNS) as u128 {
  90. return None;
  91. }
  92. Some(self.inner[(ptr as usize - BASE) / 4])
  93. }
  94. }
  95. impl SimpleRv128IHart {
  96. pub fn new(inst_memory: InstMemory<0x2000_0000, { 0x4000 / 4 }>) -> Self {
  97. Self {
  98. xregs: [0; 32],
  99. pc: 0x2000_0000,
  100. inst_memory,
  101. }
  102. }
  103. pub fn stepi(&mut self) -> Result<(), Exception> {
  104. let raw_insn = self
  105. .inst_memory
  106. .get(self.pc)
  107. .ok_or(Exception::InstructionAccessFault)?;
  108. println!("Raw insn at 0x{:x?} is 0x{:x?}", self.pc, raw_insn);
  109. let parsed_insn =
  110. Instruction::try_from(raw_insn).map_err(|_| Exception::IllegalInstruction)?;
  111. match parsed_insn {
  112. Instruction::Addi(rd, rs, simm12) => {
  113. self.xregs[rd as usize] = self.xregs[rs as usize] + simm12.0 as u128;
  114. self.pc = self.pc.wrapping_add(4);
  115. }
  116. Instruction::Lui(rd, simm20) => {
  117. self.xregs[rd as usize] = (simm20.0 as u128) << 12;
  118. self.pc = self.pc.wrapping_add(4);
  119. }
  120. Instruction::Ecall => return Err(Exception::SupervisorEcall),
  121. }
  122. Ok(())
  123. }
  124. }
  125. #[derive(Debug)]
  126. pub enum Exception {
  127. IllegalInstruction,
  128. InstructionAccessFault,
  129. SupervisorEcall,
  130. }
  131. #[derive(Debug)]
  132. pub enum Instruction {
  133. Addi(XReg, XReg, Simm12),
  134. Lui(XReg, Simm20),
  135. Ecall,
  136. }
  137. impl TryFrom<u32> for Instruction {
  138. type Error = ();
  139. fn try_from(value: u32) -> Result<Self, Self::Error> {
  140. let opcode = value & 0x7F;
  141. let rd = ((value >> 7) & 0x1F).try_into().unwrap();
  142. let rs1 = ((value >> 15) & 0x1F).try_into().unwrap();
  143. let funct3 = (value >> 12) & 0b111;
  144. let simm12 = (value >> 20).into();
  145. let simm20 = (value >> 12).into();
  146. if opcode == OPCODE_OP_IMM && funct3 == FUNCT3_OP_ADD_SUB {
  147. Ok(Self::Addi(rd, rs1, simm12))
  148. } else if opcode == OPCODE_LUI {
  149. Ok(Self::Lui(rd, simm20))
  150. } else if value == 0b000000000000_00000_000_00000_1110011 {
  151. Ok(Self::Ecall)
  152. } else {
  153. Err(())
  154. }
  155. }
  156. }
  157. #[derive(Clone, Copy, Debug)]
  158. pub enum XReg {
  159. Zero = 0,
  160. Ra = 1,
  161. Sp = 2,
  162. Gp = 3,
  163. Tp = 4,
  164. T0 = 5,
  165. T1 = 6,
  166. T2 = 7,
  167. S0 = 8,
  168. S1 = 9,
  169. A0 = 10,
  170. A1 = 11,
  171. A2 = 12,
  172. A3 = 13,
  173. A4 = 14,
  174. A5 = 15,
  175. A6 = 16,
  176. A7 = 17,
  177. S2 = 18,
  178. S3 = 19,
  179. S4 = 20,
  180. S5 = 21,
  181. S6 = 22,
  182. S7 = 23,
  183. S8 = 24,
  184. S9 = 25,
  185. S10 = 26,
  186. S11 = 27,
  187. T3 = 28,
  188. T4 = 29,
  189. T5 = 30,
  190. T6 = 31,
  191. }
  192. impl TryFrom<u32> for XReg {
  193. type Error = ();
  194. fn try_from(value: u32) -> Result<Self, Self::Error> {
  195. Ok(match value {
  196. 0 => Zero,
  197. 1 => Ra,
  198. 2 => Sp,
  199. 3 => Gp,
  200. 4 => Tp,
  201. 5 => T0,
  202. 6 => T1,
  203. 7 => T2,
  204. 8 => S0,
  205. 9 => S1,
  206. 10 => A0,
  207. 11 => A1,
  208. 12 => A2,
  209. 13 => A3,
  210. 14 => A4,
  211. 15 => A5,
  212. 16 => A6,
  213. 17 => A7,
  214. 18 => S2,
  215. 19 => S3,
  216. 20 => S4,
  217. 21 => S5,
  218. 22 => S6,
  219. 23 => S7,
  220. 24 => S8,
  221. 25 => S9,
  222. 26 => S10,
  223. 27 => S11,
  224. 28 => T3,
  225. 29 => T4,
  226. 30 => T5,
  227. 31 => T6,
  228. _ => return Err(()),
  229. })
  230. }
  231. }
  232. #[derive(Clone, Copy, Debug)]
  233. pub struct Simm12(u16);
  234. impl From<u32> for Simm12 {
  235. fn from(value: u32) -> Self {
  236. Self((value & 0x0FFF) as u16)
  237. }
  238. }
  239. #[derive(Clone, Copy, Debug)]
  240. pub struct Simm20(u32);
  241. impl From<u32> for Simm20 {
  242. fn from(value: u32) -> Self {
  243. Self(value & 0xFFFFF)
  244. }
  245. }
  246. // A simple SBI implementation without using RustSBI.
  247. fn machine_firmware_handle_ecall(hart: &mut SimpleRv128IHart) -> ControlFlow<u128> {
  248. println!("Handle ecall, registers: {:x?}", hart.xregs);
  249. let param = [
  250. hart.xregs[A0 as usize],
  251. hart.xregs[A1 as usize],
  252. hart.xregs[A2 as usize],
  253. hart.xregs[A3 as usize],
  254. hart.xregs[A4 as usize],
  255. hart.xregs[A5 as usize],
  256. ];
  257. let ret = handle_sbi_call(hart.xregs[A7 as usize], hart.xregs[A6 as usize], param);
  258. println!("SbiRet: {:?}", ret);
  259. if ret.error == 0x114514 {
  260. return ControlFlow::Break(ret.value);
  261. }
  262. hart.xregs[A0 as usize] = ret.error;
  263. hart.xregs[A1 as usize] = ret.value;
  264. hart.pc = hart.pc.wrapping_add(4);
  265. ControlFlow::Continue(())
  266. }
  267. fn handle_sbi_call(extension: u128, function: u128, param: [u128; 6]) -> SbiRet<u128> {
  268. match (extension, function) {
  269. // BASE probe_extension
  270. (0x10, 3) => {
  271. if param[0] == 0x10 {
  272. SbiRet::success(1)
  273. } else {
  274. SbiRet::success(0)
  275. }
  276. }
  277. // SRST system_reset
  278. (0x53525354, 0) => {
  279. let (reset_type, reset_reason) = (param[0], param[1]);
  280. if reset_type == sbi_spec::srst::RESET_TYPE_SHUTDOWN as u128 {
  281. // special SBI error value for platform shutdown
  282. SbiRet {
  283. value: reset_reason,
  284. error: 0x114514,
  285. }
  286. } else {
  287. SbiRet::not_supported()
  288. }
  289. }
  290. _ => SbiRet::not_supported(),
  291. }
  292. }