arm7tdmi: Small optimizations
Former-commit-id: 3348eae804c85f49231aa95e6436655c1a7c692f
This commit is contained in:
parent
8f8204a52b
commit
a18ea022cd
4 changed files with 134 additions and 61 deletions
4
build.rs
4
build.rs
|
@ -212,8 +212,8 @@ fn arm_format_to_handler(arm_fmt: &str) -> &'static str {
|
||||||
"DataProcessing" => "exec_arm_data_processing",
|
"DataProcessing" => "exec_arm_data_processing",
|
||||||
"SoftwareInterrupt" => "exec_arm_swi",
|
"SoftwareInterrupt" => "exec_arm_swi",
|
||||||
"SingleDataTransfer" => "exec_arm_ldr_str",
|
"SingleDataTransfer" => "exec_arm_ldr_str",
|
||||||
"HalfwordDataTransferImmediateOffset" => "exec_arm_ldr_str_hs",
|
"HalfwordDataTransferImmediateOffset" => "exec_arm_ldr_str_hs_imm",
|
||||||
"HalfwordDataTransferRegOffset" => "exec_arm_ldr_str_hs",
|
"HalfwordDataTransferRegOffset" => "exec_arm_ldr_str_hs_reg",
|
||||||
"BlockDataTransfer" => "exec_arm_ldm_stm",
|
"BlockDataTransfer" => "exec_arm_ldm_stm",
|
||||||
"MoveFromStatus" => "exec_arm_mrs",
|
"MoveFromStatus" => "exec_arm_mrs",
|
||||||
"MoveToStatus" => "exec_arm_transfer_to_status",
|
"MoveToStatus" => "exec_arm_transfer_to_status",
|
||||||
|
|
|
@ -145,7 +145,7 @@ impl ArmInstruction {
|
||||||
fn fmt_data_processing(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
fn fmt_data_processing(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
use AluOpCode::*;
|
use AluOpCode::*;
|
||||||
|
|
||||||
let opcode = self.opcode().unwrap();
|
let opcode = self.opcode();
|
||||||
|
|
||||||
match opcode {
|
match opcode {
|
||||||
MOV | MVN => write!(
|
MOV | MVN => write!(
|
||||||
|
|
|
@ -9,6 +9,22 @@ use crate::core::Bus;
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
|
fn get_bs_op(shift_field: u32) -> BarrelShiftOpCode {
|
||||||
|
BarrelShiftOpCode::from_u8(shift_field.bit_range(5..7) as u8).unwrap()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
|
fn get_shift_reg_by(shift_field: u32) -> ShiftRegisterBy {
|
||||||
|
if shift_field.bit(4) {
|
||||||
|
let rs = shift_field.bit_range(8..12) as usize;
|
||||||
|
ShiftRegisterBy::ByRegister(rs)
|
||||||
|
} else {
|
||||||
|
let amount = shift_field.bit_range(7..12) as u32;
|
||||||
|
ShiftRegisterBy::ByAmount(amount)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Core {
|
impl Core {
|
||||||
pub fn exec_arm(&mut self, bus: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
pub fn exec_arm(&mut self, bus: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
||||||
match insn.fmt {
|
match insn.fmt {
|
||||||
|
@ -17,8 +33,10 @@ impl Core {
|
||||||
ArmFormat::DataProcessing => self.exec_arm_data_processing(bus, insn),
|
ArmFormat::DataProcessing => self.exec_arm_data_processing(bus, insn),
|
||||||
ArmFormat::SoftwareInterrupt => self.exec_arm_swi(bus, insn),
|
ArmFormat::SoftwareInterrupt => self.exec_arm_swi(bus, insn),
|
||||||
ArmFormat::SingleDataTransfer => self.exec_arm_ldr_str(bus, insn),
|
ArmFormat::SingleDataTransfer => self.exec_arm_ldr_str(bus, insn),
|
||||||
ArmFormat::HalfwordDataTransferImmediateOffset => self.exec_arm_ldr_str_hs(bus, insn),
|
ArmFormat::HalfwordDataTransferImmediateOffset => {
|
||||||
ArmFormat::HalfwordDataTransferRegOffset => self.exec_arm_ldr_str_hs(bus, insn),
|
self.exec_arm_ldr_str_hs_imm(bus, insn)
|
||||||
|
}
|
||||||
|
ArmFormat::HalfwordDataTransferRegOffset => self.exec_arm_ldr_str_hs_reg(bus, insn),
|
||||||
ArmFormat::BlockDataTransfer => self.exec_arm_ldm_stm(bus, insn),
|
ArmFormat::BlockDataTransfer => self.exec_arm_ldm_stm(bus, insn),
|
||||||
ArmFormat::MoveFromStatus => self.exec_arm_mrs(bus, insn),
|
ArmFormat::MoveFromStatus => self.exec_arm_mrs(bus, insn),
|
||||||
ArmFormat::MoveToStatus => self.exec_arm_transfer_to_status(bus, insn),
|
ArmFormat::MoveToStatus => self.exec_arm_transfer_to_status(bus, insn),
|
||||||
|
@ -72,7 +90,7 @@ impl Core {
|
||||||
|
|
||||||
/// Cycles 2S+1N
|
/// Cycles 2S+1N
|
||||||
pub fn exec_arm_bx(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
pub fn exec_arm_bx(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
||||||
self.branch_exchange(sb, self.get_reg(insn.rn()))
|
self.branch_exchange(sb, self.get_reg(insn.raw.bit_range(0..4) as usize))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn move_from_status_register(
|
fn move_from_status_register(
|
||||||
|
@ -93,7 +111,7 @@ impl Core {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn exec_arm_mrs(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
pub fn exec_arm_mrs(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
||||||
self.move_from_status_register(sb, insn.rd(), insn.spsr_flag())
|
self.move_from_status_register(sb, insn.raw.bit_range(12..16) as usize, insn.spsr_flag())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
|
@ -162,16 +180,6 @@ impl Core {
|
||||||
CpuAction::AdvancePC
|
CpuAction::AdvancePC
|
||||||
}
|
}
|
||||||
|
|
||||||
fn decode_operand2(&mut self, op2: &BarrelShifterValue) -> u32 {
|
|
||||||
match op2 {
|
|
||||||
BarrelShifterValue::RotatedImmediate(val, amount) => {
|
|
||||||
self.ror(*val, *amount, self.cpsr.C(), false, true)
|
|
||||||
}
|
|
||||||
BarrelShifterValue::ShiftedRegister(x) => self.register_shift(&x),
|
|
||||||
_ => unreachable!(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn transfer_spsr_mode(&mut self) {
|
fn transfer_spsr_mode(&mut self) {
|
||||||
let spsr = self.spsr;
|
let spsr = self.spsr;
|
||||||
if self.cpsr.mode() != spsr.mode() {
|
if self.cpsr.mode() != spsr.mode() {
|
||||||
|
@ -191,30 +199,50 @@ impl Core {
|
||||||
) -> CpuAction {
|
) -> CpuAction {
|
||||||
use AluOpCode::*;
|
use AluOpCode::*;
|
||||||
|
|
||||||
|
let raw_insn = insn.raw;
|
||||||
self.S_cycle32(sb, self.pc);
|
self.S_cycle32(sb, self.pc);
|
||||||
let mut op1 = if insn.rn() == REG_PC {
|
|
||||||
|
let rn = raw_insn.bit_range(16..20) as usize;
|
||||||
|
let rd = raw_insn.bit_range(12..16) as usize;
|
||||||
|
|
||||||
|
let mut op1 = if rn == REG_PC {
|
||||||
insn.pc + 8
|
insn.pc + 8
|
||||||
} else {
|
} else {
|
||||||
self.get_reg(insn.rn())
|
self.get_reg(rn)
|
||||||
};
|
};
|
||||||
|
|
||||||
let mut s_flag = insn.set_cond_flag();
|
let mut s_flag = insn.set_cond_flag();
|
||||||
let opcode = insn.opcode().unwrap();
|
let opcode = insn.opcode();
|
||||||
|
|
||||||
let op2 = insn.operand2();
|
let op2 = if raw_insn.bit(25) {
|
||||||
match op2 {
|
let immediate = raw_insn & 0xff;
|
||||||
BarrelShifterValue::ShiftedRegister(shifted_reg) => {
|
let rotate = 2 * raw_insn.bit_range(8..12);
|
||||||
if insn.rn() == REG_PC && shifted_reg.is_shifted_by_reg() {
|
self.ror(immediate, rotate, self.cpsr.C(), false, true)
|
||||||
|
} else {
|
||||||
|
let reg = raw_insn & 0xf;
|
||||||
|
|
||||||
|
let shift_by = if raw_insn.bit(4) {
|
||||||
|
if rn == REG_PC {
|
||||||
op1 += 4;
|
op1 += 4;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
let op2 = self.decode_operand2(&op2);
|
|
||||||
|
|
||||||
let reg_rd = insn.rd();
|
let rs = raw_insn.bit_range(8..12) as usize;
|
||||||
|
ShiftRegisterBy::ByRegister(rs)
|
||||||
|
} else {
|
||||||
|
let amount = raw_insn.bit_range(7..12) as u32;
|
||||||
|
ShiftRegisterBy::ByAmount(amount)
|
||||||
|
};
|
||||||
|
|
||||||
if reg_rd == REG_PC && s_flag {
|
let shifted_reg = ShiftedRegister {
|
||||||
|
reg: reg as usize,
|
||||||
|
bs_op: get_bs_op(raw_insn),
|
||||||
|
shift_by: shift_by,
|
||||||
|
added: None,
|
||||||
|
};
|
||||||
|
self.register_shift(&shifted_reg)
|
||||||
|
};
|
||||||
|
|
||||||
|
if rd == REG_PC && s_flag {
|
||||||
self.transfer_spsr_mode();
|
self.transfer_spsr_mode();
|
||||||
s_flag = false;
|
s_flag = false;
|
||||||
}
|
}
|
||||||
|
@ -265,8 +293,8 @@ impl Core {
|
||||||
|
|
||||||
let mut result = CpuAction::AdvancePC;
|
let mut result = CpuAction::AdvancePC;
|
||||||
if let Some(alu_res) = alu_res {
|
if let Some(alu_res) = alu_res {
|
||||||
self.set_reg(reg_rd, alu_res as u32);
|
self.set_reg(rd, alu_res as u32);
|
||||||
if reg_rd == REG_PC {
|
if rd == REG_PC {
|
||||||
// T bit might have changed
|
// T bit might have changed
|
||||||
match self.cpsr.state() {
|
match self.cpsr.state() {
|
||||||
CpuState::ARM => self.reload_pipeline32(sb),
|
CpuState::ARM => self.reload_pipeline32(sb),
|
||||||
|
@ -292,8 +320,8 @@ impl Core {
|
||||||
let load = insn.load_flag();
|
let load = insn.load_flag();
|
||||||
let pre_index = insn.pre_index_flag();
|
let pre_index = insn.pre_index_flag();
|
||||||
let writeback = insn.write_back_flag();
|
let writeback = insn.write_back_flag();
|
||||||
let base_reg = insn.rn();
|
let base_reg = insn.raw.bit_range(16..20) as usize;
|
||||||
let dest_reg = insn.rd();
|
let dest_reg = insn.raw.bit_range(12..16) as usize;
|
||||||
let mut addr = self.get_reg(base_reg);
|
let mut addr = self.get_reg(base_reg);
|
||||||
if base_reg == REG_PC {
|
if base_reg == REG_PC {
|
||||||
addr = insn.pc + 8; // prefetching
|
addr = insn.pc + 8; // prefetching
|
||||||
|
@ -363,20 +391,49 @@ impl Core {
|
||||||
result
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn exec_arm_ldr_str_hs(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
pub fn exec_arm_ldr_str_hs_reg(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
||||||
|
self.ldr_str_hs(
|
||||||
|
sb,
|
||||||
|
insn,
|
||||||
|
BarrelShifterValue::ShiftedRegister(ShiftedRegister {
|
||||||
|
reg: (insn.raw & 0xf) as usize,
|
||||||
|
shift_by: ShiftRegisterBy::ByAmount(0),
|
||||||
|
bs_op: BarrelShiftOpCode::LSL,
|
||||||
|
added: Some(insn.add_offset_flag()),
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn exec_arm_ldr_str_hs_imm(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
||||||
|
let offset8 = (insn.raw.bit_range(8..12) << 4) + insn.raw.bit_range(0..4);
|
||||||
|
let offset8 = if insn.add_offset_flag() {
|
||||||
|
offset8
|
||||||
|
} else {
|
||||||
|
(-(offset8 as i32)) as u32
|
||||||
|
};
|
||||||
|
self.ldr_str_hs(sb, insn, BarrelShifterValue::ImmediateValue(offset8))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn ldr_str_hs(
|
||||||
|
&mut self,
|
||||||
|
sb: &mut SysBus,
|
||||||
|
insn: &ArmInstruction,
|
||||||
|
offset: BarrelShifterValue,
|
||||||
|
) -> CpuAction {
|
||||||
let mut result = CpuAction::AdvancePC;
|
let mut result = CpuAction::AdvancePC;
|
||||||
|
|
||||||
let load = insn.load_flag();
|
let load = insn.load_flag();
|
||||||
let pre_index = insn.pre_index_flag();
|
let pre_index = insn.pre_index_flag();
|
||||||
let writeback = insn.write_back_flag();
|
let writeback = insn.write_back_flag();
|
||||||
let base_reg = insn.rn();
|
let base_reg = insn.raw.bit_range(16..20) as usize;
|
||||||
let dest_reg = insn.rd();
|
let dest_reg = insn.raw.bit_range(12..16) as usize;
|
||||||
let mut addr = self.get_reg(base_reg);
|
let mut addr = self.get_reg(base_reg);
|
||||||
if base_reg == REG_PC {
|
if base_reg == REG_PC {
|
||||||
addr = insn.pc + 8; // prefetching
|
addr = insn.pc + 8; // prefetching
|
||||||
}
|
}
|
||||||
|
|
||||||
let offset = self.get_barrel_shifted_value(&insn.ldr_str_hs_offset().unwrap());
|
let offset = self.get_barrel_shifted_value(&offset);
|
||||||
|
|
||||||
// TODO - confirm this
|
// TODO - confirm this
|
||||||
let old_mode = self.cpsr.mode();
|
let old_mode = self.cpsr.mode();
|
||||||
|
@ -453,7 +510,7 @@ impl Core {
|
||||||
let s_flag = insn.raw.bit(22);
|
let s_flag = insn.raw.bit(22);
|
||||||
let is_load = insn.load_flag();
|
let is_load = insn.load_flag();
|
||||||
let mut writeback = insn.write_back_flag();
|
let mut writeback = insn.write_back_flag();
|
||||||
let base_reg = insn.rn();
|
let base_reg = insn.raw.bit_range(16..20) as usize;
|
||||||
let mut base_addr = self.get_reg(base_reg);
|
let mut base_addr = self.get_reg(base_reg);
|
||||||
|
|
||||||
let rlist = insn.register_list();
|
let rlist = insn.register_list();
|
||||||
|
@ -596,11 +653,14 @@ impl Core {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn exec_arm_mul_mla(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
pub fn exec_arm_mul_mla(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
||||||
let (rd, rn, rs, rm) = (insn.rd(), insn.rn(), insn.rs(), insn.rm());
|
let rd = insn.raw.bit_range(16..20) as usize;
|
||||||
|
let rn = insn.raw.bit_range(12..16) as usize;
|
||||||
|
let rs = insn.rs();
|
||||||
|
let rm = insn.rm();
|
||||||
|
|
||||||
// check validity
|
// // check validity
|
||||||
assert!(!(REG_PC == rd || REG_PC == rn || REG_PC == rs || REG_PC == rm));
|
// assert!(!(REG_PC == rd || REG_PC == rn || REG_PC == rs || REG_PC == rm));
|
||||||
assert!(rd != rm);
|
// assert!(rd != rm);
|
||||||
|
|
||||||
let op1 = self.get_reg(rm);
|
let op1 = self.get_reg(rm);
|
||||||
let op2 = self.get_reg(rs);
|
let op2 = self.get_reg(rs);
|
||||||
|
@ -631,14 +691,17 @@ impl Core {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn exec_arm_mull_mlal(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
pub fn exec_arm_mull_mlal(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
||||||
let (rd_hi, rd_lo, rn, rs, rm) =
|
let rd_hi = insn.rd_hi();
|
||||||
(insn.rd_hi(), insn.rd_lo(), insn.rn(), insn.rs(), insn.rm());
|
let rd_lo = insn.rd_lo();
|
||||||
|
let rn = insn.raw.bit_range(8..12) as usize;
|
||||||
|
let rs = insn.rs();
|
||||||
|
let rm = insn.rm();
|
||||||
|
|
||||||
// check validity
|
// // check validity
|
||||||
assert!(
|
// assert!(
|
||||||
!(REG_PC == rd_hi || REG_PC == rd_lo || REG_PC == rn || REG_PC == rs || REG_PC == rm)
|
// !(REG_PC == rd_hi || REG_PC == rd_lo || REG_PC == rn || REG_PC == rs || REG_PC == rm)
|
||||||
);
|
// );
|
||||||
assert!(!(rd_hi != rd_hi && rd_hi != rm && rd_lo != rm));
|
// assert!(!(rd_hi != rd_hi && rd_hi != rm && rd_lo != rm));
|
||||||
|
|
||||||
let op1 = self.get_reg(rm);
|
let op1 = self.get_reg(rm);
|
||||||
let op2 = self.get_reg(rs);
|
let op2 = self.get_reg(rs);
|
||||||
|
@ -678,19 +741,20 @@ impl Core {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn exec_arm_swp(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
pub fn exec_arm_swp(&mut self, sb: &mut SysBus, insn: &ArmInstruction) -> CpuAction {
|
||||||
let base_addr = self.get_reg(insn.rn());
|
let base_addr = self.get_reg(insn.raw.bit_range(16..20) as usize);
|
||||||
|
let rd = insn.raw.bit_range(12..16) as usize;
|
||||||
if insn.transfer_size() == 1 {
|
if insn.transfer_size() == 1 {
|
||||||
let t = sb.read_8(base_addr);
|
let t = sb.read_8(base_addr);
|
||||||
self.N_cycle8(sb, base_addr);
|
self.N_cycle8(sb, base_addr);
|
||||||
sb.write_8(base_addr, self.get_reg(insn.rm()) as u8);
|
sb.write_8(base_addr, self.get_reg(insn.rm()) as u8);
|
||||||
self.S_cycle8(sb, base_addr);
|
self.S_cycle8(sb, base_addr);
|
||||||
self.set_reg(insn.rd(), t as u32);
|
self.set_reg(rd, t as u32);
|
||||||
} else {
|
} else {
|
||||||
let t = self.ldr_word(base_addr, sb);
|
let t = self.ldr_word(base_addr, sb);
|
||||||
self.N_cycle32(sb, base_addr);
|
self.N_cycle32(sb, base_addr);
|
||||||
self.write_32(base_addr, self.get_reg(insn.rm()), sb);
|
self.write_32(base_addr, self.get_reg(insn.rm()), sb);
|
||||||
self.S_cycle32(sb, base_addr);
|
self.S_cycle32(sb, base_addr);
|
||||||
self.set_reg(insn.rd(), t as u32);
|
self.set_reg(rd, t as u32);
|
||||||
}
|
}
|
||||||
self.add_cycle();
|
self.add_cycle();
|
||||||
self.N_cycle32(sb, self.pc);
|
self.N_cycle32(sb, self.pc);
|
||||||
|
|
|
@ -205,8 +205,16 @@ impl ArmInstruction {
|
||||||
self.raw.bit_range(16..20) as usize
|
self.raw.bit_range(16..20) as usize
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn opcode(&self) -> Option<AluOpCode> {
|
pub fn opcode(&self) -> AluOpCode {
|
||||||
AluOpCode::from_u32(self.raw.bit_range(21..25))
|
use std::hint::unreachable_unchecked;
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
if let Some(opc) = AluOpCode::from_u16(self.raw.bit_range(21..25) as u16) {
|
||||||
|
opc
|
||||||
|
} else {
|
||||||
|
unreachable_unchecked()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn branch_offset(&self) -> i32 {
|
pub fn branch_offset(&self) -> i32 {
|
||||||
|
@ -290,10 +298,12 @@ impl ArmInstruction {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
fn get_bs_op(&self, shift_field: u32) -> BarrelShiftOpCode {
|
fn get_bs_op(&self, shift_field: u32) -> BarrelShiftOpCode {
|
||||||
BarrelShiftOpCode::from_u8(shift_field.bit_range(5..7) as u8).unwrap()
|
BarrelShiftOpCode::from_u8(shift_field.bit_range(5..7) as u8).unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
fn get_shift_reg_by(&self, shift_field: u32) -> ShiftRegisterBy {
|
fn get_shift_reg_by(&self, shift_field: u32) -> ShiftRegisterBy {
|
||||||
if shift_field.bit(4) {
|
if shift_field.bit(4) {
|
||||||
let rs = shift_field.bit_range(8..12) as usize;
|
let rs = shift_field.bit_range(8..12) as usize;
|
||||||
|
@ -328,17 +338,16 @@ impl ArmInstruction {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn operand2(&self) -> BarrelShifterValue {
|
pub fn operand2(&self) -> BarrelShifterValue {
|
||||||
let op2 = self.raw.bit_range(0..12);
|
|
||||||
if self.raw.bit(25) {
|
if self.raw.bit(25) {
|
||||||
let immediate = op2 & 0xff;
|
let immediate = self.raw & 0xff;
|
||||||
let rotate = 2 * op2.bit_range(8..12);
|
let rotate = 2 * self.raw.bit_range(8..12);
|
||||||
BarrelShifterValue::RotatedImmediate(immediate, rotate)
|
BarrelShifterValue::RotatedImmediate(immediate, rotate)
|
||||||
} else {
|
} else {
|
||||||
let reg = op2 & 0xf;
|
let reg = self.raw & 0xf;
|
||||||
let shifted_reg = ShiftedRegister {
|
let shifted_reg = ShiftedRegister {
|
||||||
reg: reg as usize,
|
reg: reg as usize,
|
||||||
bs_op: self.get_bs_op(op2),
|
bs_op: self.get_bs_op(self.raw),
|
||||||
shift_by: self.get_shift_reg_by(op2),
|
shift_by: self.get_shift_reg_by(self.raw),
|
||||||
added: None,
|
added: None,
|
||||||
}; // TODO error handling
|
}; // TODO error handling
|
||||||
BarrelShifterValue::ShiftedRegister(shifted_reg)
|
BarrelShifterValue::ShiftedRegister(shifted_reg)
|
||||||
|
|
Reference in a new issue