diff --git a/crates/cpu/src/instruction.rs b/crates/cpu/src/instruction.rs index 2a0d6b5..04c5957 100644 --- a/crates/cpu/src/instruction.rs +++ b/crates/cpu/src/instruction.rs @@ -10,6 +10,20 @@ use std::{fmt, marker::PhantomData, ops::Range}; pub mod power_isa; +pub trait MOpInto: MOpTrait { + fn mop_into_ty(self) -> Target; + fn mop_into(this: Expr) -> Expr; +} + +impl MOpInto for T { + fn mop_into_ty(self) -> T { + self + } + fn mop_into(this: Expr) -> Expr { + this + } +} + pub trait MOpTrait: Type { type Mapped: MOpTrait; type DestReg: Type; @@ -424,6 +438,30 @@ macro_rules! mop_enum { )* } + impl<$DestReg: Type, $SrcRegWidth: Size, Target: MOpTrait> MOpInto for $first_ty + where + $MOp<$DestReg, $SrcRegWidth>: MOpInto + { + fn mop_into_ty(self) -> Target { + MOpInto::mop_into_ty($MOp[MOpTrait::dest_reg_ty(self)][MOpTrait::src_reg_width(self)]) + } + fn mop_into(this: Expr) -> Expr { + MOpInto::mop_into(MOpInto::<$MOp<$DestReg, $SrcRegWidth>>::mop_into_ty(Expr::ty(this)).$FirstVariant(this)) + } + } + + $(impl<$DestReg: Type, $SrcRegWidth: Size, Target: MOpTrait> MOpInto for $ty + where + $MOp<$DestReg, $SrcRegWidth>: MOpInto + { + fn mop_into_ty(self) -> Target { + MOpInto::mop_into_ty($MOp[MOpTrait::dest_reg_ty(self)][MOpTrait::src_reg_width(self)]) + } + fn mop_into(this: Expr) -> Expr { + MOpInto::mop_into(MOpInto::<$MOp<$DestReg, $SrcRegWidth>>::mop_into_ty(Expr::ty(this)).$Variant(this)) + } + })* + impl<$DestReg: Type, $SrcRegWidth: Size> MOpTrait for $MOp<$DestReg, $SrcRegWidth> { type Mapped = $MOp; type DestReg = $DestReg; @@ -517,6 +555,70 @@ common_mop_struct! { } } +impl AddSubMOp> { + #[hdl] + pub fn add_sub( + dest: impl ToExpr, + src: impl ToExpr, 3>>, + imm: impl ToExpr>, + output_integer_mode: impl ToExpr, + invert_src0: impl ToExpr, + src1_is_carry_in: impl ToExpr, + invert_carry_in: impl ToExpr, + add_pc: impl ToExpr, + ) -> Expr + where + Self: MOpInto, + { + MOpInto::mop_into( + #[hdl] + AddSubMOp { + alu_common: #[hdl] + AluCommonMOp { + common: CommonMOp::new(0_hdl_u0, dest, src, Expr::as_dyn_int(imm.to_expr())), + output_integer_mode, + }, + invert_src0, + src1_is_carry_in, + invert_carry_in, + add_pc, + }, + ) + } +} + +impl AddSubMOp> { + #[hdl] + pub fn add_sub_i( + dest: impl ToExpr, + src: impl ToExpr, 2>>, + imm: impl ToExpr>, + output_integer_mode: impl ToExpr, + invert_src0: impl ToExpr, + src1_is_carry_in: impl ToExpr, + invert_carry_in: impl ToExpr, + add_pc: impl ToExpr, + ) -> Expr + where + Self: MOpInto, + { + MOpInto::mop_into( + #[hdl] + AddSubMOp { + alu_common: #[hdl] + AluCommonMOp { + common: CommonMOp::new(0_hdl_u0, dest, src, Expr::as_dyn_int(imm.to_expr())), + output_integer_mode, + }, + invert_src0, + src1_is_carry_in, + invert_carry_in, + add_pc, + }, + ) + } +} + common_mop_struct! { #[mapped( LogicalMOp)] #[hdl(cmp_eq)] @@ -527,6 +629,32 @@ common_mop_struct! { } } +impl LogicalMOp { + #[hdl] + pub fn logical( + dest: impl ToExpr, + src: impl ToExpr, 2>>, + imm: impl ToExpr>, + output_integer_mode: impl ToExpr, + lut: impl ToExpr>, + ) -> Expr + where + Self: MOpInto, + { + MOpInto::mop_into( + #[hdl] + LogicalMOp { + alu_common: #[hdl] + AluCommonMOp { + common: CommonMOp::new(0_hdl_u0, dest, src, Expr::as_dyn_int(imm.to_expr())), + output_integer_mode, + }, + lut, + }, + ) + } +} + common_mop_struct! { #[mapped( BranchMOp)] #[hdl(cmp_eq)] diff --git a/crates/cpu/src/unit.rs b/crates/cpu/src/unit.rs index d4d2d78..2f48118 100644 --- a/crates/cpu/src/unit.rs +++ b/crates/cpu/src/unit.rs @@ -4,7 +4,7 @@ use crate::{ config::CpuConfig, instruction::{ - mop_enum, AluBranchMOp, L2RegisterFileMOp, LoadStoreMOp, MOpTrait, UnitOutRegNum, + mop_enum, AluBranchMOp, L2RegisterFileMOp, LoadStoreMOp, MOpInto, MOpTrait, UnitOutRegNum, }, register::{FlagsMode, PRegValue}, unit::unit_base::UnitToRegAlloc, diff --git a/crates/cpu/tests/reg_alloc.rs b/crates/cpu/tests/reg_alloc.rs index f30eba2..2ba28fb 100644 --- a/crates/cpu/tests/reg_alloc.rs +++ b/crates/cpu/tests/reg_alloc.rs @@ -3,10 +3,7 @@ use cpu::{ config::{CpuConfig, UnitConfig}, - instruction::{ - AddSubMOp, AluCommonMOp, CommonMOp, LogicalMOp, MOp, MOpDestReg, MOpRegNum, - OutputIntegerMode, COMMON_MOP_2_IMM_WIDTH, COMMON_MOP_3_IMM_WIDTH, - }, + instruction::{AddSubMOp, LogicalMOp, MOp, MOpDestReg, MOpRegNum, OutputIntegerMode}, reg_alloc::{reg_alloc, FetchedDecodedMOp}, register::{FlagsMode, PRegFlagsPowerISA}, unit::{GlobalState, UnitKind}, @@ -47,97 +44,66 @@ fn test_reg_alloc() { ), }, ); - let insns = std::array::from_fn::<_, 4, _>(|i| { - MOp.AluBranch(MOp.AluBranch.AddSubI( + let insns_init: [Expr; 4] = std::array::from_fn(|i| { + AddSubMOp::add_sub_i( #[hdl] - AddSubMOp { - alu_common: #[hdl] - AluCommonMOp { - common: CommonMOp::new( - 0_hdl_u0, - #[hdl] - MOpDestReg { - normal_regs: #[hdl] - [ - #[hdl] - MOpRegNum { value: i as u8 + 1 }, - MOpRegNum::const_zero(), - ], - flag_regs: #[hdl] - [HdlSome(()), HdlNone()], - }, - [0u8; 2], - 0x12345678u32.cast_to(SInt[COMMON_MOP_2_IMM_WIDTH]), - ), - output_integer_mode: OutputIntegerMode.DupLow32(), - }, - invert_src0: false, - src1_is_carry_in: false, - invert_carry_in: false, - add_pc: false, + MOpDestReg { + normal_regs: #[hdl] + [ + #[hdl] + MOpRegNum { value: i as u8 + 1 }, + MOpRegNum::const_zero(), + ], + flag_regs: #[hdl] + [HdlSome(()), HdlNone()], }, - )) - }) - .into_iter() - .chain( - [ - MOp.AluBranch(MOp.AluBranch.AddSub( - #[hdl] - AddSubMOp { - alu_common: #[hdl] - AluCommonMOp { - common: CommonMOp::new( - 0_hdl_u0, - #[hdl] - MOpDestReg { - normal_regs: #[hdl] - [ - #[hdl] - MOpRegNum { value: 1u8 }, - MOpRegNum::const_zero(), - ], - flag_regs: #[hdl] - [HdlSome(()), HdlNone()], - }, - [1u8, 0, 0], - 1.cast_to(SInt[COMMON_MOP_3_IMM_WIDTH]), - ), - output_integer_mode: OutputIntegerMode.Full64(), - }, - invert_src0: false, - src1_is_carry_in: false, - invert_carry_in: false, - add_pc: false, - }, - )), - MOp.AluBranch(MOp.AluBranch.Logical( - #[hdl] - LogicalMOp { - alu_common: #[hdl] - AluCommonMOp { - common: CommonMOp::new( - 0_hdl_u0, - #[hdl] - MOpDestReg { - normal_regs: [ - #[hdl] - MOpRegNum { value: 2u8 }, - MOpRegNum::const_zero(), - ], - flag_regs: [HdlNone(), HdlSome(())], - }, - [2u8, 4u8], - SInt[COMMON_MOP_2_IMM_WIDTH].zero(), - ), - output_integer_mode: OutputIntegerMode.Full64(), - }, - lut: 0b0110_hdl_u4, - }, - )), - ] - .into_iter() - .cycle(), - ); + [0u8; 2], + 0x12345678u32.cast_to_static(), + OutputIntegerMode.DupLow32(), + false, + false, + false, + false, + ) + }); + let insns_loop = [ + AddSubMOp::add_sub( + #[hdl] + MOpDestReg { + normal_regs: #[hdl] + [ + #[hdl] + MOpRegNum { value: 1u8 }, + MOpRegNum::const_zero(), + ], + flag_regs: #[hdl] + [HdlSome(()), HdlNone()], + }, + [1u8, 0, 0], + 1.cast_to_static(), + OutputIntegerMode.Full64(), + false, + false, + false, + false, + ), + LogicalMOp::logical( + #[hdl] + MOpDestReg { + normal_regs: [ + #[hdl] + MOpRegNum { value: 2u8 }, + MOpRegNum::const_zero(), + ], + flag_regs: [HdlNone(), HdlSome(())], + }, + [2u8, 4u8], + 0.cast_to_static(), + OutputIntegerMode.Full64(), + 0b0110_hdl_u4, + ), + ]; + let insns = insns_init.into_iter().chain(insns_loop.into_iter().cycle()); let mut insn_index = 0; for cycle in 0..20 { for fetch_index in 0..config.fetch_width.get() {