mirror of
https://github.com/zoriya/ComSquare.git
synced 2025-12-20 06:05:11 +00:00
211 lines
5.4 KiB
C++
211 lines
5.4 KiB
C++
//
|
|
// Created by anonymus-raccoon on 1/24/20.
|
|
//
|
|
|
|
#ifndef COMSQUARE_CPU_HPP
|
|
#define COMSQUARE_CPU_HPP
|
|
|
|
#include "../Memory/IMemory.hpp"
|
|
#include "../Memory/MemoryBus.hpp"
|
|
#include "../Models/Ints.hpp"
|
|
#include "CommonCpu.hpp"
|
|
|
|
namespace ComSquare::CPU
|
|
{
|
|
//! @brief Struct containing registers for the main CPU.
|
|
struct Registers {
|
|
//! @brief The Accumulator
|
|
union {
|
|
struct {
|
|
uint8_t ah;
|
|
uint8_t al;
|
|
};
|
|
uint16_t a;
|
|
};
|
|
//! @brief The Data Bank Register;
|
|
uint8_t dbr;
|
|
//! @brief The Direct register;
|
|
union {
|
|
struct {
|
|
uint8_t dh;
|
|
uint8_t dl;
|
|
};
|
|
uint16_t d;
|
|
};
|
|
//! @brief The program banK register;
|
|
uint8_t k;
|
|
//! @brief The Program Counter;
|
|
union {
|
|
struct {
|
|
uint8_t pch;
|
|
uint8_t pcl;
|
|
};
|
|
uint16_t pc;
|
|
};
|
|
//! @brief The Stack pointer
|
|
union {
|
|
struct {
|
|
uint8_t sh;
|
|
uint8_t sl;
|
|
};
|
|
uint16_t s;
|
|
};
|
|
//! @brief The X index register
|
|
union {
|
|
struct {
|
|
uint8_t xh;
|
|
uint8_t xl;
|
|
};
|
|
uint16_t x;
|
|
};
|
|
//! @brief The Y index register
|
|
union {
|
|
struct {
|
|
uint8_t yh;
|
|
uint8_t yl;
|
|
};
|
|
uint16_t y;
|
|
};
|
|
|
|
//! @brief The Processor status register;
|
|
union p {
|
|
//! @brief The Negative flag
|
|
bool n : 1;
|
|
//! @brief The oVerflow flag
|
|
bool v : 1;
|
|
//! @brief The accumulator and Memory width flag (in native mode only)
|
|
bool m : 1;
|
|
union {
|
|
//! @brief The indeX register width flag (in native mode only)
|
|
bool x : 1;
|
|
//! @brief The Break flag (in emulation mode only)
|
|
bool b : 1;
|
|
};
|
|
//! @brief The Decimal mode flag
|
|
bool d : 1;
|
|
//! @brief The Interrupt disable flag
|
|
bool i : 1;
|
|
//! @brief The Zero flag
|
|
bool z : 1;
|
|
//! @brief The Carry flag
|
|
bool c : 1;
|
|
};
|
|
};
|
|
|
|
//! @brief Struct containing internal registers of the CPU.
|
|
struct InternalRegisters
|
|
{
|
|
//! @brief Interrupt Enable Register
|
|
uint8_t nmitimen;
|
|
|
|
//! @brief IO Port Write Register
|
|
uint8_t wrio;
|
|
|
|
//! @brief Multiplicand Register A
|
|
uint8_t wrmpya;
|
|
//! @brief Multiplicand Register B
|
|
uint8_t wrmpyb;
|
|
|
|
//! @brief Divisor & Dividend Registers (A - Low)
|
|
uint8_t wrdivl;
|
|
//! @brief Divisor & Dividend Registers (A - High)
|
|
uint8_t wrdivh;
|
|
//! @brief Divisor & Dividend Registers (B)
|
|
uint8_t wrdivb;
|
|
|
|
//! @brief IRQ Timer Registers (Horizontal - Low)
|
|
uint8_t htimel;
|
|
//! @brief IRQ Timer Registers (Horizontal - High)
|
|
uint8_t htimeh;
|
|
|
|
//! @brief IRQ Timer Registers (Vertical - Low)
|
|
uint8_t vtimel;
|
|
//! @brief IRQ Timer Registers (Vertical - High)
|
|
uint8_t vtimeh;
|
|
|
|
//! @brief DMA Enable Register
|
|
uint8_t mdmaen;
|
|
|
|
//! @brief HDMA Enable Register
|
|
uint8_t hdmaen;
|
|
|
|
//! @brief ROM Speed Register
|
|
uint8_t memsel;
|
|
|
|
//! @brief Interrupt Flag Registers
|
|
uint8_t rdnmi;
|
|
//! @brief Interrupt Flag Registers - TimeUp
|
|
uint8_t timeup;
|
|
|
|
//! @brief PPU Status Register
|
|
uint8_t hvbjoy;
|
|
|
|
//! @brief IO Port Read Register
|
|
uint8_t rdio;
|
|
|
|
//! @brief Divide Result Registers (can sometimes be used as multiplication result register) - LOW
|
|
uint8_t rddivl;
|
|
//! @brief Divide Result Registers (can sometimes be used as multiplication result register) - HIGH
|
|
uint8_t rddivh;
|
|
|
|
//! @brief Multiplication Result Registers (can sometimes be used as divide result register) - LOW
|
|
uint8_t rdmpyl;
|
|
//! @brief Multiplication Result Registers (can sometimes be used as divide result register) - HIGH
|
|
uint8_t rdmpyh;
|
|
|
|
//! @brief Controller Port Data Registers (Pad 1 - Low)
|
|
uint8_t joy1l;
|
|
//! @brief Controller Port Data Registers (Pad 1 - High)
|
|
uint8_t joy1h;
|
|
|
|
//! @brief Controller Port Data Registers (Pad 2 - Low)
|
|
uint8_t joy2l;
|
|
//! @brief Controller Port Data Registers (Pad 2 - High)
|
|
uint8_t joy2h;
|
|
|
|
//! @brief Controller Port Data Registers (Pad 3 - Low)
|
|
uint8_t joy3l;
|
|
//! @brief Controller Port Data Registers (Pad 3 - High)
|
|
uint8_t joy3h;
|
|
|
|
//! @brief Controller Port Data Registers (Pad 4 - Low)
|
|
uint8_t joy4l;
|
|
//! @brief Controller Port Data Registers (Pad 4 - High)
|
|
uint8_t joy4h;
|
|
};
|
|
|
|
//! @brief The main CPU
|
|
class CPU : public CommonCPU, public Memory::IMemory {
|
|
private:
|
|
//! @brief All the registers of the CPU
|
|
Registers _registers{};
|
|
//! @brief Is the CPU running in emulation mode (in 8bits)
|
|
bool _isEmulationMode = true;
|
|
//! @brief Internal registers of the CPU (accessible from the bus via addr $4200 to $421F).
|
|
InternalRegisters _internalRegisters{};
|
|
//! @brief The memory bus to use for read/write.
|
|
std::shared_ptr<Memory::MemoryBus> _bus;
|
|
|
|
//! @brief Execute a single instruction.
|
|
//! @return The number of CPU cycles that the instruction took.
|
|
int executeInstruction();
|
|
public:
|
|
explicit CPU(std::shared_ptr<Memory::MemoryBus> bus);
|
|
//! @brief This function continue to execute the Cartridge code.
|
|
//! @return The number of CPU cycles that elapsed
|
|
int update();
|
|
//! @brief Read from the internal CPU register.
|
|
//! @param addr The address to read from. The address 0x0 should refer to the first byte of the register.
|
|
//! @throw InvalidAddress will be thrown if the address is more than $1F (the number of register).
|
|
//! @return Return the value of the register.
|
|
uint8_t read(uint24_t addr) override;
|
|
//! @brief Write data to the internal CPU register.
|
|
//! @param addr The address to write to. The address 0x0 should refer to the first byte of register.
|
|
//! @param data The new value of the register.
|
|
//! @throw InvalidAddress will be thrown if the address is more than $1F (the number of register).
|
|
void write(uint24_t addr, uint8_t data) override;
|
|
};
|
|
}
|
|
|
|
#endif //COMSQUARE_CPU_HPP
|