mirror of
https://github.com/TomHarte/CLK.git
synced 2025-01-01 02:31:00 +00:00
287 lines
8.6 KiB
Plaintext
287 lines
8.6 KiB
Plaintext
//
|
|
// 68000BCDTests.m
|
|
// Clock SignalTests
|
|
//
|
|
// Created by Thomas Harte on 29/06/2019.
|
|
//
|
|
// Largely ported from the tests of the Portable 68k Emulator.
|
|
//
|
|
|
|
#import <XCTest/XCTest.h>
|
|
|
|
#include "TestRunner68000.hpp"
|
|
|
|
@interface M68000BCDTests : XCTestCase
|
|
@end
|
|
|
|
@implementation M68000BCDTests {
|
|
std::unique_ptr<RAM68000> _machine;
|
|
}
|
|
|
|
- (void)setUp {
|
|
_machine = std::make_unique<RAM68000>();
|
|
}
|
|
|
|
- (void)tearDown {
|
|
_machine.reset();
|
|
}
|
|
|
|
// MARK: ABCD
|
|
|
|
- (void)testABCD {
|
|
_machine->set_program({
|
|
0xc302, // ABCD D2, D1
|
|
});
|
|
_machine->set_registers([=](auto ®isters){
|
|
registers.data[1] = 0x1234567a;
|
|
registers.data[2] = 0xf745ff78;
|
|
});
|
|
_machine->run_for_instructions(1);
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssert(state.registers.status & ConditionCode::Carry);
|
|
XCTAssertEqual(state.registers.data[1], 0x12345658);
|
|
XCTAssertEqual(state.registers.data[2], 0xf745ff78);
|
|
}
|
|
|
|
- (void)testABCDZero {
|
|
_machine->set_program({
|
|
0xc302, // ABCD D2, D1
|
|
});
|
|
_machine->set_registers([=](auto ®isters){
|
|
registers.data[1] = 0x12345600;
|
|
registers.data[2] = 0x12345600;
|
|
registers.status = ConditionCode::Zero;
|
|
});
|
|
_machine->run_for_instructions(1);
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssert(state.registers.status & ConditionCode::Zero);
|
|
XCTAssertEqual(state.registers.data[1], 0x12345600);
|
|
XCTAssertEqual(state.registers.data[2], 0x12345600);
|
|
}
|
|
|
|
- (void)testABCDNegative {
|
|
_machine->set_program({
|
|
0xc302, // ABCD D2, D1
|
|
});
|
|
_machine->set_registers([=](auto ®isters){
|
|
registers.data[1] = 0x12345645;
|
|
registers.data[2] = 0x12345654;
|
|
registers.status = ConditionCode::Zero;
|
|
});
|
|
_machine->run_for_instructions(1);
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssert(state.registers.status & ConditionCode::Negative);
|
|
XCTAssertEqual(state.registers.data[1], 0x12345699);
|
|
XCTAssertEqual(state.registers.data[2], 0x12345654);
|
|
}
|
|
|
|
- (void)testABCDWithX {
|
|
_machine->set_program({
|
|
0xc302, // ABCD D2, D1
|
|
});
|
|
_machine->set_registers([=](auto ®isters){
|
|
registers.data[1] = 0x12345645;
|
|
registers.data[2] = 0x12345654;
|
|
registers.status = ConditionCode::Extend;
|
|
});
|
|
_machine->run_for_instructions(1);
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssert(state.registers.status & ConditionCode::Carry);
|
|
XCTAssertEqual(state.registers.data[1], 0x12345600);
|
|
XCTAssertEqual(state.registers.data[2], 0x12345654);
|
|
}
|
|
|
|
- (void)testABCDOverflow {
|
|
_machine->set_program({
|
|
0xc302, // ABCD D2, D1
|
|
});
|
|
_machine->set_registers([=](auto ®isters){
|
|
registers.data[1] = 0x1234563e;
|
|
registers.data[2] = 0x1234563e;
|
|
registers.status = ConditionCode::Extend;
|
|
});
|
|
_machine->run_for_instructions(1);
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssert(state.registers.status & ConditionCode::Overflow);
|
|
XCTAssertEqual(state.registers.data[1], 0x12345683);
|
|
XCTAssertEqual(state.registers.data[2], 0x1234563e);
|
|
}
|
|
|
|
- (void)testABCDPredecDifferent {
|
|
_machine->set_program({
|
|
0xc30a, // ABCD -(A2), -(A1)
|
|
});
|
|
_machine->set_registers([=](auto ®isters){
|
|
registers.address[1] = 0x3001;
|
|
registers.address[2] = 0x4001;
|
|
registers.status = ConditionCode::Extend;
|
|
});
|
|
*_machine->ram_at(0x3000) = 0xa200;
|
|
*_machine->ram_at(0x4000) = 0x1900;
|
|
_machine->run_for_instructions(1);
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssert(state.registers.status & ConditionCode::Carry);
|
|
XCTAssert(state.registers.status & ConditionCode::Extend);
|
|
XCTAssertEqual(state.registers.address[1], 0x3000);
|
|
XCTAssertEqual(state.registers.address[2], 0x4000);
|
|
XCTAssertEqual(*_machine->ram_at(0x3000), 0x2200);
|
|
XCTAssertEqual(*_machine->ram_at(0x4000), 0x1900);
|
|
}
|
|
|
|
- (void)testABCDPredecSame {
|
|
_machine->set_program({
|
|
0xc309, // ABCD -(A1), -(A1)
|
|
});
|
|
_machine->set_registers([=](auto ®isters){
|
|
registers.address[1] = 0x3002;
|
|
registers.status = ConditionCode::Extend;
|
|
});
|
|
*_machine->ram_at(0x3000) = 0x19a2;
|
|
_machine->run_for_instructions(1);
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssert(state.registers.status & ConditionCode::Carry);
|
|
XCTAssert(state.registers.status & ConditionCode::Extend);
|
|
XCTAssertEqual(state.registers.address[1], 0x3000);
|
|
XCTAssertEqual(*_machine->ram_at(0x3000), 0x22a2);
|
|
}
|
|
|
|
// MARK: NBCD
|
|
|
|
- (void)performNBCDd1:(uint32_t)d1 ccr:(uint8_t)ccr {
|
|
_machine->set_program({
|
|
0x4801 // NBCD D1
|
|
});
|
|
_machine->set_registers([=](auto ®isters){
|
|
registers.status |= ccr;
|
|
registers.data[1] = d1;
|
|
});
|
|
_machine->run_for_instructions(1);
|
|
|
|
XCTAssertEqual(6, _machine->get_cycle_count());
|
|
}
|
|
|
|
- (void)testNBCD_Dn {
|
|
[self performNBCDd1:0x7a ccr:0];
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(state.registers.data[1], 0x20);
|
|
XCTAssertEqual(state.registers.status & ConditionCode::AllConditions, ConditionCode::Extend | ConditionCode::Carry | ConditionCode::Overflow);
|
|
}
|
|
|
|
- (void)testNBCD_Dn_extend {
|
|
[self performNBCDd1:0x1234567a ccr:ConditionCode::Extend | ConditionCode::Zero];
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(state.registers.data[1], 0x1234561f);
|
|
XCTAssertEqual(state.registers.status & ConditionCode::AllConditions, ConditionCode::Extend | ConditionCode::Carry | ConditionCode::Overflow);
|
|
}
|
|
|
|
- (void)testNBCD_Dn_zero {
|
|
[self performNBCDd1:0x12345600 ccr:ConditionCode::Zero];
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(state.registers.data[1], 0x12345600);
|
|
XCTAssertEqual(state.registers.status & ConditionCode::AllConditions, ConditionCode::Zero);
|
|
}
|
|
|
|
- (void)testNBCD_Dn_negative {
|
|
[self performNBCDd1:0x123456ff ccr:ConditionCode::Extend | ConditionCode::Zero];
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(state.registers.data[1], 0x1234569a);
|
|
XCTAssertEqual(state.registers.status & ConditionCode::AllConditions, ConditionCode::Extend | ConditionCode::Carry | ConditionCode::Negative);
|
|
}
|
|
|
|
- (void)testNBCD_Dn_XXXw {
|
|
_machine->set_program({
|
|
0x4838, 0x3000 // NBCD ($3000).w
|
|
});
|
|
*_machine->ram_at(0x3000) = 0x0100;
|
|
|
|
_machine->run_for_instructions(1);
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(16, _machine->get_cycle_count());
|
|
XCTAssertEqual(*_machine->ram_at(0x3000), 0x9900);
|
|
XCTAssertEqual(state.registers.status & ConditionCode::AllConditions, ConditionCode::Extend | ConditionCode::Carry | ConditionCode::Negative);
|
|
}
|
|
|
|
// MARK: SBCD
|
|
|
|
- (void)performSBCDd1:(uint32_t)d1 d2:(uint32_t)d2 ccr:(uint8_t)ccr {
|
|
_machine->set_program({
|
|
0x8302 // SBCD D2, D1
|
|
});
|
|
_machine->set_registers([=](auto ®isters){
|
|
registers.status |= ccr;
|
|
registers.data[1] = d1;
|
|
registers.data[2] = d2;
|
|
});
|
|
_machine->run_for_instructions(1);
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(6, _machine->get_cycle_count());
|
|
XCTAssertEqual(state.registers.data[2], d2);
|
|
}
|
|
|
|
- (void)testSBCD_Dn {
|
|
[self performSBCDd1:0x12345689 d2:0xf745ff78 ccr:ConditionCode::Zero];
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(state.registers.data[1], 0x12345611);
|
|
XCTAssertEqual(state.registers.status & ConditionCode::AllConditions, 0);
|
|
}
|
|
|
|
- (void)testSBCD_Dn_zero {
|
|
[self performSBCDd1:0x123456ff d2:0xf745ffff ccr:ConditionCode::Zero];
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(state.registers.data[1], 0x12345600);
|
|
XCTAssertEqual(state.registers.status & ConditionCode::AllConditions, ConditionCode::Zero);
|
|
}
|
|
|
|
- (void)testSBCD_Dn_negative {
|
|
[self performSBCDd1:0x12345634 d2:0xf745ff45 ccr:ConditionCode::Extend];
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(state.registers.data[1], 0x12345688);
|
|
XCTAssertEqual(state.registers.status & ConditionCode::AllConditions, ConditionCode::Extend | ConditionCode::Carry | ConditionCode::Negative);
|
|
}
|
|
|
|
- (void)testSBCD_Dn_overflow {
|
|
[self performSBCDd1:0x123456a9 d2:0xf745ffff ccr:ConditionCode::Extend];
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(state.registers.data[1], 0x12345643);
|
|
XCTAssertEqual(state.registers.status & ConditionCode::AllConditions, ConditionCode::Extend | ConditionCode::Carry | ConditionCode::Overflow);
|
|
}
|
|
|
|
- (void)testSBCD_Dn_PreDec {
|
|
_machine->set_program({
|
|
0x830a // SBCD -(A2), -(A1)
|
|
});
|
|
*_machine->ram_at(0x3000) = 0xa200;
|
|
*_machine->ram_at(0x4000) = 0x1900;
|
|
_machine->set_registers([=](auto ®isters){
|
|
registers.address[1] = 0x3001;
|
|
registers.address[2] = 0x4001;
|
|
registers.status |= ConditionCode::Extend;
|
|
});
|
|
_machine->run_for_instructions(1);
|
|
|
|
const auto state = _machine->get_processor_state();
|
|
XCTAssertEqual(18, _machine->get_cycle_count());
|
|
XCTAssertEqual(*_machine->ram_at(0x3000), 0x8200);
|
|
XCTAssertEqual(*_machine->ram_at(0x4000), 0x1900);
|
|
XCTAssertEqual(state.registers.status & ConditionCode::AllConditions, ConditionCode::Negative);
|
|
}
|
|
|
|
@end
|