2021-10-02 02:27:05 +00:00
|
|
|
|
|
|
|
#[cfg(test)]
|
2021-10-11 22:04:39 +00:00
|
|
|
mod decode_tests {
|
2021-10-18 23:34:55 +00:00
|
|
|
use crate::error::{Error, ErrorType};
|
2021-10-11 22:04:39 +00:00
|
|
|
use crate::system::System;
|
2021-10-27 04:32:25 +00:00
|
|
|
use crate::memory::{MemoryBlock, BusPort};
|
2021-11-13 18:55:58 +00:00
|
|
|
use crate::devices::{Address, Addressable, wrap_transmutable};
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
use crate::cpus::m68k::{M68k, M68kType};
|
2021-10-18 23:34:55 +00:00
|
|
|
use crate::cpus::m68k::state::Exceptions;
|
2021-11-19 17:55:31 +00:00
|
|
|
use crate::cpus::m68k::instructions::{Instruction, Target, Size, Sign, XRegister, BaseRegister, IndexRegister, Direction, ShiftDirection};
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
const INIT_STACK: Address = 0x00002000;
|
|
|
|
const INIT_ADDR: Address = 0x00000010;
|
|
|
|
|
2021-11-13 18:55:58 +00:00
|
|
|
struct TestCase {
|
|
|
|
cpu: M68kType,
|
|
|
|
data: &'static [u16],
|
|
|
|
ins: Option<Instruction>,
|
|
|
|
}
|
|
|
|
|
|
|
|
const DECODE_TESTS: &'static [TestCase] = &[
|
|
|
|
// MC68000
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x4e71], ins: Some(Instruction::NOP) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0008, 0x00FF], ins: Some(Instruction::OR(Target::Immediate(0xFF), Target::DirectAReg(0), Size::Byte)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x003C, 0x00FF], ins: Some(Instruction::ORtoCCR(0xFF)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x007C, 0x1234], ins: Some(Instruction::ORtoSR(0x1234)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0263, 0x1234], ins: Some(Instruction::AND(Target::Immediate(0x1234), Target::IndirectARegDec(3), Size::Word)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x023C, 0x1234], ins: Some(Instruction::ANDtoCCR(0x34)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x027C, 0xF8FF], ins: Some(Instruction::ANDtoSR(0xF8FF)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0487, 0x1234, 0x5678], ins: Some(Instruction::SUB(Target::Immediate(0x12345678), Target::DirectDReg(7), Size::Long)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x063A, 0x1234, 0x0055], ins: Some(Instruction::ADD(Target::Immediate(0x34), Target::IndirectRegOffset(BaseRegister::PC, None, 0x55), Size::Byte)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0A23, 0x1234], ins: Some(Instruction::EOR(Target::Immediate(0x34), Target::IndirectARegDec(3), Size::Byte)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0A3C, 0x1234], ins: Some(Instruction::EORtoCCR(0x34)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0A7C, 0xF8FF], ins: Some(Instruction::EORtoSR(0xF8FF)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0C00, 0x0020], ins: Some(Instruction::CMP(Target::Immediate(0x20), Target::DirectDReg(0), Size::Byte)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0C00, 0x0030], ins: Some(Instruction::CMP(Target::Immediate(0x30), Target::DirectDReg(0), Size::Byte)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0C00, 0x0010], ins: Some(Instruction::CMP(Target::Immediate(0x10), Target::DirectDReg(0), Size::Byte)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x81FC, 0x0003], ins: Some(Instruction::DIVW(Target::Immediate(3), 0, Sign::Signed)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xC1FC, 0x0276], ins: Some(Instruction::MULW(Target::Immediate(0x276), 0, Sign::Signed)) },
|
2021-11-19 17:55:31 +00:00
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0108, 0x1234], ins: Some(Instruction::MOVEP(0, 0, 0x1234, Size::Word, Direction::FromTarget)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0148, 0x1234], ins: Some(Instruction::MOVEP(0, 0, 0x1234, Size::Long, Direction::FromTarget)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0188, 0x1234], ins: Some(Instruction::MOVEP(0, 0, 0x1234, Size::Word, Direction::ToTarget)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x01C8, 0x1234], ins: Some(Instruction::MOVEP(0, 0, 0x1234, Size::Long, Direction::ToTarget)) },
|
2021-11-13 18:55:58 +00:00
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xE300], ins: Some(Instruction::ASd(Target::Immediate(1), Target::DirectDReg(0), Size::Byte, ShiftDirection::Left)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xE200], ins: Some(Instruction::ASd(Target::Immediate(1), Target::DirectDReg(0), Size::Byte, ShiftDirection::Right)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xE318], ins: Some(Instruction::ROd(Target::Immediate(1), Target::DirectDReg(0), Size::Byte, ShiftDirection::Left)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xE218], ins: Some(Instruction::ROd(Target::Immediate(1), Target::DirectDReg(0), Size::Byte, ShiftDirection::Right)) },
|
2021-11-23 19:45:11 +00:00
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xA000], ins: Some(Instruction::UnimplementedA(0xA000)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xFFFF], ins: Some(Instruction::UnimplementedF(0xFFFF)) },
|
2021-11-13 18:55:58 +00:00
|
|
|
|
|
|
|
// MC68030
|
|
|
|
TestCase { cpu: M68kType::MC68030, data: &[0x4C3C, 0x0800, 0x0000, 0x0097], ins: Some(Instruction::MULL(Target::Immediate(0x97), None, 0, Sign::Signed)) },
|
|
|
|
TestCase { cpu: M68kType::MC68030, data: &[0x21BC, 0x0010, 0x14C4, 0x09B0, 0x0010, 0xDF40], ins: Some(Instruction::MOVE(Target::Immediate(1053892), Target::IndirectRegOffset(BaseRegister::None, Some(IndexRegister { xreg: XRegister::DReg(0), scale: 0, size: Size::Long }), 0x10df40), Size::Long)) },
|
|
|
|
|
|
|
|
// Should Fail
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x21BC, 0x0010, 0x14C4, 0x09B0, 0x0010, 0xDF40], ins: None },
|
|
|
|
];
|
|
|
|
|
|
|
|
|
2021-10-18 23:34:55 +00:00
|
|
|
fn init_decode_test(cputype: M68kType) -> (M68k, System) {
|
2021-10-11 22:04:39 +00:00
|
|
|
let mut system = System::new();
|
|
|
|
|
|
|
|
// Insert basic initialization
|
|
|
|
let data = vec![0; 0x00100000];
|
|
|
|
let mem = MemoryBlock::new(data);
|
2021-10-18 19:05:10 +00:00
|
|
|
system.add_addressable_device(0x00000000, wrap_transmutable(mem)).unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
system.get_bus().write_beu32(0, INIT_STACK as u32).unwrap();
|
|
|
|
system.get_bus().write_beu32(4, INIT_ADDR as u32).unwrap();
|
|
|
|
|
|
|
|
// Initialize the CPU and make sure it's in the expected state
|
2021-10-27 04:32:25 +00:00
|
|
|
let port = if cputype <= M68kType::MC68010 {
|
|
|
|
BusPort::new(0, 24, 16, system.bus.clone())
|
|
|
|
} else {
|
|
|
|
BusPort::new(0, 24, 16, system.bus.clone())
|
|
|
|
};
|
|
|
|
let mut cpu = M68k::new(cputype, 10_000_000, port);
|
2021-11-13 19:39:20 +00:00
|
|
|
cpu.init().unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
assert_eq!(cpu.state.pc, INIT_ADDR as u32);
|
2021-11-18 20:13:36 +00:00
|
|
|
assert_eq!(cpu.state.ssp, INIT_STACK as u32);
|
2021-10-27 04:32:25 +00:00
|
|
|
|
|
|
|
cpu.decoder.init(INIT_ADDR as u32);
|
|
|
|
assert_eq!(cpu.decoder.start, INIT_ADDR as u32);
|
2021-10-11 22:04:39 +00:00
|
|
|
assert_eq!(cpu.decoder.instruction, Instruction::NOP);
|
|
|
|
(cpu, system)
|
|
|
|
}
|
|
|
|
|
2021-10-26 19:17:59 +00:00
|
|
|
fn load_memory(system: &System, data: &[u16]) {
|
|
|
|
let mut addr = INIT_ADDR;
|
|
|
|
for word in data {
|
|
|
|
system.get_bus().write_beu16(addr, *word).unwrap();
|
|
|
|
addr += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-13 18:55:58 +00:00
|
|
|
fn run_decode_test(case: &TestCase) {
|
|
|
|
let (mut cpu, system) = init_decode_test(case.cpu);
|
|
|
|
load_memory(&system, case.data);
|
|
|
|
match &case.ins {
|
|
|
|
Some(ins) => {
|
2021-11-13 19:39:20 +00:00
|
|
|
cpu.decode_next().unwrap();
|
2021-11-13 18:55:58 +00:00
|
|
|
assert_eq!(cpu.decoder.instruction, ins.clone());
|
|
|
|
},
|
|
|
|
None => {
|
2021-11-13 19:39:20 +00:00
|
|
|
assert_eq!(cpu.decode_next().is_err(), true);
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn run_decode_tests() {
|
|
|
|
for case in DECODE_TESTS {
|
|
|
|
println!("Testing for {:?}", case.ins);
|
|
|
|
run_decode_test(case);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-11 22:04:39 +00:00
|
|
|
//
|
|
|
|
// Addressing Mode Target Tests
|
|
|
|
//
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_direct_d() {
|
2021-10-18 23:34:55 +00:00
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Word;
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b000, 0b001, Some(size)).unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
assert_eq!(target, Target::DirectDReg(1));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_direct_a() {
|
2021-10-18 23:34:55 +00:00
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Word;
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b001, 0b010, Some(size)).unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
assert_eq!(target, Target::DirectAReg(2));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_indirect_a() {
|
2021-10-18 23:34:55 +00:00
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Long;
|
|
|
|
let expected = 0x12345678;
|
|
|
|
|
|
|
|
system.get_bus().write_beu32(INIT_ADDR, expected).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b010, 0b010, Some(size)).unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
assert_eq!(target, Target::IndirectAReg(2));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_indirect_a_inc() {
|
2021-10-18 23:34:55 +00:00
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Long;
|
|
|
|
let expected = 0x12345678;
|
|
|
|
|
|
|
|
system.get_bus().write_beu32(INIT_ADDR, expected).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b011, 0b010, Some(size)).unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
assert_eq!(target, Target::IndirectARegInc(2));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_indirect_a_dec() {
|
2021-10-18 23:34:55 +00:00
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Long;
|
|
|
|
let expected = 0x12345678;
|
|
|
|
|
|
|
|
system.get_bus().write_beu32(INIT_ADDR, expected).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b100, 0b010, Some(size)).unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
assert_eq!(target, Target::IndirectARegDec(2));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_indirect_a_reg_offset() {
|
2021-10-18 23:34:55 +00:00
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Long;
|
|
|
|
let offset = -8;
|
|
|
|
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR, (offset as i16) as u16).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b101, 0b100, Some(size)).unwrap();
|
2021-10-18 23:34:55 +00:00
|
|
|
assert_eq!(target, Target::IndirectRegOffset(BaseRegister::AReg(4), None, offset));
|
2021-10-11 22:04:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2021-10-18 23:34:55 +00:00
|
|
|
fn target_indirect_a_reg_brief_extension_word() {
|
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Long;
|
|
|
|
let offset = -8;
|
|
|
|
let brief_extension = 0x3800 | (((offset as i8) as u8) as u16);
|
|
|
|
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR, brief_extension).unwrap();
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR + 2, (offset as i16) as u16).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b110, 0b010, Some(size)).unwrap();
|
2021-10-18 23:34:55 +00:00
|
|
|
assert_eq!(target, Target::IndirectRegOffset(BaseRegister::AReg(2), Some(IndexRegister { xreg: XRegister::DReg(3), scale: 0, size: size }), offset));
|
2021-10-11 22:04:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2021-10-18 23:34:55 +00:00
|
|
|
fn target_indirect_a_reg_full_extension_word() {
|
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68020);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Word;
|
2021-10-18 23:34:55 +00:00
|
|
|
let offset = -1843235 as i32;
|
|
|
|
let brief_extension = 0xF330;
|
2021-10-11 22:04:39 +00:00
|
|
|
|
2021-10-18 23:34:55 +00:00
|
|
|
system.get_bus().write_beu16(INIT_ADDR, brief_extension).unwrap();
|
|
|
|
system.get_bus().write_beu32(INIT_ADDR + 2, offset as u32).unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b110, 0b010, Some(size)).unwrap();
|
2021-10-18 23:34:55 +00:00
|
|
|
assert_eq!(target, Target::IndirectRegOffset(BaseRegister::AReg(2), Some(IndexRegister { xreg: XRegister::AReg(7), scale: 1, size: size }), offset));
|
2021-10-11 22:04:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2021-10-18 23:34:55 +00:00
|
|
|
fn target_indirect_a_reg_full_extension_word_no_base() {
|
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68020);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Word;
|
2021-10-18 23:34:55 +00:00
|
|
|
let offset = -1843235 as i32;
|
|
|
|
let brief_extension = 0xF3B0;
|
2021-10-11 22:04:39 +00:00
|
|
|
|
2021-10-18 23:34:55 +00:00
|
|
|
system.get_bus().write_beu16(INIT_ADDR, brief_extension).unwrap();
|
|
|
|
system.get_bus().write_beu32(INIT_ADDR + 2, offset as u32).unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b110, 0b010, Some(size)).unwrap();
|
2021-10-18 23:34:55 +00:00
|
|
|
assert_eq!(target, Target::IndirectRegOffset(BaseRegister::None, Some(IndexRegister { xreg: XRegister::AReg(7), scale: 1, size: size }), offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_indirect_a_reg_full_extension_word_no_index() {
|
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68020);
|
|
|
|
|
|
|
|
let size = Size::Word;
|
|
|
|
let offset = -1843235 as i32;
|
|
|
|
let brief_extension = 0xF370;
|
|
|
|
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR, brief_extension).unwrap();
|
|
|
|
system.get_bus().write_beu32(INIT_ADDR + 2, offset as u32).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b110, 0b010, Some(size)).unwrap();
|
2021-10-18 23:34:55 +00:00
|
|
|
assert_eq!(target, Target::IndirectRegOffset(BaseRegister::AReg(2), None, offset));
|
2021-10-11 22:04:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_indirect_pc_offset() {
|
2021-10-18 23:34:55 +00:00
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Long;
|
|
|
|
let offset = -8;
|
|
|
|
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR, (offset as i16) as u16).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b111, 0b010, Some(size)).unwrap();
|
2021-10-18 23:34:55 +00:00
|
|
|
assert_eq!(target, Target::IndirectRegOffset(BaseRegister::PC, None, offset));
|
2021-10-11 22:04:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2021-10-18 23:34:55 +00:00
|
|
|
fn target_indirect_pc_brief_extension_word() {
|
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
2021-10-11 03:47:51 +00:00
|
|
|
|
2021-10-11 22:04:39 +00:00
|
|
|
let size = Size::Word;
|
|
|
|
let offset = -8;
|
|
|
|
let brief_extension = 0x3000 | (((offset as i8) as u8) as u16);
|
|
|
|
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR, brief_extension).unwrap();
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR + 2, (offset as i16) as u16).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b111, 0b011, Some(size)).unwrap();
|
2021-10-18 23:34:55 +00:00
|
|
|
assert_eq!(target, Target::IndirectRegOffset(BaseRegister::PC, Some(IndexRegister { xreg: XRegister::DReg(3), scale: 0, size: size }), offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_indirect_pc_full_extension_word() {
|
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68020);
|
|
|
|
|
|
|
|
let size = Size::Word;
|
|
|
|
let offset = -1843235 as i32;
|
|
|
|
let brief_extension = 0xF330;
|
|
|
|
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR, brief_extension).unwrap();
|
|
|
|
system.get_bus().write_beu32(INIT_ADDR + 2, offset as u32).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b111, 0b011, Some(size)).unwrap();
|
2021-10-18 23:34:55 +00:00
|
|
|
assert_eq!(target, Target::IndirectRegOffset(BaseRegister::PC, Some(IndexRegister { xreg: XRegister::AReg(7), scale: 1, size: size }), offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_indirect_immediate_word() {
|
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
|
|
|
|
|
|
|
let size = Size::Word;
|
|
|
|
let expected = 0x1234;
|
|
|
|
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR, expected as u16).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b111, 0b000, Some(size)).unwrap();
|
2021-10-18 23:34:55 +00:00
|
|
|
assert_eq!(target, Target::IndirectMemory(expected));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_indirect_immediate_long() {
|
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
|
|
|
|
|
|
|
let size = Size::Word;
|
|
|
|
let expected = 0x12345678;
|
|
|
|
|
|
|
|
system.get_bus().write_beu32(INIT_ADDR, expected).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b111, 0b001, Some(size)).unwrap();
|
2021-10-18 23:34:55 +00:00
|
|
|
assert_eq!(target, Target::IndirectMemory(expected));
|
2021-10-11 22:04:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_immediate() {
|
2021-10-18 23:34:55 +00:00
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
let size = Size::Word;
|
|
|
|
let expected = 0x1234;
|
|
|
|
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR, expected as u16).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let target = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b111, 0b100, Some(size)).unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
assert_eq!(target, Target::Immediate(expected));
|
|
|
|
}
|
|
|
|
|
2021-10-18 23:34:55 +00:00
|
|
|
#[test]
|
|
|
|
fn target_full_extension_word_unsupported_on_mc68010() {
|
|
|
|
let (mut cpu, system) = init_decode_test(M68kType::MC68010);
|
|
|
|
|
|
|
|
let brief_extension = 0x0100;
|
|
|
|
|
|
|
|
system.get_bus().write_beu16(INIT_ADDR, brief_extension).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let result = cpu.decoder.get_mode_as_target(&mut cpu.port, 0b110, 0b010, Some(Size::Long));
|
2021-10-18 23:34:55 +00:00
|
|
|
match result {
|
|
|
|
Err(Error { err: ErrorType::Processor, native, .. }) if native == Exceptions::IllegalInstruction as u32 => { },
|
|
|
|
result => panic!("Expected illegal instruction but found: {:?}", result),
|
|
|
|
}
|
|
|
|
}
|
2021-10-11 22:04:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod execute_tests {
|
|
|
|
use crate::system::System;
|
2021-10-27 04:32:25 +00:00
|
|
|
use crate::memory::{MemoryBlock, BusPort};
|
2021-10-18 19:05:10 +00:00
|
|
|
use crate::devices::{Address, Addressable, Steppable, wrap_transmutable};
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
use crate::cpus::m68k::{M68k, M68kType};
|
2021-11-13 18:55:58 +00:00
|
|
|
use crate::cpus::m68k::state::{M68kState};
|
2021-11-19 17:55:31 +00:00
|
|
|
use crate::cpus::m68k::instructions::{Instruction, Target, Size, Sign, ShiftDirection, Direction, Condition};
|
2021-10-11 22:04:39 +00:00
|
|
|
|
|
|
|
const INIT_STACK: Address = 0x00002000;
|
|
|
|
const INIT_ADDR: Address = 0x00000010;
|
|
|
|
|
2021-11-19 17:55:31 +00:00
|
|
|
const MEM_ADDR: u32 = 0x00001234;
|
2021-11-13 18:55:58 +00:00
|
|
|
|
|
|
|
struct TestState {
|
|
|
|
pc: u32,
|
2021-11-18 20:13:36 +00:00
|
|
|
ssp: u32,
|
2021-11-13 18:55:58 +00:00
|
|
|
usp: u32,
|
|
|
|
d0: u32,
|
|
|
|
d1: u32,
|
|
|
|
a0: u32,
|
|
|
|
a1: u32,
|
|
|
|
sr: u16,
|
2021-11-23 19:45:11 +00:00
|
|
|
mem: u32,
|
2021-11-13 18:55:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct TestCase {
|
|
|
|
name: &'static str,
|
|
|
|
ins: Instruction,
|
|
|
|
data: &'static [u16],
|
|
|
|
cputype: M68kType,
|
|
|
|
init: TestState,
|
|
|
|
fini: TestState,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn init_execute_test(cputype: M68kType) -> (M68k, System) {
|
2021-10-11 22:04:39 +00:00
|
|
|
let mut system = System::new();
|
|
|
|
|
|
|
|
// Insert basic initialization
|
|
|
|
let data = vec![0; 0x00100000];
|
|
|
|
let mem = MemoryBlock::new(data);
|
2021-10-18 19:05:10 +00:00
|
|
|
system.add_addressable_device(0x00000000, wrap_transmutable(mem)).unwrap();
|
2021-10-11 22:04:39 +00:00
|
|
|
system.get_bus().write_beu32(0, INIT_STACK as u32).unwrap();
|
|
|
|
system.get_bus().write_beu32(4, INIT_ADDR as u32).unwrap();
|
|
|
|
|
2021-10-27 04:32:25 +00:00
|
|
|
let port = if cputype <= M68kType::MC68010 {
|
|
|
|
BusPort::new(0, 24, 16, system.bus.clone())
|
|
|
|
} else {
|
|
|
|
BusPort::new(0, 24, 16, system.bus.clone())
|
|
|
|
};
|
|
|
|
let mut cpu = M68k::new(cputype, 10_000_000, port);
|
2021-10-11 22:04:39 +00:00
|
|
|
cpu.step(&system).unwrap();
|
2021-11-04 06:55:50 +00:00
|
|
|
cpu.decoder.init(cpu.state.pc);
|
2021-10-11 22:04:39 +00:00
|
|
|
assert_eq!(cpu.state.pc, INIT_ADDR as u32);
|
2021-11-18 20:13:36 +00:00
|
|
|
assert_eq!(cpu.state.ssp, INIT_STACK as u32);
|
2021-10-11 22:04:39 +00:00
|
|
|
assert_eq!(cpu.decoder.instruction, Instruction::NOP);
|
|
|
|
(cpu, system)
|
|
|
|
}
|
|
|
|
|
2021-11-13 18:55:58 +00:00
|
|
|
fn build_state(state: &TestState) -> M68kState {
|
|
|
|
let mut new_state = M68kState::new();
|
|
|
|
new_state.pc = state.pc;
|
2021-11-18 20:13:36 +00:00
|
|
|
new_state.ssp = state.ssp;
|
2021-11-13 18:55:58 +00:00
|
|
|
new_state.usp = state.usp;
|
|
|
|
new_state.d_reg[0] = state.d0;
|
|
|
|
new_state.d_reg[1] = state.d1;
|
|
|
|
new_state.a_reg[0] = state.a0;
|
|
|
|
new_state.a_reg[1] = state.a1;
|
|
|
|
new_state.sr = state.sr;
|
|
|
|
new_state
|
2021-11-05 04:29:52 +00:00
|
|
|
}
|
|
|
|
|
2021-11-13 18:55:58 +00:00
|
|
|
fn load_memory(system: &System, data: &[u16]) {
|
|
|
|
for i in 0..data.len() {
|
|
|
|
system.get_bus().write_beu16((i << 1) as Address, data[i]).unwrap();
|
|
|
|
}
|
2021-10-20 02:50:42 +00:00
|
|
|
}
|
|
|
|
|
2021-11-13 18:55:58 +00:00
|
|
|
fn run_test(case: &TestCase) {
|
|
|
|
let (mut cpu, system) = init_execute_test(case.cputype);
|
2021-10-20 02:50:42 +00:00
|
|
|
|
2021-11-13 18:55:58 +00:00
|
|
|
let init_state = build_state(&case.init);
|
2021-11-13 19:39:20 +00:00
|
|
|
let expected_state = build_state(&case.fini);
|
2021-11-19 17:55:31 +00:00
|
|
|
system.get_bus().write_beu32(MEM_ADDR as Address, case.init.mem).unwrap();
|
2021-10-26 19:17:59 +00:00
|
|
|
|
2021-11-13 18:55:58 +00:00
|
|
|
load_memory(&system, case.data);
|
|
|
|
cpu.state = init_state;
|
2021-10-20 02:50:42 +00:00
|
|
|
|
2021-11-13 19:39:20 +00:00
|
|
|
cpu.decode_next().unwrap();
|
2021-11-13 18:55:58 +00:00
|
|
|
assert_eq!(cpu.decoder.instruction, case.ins);
|
2021-10-26 19:17:59 +00:00
|
|
|
|
2021-11-13 19:39:20 +00:00
|
|
|
cpu.execute_current().unwrap();
|
2021-10-26 19:17:59 +00:00
|
|
|
assert_eq!(cpu.state, expected_state);
|
2021-11-19 17:55:31 +00:00
|
|
|
|
|
|
|
let mem = system.get_bus().read_beu32(MEM_ADDR as Address).unwrap();
|
|
|
|
assert_eq!(mem, case.fini.mem);
|
2021-10-20 02:50:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2021-11-13 18:55:58 +00:00
|
|
|
pub fn run_execute_tests() {
|
|
|
|
for case in TEST_CASES {
|
|
|
|
println!("Running test {}", case.name);
|
|
|
|
run_test(case);
|
|
|
|
}
|
2021-10-26 19:17:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-13 18:55:58 +00:00
|
|
|
const TEST_CASES: &'static [TestCase] = &[
|
|
|
|
TestCase {
|
|
|
|
name: "nop",
|
|
|
|
ins: Instruction::NOP,
|
|
|
|
data: &[ 0x4e71 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "addi with no overflow or carry",
|
|
|
|
ins: Instruction::ADD(Target::Immediate(0x7f), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0600, 0x007F ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x0000007f, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "addi with no overflow but negative",
|
|
|
|
ins: Instruction::ADD(Target::Immediate(0x80), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0600, 0x0080 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000081, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2708, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "addi with overflow",
|
|
|
|
ins: Instruction::ADD(Target::Immediate(0x7f), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0600, 0x007F ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000080, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x270A, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "addi with carry",
|
|
|
|
ins: Instruction::ADD(Target::Immediate(0x80), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0600, 0x0080 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000080, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2717, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
2021-11-16 04:52:19 +00:00
|
|
|
TestCase {
|
|
|
|
name: "adda immediate",
|
|
|
|
ins: Instruction::ADDA(Target::Immediate(0xF800), 0, Size::Word),
|
|
|
|
data: &[ 0xD0FC, 0xF800 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0xFFFFF800, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "adda register",
|
|
|
|
ins: Instruction::ADDA(Target::DirectDReg(0), 0, Size::Word),
|
|
|
|
data: &[ 0xD0C0 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x0000F800, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x0000F800, d1: 0x00000000, a0: 0xFFFFF800, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
2021-11-13 18:55:58 +00:00
|
|
|
TestCase {
|
|
|
|
name: "andi with sr",
|
|
|
|
ins: Instruction::ANDtoSR(0xF8FF),
|
|
|
|
data: &[ 0x027C, 0xF8FF ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0xA7AA, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0xA0AA, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "asl",
|
|
|
|
ins: Instruction::ASd(Target::Immediate(1), Target::DirectDReg(0), Size::Byte, ShiftDirection::Left),
|
|
|
|
data: &[ 0xE300 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000002, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "asr",
|
|
|
|
ins: Instruction::ASd(Target::Immediate(1), Target::DirectDReg(0), Size::Byte, ShiftDirection::Right),
|
|
|
|
data: &[ 0xE200 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000081, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x000000C0, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2719, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "blt with jump",
|
|
|
|
ins: Instruction::Bcc(Condition::LessThan, 8),
|
|
|
|
data: &[ 0x6D08 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2709, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x0000000A, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2709, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "blt with jump",
|
|
|
|
ins: Instruction::Bcc(Condition::LessThan, 8),
|
|
|
|
data: &[ 0x6D08 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
2021-11-16 04:52:19 +00:00
|
|
|
TestCase {
|
|
|
|
name: "bchg not zero",
|
|
|
|
ins: Instruction::BCHG(Target::Immediate(7), Target::DirectDReg(1), Size::Long),
|
|
|
|
data: &[ 0x0841, 0x0007 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x000000FF, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x0000007F, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "bchg zero",
|
|
|
|
ins: Instruction::BCHG(Target::Immediate(7), Target::DirectDReg(1), Size::Long),
|
|
|
|
data: &[ 0x0841, 0x0007 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000080, a0: 0x00000000, a1: 0x00000000, sr: 0x2704, mem: 0x00000000 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "bra 8-bit",
|
|
|
|
ins: Instruction::BRA(-32),
|
|
|
|
data: &[ 0x60E0 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0xFFFFFFE2, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
2021-11-13 18:55:58 +00:00
|
|
|
TestCase {
|
|
|
|
name: "cmpi equal",
|
|
|
|
ins: Instruction::CMP(Target::Immediate(0x20), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0C00, 0x0020 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000020, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000020, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2704, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "cmpi greater than",
|
|
|
|
ins: Instruction::CMP(Target::Immediate(0x30), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0C00, 0x0030 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000020, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000020, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2709, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "cmpi less than",
|
|
|
|
ins: Instruction::CMP(Target::Immediate(0x10), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0C00, 0x0010 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000020, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000020, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "cmpi no overflow",
|
|
|
|
ins: Instruction::CMP(Target::Immediate(0x7F), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0C00, 0x007F ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2709, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "cmpi no overflow, already negative",
|
|
|
|
ins: Instruction::CMP(Target::Immediate(0x8001), Target::DirectDReg(0), Size::Word),
|
|
|
|
data: &[ 0x0C40, 0x8001 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2701, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "cmpi with overflow",
|
|
|
|
ins: Instruction::CMP(Target::Immediate(0x80), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0C00, 0x0080 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x270B, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "cmpi with overflow 2",
|
|
|
|
ins: Instruction::CMP(Target::Immediate(0x8001), Target::DirectDReg(0), Size::Word),
|
|
|
|
data: &[ 0x0C40, 0x8001 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x270B, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "cmpi no carry",
|
|
|
|
ins: Instruction::CMP(Target::Immediate(0x01), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0C00, 0x0001 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x000000FF, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x000000FF, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2708, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "cmpi with carry",
|
|
|
|
ins: Instruction::CMP(Target::Immediate(0xFF), Target::DirectDReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0C00, 0x00FF ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2701, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "divu",
|
|
|
|
ins: Instruction::DIVW(Target::Immediate(0x0245), 0, Sign::Unsigned),
|
|
|
|
data: &[ 0x80FC, 0x0245 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00040000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x007101C3, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
2021-11-16 04:52:19 +00:00
|
|
|
TestCase {
|
|
|
|
name: "eori",
|
|
|
|
ins: Instruction::EOR(Target::DirectDReg(1), Target::DirectDReg(0), Size::Long),
|
|
|
|
data: &[ 0xB380 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0xAAAA5555, d1: 0x55AA55AA, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0xFF0000FF, d1: 0x55AA55AA, a0: 0x00000000, a1: 0x00000000, sr: 0x2708, mem: 0x00000000 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "exg",
|
|
|
|
ins: Instruction::EXG(Target::DirectDReg(0), Target::DirectAReg(1)),
|
|
|
|
data: &[ 0xC189 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x12345678, d1: 0x00000000, a0: 0x00000000, a1: 0x87654321, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x87654321, d1: 0x00000000, a0: 0x00000000, a1: 0x12345678, sr: 0x2700, mem: 0x00000000 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "ext",
|
|
|
|
ins: Instruction::EXT(0, Size::Byte, Size::Word),
|
|
|
|
data: &[ 0x4880 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x000000CB, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x0000FFCB, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x27F8, mem: 0x00000000 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "ext",
|
|
|
|
ins: Instruction::EXT(0, Size::Word, Size::Long),
|
|
|
|
data: &[ 0x48C0 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x000000CB, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x000000CB, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x27F0, mem: 0x00000000 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
2021-11-13 18:55:58 +00:00
|
|
|
TestCase {
|
|
|
|
name: "muls",
|
|
|
|
ins: Instruction::MULW(Target::Immediate(0x0276), 0, Sign::Signed),
|
|
|
|
data: &[ 0xC1FC, 0x0276 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000200, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x0004ec00, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
2021-11-16 04:52:19 +00:00
|
|
|
TestCase {
|
|
|
|
name: "movel",
|
|
|
|
ins: Instruction::MOVE(Target::DirectDReg(0), Target::DirectDReg(1), Size::Long),
|
|
|
|
data: &[ 0x2200 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0xFEDCBA98, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0xFEDCBA98, d1: 0xFEDCBA98, a0: 0x00000000, a1: 0x00000000, sr: 0x2708, mem: 0x00000000 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movea",
|
|
|
|
ins: Instruction::MOVEA(Target::DirectDReg(0), 0, Size::Long),
|
|
|
|
data: &[ 0x2040 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0xFEDCBA98, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0xFEDCBA98, d1: 0x00000000, a0: 0xFEDCBA98, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
|
|
|
},
|
|
|
|
|
|
|
|
// MOVEM
|
|
|
|
TestCase {
|
|
|
|
name: "movem word to target",
|
|
|
|
ins: Instruction::MOVEM(Target::IndirectAReg(0), Size::Word, Direction::ToTarget, 0x0003),
|
|
|
|
data: &[ 0x4890, 0x0003 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0xABCD1234, d1: 0xEFEF5678, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0xABCD1234, d1: 0xEFEF5678, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0x12345678 },
|
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movem long to target",
|
|
|
|
ins: Instruction::MOVEM(Target::IndirectAReg(0), Size::Long, Direction::ToTarget, 0x0001),
|
|
|
|
data: &[ 0x48D0, 0x0001 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0xABCD1234, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0xABCD1234, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xABCD1234 },
|
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movem long from target",
|
|
|
|
ins: Instruction::MOVEM(Target::IndirectAReg(0), Size::Long, Direction::FromTarget, 0x0001),
|
|
|
|
data: &[ 0x4CD0, 0x0001 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xABCD1234 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0xABCD1234, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xABCD1234 },
|
2021-11-16 04:52:19 +00:00
|
|
|
},
|
2021-11-19 17:55:31 +00:00
|
|
|
TestCase {
|
|
|
|
name: "movem word from target inc",
|
|
|
|
ins: Instruction::MOVEM(Target::IndirectARegInc(0), Size::Word, Direction::FromTarget, 0x0001),
|
|
|
|
data: &[ 0x4C98, 0x0001 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0xFFFFFFFF, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xABCD1234 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0xFFFFABCD, d1: 0x00000000, a0: MEM_ADDR+2, a1: 0x00000000, sr: 0x27FF, mem: 0xABCD1234 },
|
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movem long to target dec",
|
|
|
|
ins: Instruction::MOVEM(Target::IndirectARegDec(0), Size::Long, Direction::ToTarget, 0x8000),
|
|
|
|
data: &[ 0x48E0, 0x8000 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0xABCD1234, d1: 0x00000000, a0: MEM_ADDR+4, a1: 0x00000000, sr: 0x27FF, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0xABCD1234, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xABCD1234 },
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
// MOVEP
|
|
|
|
TestCase {
|
|
|
|
name: "movep word to even memory",
|
|
|
|
ins: Instruction::MOVEP(0, 0, 0, Size::Word, Direction::ToTarget),
|
|
|
|
data: &[ 0x0188, 0x0000 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x000055AA, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xFFFFFFFF },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x000055AA, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0x55FFAAFF },
|
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movep word to odd memory",
|
|
|
|
ins: Instruction::MOVEP(0, 0, 1, Size::Word, Direction::ToTarget),
|
|
|
|
data: &[ 0x0188, 0x0001 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x000055AA, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xFFFFFFFF },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x000055AA, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xFF55FFAA },
|
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movep long to even memory upper",
|
|
|
|
ins: Instruction::MOVEP(0, 0, 0, Size::Long, Direction::ToTarget),
|
|
|
|
data: &[ 0x01C8, 0x0000 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0xAABBCCDD, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xFFFFFFFF },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0xAABBCCDD, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xAAFFBBFF },
|
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movep long to even memory lower",
|
|
|
|
ins: Instruction::MOVEP(0, 0, 0, Size::Long, Direction::ToTarget),
|
|
|
|
data: &[ 0x01C8, 0x0000 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0xAABBCCDD, d1: 0x00000000, a0: MEM_ADDR-4, a1: 0x00000000, sr: 0x27FF, mem: 0xFFFFFFFF },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0xAABBCCDD, d1: 0x00000000, a0: MEM_ADDR-4, a1: 0x00000000, sr: 0x27FF, mem: 0xCCFFDDFF },
|
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movep word from even memory",
|
|
|
|
ins: Instruction::MOVEP(0, 0, 0, Size::Word, Direction::FromTarget),
|
|
|
|
data: &[ 0x0108, 0x0000 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0x55FFAAFF },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x000055AA, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0x55FFAAFF },
|
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movep word from odd memory",
|
|
|
|
ins: Instruction::MOVEP(0, 0, 1, Size::Word, Direction::FromTarget),
|
|
|
|
data: &[ 0x0108, 0x0001 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xFF55FFAA },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x000055AA, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xFF55FFAA },
|
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movep long from even memory upper",
|
|
|
|
ins: Instruction::MOVEP(0, 0, 0, Size::Long, Direction::FromTarget),
|
|
|
|
data: &[ 0x0148, 0x0000 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xAAFFBBFF },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0xAABB0000, d1: 0x00000000, a0: MEM_ADDR, a1: 0x00000000, sr: 0x27FF, mem: 0xAAFFBBFF },
|
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "movep long from even memory lower",
|
|
|
|
ins: Instruction::MOVEP(0, 0, 0, Size::Long, Direction::FromTarget),
|
|
|
|
data: &[ 0x0148, 0x0000 ],
|
|
|
|
cputype: M68kType::MC68010,
|
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: MEM_ADDR-4, a1: 0x00000000, sr: 0x27FF, mem: 0xCCFFDDFF },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x0000CCDD, d1: 0x00000000, a0: MEM_ADDR-4, a1: 0x00000000, sr: 0x27FF, mem: 0xCCFFDDFF },
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
// NEG
|
2021-11-13 18:55:58 +00:00
|
|
|
TestCase {
|
|
|
|
name: "neg",
|
|
|
|
ins: Instruction::NEG(Target::DirectDReg(0), Size::Word),
|
|
|
|
data: &[ 0x4440 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000080, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x0000FF80, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2719, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
TestCase {
|
|
|
|
name: "ori",
|
|
|
|
ins: Instruction::OR(Target::Immediate(0xFF), Target::DirectAReg(0), Size::Byte),
|
|
|
|
data: &[ 0x0008, 0x00FF ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x000000FF, a1: 0x00000000, sr: 0x2708, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "ori with sr",
|
|
|
|
ins: Instruction::ORtoSR(0x00AA),
|
|
|
|
data: &[ 0x007C, 0x00AA ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0xA755, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000004, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0xA7FF, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TestCase {
|
|
|
|
name: "rol",
|
|
|
|
ins: Instruction::ROd(Target::Immediate(1), Target::DirectDReg(0), Size::Byte, ShiftDirection::Left),
|
|
|
|
data: &[ 0xE318 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000080, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2701, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "ror",
|
|
|
|
ins: Instruction::ROd(Target::Immediate(1), Target::DirectDReg(0), Size::Byte, ShiftDirection::Right),
|
|
|
|
data: &[ 0xE218 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000080, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2709, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "roxl",
|
|
|
|
ins: Instruction::ROXd(Target::Immediate(1), Target::DirectDReg(0), Size::Byte, ShiftDirection::Left),
|
|
|
|
data: &[ 0xE310 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000080, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2715, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "roxr",
|
|
|
|
ins: Instruction::ROXd(Target::Immediate(1), Target::DirectDReg(0), Size::Byte, ShiftDirection::Right),
|
|
|
|
data: &[ 0xE210 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000000, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2715, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "roxl two bits",
|
|
|
|
ins: Instruction::ROXd(Target::Immediate(2), Target::DirectDReg(0), Size::Byte, ShiftDirection::Left),
|
|
|
|
data: &[ 0xE510 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000080, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
TestCase {
|
|
|
|
name: "roxr two bits",
|
|
|
|
ins: Instruction::ROXd(Target::Immediate(2), Target::DirectDReg(0), Size::Byte, ShiftDirection::Right),
|
|
|
|
data: &[ 0xE410 ],
|
|
|
|
cputype: M68kType::MC68010,
|
2021-11-19 17:55:31 +00:00
|
|
|
init: TestState { pc: 0x00000000, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000001, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2700, mem: 0x00000000 },
|
|
|
|
fini: TestState { pc: 0x00000002, ssp: 0x00000000, usp: 0x00000000, d0: 0x00000080, d1: 0x00000000, a0: 0x00000000, a1: 0x00000000, sr: 0x2708, mem: 0x00000000 },
|
2021-11-13 18:55:58 +00:00
|
|
|
},
|
|
|
|
];
|
2021-10-20 02:50:42 +00:00
|
|
|
|
2021-10-05 23:22:21 +00:00
|
|
|
|
2021-11-13 18:55:58 +00:00
|
|
|
//
|
|
|
|
// Addressing Mode Target Tests
|
|
|
|
//
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_value_direct_d() {
|
2021-11-13 19:39:20 +00:00
|
|
|
let (mut cpu, _) = init_execute_test(M68kType::MC68010);
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
let size = Size::Word;
|
|
|
|
let expected = 0x1234;
|
2021-10-11 22:04:39 +00:00
|
|
|
let target = Target::DirectDReg(1);
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
cpu.state.d_reg[1] = expected;
|
2021-11-13 19:39:20 +00:00
|
|
|
let result = cpu.get_target_value(target, size).unwrap();
|
2021-10-05 23:22:21 +00:00
|
|
|
assert_eq!(result, expected);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_value_direct_a() {
|
2021-11-13 19:39:20 +00:00
|
|
|
let (mut cpu, _) = init_execute_test(M68kType::MC68010);
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
let size = Size::Word;
|
|
|
|
let expected = 0x1234;
|
2021-10-11 22:04:39 +00:00
|
|
|
let target = Target::DirectAReg(2);
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
cpu.state.a_reg[2] = expected;
|
2021-11-13 19:39:20 +00:00
|
|
|
let result = cpu.get_target_value(target, size).unwrap();
|
2021-10-05 23:22:21 +00:00
|
|
|
assert_eq!(result, expected);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_value_indirect_a() {
|
2021-11-13 19:39:20 +00:00
|
|
|
let (mut cpu, _) = init_execute_test(M68kType::MC68010);
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
let size = Size::Long;
|
|
|
|
let expected = 0x12345678;
|
2021-10-11 22:04:39 +00:00
|
|
|
let target = Target::IndirectAReg(2);
|
2021-11-13 19:39:20 +00:00
|
|
|
cpu.port.write_beu32(INIT_ADDR, expected).unwrap();
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
cpu.state.a_reg[2] = INIT_ADDR as u32;
|
2021-11-13 19:39:20 +00:00
|
|
|
let result = cpu.get_target_value(target, size).unwrap();
|
2021-10-05 23:22:21 +00:00
|
|
|
assert_eq!(result, expected);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_value_indirect_a_inc() {
|
2021-11-13 19:39:20 +00:00
|
|
|
let (mut cpu, _) = init_execute_test(M68kType::MC68010);
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
let size = Size::Long;
|
|
|
|
let expected = 0x12345678;
|
2021-10-11 22:04:39 +00:00
|
|
|
let target = Target::IndirectARegInc(2);
|
2021-11-13 19:39:20 +00:00
|
|
|
cpu.port.write_beu32(INIT_ADDR, expected).unwrap();
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
cpu.state.a_reg[2] = INIT_ADDR as u32;
|
2021-11-13 19:39:20 +00:00
|
|
|
let result = cpu.get_target_value(target, size).unwrap();
|
2021-10-05 23:22:21 +00:00
|
|
|
assert_eq!(result, expected);
|
|
|
|
assert_eq!(cpu.state.a_reg[2], (INIT_ADDR as u32) + 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_value_indirect_a_dec() {
|
2021-11-13 19:39:20 +00:00
|
|
|
let (mut cpu, _) = init_execute_test(M68kType::MC68010);
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
let size = Size::Long;
|
|
|
|
let expected = 0x12345678;
|
2021-10-11 22:04:39 +00:00
|
|
|
let target = Target::IndirectARegDec(2);
|
2021-11-13 19:39:20 +00:00
|
|
|
cpu.port.write_beu32(INIT_ADDR, expected).unwrap();
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
cpu.state.a_reg[2] = (INIT_ADDR as u32) + 4;
|
2021-11-13 19:39:20 +00:00
|
|
|
let result = cpu.get_target_value(target, size).unwrap();
|
2021-10-05 23:22:21 +00:00
|
|
|
assert_eq!(result, expected);
|
|
|
|
assert_eq!(cpu.state.a_reg[2], INIT_ADDR as u32);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn target_value_immediate() {
|
2021-11-13 19:39:20 +00:00
|
|
|
let (mut cpu, _) = init_execute_test(M68kType::MC68010);
|
2021-10-05 23:22:21 +00:00
|
|
|
|
|
|
|
let size = Size::Word;
|
|
|
|
let expected = 0x1234;
|
|
|
|
|
2021-10-11 22:04:39 +00:00
|
|
|
let target = Target::Immediate(expected);
|
2021-10-05 23:22:21 +00:00
|
|
|
|
2021-11-13 19:39:20 +00:00
|
|
|
let result = cpu.get_target_value(target, size).unwrap();
|
2021-10-05 23:22:21 +00:00
|
|
|
assert_eq!(result, expected);
|
|
|
|
}
|
2021-10-02 02:27:05 +00:00
|
|
|
}
|
|
|
|
|
2021-10-11 22:04:39 +00:00
|
|
|
|