2017-06-04 21:38:34 +01:00
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
|
|
#include <cstdint>
|
|
|
|
|
#include <string>
|
|
|
|
|
#include <array>
|
|
|
|
|
#include <functional>
|
|
|
|
|
|
2017-07-05 17:46:02 +01:00
|
|
|
|
#include "Memory.h"
|
|
|
|
|
#include "Processor.h"
|
2017-07-02 22:03:33 +01:00
|
|
|
|
#include "Signal.h"
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
namespace EightBit {
|
2017-07-05 17:46:02 +01:00
|
|
|
|
class MOS6502 : public Processor {
|
2017-07-02 22:03:33 +01:00
|
|
|
|
public:
|
2017-07-17 13:46:06 +01:00
|
|
|
|
struct opcode_decoded_t {
|
|
|
|
|
|
|
|
|
|
int aaa;
|
|
|
|
|
int bbb;
|
|
|
|
|
int cc;
|
|
|
|
|
|
|
|
|
|
opcode_decoded_t() {
|
|
|
|
|
aaa = bbb = cc = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
opcode_decoded_t(uint8_t opcode) {
|
|
|
|
|
aaa = (opcode & 0b11100000) >> 5; // 0 - 7
|
|
|
|
|
bbb = (opcode & 0b00011100) >> 2; // 0 - 7
|
|
|
|
|
cc = (opcode & 0b00000011); // 0 - 3
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2017-07-10 15:51:33 +01:00
|
|
|
|
enum StatusBits {
|
2017-07-17 13:46:06 +01:00
|
|
|
|
NF = Bit7, // Negative
|
|
|
|
|
VF = Bit6, // Overflow
|
|
|
|
|
RF = Bit5, // reserved
|
|
|
|
|
BF = Bit4, // Brk
|
|
|
|
|
DF = Bit3, // D (use BCD for arithmetic)
|
|
|
|
|
IF = Bit2, // I (IRQ disable)
|
|
|
|
|
ZF = Bit1, // Zero
|
|
|
|
|
CF = Bit0, // Carry
|
2017-07-10 15:51:33 +01:00
|
|
|
|
};
|
|
|
|
|
|
2017-07-14 17:22:28 +01:00
|
|
|
|
MOS6502(Memory& memory);
|
2017-07-17 13:46:06 +01:00
|
|
|
|
virtual ~MOS6502();
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
|
|
|
|
Signal<MOS6502> ExecutingInstruction;
|
2017-07-05 17:46:02 +01:00
|
|
|
|
Signal<MOS6502> ExecutedInstruction;
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-07 09:24:58 +01:00
|
|
|
|
uint8_t& X() { return x; }
|
|
|
|
|
uint8_t& Y() { return y; }
|
|
|
|
|
uint8_t& A() { return a; }
|
|
|
|
|
uint8_t& S() { return s; }
|
2017-07-10 15:51:33 +01:00
|
|
|
|
uint8_t& P() { return p; }
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
2017-07-05 17:46:02 +01:00
|
|
|
|
virtual void initialise();
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
2017-07-05 17:46:02 +01:00
|
|
|
|
virtual int step();
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
virtual void Reset();
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
virtual void TriggerIRQ();
|
|
|
|
|
virtual void TriggerNMI();
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-06 21:32:52 +01:00
|
|
|
|
void GetWord(register16_t& output);
|
|
|
|
|
void GetWord(uint16_t offset, register16_t& output);
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-06 21:32:52 +01:00
|
|
|
|
uint8_t GetByte() { return m_memory.read(); }
|
2017-07-05 17:46:02 +01:00
|
|
|
|
uint8_t GetByte(uint16_t offset) { return m_memory.read(offset); }
|
2017-07-06 21:32:52 +01:00
|
|
|
|
|
|
|
|
|
void SetByte(uint8_t value) { m_memory.write(value); }
|
2017-07-05 17:46:02 +01:00
|
|
|
|
void SetByte(uint16_t offset, uint8_t value) { m_memory.write(offset, value); }
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
protected:
|
|
|
|
|
virtual void Interrupt(uint16_t vector);
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-05 17:46:02 +01:00
|
|
|
|
virtual int Execute(uint8_t cell);
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
private:
|
2017-07-17 13:46:06 +01:00
|
|
|
|
register16_t& MEMPTR() { return m_memptr; }
|
|
|
|
|
|
2017-07-16 10:05:49 +01:00
|
|
|
|
void adjustZero(uint8_t datum) { clearFlag(P(), ZF, datum); }
|
|
|
|
|
void adjustNegative(uint8_t datum) { setFlag(P(), NF, datum & NF); }
|
2017-07-06 21:32:52 +01:00
|
|
|
|
|
2017-07-16 10:05:49 +01:00
|
|
|
|
void adjustNZ(uint8_t datum) {
|
|
|
|
|
adjustZero(datum);
|
|
|
|
|
adjustNegative(datum);
|
2017-07-06 21:32:52 +01:00
|
|
|
|
}
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
void PushByte(uint8_t value);
|
|
|
|
|
uint8_t PopByte();
|
2017-07-05 17:46:02 +01:00
|
|
|
|
void PushWord(register16_t value);
|
2017-07-06 21:32:52 +01:00
|
|
|
|
void PopWord(register16_t& output);
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
uint8_t FetchByte();
|
2017-07-06 21:32:52 +01:00
|
|
|
|
void FetchWord(register16_t& output);
|
|
|
|
|
|
2017-07-11 21:34:01 +01:00
|
|
|
|
#pragma region 6502 addressing modes
|
|
|
|
|
|
|
|
|
|
#pragma region Addresses
|
|
|
|
|
|
|
|
|
|
void Address_Absolute() {
|
2017-07-17 13:46:06 +01:00
|
|
|
|
FetchWord(MEMPTR());
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Address_ZeroPage() {
|
2017-07-17 13:46:06 +01:00
|
|
|
|
MEMPTR().low = FetchByte();
|
|
|
|
|
MEMPTR().high = 0;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Address_ZeroPageIndirect() {
|
|
|
|
|
Address_ZeroPage();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
|
|
|
|
GetWord(MEMPTR());
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Address_Indirect() {
|
|
|
|
|
Address_Absolute();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
|
|
|
|
GetWord(MEMPTR());
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Address_ZeroPageX() {
|
|
|
|
|
Address_ZeroPage();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
MEMPTR().low += X();
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Address_ZeroPageY() {
|
|
|
|
|
Address_ZeroPage();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
MEMPTR().low += Y();
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Address_AbsoluteX() {
|
|
|
|
|
Address_Absolute();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
MEMPTR().word += X();
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Address_AbsoluteY() {
|
|
|
|
|
Address_Absolute();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
MEMPTR().word += Y();
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Address_IndexedIndirectX() {
|
|
|
|
|
Address_ZeroPageX();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
|
|
|
|
GetWord(MEMPTR());
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Address_IndirectIndexedY() {
|
|
|
|
|
Address_ZeroPageIndirect();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
MEMPTR().word += Y();
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#pragma endregion Addresses
|
|
|
|
|
|
|
|
|
|
#pragma region References
|
|
|
|
|
|
2017-07-16 10:40:38 +01:00
|
|
|
|
uint8_t& AM_A() {
|
|
|
|
|
m_busRW = false;
|
|
|
|
|
return A();
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-11 21:34:01 +01:00
|
|
|
|
uint8_t& AM_Immediate() {
|
2017-07-16 10:40:38 +01:00
|
|
|
|
m_busRW = false;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
FetchByte();
|
|
|
|
|
return m_memory.reference();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t& AM_Absolute() {
|
2017-07-16 10:40:38 +01:00
|
|
|
|
m_busRW = true;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
Address_Absolute();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
2017-07-11 21:34:01 +01:00
|
|
|
|
return m_memory.reference();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t& AM_ZeroPage() {
|
2017-07-16 10:40:38 +01:00
|
|
|
|
m_busRW = true;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
Address_ZeroPage();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
2017-07-11 21:34:01 +01:00
|
|
|
|
return m_memory.reference();
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-15 23:19:46 +01:00
|
|
|
|
uint8_t& AM_AbsoluteX(bool read = true) {
|
2017-07-16 10:40:38 +01:00
|
|
|
|
m_busRW = true;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
Address_AbsoluteX();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
|
|
|
|
if (read && (m_memory.ADDRESS().low == Mask8))
|
2017-07-15 23:19:46 +01:00
|
|
|
|
++cycles;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
return m_memory.reference();
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-15 23:19:46 +01:00
|
|
|
|
uint8_t& AM_AbsoluteY(bool read = true) {
|
2017-07-16 10:40:38 +01:00
|
|
|
|
m_busRW = true;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
Address_AbsoluteY();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
|
|
|
|
if (read && (m_memory.ADDRESS().low == Mask8))
|
2017-07-15 23:19:46 +01:00
|
|
|
|
++cycles;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
return m_memory.reference();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t& AM_ZeroPageX() {
|
2017-07-16 10:40:38 +01:00
|
|
|
|
m_busRW = true;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
Address_ZeroPageX();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
2017-07-11 21:34:01 +01:00
|
|
|
|
return m_memory.reference();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t& AM_ZeroPageY() {
|
2017-07-16 10:40:38 +01:00
|
|
|
|
m_busRW = true;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
Address_ZeroPageY();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
2017-07-11 21:34:01 +01:00
|
|
|
|
return m_memory.reference();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t& AM_IndexedIndirectX() {
|
2017-07-16 10:40:38 +01:00
|
|
|
|
m_busRW = true;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
Address_IndexedIndirectX();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
2017-07-11 21:34:01 +01:00
|
|
|
|
return m_memory.reference();
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-15 23:19:46 +01:00
|
|
|
|
uint8_t& AM_IndirectIndexedY(bool read = true) {
|
2017-07-16 10:40:38 +01:00
|
|
|
|
m_busRW = true;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
Address_IndirectIndexedY();
|
2017-07-17 13:46:06 +01:00
|
|
|
|
m_memory.ADDRESS() = MEMPTR();
|
|
|
|
|
if (read && (m_memory.ADDRESS().low == Mask8))
|
2017-07-15 23:19:46 +01:00
|
|
|
|
++cycles;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
return m_memory.reference();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#pragma endregion References
|
|
|
|
|
|
2017-07-13 12:02:44 +01:00
|
|
|
|
#pragma region 6502 addressing mode switching
|
|
|
|
|
|
2017-07-15 23:19:46 +01:00
|
|
|
|
uint8_t& AM_00(int bbb, bool read = true) {
|
2017-07-13 12:02:44 +01:00
|
|
|
|
switch (bbb) {
|
|
|
|
|
case 0b000:
|
|
|
|
|
return AM_Immediate();
|
|
|
|
|
case 0b001:
|
|
|
|
|
return AM_ZeroPage();
|
|
|
|
|
case 0b011:
|
|
|
|
|
return AM_Absolute();
|
|
|
|
|
case 0b101:
|
|
|
|
|
return AM_ZeroPageX();
|
|
|
|
|
case 0b111:
|
2017-07-15 23:19:46 +01:00
|
|
|
|
return AM_AbsoluteX(read);
|
2017-07-13 12:02:44 +01:00
|
|
|
|
case 0b010:
|
|
|
|
|
case 0b100:
|
|
|
|
|
case 0b110:
|
|
|
|
|
throw std::domain_error("Illegal addressing mode");
|
|
|
|
|
default:
|
|
|
|
|
__assume(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-15 23:19:46 +01:00
|
|
|
|
uint8_t& AM_01(int bbb, bool read = true) {
|
2017-07-13 12:02:44 +01:00
|
|
|
|
switch (bbb) {
|
|
|
|
|
case 0b000:
|
|
|
|
|
return AM_IndexedIndirectX();
|
|
|
|
|
case 0b001:
|
|
|
|
|
return AM_ZeroPage();
|
|
|
|
|
case 0b010:
|
|
|
|
|
return AM_Immediate();
|
|
|
|
|
case 0b011:
|
|
|
|
|
return AM_Absolute();
|
|
|
|
|
case 0b100:
|
2017-07-15 23:19:46 +01:00
|
|
|
|
return AM_IndirectIndexedY(read);
|
2017-07-13 12:02:44 +01:00
|
|
|
|
case 0b101:
|
|
|
|
|
return AM_ZeroPageX();
|
|
|
|
|
case 0b110:
|
2017-07-15 23:19:46 +01:00
|
|
|
|
return AM_AbsoluteY(read);
|
2017-07-13 12:02:44 +01:00
|
|
|
|
case 0b111:
|
2017-07-15 23:19:46 +01:00
|
|
|
|
return AM_AbsoluteX(read);
|
2017-07-13 12:02:44 +01:00
|
|
|
|
default:
|
|
|
|
|
__assume(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-15 23:19:46 +01:00
|
|
|
|
uint8_t& AM_10(int bbb, bool read = true) {
|
2017-07-13 12:02:44 +01:00
|
|
|
|
switch (bbb) {
|
|
|
|
|
case 0b000:
|
|
|
|
|
return AM_Immediate();
|
|
|
|
|
case 0b001:
|
|
|
|
|
return AM_ZeroPage();
|
|
|
|
|
case 0b010:
|
2017-07-16 10:40:38 +01:00
|
|
|
|
return AM_A();
|
2017-07-13 12:02:44 +01:00
|
|
|
|
case 0b011:
|
|
|
|
|
return AM_Absolute();
|
|
|
|
|
case 0b101:
|
|
|
|
|
return AM_ZeroPageX();
|
|
|
|
|
case 0b111:
|
2017-07-15 23:19:46 +01:00
|
|
|
|
return AM_AbsoluteX(read);
|
2017-07-13 12:02:44 +01:00
|
|
|
|
case 0b100:
|
|
|
|
|
case 0b110:
|
|
|
|
|
throw std::domain_error("Illegal addressing mode");
|
|
|
|
|
default:
|
|
|
|
|
__assume(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-15 23:19:46 +01:00
|
|
|
|
uint8_t& AM_10_x(int bbb, bool read = true) {
|
2017-07-13 12:02:44 +01:00
|
|
|
|
switch (bbb) {
|
|
|
|
|
case 0b000:
|
|
|
|
|
return AM_Immediate();
|
|
|
|
|
case 0b001:
|
|
|
|
|
return AM_ZeroPage();
|
|
|
|
|
case 0b010:
|
2017-07-16 10:40:38 +01:00
|
|
|
|
return AM_A();
|
2017-07-13 12:02:44 +01:00
|
|
|
|
case 0b011:
|
|
|
|
|
return AM_Absolute();
|
|
|
|
|
case 0b101:
|
|
|
|
|
return AM_ZeroPageY();
|
|
|
|
|
case 0b111:
|
2017-07-15 23:19:46 +01:00
|
|
|
|
return AM_AbsoluteY(read);
|
2017-07-13 12:02:44 +01:00
|
|
|
|
case 0b100:
|
|
|
|
|
case 0b110:
|
|
|
|
|
throw std::domain_error("Illegal addressing mode");
|
|
|
|
|
default:
|
|
|
|
|
__assume(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#pragma endregion 6502 addressing mode switching
|
|
|
|
|
|
2017-07-11 21:34:01 +01:00
|
|
|
|
#pragma endregion 6502 addressing modes
|
|
|
|
|
|
|
|
|
|
void ROR(uint8_t& output);
|
|
|
|
|
|
|
|
|
|
void LSR(uint8_t& output);
|
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
void BIT(uint8_t data);
|
|
|
|
|
|
2017-07-11 21:34:01 +01:00
|
|
|
|
void ROL(uint8_t& output);
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
2017-07-11 21:34:01 +01:00
|
|
|
|
void ASL(uint8_t& output);
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
|
|
|
|
void SBC(uint8_t data);
|
|
|
|
|
void SBC_b(uint8_t data);
|
|
|
|
|
void SBC_d(uint8_t data);
|
|
|
|
|
|
|
|
|
|
void CMP(uint8_t first, uint8_t second);
|
|
|
|
|
|
|
|
|
|
void ADC(uint8_t data);
|
|
|
|
|
void ADC_b(uint8_t data);
|
|
|
|
|
void ADC_d(uint8_t data);
|
|
|
|
|
|
|
|
|
|
void Branch(int8_t displacement);
|
2017-07-14 17:22:28 +01:00
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
void Branch(bool flag);
|
2017-07-14 17:22:28 +01:00
|
|
|
|
|
|
|
|
|
void PHP();
|
|
|
|
|
void PLP();
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
|
|
|
|
void JSR_abs();
|
2017-07-14 17:22:28 +01:00
|
|
|
|
void RTI();
|
|
|
|
|
void RTS();
|
2017-07-02 22:03:33 +01:00
|
|
|
|
void JMP_abs();
|
|
|
|
|
void JMP_ind();
|
2017-07-14 17:22:28 +01:00
|
|
|
|
void BRK();
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
|
|
|
|
const uint16_t PageOne = 0x100;
|
|
|
|
|
const uint16_t IRQvector = 0xfffe;
|
|
|
|
|
const uint16_t RSTvector = 0xfffc;
|
|
|
|
|
const uint16_t NMIvector = 0xfffa;
|
|
|
|
|
|
|
|
|
|
uint8_t x; // index register X
|
|
|
|
|
uint8_t y; // index register Y
|
|
|
|
|
uint8_t a; // accumulator
|
|
|
|
|
uint8_t s; // stack pointer
|
2017-07-10 15:51:33 +01:00
|
|
|
|
uint8_t p; // processor status
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
2017-07-11 21:34:01 +01:00
|
|
|
|
register16_t m_memptr;
|
2017-07-15 23:19:46 +01:00
|
|
|
|
|
|
|
|
|
std::array<int, 0x100> m_timings;
|
2017-07-17 13:46:06 +01:00
|
|
|
|
std::array<opcode_decoded_t, 0x100> m_decodedOpcodes;
|
2017-07-16 10:40:38 +01:00
|
|
|
|
|
|
|
|
|
bool m_busRW;
|
2017-07-02 22:03:33 +01:00
|
|
|
|
};
|
|
|
|
|
}
|