2024-02-24 13:02:09 -08:00
|
|
|
use femtos::{Instant, Frequency};
|
2024-03-15 23:01:41 -07:00
|
|
|
use emulator_hal::bus::BusAccess;
|
|
|
|
use emulator_hal_memory::MemoryBlock;
|
2024-02-24 13:02:09 -08:00
|
|
|
|
2024-03-15 23:01:41 -07:00
|
|
|
use moa_m68k::{M68k, M68kType, M68kAddress};
|
2023-06-10 17:39:20 -07:00
|
|
|
use moa_m68k::instructions::{Instruction, Target, Size, Sign, XRegister, BaseRegister, IndexRegister, Direction};
|
2023-03-25 21:27:02 -07:00
|
|
|
use moa_m68k::assembler::M68kAssembler;
|
2024-03-02 23:48:19 -08:00
|
|
|
use moa_m68k::execute::M68kCycle;
|
2023-03-25 21:27:02 -07:00
|
|
|
|
2024-03-15 23:01:41 -07:00
|
|
|
const INIT_STACK: M68kAddress = 0x00002000;
|
|
|
|
const INIT_ADDR: M68kAddress = 0x00000010;
|
2023-03-25 21:27:02 -07:00
|
|
|
|
|
|
|
struct TestCase {
|
|
|
|
cpu: M68kType,
|
|
|
|
data: &'static [u16],
|
|
|
|
ins: Option<Instruction>,
|
|
|
|
}
|
|
|
|
|
2024-03-16 13:15:34 -07:00
|
|
|
#[rustfmt::skip]
|
2023-03-25 21:27:02 -07:00
|
|
|
const DECODE_TESTS: &'static [TestCase] = &[
|
|
|
|
// MC68000
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x4e71], ins: Some(Instruction::NOP) },
|
|
|
|
// TODO I think this one is illegal (which is causing problems for the assembler)
|
|
|
|
//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: &[0x0240, 0x1234], ins: Some(Instruction::AND(Target::Immediate(0x1234), Target::DirectDReg(0), Size::Word)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x02A3, 0x1234, 0x5678], ins: Some(Instruction::AND(Target::Immediate(0x12345678), Target::IndirectARegDec(3), Size::Long)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x0280, 0x1234, 0x5678], ins: Some(Instruction::AND(Target::Immediate(0x12345678), Target::DirectDReg(0), Size::Long)) },
|
|
|
|
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: &[0x4240], ins: Some(Instruction::CLR(Target::DirectDReg(0), Size::Word)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x4280], ins: Some(Instruction::CLR(Target::DirectDReg(0), Size::Long)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x4250], ins: Some(Instruction::CLR(Target::IndirectAReg(0), Size::Word)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0x4290], ins: Some(Instruction::CLR(Target::IndirectAReg(0), Size::Long)) },
|
|
|
|
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)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xCDC5], ins: Some(Instruction::MULW(Target::DirectDReg(5), 6, Sign::Signed)) },
|
|
|
|
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)) },
|
2023-05-21 23:14:26 -07:00
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xE300], ins: Some(Instruction::ASL(Target::Immediate(1), Target::DirectDReg(0), Size::Byte)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xE200], ins: Some(Instruction::ASR(Target::Immediate(1), Target::DirectDReg(0), Size::Byte)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xE318], ins: Some(Instruction::ROL(Target::Immediate(1), Target::DirectDReg(0), Size::Byte)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xE218], ins: Some(Instruction::ROR(Target::Immediate(1), Target::DirectDReg(0), Size::Byte)) },
|
2023-03-25 21:27:02 -07:00
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xA000], ins: Some(Instruction::UnimplementedA(0xA000)) },
|
|
|
|
TestCase { cpu: M68kType::MC68000, data: &[0xFFFF], ins: Some(Instruction::UnimplementedF(0xFFFF)) },
|
|
|
|
|
|
|
|
// 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
|
|
|
|
];
|
|
|
|
|
|
|
|
|
2024-03-20 21:00:33 -07:00
|
|
|
fn init_decode_test(cputype: M68kType) -> (M68k<Instant>, M68kCycle<Instant>, MemoryBlock<u32, Instant>) {
|
2023-03-25 21:27:02 -07:00
|
|
|
// Insert basic initialization
|
2024-03-16 13:15:34 -07:00
|
|
|
let len = 0x2000;
|
2024-03-15 23:01:41 -07:00
|
|
|
let mut data = Vec::with_capacity(len);
|
2024-03-17 11:03:52 -07:00
|
|
|
unsafe {
|
|
|
|
data.set_len(len);
|
|
|
|
}
|
2024-03-15 23:01:41 -07:00
|
|
|
let mut memory = MemoryBlock::from(data);
|
|
|
|
memory.write_beu32(Instant::START, 0, INIT_STACK).unwrap();
|
|
|
|
memory.write_beu32(Instant::START, 4, INIT_ADDR).unwrap();
|
2023-03-25 21:27:02 -07:00
|
|
|
|
|
|
|
// Initialize the CPU and make sure it's in the expected state
|
2024-03-16 13:15:34 -07:00
|
|
|
let cpu = M68k::from_type(cputype, Frequency::from_mhz(10));
|
2024-03-15 23:01:41 -07:00
|
|
|
let cycle = M68kCycle::new(&cpu, Instant::START);
|
|
|
|
(cpu, cycle, memory)
|
2023-03-25 21:27:02 -07:00
|
|
|
}
|
|
|
|
|
2024-03-22 18:51:17 -07:00
|
|
|
fn load_memory<Bus: BusAccess<u32, Instant = Instant>>(memory: &mut Bus, data: &[u16]) {
|
2023-03-25 21:27:02 -07:00
|
|
|
let mut addr = INIT_ADDR;
|
|
|
|
for word in data {
|
2024-03-22 18:51:17 -07:00
|
|
|
memory.write_beu16(Bus::Instant::START, addr, *word).unwrap();
|
2023-03-25 21:27:02 -07:00
|
|
|
addr += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn run_decode_test(case: &TestCase) {
|
2024-03-15 23:01:41 -07:00
|
|
|
let (mut cpu, cycle, mut memory) = init_decode_test(case.cpu);
|
|
|
|
load_memory(&mut memory, case.data);
|
2024-03-13 21:49:04 -07:00
|
|
|
|
2023-03-25 21:27:02 -07:00
|
|
|
match &case.ins {
|
|
|
|
Some(ins) => {
|
2024-03-15 23:01:41 -07:00
|
|
|
let mut executor = cycle.begin(&mut cpu, &mut memory);
|
2024-03-13 21:49:04 -07:00
|
|
|
executor.reset_cpu().unwrap();
|
2024-03-16 13:15:34 -07:00
|
|
|
assert_eq!(executor.state.pc, INIT_ADDR);
|
|
|
|
assert_eq!(executor.state.ssp, INIT_STACK);
|
2024-03-03 13:26:15 -08:00
|
|
|
executor.decode_next().unwrap();
|
|
|
|
assert_eq!(executor.cycle.decoder.instruction, ins.clone());
|
2023-03-25 21:27:02 -07:00
|
|
|
},
|
|
|
|
None => {
|
2024-03-15 23:01:41 -07:00
|
|
|
let mut executor = cycle.begin(&mut cpu, &mut memory);
|
2024-03-13 21:49:04 -07:00
|
|
|
executor.reset_cpu().unwrap();
|
2024-03-16 13:15:34 -07:00
|
|
|
assert_eq!(executor.state.pc, INIT_ADDR);
|
|
|
|
assert_eq!(executor.state.ssp, INIT_STACK);
|
2024-03-03 13:26:15 -08:00
|
|
|
let next = executor.decode_next();
|
|
|
|
println!("{:?}", executor.cycle.decoder.instruction);
|
2023-03-25 21:27:02 -07:00
|
|
|
assert!(next.is_err());
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn run_decode_tests() {
|
|
|
|
for case in DECODE_TESTS {
|
|
|
|
println!("Testing for {:?}", case.ins);
|
|
|
|
run_decode_test(case);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2024-03-16 13:15:34 -07:00
|
|
|
#[ignore]
|
2023-03-25 21:27:02 -07:00
|
|
|
pub fn run_assembler_tests() {
|
|
|
|
let mut tests = 0;
|
|
|
|
let mut errors = 0;
|
|
|
|
|
|
|
|
for case in DECODE_TESTS {
|
|
|
|
if case.ins.is_some() {
|
|
|
|
tests += 1;
|
|
|
|
let assembly_text = format!("{}", case.ins.as_ref().unwrap());
|
|
|
|
print!("Testing assembling of {:?} ", assembly_text);
|
|
|
|
let mut assembler = M68kAssembler::new(M68kType::MC68000);
|
|
|
|
match assembler.assemble_words(&assembly_text) {
|
|
|
|
Ok(data) => {
|
|
|
|
if data == case.data {
|
|
|
|
print!("pass");
|
|
|
|
} else {
|
|
|
|
errors += 1;
|
|
|
|
print!("FAILED");
|
|
|
|
print!("\nleft: {:?}, right: {:?}", data, case.data);
|
|
|
|
}
|
|
|
|
println!();
|
|
|
|
},
|
|
|
|
Err(err) => {
|
|
|
|
println!("FAILED\n{:?}", err);
|
|
|
|
errors += 1;
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if errors > 0 {
|
|
|
|
panic!("{} errors out of {} tests", errors, tests);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
#[test]
|
|
|
|
pub fn run_assembler_opcode_tests() {
|
|
|
|
let mut tests = 0;
|
|
|
|
let mut errors = 0;
|
|
|
|
|
|
|
|
use super::super::testcases::{TimingCase, TIMING_TESTS};
|
|
|
|
for case in TIMING_TESTS {
|
|
|
|
tests += 1;
|
|
|
|
let assembly_text = format!("{}", case.ins);
|
|
|
|
print!("Testing assembling of {:?} from {:?}", assembly_text, case.ins);
|
|
|
|
|
|
|
|
let mut assembler = M68kAssembler::new(M68kType::MC68000);
|
|
|
|
match assembler.assemble_words(&assembly_text) {
|
|
|
|
Ok(data) => {
|
|
|
|
if data[0] == case.data[0] {
|
|
|
|
print!("pass");
|
|
|
|
} else {
|
|
|
|
errors += 1;
|
|
|
|
print!("FAILED");
|
|
|
|
print!("\nleft: {:#06x}, right: {:#06x}", data[0], case.data[0]);
|
|
|
|
}
|
|
|
|
println!();
|
|
|
|
},
|
|
|
|
Err(err) => {
|
|
|
|
println!("FAILED\n{:?}", err);
|
|
|
|
errors += 1;
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if errors > 0 {
|
|
|
|
panic!("{} errors out of {} tests", errors, tests);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*/
|