2017-06-05 22:39:15 +01:00
|
|
|
#include "stdafx.h"
|
|
|
|
#include "Board.h"
|
|
|
|
#include "Disassembler.h"
|
|
|
|
|
|
|
|
Board::Board(const Configuration& configuration)
|
|
|
|
: m_configuration(configuration),
|
|
|
|
m_memory(0x3fff),
|
2017-06-19 18:08:13 +01:00
|
|
|
m_cpu(EightBit::Z80(m_memory, m_ports)),
|
|
|
|
m_profiler(m_cpu, m_disassembler) {
|
2017-06-05 22:39:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Board::initialise() {
|
|
|
|
|
|
|
|
m_memory.clear();
|
|
|
|
auto romDirectory = m_configuration.getRomDirectory();
|
|
|
|
|
|
|
|
//m_memory.loadRam(romDirectory + "/prelim.com", 0x100); // Bartholomew preliminary
|
|
|
|
//m_memory.loadRam(romDirectory + "/zexdoc.com", 0x100); // Cringle/Bartholomew
|
|
|
|
m_memory.loadRam(romDirectory + "/zexall.com", 0x100); // Cringle/Bartholomew
|
|
|
|
//m_memory.loadRam(romDirectory + "/CPUTEST.COM", 0x100); // SuperSoft diagnostics
|
|
|
|
//m_memory.loadRam(romDirectory + "/TEST.COM", 0x100); // Microcosm
|
|
|
|
|
|
|
|
m_memory.ADDRESS().word = 5;
|
|
|
|
m_memory.reference() = 0xc9; // ret
|
|
|
|
m_cpu.ExecutingInstruction.connect(std::bind(&Board::Cpu_ExecutingInstruction_Cpm, this, std::placeholders::_1));
|
|
|
|
|
|
|
|
if (m_configuration.isProfileMode()) {
|
|
|
|
m_cpu.ExecutingInstruction.connect(std::bind(&Board::Cpu_ExecutingInstruction_Profile, this, std::placeholders::_1));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_configuration.isDebugMode()) {
|
|
|
|
m_cpu.ExecutingInstruction.connect(std::bind(&Board::Cpu_ExecutingInstruction_Debug, this, std::placeholders::_1));
|
|
|
|
}
|
|
|
|
|
|
|
|
m_cpu.initialise();
|
2017-06-19 13:53:00 +01:00
|
|
|
m_cpu.PC() = m_configuration.getStartAddress();
|
2017-06-05 22:39:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Board::Cpu_ExecutingInstruction_Cpm(const EightBit::Z80&) {
|
2017-06-19 13:53:00 +01:00
|
|
|
auto pc = m_cpu.PC();
|
2017-06-05 22:39:15 +01:00
|
|
|
switch (pc.word) {
|
|
|
|
case 0x0: // CP/M warm start
|
|
|
|
m_cpu.halt();
|
2017-06-22 10:27:19 +01:00
|
|
|
if (m_configuration.isProfileMode()) {
|
|
|
|
m_profiler.dump();
|
|
|
|
}
|
2017-06-05 22:39:15 +01:00
|
|
|
break;
|
|
|
|
case 0x5: // BDOS
|
|
|
|
bdos();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Board::bdos() {
|
|
|
|
auto c = m_cpu.C();
|
|
|
|
switch (c) {
|
|
|
|
case 0x2: {
|
|
|
|
auto character = m_cpu.E();
|
|
|
|
std::cout << character;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 0x9:
|
|
|
|
for (uint16_t i = m_cpu.DE().word; m_memory.peek(i) != '$'; ++i) {
|
|
|
|
std::cout << m_memory.peek(i);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Board::Cpu_ExecutingInstruction_Profile(const EightBit::Z80& cpu) {
|
|
|
|
|
2017-06-19 13:53:00 +01:00
|
|
|
const auto pc = m_cpu.PC();
|
2017-06-05 22:39:15 +01:00
|
|
|
|
|
|
|
m_profiler.addAddress(pc.word);
|
|
|
|
m_profiler.addInstruction(m_memory.peek(pc.word));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Board::Cpu_ExecutingInstruction_Debug(const EightBit::Z80& cpu) {
|
|
|
|
|
|
|
|
std::cerr
|
|
|
|
<< EightBit::Disassembler::state(m_cpu)
|
|
|
|
<< "\t"
|
2017-06-19 13:53:00 +01:00
|
|
|
<< m_disassembler.disassemble(m_cpu)
|
2017-06-05 22:39:15 +01:00
|
|
|
<< '\n';
|
|
|
|
}
|