Skip to main content

ab_riscv_primitives/instructions/rv32/
m.rs

1//! RV32 M extension
2
3#[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/// RISC-V RV32 M instruction
13#[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            // R-type
44            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}