2017-06-04 21:38:34 +01:00
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
|
|
#include <cstdint>
|
2018-10-31 23:29:13 +00:00
|
|
|
|
#include <utility>
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2018-10-31 23:29:13 +00:00
|
|
|
|
#include <EightBitCompilerDefinitions.h>
|
2018-08-17 21:53:49 +01:00
|
|
|
|
#include <LittleEndianProcessor.h>
|
2018-10-31 23:29:13 +00:00
|
|
|
|
#include <Register.h>
|
2017-09-07 00:58:56 +01:00
|
|
|
|
#include <Signal.h>
|
2019-01-14 02:10:17 +00:00
|
|
|
|
#include <EventArgs.h>
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
namespace EightBit {
|
2018-10-31 23:29:13 +00:00
|
|
|
|
|
|
|
|
|
class Bus;
|
|
|
|
|
|
2018-08-17 21:53:49 +01:00
|
|
|
|
class MOS6502 : public LittleEndianProcessor {
|
2021-12-08 19:47:35 +00:00
|
|
|
|
public:
|
|
|
|
|
DECLARE_PIN_INPUT(NMI)
|
|
|
|
|
DECLARE_PIN_INPUT(SO)
|
|
|
|
|
DECLARE_PIN_OUTPUT(SYNC)
|
|
|
|
|
DECLARE_PIN_INPUT(RDY)
|
|
|
|
|
DECLARE_PIN_OUTPUT(RW)
|
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
public:
|
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
|
|
|
|
};
|
|
|
|
|
|
2021-10-24 11:12:23 +01:00
|
|
|
|
MOS6502(Bus& bus) noexcept;
|
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
|
|
|
|
|
2022-01-17 19:10:15 +00:00
|
|
|
|
int execute() noexcept final;
|
|
|
|
|
[[nodiscard]] int step() noexcept final;
|
2019-01-14 02:10:17 +00:00
|
|
|
|
|
2021-07-18 14:28:40 +01:00
|
|
|
|
[[nodiscard]] constexpr auto& X() noexcept { return x; }
|
|
|
|
|
[[nodiscard]] constexpr auto& Y() noexcept { return y; }
|
|
|
|
|
[[nodiscard]] constexpr auto& A() noexcept { return a; }
|
|
|
|
|
[[nodiscard]] constexpr auto& S() noexcept { return s; }
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
2021-07-18 14:28:40 +01:00
|
|
|
|
[[nodiscard]] constexpr auto& P() noexcept { return p; }
|
|
|
|
|
[[nodiscard]] constexpr const auto& P() const noexcept { return p; }
|
2018-10-31 23:29:13 +00:00
|
|
|
|
|
2017-12-10 21:41:48 +00:00
|
|
|
|
protected:
|
2022-01-17 19:10:15 +00:00
|
|
|
|
void handleRESET() noexcept final;
|
|
|
|
|
void handleINT() noexcept final;
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2022-01-17 19:10:15 +00:00
|
|
|
|
void busWrite() noexcept final;
|
|
|
|
|
[[nodiscard]] uint8_t busRead() noexcept final;
|
2018-12-29 22:18:01 +00:00
|
|
|
|
|
2024-01-06 12:19:02 +00:00
|
|
|
|
// Instructions with BCD effects
|
|
|
|
|
|
2021-07-18 14:28:40 +01:00
|
|
|
|
[[nodiscard]] virtual uint8_t sub(uint8_t operand, uint8_t data, int borrow = 0) noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t sbc(uint8_t operand, uint8_t data) noexcept;
|
2024-01-05 12:52:27 +00:00
|
|
|
|
[[nodiscard]] uint8_t sub_b(uint8_t operand, uint8_t data, int borrow = 0) noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t sub_d(uint8_t operand, uint8_t data, int borrow = 0) noexcept;
|
2017-12-25 23:34:56 +00:00
|
|
|
|
|
2021-07-18 14:28:40 +01:00
|
|
|
|
[[nodiscard]] virtual uint8_t add(uint8_t operand, uint8_t data, int carry = 0) noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t adc(uint8_t operand, uint8_t data) noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t add_b(uint8_t operand, uint8_t data, int carry) noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t add_d(uint8_t operand, uint8_t data, int carry) noexcept;
|
2017-12-25 23:34:56 +00:00
|
|
|
|
|
2024-01-06 12:19:02 +00:00
|
|
|
|
// Undocumented compound instructions (with BCD effects)
|
|
|
|
|
|
|
|
|
|
virtual void arr(uint8_t value) noexcept;
|
|
|
|
|
virtual void arr_b(uint8_t value) noexcept;
|
|
|
|
|
virtual void arr_d(uint8_t value) noexcept;
|
|
|
|
|
|
2017-07-02 22:03:33 +01:00
|
|
|
|
private:
|
2018-10-31 23:29:13 +00:00
|
|
|
|
const uint8_t IRQvector = 0xfe; // IRQ vector
|
|
|
|
|
const uint8_t RSTvector = 0xfc; // RST vector
|
|
|
|
|
const uint8_t NMIvector = 0xfa; // NMI vector
|
|
|
|
|
|
2022-01-17 19:10:15 +00:00
|
|
|
|
void handleNMI() noexcept;
|
|
|
|
|
void handleSO() noexcept;
|
2018-08-25 12:09:26 +01:00
|
|
|
|
|
2022-01-24 23:00:25 +00:00
|
|
|
|
void interrupt() noexcept;
|
2017-12-10 21:41:48 +00:00
|
|
|
|
|
2022-01-17 19:10:15 +00:00
|
|
|
|
void push(uint8_t value) noexcept final;
|
|
|
|
|
[[nodiscard]] uint8_t pop() noexcept final;
|
2017-06-04 21:38:34 +01:00
|
|
|
|
|
2019-01-08 23:09:52 +00:00
|
|
|
|
// Dummy stack push, used during RESET
|
2022-01-17 19:10:15 +00:00
|
|
|
|
void dummyPush(uint8_t value) noexcept;
|
2019-01-08 23:09:52 +00:00
|
|
|
|
|
2018-10-31 23:29:13 +00:00
|
|
|
|
// Addressing modes
|
2017-07-11 21:34:01 +01:00
|
|
|
|
|
2022-01-24 23:00:25 +00:00
|
|
|
|
[[nodiscard]] register16_t Address_Absolute() noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t Address_ZeroPage() noexcept;
|
|
|
|
|
[[nodiscard]] register16_t Address_ZeroPageIndirect() noexcept;
|
|
|
|
|
[[nodiscard]] register16_t Address_Indirect() noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t Address_ZeroPageX() noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t Address_ZeroPageY() noexcept;
|
|
|
|
|
[[nodiscard]] std::pair<register16_t, uint8_t> Address_AbsoluteX() noexcept;
|
|
|
|
|
[[nodiscard]] std::pair<register16_t, uint8_t> Address_AbsoluteY() noexcept;
|
|
|
|
|
[[nodiscard]] register16_t Address_IndexedIndirectX() noexcept;
|
|
|
|
|
[[nodiscard]] std::pair<register16_t, uint8_t> Address_IndirectIndexedY() noexcept;
|
|
|
|
|
[[nodiscard]] register16_t Address_relative_byte() noexcept;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
|
2017-12-30 15:22:27 +00:00
|
|
|
|
// Addressing modes, read
|
2017-07-11 21:34:01 +01:00
|
|
|
|
|
2019-11-09 18:58:23 +00:00
|
|
|
|
enum class PageCrossingBehavior { AlwaysReadTwice, MaybeReadTwice };
|
2019-01-06 12:58:13 +00:00
|
|
|
|
|
2022-01-24 23:00:25 +00:00
|
|
|
|
uint8_t AM_Immediate() noexcept;
|
|
|
|
|
uint8_t AM_Absolute() noexcept;
|
|
|
|
|
uint8_t AM_ZeroPage() noexcept;
|
|
|
|
|
uint8_t AM_AbsoluteX(PageCrossingBehavior behaviour = PageCrossingBehavior::MaybeReadTwice) noexcept;
|
|
|
|
|
uint8_t AM_AbsoluteY() noexcept;
|
|
|
|
|
uint8_t AM_ZeroPageX() noexcept;
|
|
|
|
|
uint8_t AM_ZeroPageY() noexcept;
|
|
|
|
|
uint8_t AM_IndexedIndirectX() noexcept;
|
|
|
|
|
uint8_t AM_IndirectIndexedY() noexcept;
|
2017-07-11 21:34:01 +01:00
|
|
|
|
|
2021-12-08 19:47:35 +00:00
|
|
|
|
// Flag checking
|
|
|
|
|
|
|
|
|
|
[[nodiscard]] constexpr auto interruptMasked() const noexcept { return P() & IF; }
|
|
|
|
|
[[nodiscard]] constexpr auto decimal() const noexcept { return P() & DF; }
|
|
|
|
|
|
|
|
|
|
[[nodiscard]] static constexpr auto negative(uint8_t data) noexcept { return data & NF; }
|
|
|
|
|
[[nodiscard]] constexpr auto negative() const noexcept { return negative(P()); }
|
|
|
|
|
|
|
|
|
|
[[nodiscard]] static constexpr auto zero(uint8_t data) noexcept { return data & ZF; }
|
|
|
|
|
[[nodiscard]] constexpr auto zero() const noexcept { return zero(P()); }
|
|
|
|
|
|
|
|
|
|
[[nodiscard]] static constexpr auto overflow(uint8_t data) noexcept { return data & VF; }
|
|
|
|
|
[[nodiscard]] constexpr auto overflow() const noexcept { return overflow(P()); }
|
|
|
|
|
|
|
|
|
|
[[nodiscard]] static constexpr auto carry(uint8_t data) noexcept { return data & CF; }
|
|
|
|
|
[[nodiscard]] constexpr auto carry() const noexcept { return carry(P()); }
|
|
|
|
|
|
2018-10-31 23:29:13 +00:00
|
|
|
|
// Flag adjustment
|
2017-07-11 21:34:01 +01:00
|
|
|
|
|
2024-01-07 14:04:41 +00:00
|
|
|
|
constexpr void adjustZero(const uint8_t datum) noexcept { reset_flag(ZF, datum); }
|
|
|
|
|
constexpr void adjustNegative(const uint8_t datum) noexcept { set_flag(NF, negative(datum)); }
|
2021-12-08 19:47:35 +00:00
|
|
|
|
|
2021-10-24 11:12:23 +01:00
|
|
|
|
constexpr void adjustNZ(const uint8_t datum) noexcept {
|
2018-10-31 23:29:13 +00:00
|
|
|
|
adjustZero(datum);
|
|
|
|
|
adjustNegative(datum);
|
2017-07-11 21:34:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
2024-01-07 14:35:12 +00:00
|
|
|
|
constexpr void adjustOverflow_add(uint8_t operand, uint8_t data, uint8_t intermediate) noexcept {
|
|
|
|
|
set_flag(VF, negative(~(operand ^ data) & (operand ^ intermediate)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
constexpr void adjustOverflow_subtract(uint8_t operand, uint8_t data, uint8_t intermediate) noexcept {
|
|
|
|
|
set_flag(VF, negative((operand ^ data) & (operand ^ intermediate)));
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-31 23:29:13 +00:00
|
|
|
|
// Miscellaneous
|
|
|
|
|
|
2022-01-24 23:00:25 +00:00
|
|
|
|
void branch(int condition) noexcept;
|
2019-01-05 17:23:50 +00:00
|
|
|
|
|
2021-10-24 11:12:23 +01:00
|
|
|
|
[[nodiscard]] constexpr auto through(const uint8_t data) noexcept {
|
2018-10-31 23:29:13 +00:00
|
|
|
|
adjustNZ(data);
|
|
|
|
|
return data;
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-24 23:00:25 +00:00
|
|
|
|
void memoryReadModifyWrite(const uint8_t data) noexcept {
|
2019-01-03 01:04:12 +00:00
|
|
|
|
// The read will have already taken place...
|
2020-02-09 11:51:58 +00:00
|
|
|
|
memoryWrite();
|
|
|
|
|
memoryWrite(data);
|
2019-01-03 01:04:12 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-01-07 12:15:11 +00:00
|
|
|
|
// Unconditional page fixup cycle required
|
|
|
|
|
void fixup(const register16_t address, const uint8_t unfixed_page) noexcept {
|
|
|
|
|
getBytePaged(unfixed_page, address.low); // Possible fixup for page boundary crossing
|
|
|
|
|
BUS().ADDRESS() = address;
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-07 14:04:41 +00:00
|
|
|
|
// Status flag operations
|
|
|
|
|
|
|
|
|
|
constexpr void set_flag(int which, int condition) noexcept { P() = setBit(P(), which, condition); }
|
|
|
|
|
constexpr void set_flag(int which) noexcept { P() = setBit(P(), which); }
|
|
|
|
|
|
|
|
|
|
constexpr void reset_flag(int which, int condition) noexcept { P() = clearBit(P(), which, condition); }
|
|
|
|
|
constexpr void reset_flag(int which) noexcept { P() = clearBit(P(), which); }
|
|
|
|
|
|
2024-01-07 12:15:11 +00:00
|
|
|
|
// Chew up a cycle
|
|
|
|
|
void swallow() noexcept { memoryRead(PC()); }
|
|
|
|
|
void swallow_stack() noexcept { getBytePaged(1, S()); }
|
|
|
|
|
void swallow_fetch() noexcept { fetchByte(); }
|
|
|
|
|
|
2018-10-31 23:29:13 +00:00
|
|
|
|
// Instruction implementations
|
|
|
|
|
|
2021-07-18 14:28:40 +01:00
|
|
|
|
[[nodiscard]] uint8_t andr(uint8_t operand, uint8_t data) noexcept;
|
|
|
|
|
void bit(uint8_t operand, uint8_t data) noexcept;
|
|
|
|
|
void cmp(uint8_t first, uint8_t second) noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t dec(uint8_t value) noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t eorr(uint8_t operand, uint8_t data) noexcept;
|
|
|
|
|
[[nodiscard]] uint8_t inc(uint8_t value) noexcept;
|
2022-01-24 23:00:25 +00:00
|
|
|
|
void jsr() noexcept;
|
2021-07-18 14:28:40 +01:00
|
|
|
|
[[nodiscard]] uint8_t orr(uint8_t operand, uint8_t data) noexcept;
|
2022-01-24 23:00:25 +00:00
|
|
|
|
void php() noexcept;
|
|
|
|
|
void plp() noexcept;
|
|
|
|
|
void rti() noexcept;
|
|
|
|
|
void rts() noexcept;
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
2024-01-07 14:04:41 +00:00
|
|
|
|
[[nodiscard]] constexpr uint8_t asl(uint8_t value) noexcept {
|
|
|
|
|
set_flag(CF, value & Bit7);
|
|
|
|
|
return through(value << 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[[nodiscard]] constexpr uint8_t rol(uint8_t operand) noexcept {
|
|
|
|
|
const auto carryIn = carry();
|
|
|
|
|
return through(asl(operand) | carryIn);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[[nodiscard]] constexpr uint8_t lsr(uint8_t value) noexcept {
|
|
|
|
|
set_flag(CF, value & Bit0);
|
|
|
|
|
return through(value >> 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[[nodiscard]] constexpr uint8_t ror(uint8_t operand) noexcept {
|
|
|
|
|
const auto carryIn = carry();
|
|
|
|
|
return through(lsr(operand) | (carryIn << 7));
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-29 22:22:31 +00:00
|
|
|
|
// Undocumented compound instructions
|
|
|
|
|
|
2021-07-18 14:28:40 +01:00
|
|
|
|
void anc(uint8_t value) noexcept;
|
|
|
|
|
void asr(uint8_t value) noexcept;
|
|
|
|
|
void axs(uint8_t value) noexcept;
|
2022-01-24 23:00:25 +00:00
|
|
|
|
void dcp(uint8_t value) noexcept;
|
|
|
|
|
void isb(uint8_t value) noexcept;
|
|
|
|
|
void rla(uint8_t value) noexcept;
|
|
|
|
|
void rra(uint8_t value) noexcept;
|
|
|
|
|
void slo(uint8_t value) noexcept;
|
|
|
|
|
void sre(uint8_t value) noexcept;
|
2024-01-06 22:44:59 +00:00
|
|
|
|
void jam() noexcept;
|
2018-12-29 22:22:31 +00:00
|
|
|
|
|
2019-01-06 20:39:37 +00:00
|
|
|
|
// Complicated addressing mode implementations
|
|
|
|
|
|
2022-01-24 23:00:25 +00:00
|
|
|
|
void sta_AbsoluteX() noexcept;
|
|
|
|
|
void sta_AbsoluteY() noexcept;
|
2023-12-31 14:32:16 +00:00
|
|
|
|
void sta_IndirectIndexedY() noexcept;
|
2019-01-06 20:39:37 +00:00
|
|
|
|
|
2024-01-05 12:52:27 +00:00
|
|
|
|
void sta_with_fixup(const register16_t address, const uint8_t unfixed_page) noexcept {
|
|
|
|
|
fixup(address, unfixed_page);
|
|
|
|
|
memoryWrite(A());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Undocumented complicated mode implementations
|
|
|
|
|
|
|
|
|
|
// SLO
|
|
|
|
|
void slo_AbsoluteX() noexcept;
|
|
|
|
|
void slo_AbsoluteY() noexcept;
|
|
|
|
|
void slo_IndirectIndexedY() noexcept;
|
|
|
|
|
void slo_with_fixup(const register16_t address, const uint8_t unfixed_page) noexcept {
|
|
|
|
|
fixup(address, unfixed_page);
|
|
|
|
|
slo(memoryRead());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ISB
|
|
|
|
|
void isb_AbsoluteX() noexcept;
|
|
|
|
|
void isb_AbsoluteY() noexcept;
|
|
|
|
|
void isb_IndirectIndexedY() noexcept;
|
|
|
|
|
void isb_with_fixup(const register16_t address, const uint8_t unfixed_page) noexcept {
|
|
|
|
|
fixup(address, unfixed_page);
|
|
|
|
|
isb(memoryRead());
|
2023-12-31 14:58:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-01-05 12:52:27 +00:00
|
|
|
|
// RLA
|
|
|
|
|
void rla_AbsoluteX() noexcept;
|
|
|
|
|
void rla_AbsoluteY() noexcept;
|
|
|
|
|
void rla_IndirectIndexedY() noexcept;
|
|
|
|
|
void rla_with_fixup(const register16_t address, const uint8_t unfixed_page) noexcept {
|
|
|
|
|
fixup(address, unfixed_page);
|
|
|
|
|
rla(memoryRead());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// RRA
|
|
|
|
|
void rra_AbsoluteX() noexcept;
|
|
|
|
|
void rra_AbsoluteY() noexcept;
|
|
|
|
|
void rra_IndirectIndexedY() noexcept;
|
|
|
|
|
void rra_with_fixup(const register16_t address, const uint8_t unfixed_page) noexcept {
|
|
|
|
|
fixup(address, unfixed_page);
|
|
|
|
|
rra(memoryRead());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DCP
|
|
|
|
|
void dcp_AbsoluteX() noexcept;
|
|
|
|
|
void dcp_AbsoluteY() noexcept;
|
|
|
|
|
void dcp_IndirectIndexedY() noexcept;
|
|
|
|
|
void dcp_with_fixup(const register16_t address, const uint8_t unfixed_page) noexcept {
|
|
|
|
|
fixup(address, unfixed_page);
|
|
|
|
|
dcp(memoryRead());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// SRE
|
|
|
|
|
void sre_AbsoluteX() noexcept;
|
|
|
|
|
void sre_AbsoluteY() noexcept;
|
|
|
|
|
void sre_IndirectIndexedY() noexcept;
|
|
|
|
|
void sre_with_fixup(const register16_t address, const uint8_t unfixed_page) noexcept {
|
|
|
|
|
fixup(address, unfixed_page);
|
|
|
|
|
sre(memoryRead());
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-06 09:52:17 +00:00
|
|
|
|
// SYA
|
|
|
|
|
void sya_AbsoluteX() noexcept;
|
|
|
|
|
|
|
|
|
|
// SXA
|
|
|
|
|
void sxa_AbsoluteY() noexcept;
|
|
|
|
|
|
2024-01-05 12:52:27 +00:00
|
|
|
|
// NOP
|
|
|
|
|
void nop_AbsoluteX() noexcept;
|
|
|
|
|
|
2018-01-18 17:50:15 +00:00
|
|
|
|
uint8_t x = 0; // index register X
|
|
|
|
|
uint8_t y = 0; // index register Y
|
|
|
|
|
uint8_t a = 0; // accumulator
|
|
|
|
|
uint8_t s = 0; // stack pointer
|
|
|
|
|
uint8_t p = 0; // processor status
|
2017-07-02 22:03:33 +01:00
|
|
|
|
|
2018-06-16 00:55:32 +01:00
|
|
|
|
register16_t m_intermediate;
|
2019-01-08 01:32:43 +00:00
|
|
|
|
|
|
|
|
|
bool m_handlingRESET = false;
|
|
|
|
|
bool m_handlingNMI = false;
|
2019-01-14 02:10:17 +00:00
|
|
|
|
bool m_handlingINT = false;
|
2017-07-02 22:03:33 +01:00
|
|
|
|
};
|
|
|
|
|
}
|