-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathCPU.cpp
executable file
·154 lines (139 loc) · 2.91 KB
/
CPU.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
#include "stdafx.h"
#include "CPU.h"
CPU::CPU(Memory &memory, MemoryMap &mmap) : Logger("CPU"), m_memory(memory), m_mmap(mmap)
{
for (int i=0; i<256; i++)
{
m_opcodesTable[i] = &CPU::UnknownOpcode;
}
}
CPU::~CPU()
{
}
void CPU::Reset()
{
m_programCounter = 0;
m_state = STOP;
m_timeTicks = 0;
}
void CPU::Run()
{
m_state = RUN;
while (Step() && m_state == RUN);
}
bool CPU::Step()
{
try
{
// Fetch opcode
unsigned char opcode;
m_state = RUN;
m_memory.Read(m_programCounter, opcode);
// Execute instruction
(this->*m_opcodesTable[opcode])(opcode);
}
catch (std::exception e)
{
LogPrintf(LOG_ERROR, "Error processing instruction at 0x%04X! Stopping CPU.\n", m_programCounter);
m_state = STOP;
}
return (m_state == RUN);
}
void CPU::DumpUnassignedOpcodes()
{
for (int i = 0; i < 256; ++i)
{
if (m_opcodesTable[i] == &CPU::UnknownOpcode)
{
LogPrintf(LOG_INFO, "Unassigned: 0x%02X\t0%03o\n", i, i);
}
}
}
void CPU::AddOpcode(BYTE opcode, OPCodeFunction f)
{
if (m_opcodesTable[opcode] != &CPU::UnknownOpcode)
{
LogPrintf(LOG_ERROR, "CPU: Opcode (0x%02X) already defined!\n", opcode);
}
m_opcodesTable[opcode] = f;
}
void CPU::UnknownOpcode(BYTE opcode)
{
LogPrintf(LOG_ERROR, "CPU: Unknown Opcode (0x%02X) at address 0x%04X! Stopping CPU.\n", opcode, m_programCounter);
m_state = STOP;
}
bool CPU::isParityOdd(BYTE b)
{
BYTE parity = 0;
for (int i=0; i<8; i++)
{
parity ^= (b&1);
b = b >> 1;
}
return (parity != 0);
}
void CPU::AddWatch(WORD address, CPUCallbackFunc onCall, CPUCallbackFunc onRet)
{
LogPrintf(LOG_INFO, "Adding watch at address 0x%04X", address);
WatchItem item;
item.addr = address;
item.onCall = onCall;
item.onRet = onRet;
m_callWatches[address] = item;
}
void CPU::AddWatch(const char* label, CPUCallbackFunc onCall, CPUCallbackFunc onRet)
{
MemoryMapItem* item = m_mmap.Get(label);
if (item)
{
AddWatch(item->address, onCall, onRet);
}
else
{
LogPrintf(LOG_WARNING, "AddWatch: Undefined label %s", label);
}
}
void CPU::RemoveWatch(WORD address)
{
LogPrintf(LOG_INFO, "Removing watch at address 0x%04X", address);
m_callWatches.erase(address);
m_returnWatches.erase(address);
}
void CPU::RemoveWatch(const char* label)
{
MemoryMapItem* item = m_mmap.Get(label);
if (item)
{
RemoveWatch(item->address);
}
else
{
LogPrintf(LOG_WARNING, "RemoveWatch: Undefined label %s", label);
}
}
void CPU::OnCall(WORD caller, WORD target)
{
auto it = m_callWatches.find(target);
if (it != m_callWatches.end())
{
WatchItem& item = it->second;
if (item.onCall)
{
m_returnWatches[caller] = item;
item.onCall(this, target);
}
}
}
void CPU::OnReturn(WORD address)
{
auto it = m_returnWatches.find(address);
if (it != m_returnWatches.end())
{
WatchItem& item = it->second;
if (item.onRet)
{
item.onRet(this, address);
}
m_returnWatches.erase(it);
}
}