mirror of
https://github.com/MoleskiCoder/EightBit.git
synced 2025-01-13 15:30:33 +00:00
7c3fc469a8
Signed-off-by: Adrian.Conlon <adrian.conlon@gmail.com>
207 lines
4.9 KiB
C++
207 lines
4.9 KiB
C++
#pragma once
|
|
|
|
#include <cstdint>
|
|
|
|
#include "IntelProcessor.h"
|
|
#include "Bus.h"
|
|
#include "Signal.h"
|
|
|
|
namespace EightBit {
|
|
class LR35902 : public IntelProcessor {
|
|
public:
|
|
enum StatusBits {
|
|
ZF = Bit7,
|
|
NF = Bit6,
|
|
HC = Bit5,
|
|
CF = Bit4,
|
|
};
|
|
|
|
LR35902(Bus& memory);
|
|
|
|
Signal<LR35902> ExecutingInstruction;
|
|
|
|
void stop() { m_stopped = true; }
|
|
void start() { m_stopped = false; }
|
|
bool stopped() const { return m_stopped; }
|
|
|
|
bool& IME() { return m_ime; }
|
|
|
|
void di();
|
|
void ei();
|
|
|
|
int interrupt(uint8_t value);
|
|
|
|
int execute(uint8_t opcode);
|
|
int step();
|
|
|
|
// Mutable access to processor!!
|
|
|
|
virtual register16_t& AF() override {
|
|
m_accumulatorFlag.low &= 0xf0;
|
|
return m_accumulatorFlag;
|
|
}
|
|
|
|
virtual register16_t& BC() override {
|
|
return m_registers[BC_IDX];
|
|
}
|
|
|
|
virtual register16_t& DE() override {
|
|
return m_registers[DE_IDX];
|
|
}
|
|
|
|
virtual register16_t& HL() override {
|
|
return m_registers[HL_IDX];
|
|
}
|
|
|
|
virtual void reset();
|
|
virtual void initialise();
|
|
|
|
protected:
|
|
virtual uint8_t fetchByte() {
|
|
auto returned = IntelProcessor::fetchByte();
|
|
m_memory.fireReadBusEvent();
|
|
return returned;
|
|
}
|
|
|
|
virtual void push(uint8_t value) {
|
|
IntelProcessor::push(value);
|
|
m_memory.fireWriteBusEvent();
|
|
}
|
|
|
|
virtual uint8_t pop() {
|
|
auto returned = IntelProcessor::pop();
|
|
m_memory.fireReadBusEvent();
|
|
return returned;
|
|
}
|
|
|
|
virtual void getWordViaMemptr(register16_t& value) {
|
|
value.low = memptrReference();
|
|
m_memory.fireReadBusEvent();
|
|
m_memory.ADDRESS().word++;
|
|
value.high = m_memory.reference();
|
|
m_memory.fireReadBusEvent();
|
|
}
|
|
|
|
virtual void setWordViaMemptr(register16_t value) {
|
|
memptrReference() = value.low;
|
|
m_memory.fireWriteBusEvent();
|
|
m_memory.ADDRESS().word++;
|
|
m_memory.reference() = value.high;
|
|
m_memory.fireWriteBusEvent();
|
|
}
|
|
|
|
private:
|
|
enum { BC_IDX, DE_IDX, HL_IDX };
|
|
|
|
Bus& m_bus;
|
|
|
|
std::array<register16_t, 3> m_registers;
|
|
register16_t m_accumulatorFlag;
|
|
|
|
bool m_ime;
|
|
|
|
bool m_prefixCB;
|
|
|
|
bool m_stopped;
|
|
|
|
int fetchExecute() {
|
|
return execute(fetchByte());
|
|
}
|
|
|
|
uint8_t& R(int r, uint8_t& a) {
|
|
switch (r) {
|
|
case 0:
|
|
return B();
|
|
case 1:
|
|
return C();
|
|
case 2:
|
|
return D();
|
|
case 3:
|
|
return E();
|
|
case 4:
|
|
return H();
|
|
case 5:
|
|
return L();
|
|
case 6:
|
|
m_memory.ADDRESS() = HL();
|
|
return m_memory.reference();
|
|
case 7:
|
|
return a;
|
|
}
|
|
throw std::logic_error("Unhandled registry mechanism");
|
|
}
|
|
|
|
register16_t& RP(int rp) {
|
|
switch (rp) {
|
|
case 3:
|
|
return SP();
|
|
default:
|
|
return m_registers[rp];
|
|
}
|
|
}
|
|
|
|
register16_t& RP2(int rp) {
|
|
switch (rp) {
|
|
case 3:
|
|
return AF();
|
|
default:
|
|
return m_registers[rp];
|
|
}
|
|
}
|
|
|
|
static void adjustHalfCarryAdd(uint8_t& f, uint8_t before, uint8_t value, int calculation) {
|
|
setFlag(f, HC, calculateHalfCarryAdd(before, value, calculation));
|
|
}
|
|
|
|
static void adjustHalfCarrySub(uint8_t& f, uint8_t before, uint8_t value, int calculation) {
|
|
setFlag(f, HC, calculateHalfCarrySub(before, value, calculation));
|
|
}
|
|
|
|
static void subtract(uint8_t& f, uint8_t& operand, uint8_t value, int carry = 0);
|
|
|
|
void executeCB(int x, int y, int z, int p, int q);
|
|
void executeOther(int x, int y, int z, int p, int q);
|
|
|
|
static void postIncrement(uint8_t& f, uint8_t value);
|
|
static void postDecrement(uint8_t& f, uint8_t value);
|
|
|
|
void reti();
|
|
|
|
bool jrConditionalFlag(uint8_t& f, int flag);
|
|
bool returnConditionalFlag(uint8_t& f, int flag);
|
|
bool jumpConditionalFlag(uint8_t& f, int flag);
|
|
bool callConditionalFlag(uint8_t& f, int flag);
|
|
|
|
void sbc(uint8_t& f, register16_t& operand, register16_t value);
|
|
void adc(uint8_t& f, register16_t& operand, register16_t value);
|
|
void add(uint8_t& f, register16_t& operand, register16_t value);
|
|
|
|
static void add(uint8_t& f, uint8_t& operand, uint8_t value, int carry = 0);
|
|
static void adc(uint8_t& f, uint8_t& operand, uint8_t value);
|
|
static void sbc(uint8_t& f, uint8_t& operand, uint8_t value);
|
|
static void andr(uint8_t& f, uint8_t& operand, uint8_t value);
|
|
static void xorr(uint8_t& f, uint8_t& operand, uint8_t value);
|
|
static void orr(uint8_t& f, uint8_t& operand, uint8_t value);
|
|
static void compare(uint8_t& f, uint8_t check, uint8_t value);
|
|
|
|
static uint8_t& rlc(uint8_t& f, uint8_t& operand);
|
|
static uint8_t& rrc(uint8_t& f, uint8_t& operand);
|
|
static uint8_t& rl(uint8_t& f, uint8_t& operand);
|
|
static uint8_t& rr(uint8_t& f, uint8_t& operand);
|
|
static uint8_t& sla(uint8_t& f, uint8_t& operand);
|
|
static uint8_t& sra(uint8_t& f, uint8_t& operand);
|
|
static uint8_t& srl(uint8_t& f, uint8_t& operand);
|
|
|
|
static uint8_t& bit(uint8_t& f, int n, uint8_t& operand);
|
|
static uint8_t& res(int n, uint8_t& operand);
|
|
static uint8_t& set(int n, uint8_t& operand);
|
|
|
|
static void daa(uint8_t& a, uint8_t& f);
|
|
|
|
static void scf(uint8_t& a, uint8_t& f);
|
|
static void ccf(uint8_t& a, uint8_t& f);
|
|
static void cpl(uint8_t& a, uint8_t& f);
|
|
|
|
uint8_t& swap(uint8_t& f, uint8_t& operand);
|
|
};
|
|
} |