ab_riscv_primitives/instructions/rv32/zk/zkn/
zknh.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]
26#[derive(Debug, Clone, Copy, PartialEq, Eq)]
27pub enum Rv32ZknhInstruction<Reg> {
28 Sha256Sig0 { rd: Reg, rs1: Reg },
30 Sha256Sig1 { rd: Reg, rs1: Reg },
31 Sha256Sum0 { rd: Reg, rs1: Reg },
32 Sha256Sum1 { rd: Reg, rs1: Reg },
33 Sha512Sig0h { rd: Reg, rs1: Reg, rs2: Reg },
35 Sha512Sig0l { rd: Reg, rs1: Reg, rs2: Reg },
36 Sha512Sig1h { rd: Reg, rs1: Reg, rs2: Reg },
37 Sha512Sig1l { rd: Reg, rs1: Reg, rs2: Reg },
38 Sha512Sum0r { rd: Reg, rs1: Reg, rs2: Reg },
39 Sha512Sum1r { rd: Reg, rs1: Reg, rs2: Reg },
40}
41
42#[instruction]
43impl<Reg> const Instruction for Rv32ZknhInstruction<Reg>
44where
45 Reg: [const] Register<Type = u32>,
46{
47 type Reg = Reg;
48
49 #[inline(always)]
50 fn try_decode(instruction: u32) -> Option<Self> {
51 let opcode = (instruction & 0b111_1111) as u8;
52 let rd_bits = ((instruction >> 7) & 0x1f) as u8;
53 let funct3 = ((instruction >> 12) & 0b111) as u8;
54 let rs1_bits = ((instruction >> 15) & 0x1f) as u8;
55 let rs2_bits = ((instruction >> 20) & 0x1f) as u8;
56 let funct5 = ((instruction >> 20) & 0x1f) as u8;
58 let funct7 = ((instruction >> 25) & 0b111_1111) as u8;
59
60 match opcode {
61 0b001_0011 => {
63 if funct3 != 0b001 || funct7 != 0b000_1000 {
64 None
65 } else {
66 let rd = Reg::from_bits(rd_bits)?;
67 let rs1 = Reg::from_bits(rs1_bits)?;
68 match funct5 {
69 0b0_0010 => Some(Self::Sha256Sig0 { rd, rs1 }),
70 0b0_0011 => Some(Self::Sha256Sig1 { rd, rs1 }),
71 0b0_0000 => Some(Self::Sha256Sum0 { rd, rs1 }),
72 0b0_0001 => Some(Self::Sha256Sum1 { rd, rs1 }),
73 _ => None,
74 }
75 }
76 }
77 0b011_0011 => {
80 if funct3 == 0b000 {
81 let rd = Reg::from_bits(rd_bits)?;
82 let rs1 = Reg::from_bits(rs1_bits)?;
83 let rs2 = Reg::from_bits(rs2_bits)?;
84 match funct7 {
85 0b010_1000 => Some(Self::Sha512Sum0r { rd, rs1, rs2 }),
87 0b010_1001 => Some(Self::Sha512Sum1r { rd, rs1, rs2 }),
89 0b010_1010 => Some(Self::Sha512Sig0l { rd, rs1, rs2 }),
91 0b010_1011 => Some(Self::Sha512Sig1l { rd, rs1, rs2 }),
93 0b010_1110 => Some(Self::Sha512Sig0h { rd, rs1, rs2 }),
95 0b010_1111 => Some(Self::Sha512Sig1h { rd, rs1, rs2 }),
97 _ => None,
98 }
99 } else {
100 None
101 }
102 }
103 _ => None,
104 }
105 }
106
107 #[inline(always)]
108 fn alignment() -> u8 {
109 align_of::<u32>() as u8
110 }
111
112 #[inline(always)]
113 fn size(&self) -> u8 {
114 size_of::<u32>() as u8
115 }
116}
117
118#[instruction]
119impl<Reg> fmt::Display for Rv32ZknhInstruction<Reg>
120where
121 Reg: fmt::Display,
122{
123 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
124 match self {
125 Self::Sha256Sig0 { rd, rs1 } => write!(f, "sha256sig0 {rd}, {rs1}"),
126 Self::Sha256Sig1 { rd, rs1 } => write!(f, "sha256sig1 {rd}, {rs1}"),
127 Self::Sha256Sum0 { rd, rs1 } => write!(f, "sha256sum0 {rd}, {rs1}"),
128 Self::Sha256Sum1 { rd, rs1 } => write!(f, "sha256sum1 {rd}, {rs1}"),
129 Self::Sha512Sig0h { rd, rs1, rs2 } => write!(f, "sha512sig0h {rd}, {rs1}, {rs2}"),
130 Self::Sha512Sig0l { rd, rs1, rs2 } => write!(f, "sha512sig0l {rd}, {rs1}, {rs2}"),
131 Self::Sha512Sig1h { rd, rs1, rs2 } => write!(f, "sha512sig1h {rd}, {rs1}, {rs2}"),
132 Self::Sha512Sig1l { rd, rs1, rs2 } => write!(f, "sha512sig1l {rd}, {rs1}, {rs2}"),
133 Self::Sha512Sum0r { rd, rs1, rs2 } => write!(f, "sha512sum0r {rd}, {rs1}, {rs2}"),
134 Self::Sha512Sum1r { rd, rs1, rs2 } => write!(f, "sha512sum1r {rd}, {rs1}, {rs2}"),
135 }
136 }
137}