ab_riscv_primitives/instructions/
zvbb.rs1#[cfg(test)]
4mod tests;
5pub mod zvkb;
6
7use crate::instructions::Instruction;
8use crate::instructions::v::Eew;
9use crate::instructions::v::zvexx::arith::ZveXxArithInstruction;
10use crate::instructions::v::zvexx::carry::ZveXxCarryInstruction;
11use crate::instructions::v::zvexx::config::ZveXxConfigInstruction;
12use crate::instructions::v::zvexx::fixed_point::ZveXxFixedPointInstruction;
13use crate::instructions::v::zvexx::load::{LoadStoreNreg, Nf, SegVmNf, ZveXxLoadInstruction};
14use crate::instructions::v::zvexx::mask::ZveXxMaskInstruction;
15use crate::instructions::v::zvexx::muldiv::ZveXxMulDivInstruction;
16use crate::instructions::v::zvexx::perm::ZveXxPermInstruction;
17use crate::instructions::v::zvexx::reduction::ZveXxReductionInstruction;
18use crate::instructions::v::zvexx::store::ZveXxStoreInstruction;
19use crate::instructions::v::zvexx::widen_narrow::ZveXxWidenNarrowInstruction;
20use crate::instructions::zicsr::ZicsrInstruction;
21use crate::instructions::zvbb::zvkb::ZvkbInstruction;
22use crate::registers::general_purpose::Register;
23use crate::registers::vector::VReg;
24use ab_riscv_macros::instruction;
25use core::fmt;
26
27#[instruction(
52 inherit = [ZvkbInstruction],
53)]
54#[derive(Debug, Clone, Copy, PartialEq, Eq)]
55#[rustfmt::skip]
56pub enum ZvbbInstruction<Reg> {
57 VbrevV { vd: VReg, vs2: VReg, vm: bool },
60 VclzV { vd: VReg, vs2: VReg, vm: bool },
63 VctzV { vd: VReg, vs2: VReg, vm: bool },
66 VcpopV { vd: VReg, vs2: VReg, vm: bool },
69 VwsllVv { vd: VReg, vs2: VReg, vs1: VReg, vm: bool },
72 VwsllVx { vd: VReg, vs2: VReg, rs1: Reg, vm: bool },
74 VwsllVi { vd: VReg, vs2: VReg, uimm: u8, vm: bool },
77}
78
79#[instruction]
80impl<Reg> const Instruction for ZvbbInstruction<Reg>
81where
82 Reg: [const] Register,
83{
84 type Reg = Reg;
85
86 #[inline(always)]
87 fn try_decode(instruction: u32) -> Option<Self> {
88 let opcode = (instruction & 0b111_1111) as u8;
89
90 if opcode != 0b101_0111 {
91 None?;
92 }
93
94 let vd_bits = ((instruction >> 7) & 0x1f) as u8;
95 let funct3 = ((instruction >> 12) & 0b111) as u8;
96 let vs1_bits = ((instruction >> 15) & 0x1f) as u8;
97 let vs2_bits = ((instruction >> 20) & 0x1f) as u8;
98 let vm = ((instruction >> 25) & 1) as u8 == 1;
100 let funct6 = ((instruction >> 26) & 0b11_1111) as u8;
101
102 let vd = VReg::from_bits(vd_bits)?;
103 let vs2 = VReg::from_bits(vs2_bits)?;
104
105 match funct3 {
106 0b000 => {
108 if funct6 != 0b11_0101 {
109 None?;
110 }
111 let vs1 = VReg::from_bits(vs1_bits)?;
112 Some(Self::VwsllVv { vd, vs2, vs1, vm })
113 }
114 0b100 => {
116 if funct6 != 0b11_0101 {
117 None?;
118 }
119 let rs1 = Reg::from_bits(vs1_bits)?;
120 Some(Self::VwsllVx { vd, vs2, rs1, vm })
121 }
122 0b011 => {
124 if funct6 != 0b11_0101 {
125 None?;
126 }
127 let uimm = vs1_bits;
128 Some(Self::VwsllVi { vd, vs2, uimm, vm })
129 }
130 0b010 => {
134 if funct6 != 0b01_0010 {
135 None?;
136 }
137 match vs1_bits {
138 0b01010 => Some(Self::VbrevV { vd, vs2, vm }),
139 0b01100 => Some(Self::VclzV { vd, vs2, vm }),
140 0b01101 => Some(Self::VctzV { vd, vs2, vm }),
141 0b01110 => Some(Self::VcpopV { vd, vs2, vm }),
142 _ => None,
143 }
144 }
145 _ => None,
146 }
147 }
148
149 #[inline(always)]
150 fn alignment() -> u8 {
151 align_of::<u32>() as u8
152 }
153
154 #[inline(always)]
155 fn size(&self) -> u8 {
156 size_of::<u32>() as u8
157 }
158}
159
160#[instruction]
161impl<Reg> fmt::Display for ZvbbInstruction<Reg>
162where
163 Reg: fmt::Display + Copy,
164{
165 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
166 #[rustfmt::skip]
167 match self {
168 Self::VbrevV { vd, vs2, vm } => write!(f, "vbrev.v {vd}, {vs2}{}", mask_suffix(vm)),
169 Self::VclzV { vd, vs2, vm } => write!(f, "vclz.v {vd}, {vs2}{}", mask_suffix(vm)),
170 Self::VctzV { vd, vs2, vm } => write!(f, "vctz.v {vd}, {vs2}{}", mask_suffix(vm)),
171 Self::VcpopV { vd, vs2, vm } => write!(f, "vcpop.v {vd}, {vs2}{}", mask_suffix(vm)),
172 Self::VwsllVv { vd, vs2, vs1, vm } => write!(f, "vwsll.vv {vd}, {vs2}, {vs1}{}", mask_suffix(vm)),
173 Self::VwsllVx { vd, vs2, rs1, vm } => write!(f, "vwsll.vx {vd}, {vs2}, {rs1}{}", mask_suffix(vm)),
174 Self::VwsllVi { vd, vs2, uimm, vm } => write!(f, "vwsll.vi {vd}, {vs2}, {uimm}{}", mask_suffix(vm)),
175 }
176 }
177}
178
179#[inline(always)]
181fn mask_suffix(vm: &bool) -> &'static str {
182 if *vm { "" } else { ", v0.t" }
183}