// // 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 _bus; //! @brief Execute a single instruction. //! @return The number of CPU cycles that the instruction took. int executeInstruction(); public: explicit CPU(std::shared_ptr 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