use rrs_lib::{
instruction_formats::{
BType, IType, ITypeCSR, ITypeShamt, ITypeShamtW, JType, RType, SType, UType,
},
process_instruction, InstructionProcessor,
};
use crate::{Instruction, Opcode, Register};
impl Instruction {
#[must_use]
pub const fn from_r_type(opcode: Opcode, dec_insn: &RType) -> Self {
Self::new(opcode, dec_insn.rd as u8, dec_insn.rs1, dec_insn.rs2, false, false)
}
#[must_use]
pub const fn from_i_type(opcode: Opcode, dec_insn: &IType) -> Self {
Self::new(opcode, dec_insn.rd as u8, dec_insn.rs1, dec_insn.imm as u64, false, true)
}
#[must_use]
pub const fn from_i_type_shamt(opcode: Opcode, dec_insn: &ITypeShamt) -> Self {
Self::new(opcode, dec_insn.rd as u8, dec_insn.rs1, dec_insn.shamt, false, true)
}
#[must_use]
pub const fn from_i_type_shamt_w(opcode: Opcode, dec_insn: &ITypeShamtW) -> Self {
Self::new(opcode, dec_insn.rd as u8, dec_insn.rs1, dec_insn.shamt, false, true)
}
#[must_use]
pub const fn from_s_type(opcode: Opcode, dec_insn: &SType) -> Self {
Self::new(opcode, dec_insn.rs2 as u8, dec_insn.rs1, dec_insn.imm as u64, false, true)
}
#[must_use]
pub const fn from_b_type(opcode: Opcode, dec_insn: &BType) -> Self {
Self::new(opcode, dec_insn.rs1 as u8, dec_insn.rs2, dec_insn.imm as u64, false, true)
}
#[must_use]
pub const fn from_u_type(opcode: Opcode, dec_insn: &UType) -> Self {
Self::new(opcode, dec_insn.rd as u8, dec_insn.imm, dec_insn.imm, true, true)
}
#[must_use]
pub const fn unimp() -> Self {
Self::new(Opcode::UNIMP, 0, 0, 0, true, true)
}
#[inline]
#[must_use]
pub const fn is_r_type(&self) -> bool {
!self.imm_c
}
#[inline]
#[must_use]
pub const fn is_i_type(&self) -> bool {
self.imm_c
}
#[inline]
#[must_use]
pub fn r_type(&self) -> (Register, Register, Register) {
(
Register::from_u8(self.op_a),
Register::from_u8(self.op_b as u8),
Register::from_u8(self.op_c as u8),
)
}
#[inline]
#[must_use]
pub fn i_type(&self) -> (Register, Register, u64) {
(Register::from_u8(self.op_a), Register::from_u8(self.op_b as u8), self.op_c)
}
#[inline]
#[must_use]
pub fn s_type(&self) -> (Register, Register, u64) {
(Register::from_u8(self.op_a), Register::from_u8(self.op_b as u8), self.op_c)
}
#[inline]
#[must_use]
pub fn b_type(&self) -> (Register, Register, u64) {
(Register::from_u8(self.op_a), Register::from_u8(self.op_b as u8), self.op_c)
}
#[inline]
#[must_use]
pub fn j_type(&self) -> (Register, u64) {
(Register::from_u8(self.op_a), self.op_b)
}
#[inline]
#[must_use]
pub fn u_type(&self) -> (Register, u64) {
(Register::from_u8(self.op_a), self.op_b)
}
}
pub(crate) struct InstructionTranspiler;
impl InstructionProcessor for InstructionTranspiler {
type InstructionResult = Instruction;
fn process_add(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::ADD, &dec_insn)
}
fn process_addi(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::ADDI, &dec_insn)
}
fn process_sub(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::SUB, &dec_insn)
}
fn process_xor(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::XOR, &dec_insn)
}
fn process_xori(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::XOR, &dec_insn)
}
fn process_or(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::OR, &dec_insn)
}
fn process_ori(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::OR, &dec_insn)
}
fn process_and(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::AND, &dec_insn)
}
fn process_andi(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::AND, &dec_insn)
}
fn process_sll(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::SLL, &dec_insn)
}
fn process_slli(&mut self, dec_insn: ITypeShamt) -> Self::InstructionResult {
Instruction::from_i_type_shamt(Opcode::SLL, &dec_insn)
}
fn process_srl(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::SRL, &dec_insn)
}
fn process_srli(&mut self, dec_insn: ITypeShamt) -> Self::InstructionResult {
Instruction::from_i_type_shamt(Opcode::SRL, &dec_insn)
}
fn process_sra(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::SRA, &dec_insn)
}
fn process_srai(&mut self, dec_insn: ITypeShamt) -> Self::InstructionResult {
Instruction::from_i_type_shamt(Opcode::SRA, &dec_insn)
}
fn process_slt(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::SLT, &dec_insn)
}
fn process_slti(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::SLT, &dec_insn)
}
fn process_sltu(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::SLTU, &dec_insn)
}
fn process_sltui(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::SLTU, &dec_insn)
}
fn process_lb(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::LB, &dec_insn)
}
fn process_lh(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::LH, &dec_insn)
}
fn process_lw(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::LW, &dec_insn)
}
fn process_lbu(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::LBU, &dec_insn)
}
fn process_lhu(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::LHU, &dec_insn)
}
fn process_sb(&mut self, dec_insn: SType) -> Self::InstructionResult {
Instruction::from_s_type(Opcode::SB, &dec_insn)
}
fn process_sh(&mut self, dec_insn: SType) -> Self::InstructionResult {
Instruction::from_s_type(Opcode::SH, &dec_insn)
}
fn process_sw(&mut self, dec_insn: SType) -> Self::InstructionResult {
Instruction::from_s_type(Opcode::SW, &dec_insn)
}
fn process_beq(&mut self, dec_insn: BType) -> Self::InstructionResult {
Instruction::from_b_type(Opcode::BEQ, &dec_insn)
}
fn process_bne(&mut self, dec_insn: BType) -> Self::InstructionResult {
Instruction::from_b_type(Opcode::BNE, &dec_insn)
}
fn process_blt(&mut self, dec_insn: BType) -> Self::InstructionResult {
Instruction::from_b_type(Opcode::BLT, &dec_insn)
}
fn process_bge(&mut self, dec_insn: BType) -> Self::InstructionResult {
Instruction::from_b_type(Opcode::BGE, &dec_insn)
}
fn process_bltu(&mut self, dec_insn: BType) -> Self::InstructionResult {
Instruction::from_b_type(Opcode::BLTU, &dec_insn)
}
fn process_bgeu(&mut self, dec_insn: BType) -> Self::InstructionResult {
Instruction::from_b_type(Opcode::BGEU, &dec_insn)
}
fn process_jal(&mut self, dec_insn: JType) -> Self::InstructionResult {
Instruction::new(Opcode::JAL, dec_insn.rd as u8, dec_insn.imm as u64, 0, true, true)
}
fn process_jalr(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::new(
Opcode::JALR,
dec_insn.rd as u8,
dec_insn.rs1,
dec_insn.imm as u64,
false,
true,
)
}
fn process_lui(&mut self, dec_insn: UType) -> Self::InstructionResult {
Instruction::from_u_type(Opcode::LUI, &dec_insn)
}
fn process_auipc(&mut self, dec_insn: UType) -> Self::InstructionResult {
Instruction::from_u_type(Opcode::AUIPC, &dec_insn)
}
fn process_ecall(&mut self) -> Self::InstructionResult {
Instruction::new(
Opcode::ECALL,
Register::X5 as u8,
Register::X10 as u64,
Register::X11 as u64,
false,
false,
)
}
fn process_ebreak(&mut self) -> Self::InstructionResult {
Instruction::new(Opcode::EBREAK, 0, 0, 0, false, false)
}
fn process_mul(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::MUL, &dec_insn)
}
fn process_mulh(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::MULH, &dec_insn)
}
fn process_mulhu(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::MULHU, &dec_insn)
}
fn process_mulhsu(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::MULHSU, &dec_insn)
}
fn process_div(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::DIV, &dec_insn)
}
fn process_divu(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::DIVU, &dec_insn)
}
fn process_rem(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::REM, &dec_insn)
}
fn process_remu(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::REMU, &dec_insn)
}
fn process_csrrc(&mut self, _: ITypeCSR) -> Self::InstructionResult {
Instruction::unimp()
}
fn process_csrrci(&mut self, _: ITypeCSR) -> Self::InstructionResult {
Instruction::unimp()
}
fn process_csrrs(&mut self, _: ITypeCSR) -> Self::InstructionResult {
Instruction::unimp()
}
fn process_csrrsi(&mut self, _: ITypeCSR) -> Self::InstructionResult {
Instruction::unimp()
}
fn process_csrrw(&mut self, _: ITypeCSR) -> Self::InstructionResult {
Instruction::unimp()
}
fn process_csrrwi(&mut self, _: ITypeCSR) -> Self::InstructionResult {
Instruction::unimp()
}
fn process_fence(&mut self, _: IType) -> Self::InstructionResult {
Instruction::unimp()
}
fn process_mret(&mut self) -> Self::InstructionResult {
Instruction::unimp()
}
fn process_wfi(&mut self) -> Self::InstructionResult {
Instruction::unimp()
}
fn process_addw(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::ADDW, &dec_insn)
}
fn process_subw(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::SUBW, &dec_insn)
}
fn process_sllw(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::SLLW, &dec_insn)
}
fn process_srlw(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::SRLW, &dec_insn)
}
fn process_sraw(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::SRAW, &dec_insn)
}
fn process_addiw(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::ADDW, &dec_insn)
}
fn process_slliw(&mut self, dec_insn: ITypeShamtW) -> Self::InstructionResult {
Instruction::from_i_type_shamt_w(Opcode::SLLW, &dec_insn)
}
fn process_srliw(&mut self, dec_insn: ITypeShamtW) -> Self::InstructionResult {
Instruction::from_i_type_shamt_w(Opcode::SRLW, &dec_insn)
}
fn process_sraiw(&mut self, dec_insn: ITypeShamtW) -> Self::InstructionResult {
Instruction::from_i_type_shamt_w(Opcode::SRAW, &dec_insn)
}
fn process_lwu(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::LWU, &dec_insn)
}
fn process_ld(&mut self, dec_insn: IType) -> Self::InstructionResult {
Instruction::from_i_type(Opcode::LD, &dec_insn)
}
fn process_sd(&mut self, dec_insn: SType) -> Self::InstructionResult {
Instruction::from_s_type(Opcode::SD, &dec_insn)
}
fn process_mulw(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::MULW, &dec_insn)
}
fn process_divw(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::DIVW, &dec_insn)
}
fn process_divuw(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::DIVUW, &dec_insn)
}
fn process_remw(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::REMW, &dec_insn)
}
fn process_remuw(&mut self, dec_insn: RType) -> Self::InstructionResult {
Instruction::from_r_type(Opcode::REMUW, &dec_insn)
}
}
#[must_use]
pub(crate) fn transpile(instructions_u32: &[u32]) -> Vec<(Instruction, u32)> {
let mut instructions: Vec<(Instruction, u32)> = Vec::new();
let mut transpiler = InstructionTranspiler;
for instruction_u32 in instructions_u32 {
let instruction = process_instruction(&mut transpiler, *instruction_u32).unwrap();
instructions.push((instruction, *instruction_u32));
}
instructions
}