ab_riscv_primitives/instructions/rv32/
m.rs1#[cfg(test)]
4mod tests;
5pub mod zmmul;
6
7use crate::instructions::Instruction;
8use crate::registers::general_purpose::Register;
9use ab_riscv_macros::instruction;
10use core::fmt;
11
12#[instruction]
14#[derive(Debug, Clone, Copy, PartialEq, Eq)]
15pub enum Rv32MInstruction<Reg> {
16 Mul { rd: Reg, rs1: Reg, rs2: Reg },
17 Mulh { rd: Reg, rs1: Reg, rs2: Reg },
18 Mulhsu { rd: Reg, rs1: Reg, rs2: Reg },
19 Mulhu { rd: Reg, rs1: Reg, rs2: Reg },
20 Div { rd: Reg, rs1: Reg, rs2: Reg },
21 Divu { rd: Reg, rs1: Reg, rs2: Reg },
22 Rem { rd: Reg, rs1: Reg, rs2: Reg },
23 Remu { rd: Reg, rs1: Reg, rs2: Reg },
24}
25
26#[instruction]
27impl<Reg> const Instruction for Rv32MInstruction<Reg>
28where
29 Reg: [const] Register<Type = u32>,
30{
31 type Reg = Reg;
32
33 #[inline(always)]
34 fn try_decode(instruction: u32) -> Option<Self> {
35 let opcode = (instruction & 0b111_1111) as u8;
36 let rd_bits = ((instruction >> 7) & 0x1f) as u8;
37 let funct3 = ((instruction >> 12) & 0b111) as u8;
38 let rs1_bits = ((instruction >> 15) & 0x1f) as u8;
39 let rs2_bits = ((instruction >> 20) & 0x1f) as u8;
40 let funct7 = ((instruction >> 25) & 0b111_1111) as u8;
41
42 match opcode {
43 0b0110011 => {
45 let rd = Reg::from_bits(rd_bits)?;
46 let rs1 = Reg::from_bits(rs1_bits)?;
47 let rs2 = Reg::from_bits(rs2_bits)?;
48 match (funct3, funct7) {
49 (0b000, 0b0000001) => Some(Self::Mul { rd, rs1, rs2 }),
50 (0b001, 0b0000001) => Some(Self::Mulh { rd, rs1, rs2 }),
51 (0b010, 0b0000001) => Some(Self::Mulhsu { rd, rs1, rs2 }),
52 (0b011, 0b0000001) => Some(Self::Mulhu { rd, rs1, rs2 }),
53 (0b100, 0b0000001) => Some(Self::Div { rd, rs1, rs2 }),
54 (0b101, 0b0000001) => Some(Self::Divu { rd, rs1, rs2 }),
55 (0b110, 0b0000001) => Some(Self::Rem { rd, rs1, rs2 }),
56 (0b111, 0b0000001) => Some(Self::Remu { rd, rs1, rs2 }),
57 _ => None,
58 }
59 }
60 _ => None,
61 }
62 }
63
64 #[inline(always)]
65 fn alignment() -> u8 {
66 size_of::<u32>() as u8
67 }
68
69 #[inline(always)]
70 fn size(&self) -> u8 {
71 size_of::<u32>() as u8
72 }
73}
74
75impl<Reg> fmt::Display for Rv32MInstruction<Reg>
76where
77 Reg: fmt::Display,
78{
79 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
80 match self {
81 Self::Mul { rd, rs1, rs2 } => write!(f, "mul {}, {}, {}", rd, rs1, rs2),
82 Self::Mulh { rd, rs1, rs2 } => write!(f, "mulh {}, {}, {}", rd, rs1, rs2),
83 Self::Mulhsu { rd, rs1, rs2 } => write!(f, "mulhsu {}, {}, {}", rd, rs1, rs2),
84 Self::Mulhu { rd, rs1, rs2 } => write!(f, "mulhu {}, {}, {}", rd, rs1, rs2),
85 Self::Div { rd, rs1, rs2 } => write!(f, "div {}, {}, {}", rd, rs1, rs2),
86 Self::Divu { rd, rs1, rs2 } => write!(f, "divu {}, {}, {}", rd, rs1, rs2),
87 Self::Rem { rd, rs1, rs2 } => write!(f, "rem {}, {}, {}", rd, rs1, rs2),
88 Self::Remu { rd, rs1, rs2 } => write!(f, "remu {}, {}, {}", rd, rs1, rs2),
89 }
90 }
91}