mirror of
https://github.com/TomHarte/CLK.git
synced 2024-11-20 15:31:10 +00:00
d2e868ea2b
Now covered: 146/256 opcodes, 4/7 pages, 25/47 bus programs.
163 lines
4.7 KiB
C++
163 lines
4.7 KiB
C++
//
|
||
// 65816Implementation.hpp
|
||
// Clock Signal
|
||
//
|
||
// Created by Thomas Harte on 23/09/2020.
|
||
// Copyright © 2020 Thomas Harte. All rights reserved.
|
||
//
|
||
|
||
#ifndef WDC65816Implementation_h
|
||
#define WDC65816Implementation_h
|
||
|
||
enum MicroOp: uint8_t {
|
||
/// Fetches a byte from the program counter to the instruction buffer and increments the program counter.
|
||
CycleFetchIncrementPC,
|
||
/// Fetches a byte from the program counter without incrementing it, and throws it away.
|
||
CycleFetchPC,
|
||
|
||
/// Fetches a byte from the data address to the data buffer.
|
||
CycleFetchData,
|
||
/// Fetches a byte from the data address to the data buffer and increments the data address.
|
||
CycleFetchIncrementData,
|
||
/// Fetches from the address formed by the low byte of the data address and the high byte
|
||
/// of the instruction buffer, throwing the result away.
|
||
CycleFetchIncorrectDataAddress,
|
||
|
||
// Dedicated block-move cycles; these use the data buffer as an intermediary.
|
||
CycleFetchBlockX,
|
||
CycleFetchBlockY,
|
||
CycleStoreBlockY,
|
||
|
||
/// Stores a byte from the data buffer.
|
||
CycleStoreData,
|
||
/// Stores a byte to the data address from the data buffer and increments the data address.
|
||
CycleStoreIncrementData,
|
||
/// Stores a byte to the data address from the data buffer and decrements the data address.
|
||
CycleStoreDecrementData,
|
||
|
||
/// Pushes a single byte from the data buffer to the stack.
|
||
CyclePush,
|
||
/// Fetches from the current stack location and throws the result away.
|
||
CycleAccessStack,
|
||
|
||
/// Sets the data address by copying the final two bytes of the instruction buffer.
|
||
OperationConstructAbsolute,
|
||
/// Sets the data address to the result of (a, x).
|
||
/// TODO: explain better once implemented.
|
||
OperationConstructAbsoluteIndexedIndirect,
|
||
OperationConstructAbsoluteLongX,
|
||
|
||
/// Calculates an a, x address; if:
|
||
/// there was no carry into the top byte of the address; and
|
||
/// the process or in emulation or 8-bit index mode;
|
||
/// then it also skips the next micro-op.
|
||
OperationConstructAbsoluteXRead,
|
||
|
||
/// Calculates an a, x address.
|
||
OperationConstructAbsoluteX,
|
||
|
||
// These are analogous to the X versions above.
|
||
OperationConstructAbsoluteY,
|
||
OperationConstructAbsoluteYRead,
|
||
|
||
/// Constructs the current direct address using the value in the instruction buffer.
|
||
/// Skips the next micro-op if the low byte of the direct register is 0.
|
||
OperationConstructDirect,
|
||
|
||
// These follow similar skip-one-if-possible logic to OperationConstructDirect.
|
||
OperationConstructDirectIndexedIndirect,
|
||
OperationConstructDirectIndirect,
|
||
OperationConstructDirectIndirectIndexed,
|
||
OperationConstructDirectIndirectIndexedLong,
|
||
OperationConstructDirectIndirectLong,
|
||
|
||
/// Performs whatever operation goes with this program.
|
||
OperationPerform,
|
||
|
||
/// Copies the current program counter to the data buffer.
|
||
OperationCopyPCToData,
|
||
|
||
/// Copies the current PBR to the data buffer.
|
||
OperationCopyPBRToData,
|
||
|
||
OperationCopyAToData,
|
||
OperationCopyDataToA,
|
||
|
||
/// Complete this set of micr-ops.
|
||
OperationMoveToNextProgram
|
||
};
|
||
|
||
enum Operation: uint8_t {
|
||
// These perform the named operation using the value in the data buffer;
|
||
// they are implicitly AccessType::Read.
|
||
ADC, AND, BIT, CMP, CPX, CPY, EOR, ORA, SBC,
|
||
|
||
// These load the respective register from the data buffer;
|
||
// they are implicitly AccessType::Read.
|
||
LDA, LDX, LDY,
|
||
|
||
// These move the respective register (or value) to the data buffer;
|
||
// they are implicitly AccessType::Write.
|
||
STA, STX, STY, STZ,
|
||
|
||
// These modify the value in the data buffer as part of a read-modify-write.
|
||
ASL, DEC, INC, LSR, ROL, ROR, TRB, TSB,
|
||
|
||
// These merely decrement A, increment or decrement X and Y, and regress
|
||
// the program counter only if appropriate.
|
||
MVN, MVP,
|
||
|
||
/// Loads the PC with the operand from the data buffer.
|
||
JMP,
|
||
|
||
/// Loads the PC and PBR with the operand from the data buffer.
|
||
JML,
|
||
|
||
/// Loads the PC with the operand from the data buffer, replacing
|
||
/// it with the old PC.
|
||
JSR,
|
||
|
||
/// Loads the PC and the PBR with the operand from the data buffer,
|
||
/// replacing it with the old PC (and only the PC; PBR not included).
|
||
JSL,
|
||
};
|
||
|
||
class ProcessorStorageConstructor;
|
||
|
||
class ProcessorStorage {
|
||
public:
|
||
ProcessorStorage();
|
||
|
||
struct Instruction {
|
||
size_t program_offset;
|
||
Operation operation;
|
||
};
|
||
Instruction instructions[512 + 3]; // Arranged as:
|
||
// 256 entries: emulation-mode instructions;
|
||
// 256 entries: 16-bit instructions;
|
||
// reset
|
||
// NMI
|
||
// IRQ
|
||
|
||
private:
|
||
friend ProcessorStorageConstructor;
|
||
|
||
// Registers.
|
||
RegisterPair16 a_;
|
||
RegisterPair16 x_, y_;
|
||
uint16_t pc_, s_;
|
||
|
||
// Not
|
||
uint16_t direct_;
|
||
|
||
// Banking registers are all stored with the relevant byte
|
||
// shifted up bits 16–23.
|
||
uint32_t data_bank_; // i.e. DBR.
|
||
uint32_t program_bank_; // i.e. PBR.
|
||
|
||
|
||
std::vector<MicroOp> micro_ops_;
|
||
};
|
||
|
||
#endif /* WDC65816Implementation_h */
|