mirror of
https://github.com/MoleskiCoder/EightBit.git
synced 2024-11-12 17:04:46 +00:00
22506ea56c
Signed-off-by: Adrian Conlon <adrian.conlon@gmail.com>
474 lines
12 KiB
C++
474 lines
12 KiB
C++
#include "stdafx.h"
|
|
#include "FuseTestRunner.h"
|
|
#include "Disassembler.h"
|
|
|
|
#include <stdexcept>
|
|
#include <boost/algorithm/string/predicate.hpp>
|
|
|
|
Fuse::TestRunner::TestRunner(const Test& test, const ExpectedTestResult& result)
|
|
: m_test(test),
|
|
m_result(result),
|
|
m_cpu(*this),
|
|
m_totalCycles(0) {
|
|
|
|
for (const auto& event : m_result.events.events) {
|
|
if (!boost::algorithm::ends_with(event.specifier, "C"))
|
|
m_expectedEvents.events.push_back(event);
|
|
}
|
|
|
|
m_cpu.ExecutedInstruction.connect([this](EightBit::Z80& cpu) {
|
|
m_totalCycles += cpu.cycles();
|
|
std::cout << "**** Cycle count: " << cpu.cycles() << std::endl;
|
|
});
|
|
|
|
m_cpu.ReadMemory.connect([this](EightBit::EventArgs&) {
|
|
addActualEvent("MR");
|
|
});
|
|
|
|
m_cpu.WrittenMemory.connect([this](EightBit::EventArgs&) {
|
|
addActualEvent("MW");
|
|
});
|
|
|
|
m_cpu.ReadIO.connect([this](EightBit::EventArgs&) {
|
|
addActualEvent("PR");
|
|
});
|
|
|
|
m_cpu.WrittenIO.connect([this](EightBit::EventArgs&) {
|
|
addActualEvent("PW");
|
|
});
|
|
}
|
|
|
|
void Fuse::TestRunner::addActualEvent(const std::string& specifier) {
|
|
TestEvent actual;
|
|
actual.address = ADDRESS().word;
|
|
actual.cycles = m_totalCycles + m_cpu.cycles();
|
|
actual.specifier = specifier;
|
|
if (!boost::algorithm::ends_with(specifier, "C"))
|
|
actual.value = DATA();
|
|
actual.valid = true;
|
|
m_actualEvents.events.push_back(actual);
|
|
}
|
|
|
|
//
|
|
|
|
void Fuse::TestRunner::raisePOWER() {
|
|
EightBit::Bus::raisePOWER();
|
|
m_cpu.raisePOWER();
|
|
m_cpu.raiseRESET();
|
|
m_cpu.raiseINT();
|
|
m_cpu.raiseNMI();
|
|
initialiseRegisters();
|
|
}
|
|
|
|
void Fuse::TestRunner::lowerPOWER() {
|
|
m_cpu.lowerPOWER();
|
|
EightBit::Bus::lowerPOWER();
|
|
}
|
|
|
|
EightBit::MemoryMapping Fuse::TestRunner::mapping(uint16_t address) noexcept {
|
|
|
|
const bool memory = m_cpu.requestingMemory();
|
|
if (memory)
|
|
return {
|
|
m_ram,
|
|
0x0000,
|
|
0xffff,
|
|
EightBit::MemoryMapping::AccessLevel::ReadWrite
|
|
};
|
|
|
|
const bool io = m_cpu.requestingIO();
|
|
if (io) {
|
|
|
|
m_ports.accessType() = EightBit::InputOutput::AccessType::Unknown;
|
|
|
|
const bool reading = m_cpu.requestingRead();
|
|
if (reading)
|
|
m_ports.accessType() = EightBit::InputOutput::AccessType::Reading;
|
|
|
|
const bool writing = m_cpu.requestingWrite();
|
|
if (writing)
|
|
m_ports.accessType() = EightBit::InputOutput::AccessType::Writing;
|
|
|
|
return {
|
|
m_ports,
|
|
0x0000,
|
|
0xff,
|
|
EightBit::MemoryMapping::AccessLevel::ReadWrite
|
|
};
|
|
}
|
|
}
|
|
|
|
void Fuse::TestRunner::initialise() {
|
|
initialiseMemory();
|
|
}
|
|
|
|
void Fuse::TestRunner::initialiseRegisters() {
|
|
|
|
const auto& testState = m_test.registerState;
|
|
const auto& inputRegisters = testState.registers;
|
|
|
|
m_cpu.AF() = inputRegisters[Fuse::RegisterState::AF_];
|
|
m_cpu.BC() = inputRegisters[Fuse::RegisterState::BC_];
|
|
m_cpu.DE() = inputRegisters[Fuse::RegisterState::DE_];
|
|
m_cpu.HL() = inputRegisters[Fuse::RegisterState::HL_];
|
|
m_cpu.exx();
|
|
m_cpu.exxAF();
|
|
m_cpu.AF() = inputRegisters[Fuse::RegisterState::AF];
|
|
m_cpu.BC() = inputRegisters[Fuse::RegisterState::BC];
|
|
m_cpu.DE() = inputRegisters[Fuse::RegisterState::DE];
|
|
m_cpu.HL() = inputRegisters[Fuse::RegisterState::HL];
|
|
|
|
m_cpu.IX() = inputRegisters[Fuse::RegisterState::IX];
|
|
m_cpu.IY() = inputRegisters[Fuse::RegisterState::IY];
|
|
|
|
m_cpu.SP() = inputRegisters[Fuse::RegisterState::SP];
|
|
m_cpu.PC() = inputRegisters[Fuse::RegisterState::PC];
|
|
|
|
m_cpu.MEMPTR() = inputRegisters[Fuse::RegisterState::MEMPTR];
|
|
|
|
m_cpu.IV() = testState.i;
|
|
m_cpu.REFRESH() = testState.r;
|
|
m_cpu.IFF1() = testState.iff1;
|
|
m_cpu.IFF2() = testState.iff2;
|
|
m_cpu.IM() = testState.im;
|
|
}
|
|
|
|
void Fuse::TestRunner::initialiseMemory() {
|
|
for (auto memoryDatum : m_test.memoryData) {
|
|
auto address = memoryDatum.address;
|
|
auto bytes = memoryDatum.bytes;
|
|
for (int i = 0; i < bytes.size(); ++i)
|
|
m_ram.poke(address + i, bytes[i]);
|
|
}
|
|
}
|
|
|
|
//
|
|
|
|
void Fuse::TestRunner::check() {
|
|
checkRegisters();
|
|
checkMemory();
|
|
checkEvents();
|
|
}
|
|
|
|
void Fuse::TestRunner::dumpDifference(const std::string& description, uint8_t actual, uint8_t expected) const {
|
|
std::cerr
|
|
<< "**** " << description << ", Expected: "
|
|
<< EightBit::Disassembler::hex(expected)
|
|
<< ", Got: "
|
|
<< EightBit::Disassembler::hex(actual)
|
|
<< std::endl;
|
|
}
|
|
|
|
void Fuse::TestRunner::dumpDifference(
|
|
const std::string& highDescription,
|
|
const std::string& lowDescription,
|
|
EightBit::register16_t actual, EightBit::register16_t expected) const {
|
|
|
|
auto expectedHigh = expected.high;
|
|
auto expectedLow = expected.low;
|
|
|
|
auto actualHigh = actual.high;
|
|
auto actualLow = actual.low;
|
|
|
|
if (expectedHigh != actualHigh)
|
|
dumpDifference(highDescription, actualHigh, expectedHigh);
|
|
|
|
if (expectedLow != actualLow)
|
|
dumpDifference(lowDescription, actualLow, expectedLow);
|
|
}
|
|
|
|
void Fuse::TestRunner::checkRegisters() {
|
|
|
|
const auto& expectedState = m_result.registerState;
|
|
const auto& expectedRegisters = expectedState.registers;
|
|
|
|
auto af = m_cpu.AF() == expectedRegisters[Fuse::RegisterState::AF];
|
|
auto bc = m_cpu.BC() == expectedRegisters[Fuse::RegisterState::BC];
|
|
auto de = m_cpu.DE() == expectedRegisters[Fuse::RegisterState::DE];
|
|
auto hl = m_cpu.HL() == expectedRegisters[Fuse::RegisterState::HL];
|
|
m_cpu.exx();
|
|
m_cpu.exxAF();
|
|
auto af_ = m_cpu.AF() == expectedRegisters[Fuse::RegisterState::AF_];
|
|
auto bc_ = m_cpu.BC() == expectedRegisters[Fuse::RegisterState::BC_];
|
|
auto de_ = m_cpu.DE() == expectedRegisters[Fuse::RegisterState::DE_];
|
|
auto hl_ = m_cpu.HL() == expectedRegisters[Fuse::RegisterState::HL_];
|
|
|
|
auto ix = m_cpu.IX() == expectedRegisters[Fuse::RegisterState::IX];
|
|
auto iy = m_cpu.IY() == expectedRegisters[Fuse::RegisterState::IY];
|
|
|
|
auto sp = m_cpu.SP() == expectedRegisters[Fuse::RegisterState::SP];
|
|
auto pc = m_cpu.PC() == expectedRegisters[Fuse::RegisterState::PC];
|
|
|
|
auto memptr = m_cpu.MEMPTR() == expectedRegisters[Fuse::RegisterState::MEMPTR];
|
|
|
|
auto iv = m_cpu.IV() == expectedState.i;
|
|
auto refresh = m_cpu.REFRESH() == expectedState.r;
|
|
auto iff1 = m_cpu.IFF1() == expectedState.iff1;
|
|
auto iff2 = m_cpu.IFF2() == expectedState.iff2;
|
|
auto im = m_cpu.IM() == expectedState.im;
|
|
|
|
// And back again, so the following works as expected...
|
|
m_cpu.exx();
|
|
m_cpu.exxAF();
|
|
|
|
auto success =
|
|
af && bc && de && hl
|
|
&& af_ && bc_ && de_ && hl_
|
|
&& ix && iy
|
|
&& sp && pc
|
|
&& iv && refresh
|
|
&& iff1 && iff2
|
|
&& im
|
|
&& memptr;
|
|
|
|
if (!success) {
|
|
m_failed = true;
|
|
std::cerr << "**** Failed test (Register): " << m_test.description << std::endl;
|
|
|
|
if (!af) {
|
|
auto expectedA = expectedRegisters[Fuse::RegisterState::AF].high;
|
|
auto gotA = m_cpu.A();
|
|
if (expectedA != gotA)
|
|
dumpDifference("A", gotA, expectedA);
|
|
|
|
auto expectedF = expectedRegisters[Fuse::RegisterState::AF].low;
|
|
auto gotF = m_cpu.F();
|
|
if (expectedF != gotF) {
|
|
std::cerr
|
|
<< "**** F, Expected: "
|
|
<< EightBit::Disassembler::flags(expectedF)
|
|
<< ", Got: "
|
|
<< EightBit::Disassembler::flags(gotF)
|
|
<< std::endl;
|
|
}
|
|
}
|
|
|
|
if (!bc) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::BC];
|
|
auto actualWord = m_cpu.BC();
|
|
dumpDifference("B", "C", actualWord, expectedWord);
|
|
}
|
|
|
|
if (!de) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::DE];
|
|
auto actualWord = m_cpu.DE();
|
|
dumpDifference("D", "E", actualWord, expectedWord);
|
|
}
|
|
|
|
if (!hl) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::HL];
|
|
auto actualWord = m_cpu.HL();
|
|
dumpDifference("H", "L", actualWord, expectedWord);
|
|
}
|
|
|
|
if (!ix) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::IX];
|
|
auto actualWord = m_cpu.IX();
|
|
dumpDifference("IXH", "IXL", actualWord, expectedWord);
|
|
}
|
|
|
|
if (!iy) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::IY];
|
|
auto actualWord = m_cpu.IY();
|
|
dumpDifference("IYH", "IYL", actualWord, expectedWord);
|
|
}
|
|
|
|
if (!sp) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::SP];
|
|
auto actualWord = m_cpu.SP();
|
|
dumpDifference("SPH", "SPL", actualWord, expectedWord);
|
|
}
|
|
|
|
if (!pc) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::PC];
|
|
auto actualWord = m_cpu.PC();
|
|
dumpDifference("PCH", "PCL", actualWord, expectedWord);
|
|
}
|
|
|
|
if (!memptr) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::MEMPTR];
|
|
auto actualWord = m_cpu.MEMPTR();
|
|
dumpDifference("MEMPTRH", "MEMPTRL", actualWord, expectedWord);
|
|
}
|
|
|
|
m_cpu.exxAF();
|
|
m_cpu.exx();
|
|
|
|
if (!af_) {
|
|
auto expectedA_ = expectedRegisters[Fuse::RegisterState::AF_].high;
|
|
auto gotA_ = m_cpu.A();
|
|
if (expectedA_ != gotA_)
|
|
dumpDifference("A'", gotA_, expectedA_);
|
|
|
|
auto expectedF_ = expectedRegisters[Fuse::RegisterState::AF_].low;
|
|
auto gotF_ = m_cpu.F();
|
|
if (expectedF_ != gotF_) {
|
|
std::cerr
|
|
<< "**** F', Expected: "
|
|
<< EightBit::Disassembler::flags(expectedF_)
|
|
<< ", Got: "
|
|
<< EightBit::Disassembler::flags(gotF_)
|
|
<< std::endl;
|
|
}
|
|
}
|
|
|
|
if (!bc_) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::BC_];
|
|
auto actualWord = m_cpu.BC();
|
|
dumpDifference("B'", "C'", actualWord, expectedWord);
|
|
}
|
|
|
|
if (!de_) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::DE_];
|
|
auto actualWord = m_cpu.DE();
|
|
dumpDifference("D'", "E'", actualWord, expectedWord);
|
|
}
|
|
|
|
if (!hl_) {
|
|
auto expectedWord = expectedRegisters[Fuse::RegisterState::HL_];
|
|
auto actualWord = m_cpu.HL();
|
|
dumpDifference("H'", "L'", actualWord, expectedWord);
|
|
}
|
|
|
|
if (!iv) {
|
|
std::cerr
|
|
<< "**** IV, Expected: "
|
|
<< EightBit::Disassembler::hex((uint8_t)expectedState.i)
|
|
<< ", Got: "
|
|
<< EightBit::Disassembler::hex(m_cpu.IV())
|
|
<< std::endl;
|
|
}
|
|
|
|
if (!refresh) {
|
|
std::cerr
|
|
<< "**** R, Expected: "
|
|
<< EightBit::Disassembler::hex((uint8_t)expectedState.r)
|
|
<< ", Got: "
|
|
<< EightBit::Disassembler::hex((uint8_t)m_cpu.REFRESH())
|
|
<< std::endl;
|
|
}
|
|
|
|
if (!iff1) {
|
|
std::cerr
|
|
<< "**** IFF1, Expected: "
|
|
<< (bool)expectedState.iff1
|
|
<< ", Got: "
|
|
<< m_cpu.IFF1()
|
|
<< std::endl;
|
|
}
|
|
|
|
if (!iff2) {
|
|
std::cerr
|
|
<< "**** IFF2, Expected: "
|
|
<< (bool)expectedState.iff2
|
|
<< ", Got: "
|
|
<< m_cpu.IFF2()
|
|
<< std::endl;
|
|
}
|
|
|
|
if (!im) {
|
|
std::cerr
|
|
<< "**** IM, Expected: "
|
|
<< expectedState.im
|
|
<< ", Got: "
|
|
<< m_cpu.IM()
|
|
<< std::endl;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Fuse::TestRunner::checkMemory() {
|
|
|
|
bool first = true;
|
|
|
|
for (auto memoryDatum : m_result.memoryData) {
|
|
auto bytes = memoryDatum.bytes;
|
|
for (int i = 0; i < bytes.size(); ++i) {
|
|
auto expected = bytes[i];
|
|
uint16_t address = memoryDatum.address + i;
|
|
auto actual = m_ram.peek(address);
|
|
if (expected != actual) {
|
|
m_failed = true;
|
|
if (first) {
|
|
first = false;
|
|
std::cerr << "**** Failed test (Memory): " << m_test.description << std::endl;
|
|
}
|
|
std::cerr
|
|
<< "**** Difference: "
|
|
<< "Address: " << EightBit::Disassembler::hex(address)
|
|
<< " Expected: " << EightBit::Disassembler::hex(expected)
|
|
<< " Actual: " << EightBit::Disassembler::hex(actual)
|
|
<< std::endl;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Fuse::TestRunner::checkEvents() {
|
|
|
|
const auto& expectations = m_expectedEvents.events;
|
|
const auto& actuals = m_actualEvents.events;
|
|
|
|
auto eventFailure = expectations.size() != actuals.size();
|
|
for (auto i = 0; !eventFailure && (i < expectations.size()); ++i) {
|
|
|
|
const auto& expectation = expectations[i];
|
|
const auto& actual = actuals[i];
|
|
|
|
const auto equalCycles = expectation.cycles == actual.cycles;
|
|
const auto equalSpecifier = expectation.specifier == actual.specifier;
|
|
const auto equalAddress = expectation.address == actual.address;
|
|
const auto equalValue = expectation.value == actual.value;
|
|
|
|
const auto equal = equalCycles && equalSpecifier && equalAddress && equalValue;
|
|
eventFailure = !equal;
|
|
}
|
|
|
|
if (eventFailure) {
|
|
dumpExpectedEvents();
|
|
dumpActualEvents();
|
|
}
|
|
|
|
if (!m_failed)
|
|
m_failed = eventFailure;
|
|
}
|
|
|
|
void Fuse::TestRunner::dumpExpectedEvents() const {
|
|
std::cerr << "++++ Dumping expected events:" << std::endl;
|
|
dumpEvents(m_expectedEvents.events);
|
|
}
|
|
|
|
void Fuse::TestRunner::dumpActualEvents() const {
|
|
std::cerr << "++++ Dumping actual events:" << std::endl;
|
|
dumpEvents(m_actualEvents.events);
|
|
}
|
|
|
|
void Fuse::TestRunner::dumpEvents(const std::vector<TestEvent>& events) {
|
|
for (const auto& event : events) {
|
|
dumpEvent(event);
|
|
}
|
|
}
|
|
|
|
void Fuse::TestRunner::dumpEvent(const TestEvent& event) {
|
|
std::cerr << " Event issue " <<
|
|
"Cycles = " << event.cycles <<
|
|
", Specifier = " << event.specifier <<
|
|
", Address = " << EightBit::Disassembler::hex((uint16_t)event.address);
|
|
if (!boost::algorithm::ends_with(event.specifier, "C"))
|
|
std::cerr << ", Value=" << EightBit::Disassembler::hex((uint8_t)event.value);
|
|
std::cerr << std::endl;
|
|
}
|
|
|
|
void Fuse::TestRunner::run() {
|
|
raisePOWER();
|
|
initialise();
|
|
auto allowedCycles = m_test.registerState.tstates;
|
|
try {
|
|
m_cpu.run(allowedCycles);
|
|
check();
|
|
} catch (std::logic_error& error) {
|
|
m_unimplemented = true;
|
|
std::cerr << "**** Error: " << error.what() << std::endl;
|
|
}
|
|
}
|