ab_riscv_primitives/instructions/rv64/b/
zba.rs1#[cfg(test)]
4mod tests;
5
6use crate::instructions::Instruction;
7use crate::registers::general_purpose::Register;
8use ab_riscv_macros::instruction;
9use core::fmt;
10
11#[instruction]
13#[derive(Debug, Clone, Copy, PartialEq, Eq)]
14pub enum Rv64ZbaInstruction<Reg> {
15 AddUw { rd: Reg, rs1: Reg, rs2: Reg },
16 Sh1add { rd: Reg, rs1: Reg, rs2: Reg },
17 Sh1addUw { rd: Reg, rs1: Reg, rs2: Reg },
18 Sh2add { rd: Reg, rs1: Reg, rs2: Reg },
19 Sh2addUw { rd: Reg, rs1: Reg, rs2: Reg },
20 Sh3add { rd: Reg, rs1: Reg, rs2: Reg },
21 Sh3addUw { rd: Reg, rs1: Reg, rs2: Reg },
22 SlliUw { rd: Reg, rs1: Reg, shamt: u8 },
23}
24
25#[instruction]
26impl<Reg> const Instruction for Rv64ZbaInstruction<Reg>
27where
28 Reg: [const] Register<Type = u64>,
29{
30 type Reg = Reg;
31
32 #[inline(always)]
33 fn try_decode(instruction: u32) -> Option<Self> {
34 let opcode = (instruction & 0b111_1111) as u8;
35 let rd_bits = ((instruction >> 7) & 0x1f) as u8;
36 let funct3 = ((instruction >> 12) & 0b111) as u8;
37 let rs1_bits = ((instruction >> 15) & 0x1f) as u8;
38 let rs2_bits = ((instruction >> 20) & 0x1f) as u8;
39 let funct7 = ((instruction >> 25) & 0b111_1111) as u8;
40 let funct6 = ((instruction >> 26) & 0b11_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 (0b010, 0b0010000) => Some(Self::Sh1add { rd, rs1, rs2 }),
50 (0b100, 0b0010000) => Some(Self::Sh2add { rd, rs1, rs2 }),
51 (0b110, 0b0010000) => Some(Self::Sh3add { rd, rs1, rs2 }),
52 _ => None,
53 }
54 }
55 0b0111011 => {
57 let rd = Reg::from_bits(rd_bits)?;
58 let rs1 = Reg::from_bits(rs1_bits)?;
59 match funct3 {
60 0b000 => {
61 let rs2 = Reg::from_bits(rs2_bits)?;
62 match funct7 {
63 0b0000100 => Some(Self::AddUw { rd, rs1, rs2 }),
64 _ => None,
65 }
66 }
67 0b001 => {
68 let shamt = rs2_bits;
69 match funct6 {
70 0b000010 => Some(Self::SlliUw { rd, rs1, shamt }),
71 _ => None,
72 }
73 }
74 0b010 => {
75 let rs2 = Reg::from_bits(rs2_bits)?;
76 match funct7 {
77 0b0010000 => Some(Self::Sh1addUw { rd, rs1, rs2 }),
78 _ => None,
79 }
80 }
81 0b100 => {
82 let rs2 = Reg::from_bits(rs2_bits)?;
83 match funct7 {
84 0b0010000 => Some(Self::Sh2addUw { rd, rs1, rs2 }),
85 _ => None,
86 }
87 }
88 0b110 => {
89 let rs2 = Reg::from_bits(rs2_bits)?;
90 match funct7 {
91 0b0010000 => Some(Self::Sh3addUw { rd, rs1, rs2 }),
92 _ => None,
93 }
94 }
95 _ => None,
96 }
97 }
98 _ => None,
99 }
100 }
101
102 #[inline(always)]
103 fn alignment() -> u8 {
104 size_of::<u32>() as u8
105 }
106
107 #[inline(always)]
108 fn size(&self) -> u8 {
109 size_of::<u32>() as u8
110 }
111}
112
113impl<Reg> fmt::Display for Rv64ZbaInstruction<Reg>
114where
115 Reg: fmt::Display,
116{
117 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
118 match self {
119 Self::AddUw { rd, rs1, rs2 } => write!(f, "add.uw {}, {}, {}", rd, rs1, rs2),
120 Self::Sh1add { rd, rs1, rs2 } => write!(f, "sh1add {}, {}, {}", rd, rs1, rs2),
121 Self::Sh1addUw { rd, rs1, rs2 } => write!(f, "sh1add.uw {}, {}, {}", rd, rs1, rs2),
122 Self::Sh2add { rd, rs1, rs2 } => write!(f, "sh2add {}, {}, {}", rd, rs1, rs2),
123 Self::Sh2addUw { rd, rs1, rs2 } => write!(f, "sh2add.uw {}, {}, {}", rd, rs1, rs2),
124 Self::Sh3add { rd, rs1, rs2 } => write!(f, "sh3add {}, {}, {}", rd, rs1, rs2),
125 Self::Sh3addUw { rd, rs1, rs2 } => write!(f, "sh3add.uw {}, {}, {}", rd, rs1, rs2),
126 Self::SlliUw { rd, rs1, shamt } => write!(f, "slli.uw {}, {}, {}", rd, rs1, shamt),
127 }
128 }
129}