forked from kstenerud/Musashi
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmc68k.h
113 lines (81 loc) · 2.82 KB
/
mc68k.h
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
#pragma once
#include <array>
#include <string>
#include "endian.h"
#include "gpt.h"
#include "qsm.h"
#include "sim.h"
namespace mc68k
{
struct CpuState;
class Mc68k
{
public:
static constexpr uint32_t CpuStateSize = 600;
Mc68k();
virtual ~Mc68k();
virtual uint32_t exec();
void injectInterrupt(uint8_t _vector, uint8_t _level);
bool hasPendingInterrupt(uint8_t _vector, uint8_t _level) const;
virtual void onReset() {}
virtual uint32_t onIllegalInstruction(uint32_t _opcode);
virtual uint8_t read8(const uint32_t _addr)
{
const auto addr = static_cast<PeriphAddress>(_addr & g_peripheralMask);
if(m_gpt.isInRange(addr)) return m_gpt.read8(addr);
if(m_sim.isInRange(addr)) return m_sim.read8(addr);
if(m_qsm.isInRange(addr)) return m_qsm.read8(addr);
return 0;
}
virtual uint16_t read16(const uint32_t _addr)
{
const auto addr = static_cast<PeriphAddress>(_addr & g_peripheralMask);
if(m_gpt.isInRange(addr)) return m_gpt.read16(addr);
if(m_sim.isInRange(addr)) return m_sim.read16(addr);
if(m_qsm.isInRange(addr)) return m_qsm.read16(addr);
return 0;
}
virtual void write8(const uint32_t _addr, const uint8_t _val)
{
const auto addr = static_cast<PeriphAddress>(_addr & g_peripheralMask);
if(m_gpt.isInRange(addr)) m_gpt.write8(addr, _val);
else if(m_sim.isInRange(addr)) m_sim.write8(addr, _val);
else if(m_qsm.isInRange(addr)) m_qsm.write8(addr, _val);
}
virtual void write16(uint32_t _addr, uint16_t _val)
{
const auto addr = static_cast<PeriphAddress>(_addr & g_peripheralMask);
if(m_gpt.isInRange(addr)) m_gpt.write16(addr, _val);
else if(m_sim.isInRange(addr)) m_sim.write16(addr, _val);
else if(m_qsm.isInRange(addr)) m_qsm.write16(addr, _val);
}
virtual uint16_t readImm16(uint32_t _addr) = 0;
uint32_t readIrqUserVector(uint8_t _level);
void reset();
void setPC(uint32_t _pc);
uint32_t getPC() const;
virtual uint32_t getResetPC() { return 0; }
virtual uint32_t getResetSP() { return 0; }
uint32_t disassemble(uint32_t _pc, char* _buffer);
uint64_t getCycles() const { return m_cycles; }
Port& getPortE() { return m_sim.getPortE(); }
Port& getPortF() { return m_sim.getPortF(); }
Port& getPortGP() { return m_gpt.getPortGP(); }
Port& getPortQS() { return m_qsm.getPortQS(); }
Gpt& getGPT() { return m_gpt; }
Qsm& getQSM() { return m_qsm; }
Sim& getSim() { return m_sim; }
CpuState* getCpuState();
const CpuState* getCpuState() const;
bool dumpAssembly(const std::string& _filename, uint32_t _first, uint32_t _count, bool _splitFunctions = true);
private:
void raiseIPL();
std::array<uint8_t, CpuStateSize> m_cpuStateBuf;
CpuState* m_cpuState;
Gpt m_gpt;
Sim m_sim;
Qsm m_qsm;
std::array<std::deque<uint8_t>, 8> m_pendingInterrupts;
uint64_t m_cycles = 0;
};
}