AxibugEmuOnline/AxibugEmuOnline.Client/Assets/Runtime/Core/CPU.Instructions.cs

464 lines
18 KiB
C#
Raw Normal View History

2024-06-28 18:08:25 +08:00
using System;
using static AxibugEmuOnline.Client.UNES.CPU.AddressingMode;
namespace AxibugEmuOnline.Client.UNES
{
public sealed partial class CPU
{
[AttributeUsage(AttributeTargets.Method, Inherited = false, AllowMultiple = true)]
public class OpCodeDef : Attribute
{
public int OpCode;
public int Cycles = 1;
public bool PageBoundary;
public bool RMW;
public AddressingMode Mode = None;
}
[OpCodeDef(OpCode = 0x20, Cycles = 6)]
private void JSR()
{
PushWord(PC + 1);
PC = NextWord();
}
[OpCodeDef(OpCode = 0x40, Cycles = 6)]
private void RTI()
{
// TODO: this dummy fetch should happen for all single-byte instructions
NextByte();
P = Pop();
PC = PopWord();
}
[OpCodeDef(OpCode = 0x60, Cycles = 6)]
private void RTS()
{
NextByte();
PC = PopWord() + 1;
}
[OpCodeDef(OpCode = 0xC8, Cycles = 2)]
private void INY() => Y++;
[OpCodeDef(OpCode = 0x88, Cycles = 2)]
private void DEY() => Y--;
[OpCodeDef(OpCode = 0xE8, Cycles = 2)]
private void INX() => X++;
[OpCodeDef(OpCode = 0xCA, Cycles = 2, RMW = true)]
private void DEX() => X--;
[OpCodeDef(OpCode = 0xA8, Cycles = 2)]
private void TAY() => Y = A;
[OpCodeDef(OpCode = 0x98, Cycles = 2)]
private void TYA() => A = Y;
[OpCodeDef(OpCode = 0xAA, Cycles = 2, RMW = true)]
private void TAX() => X = A;
[OpCodeDef(OpCode = 0x8A, Cycles = 2, RMW = true)]
private void TXA() => A = X;
[OpCodeDef(OpCode = 0xBA, Cycles = 2)]
private void TSX() => X = SP;
[OpCodeDef(OpCode = 0x9A, Cycles = 2, RMW = true)]
private void TXS() => SP = X;
[OpCodeDef(OpCode = 0x08, Cycles = 3)]
private void PHP() => Push(P | BreakSourceBit);
[OpCodeDef(OpCode = 0x28, Cycles = 4)]
private void PLP() => P = (uint)(Pop() & ~BreakSourceBit);
[OpCodeDef(OpCode = 0x68, Cycles = 4)]
private void PLA() => A = Pop();
[OpCodeDef(OpCode = 0x48, Cycles = 3)]
private void PHA() => Push(A);
[OpCodeDef(OpCode = 0x24, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0x2C, Mode = Absolute, Cycles = 4)]
private void BIT()
{
uint val = AddressRead();
F.Overflow = (val & 0x40) > 0;
F.Zero = (val & A) == 0;
F.Negative = (val & 0x80) > 0;
}
private void Branch(bool cond)
{
uint nPC = (uint)(PC + NextSByte() + 1);
if (cond)
{
PC = nPC;
Cycle++;
}
}
[OpCodeDef(OpCode = 0x4C, Cycles = 3)]
[OpCodeDef(OpCode = 0x6C, Cycles = 5)]
private void JMP()
{
if (_currentInstruction == 0x4C)
{
PC = NextWord();
}
else if (_currentInstruction == 0x6C)
{
uint off = NextWord();
// AN INDIRECT JUMP MUST NEVER USE A VECTOR BEGINNING ON THE LAST BYTE OF A PAGE
//
// If address $3000 contains $40, $30FF contains $80, and $3100 contains $50,
// the result of JMP ($30FF) will be a transfer of control to $4080 rather than
// $5080 as you intended i.e. the 6502 took the low byte of the address from
// $30FF and the high byte from $3000.
//
// http://www.6502.org/tutorials/6502opcodes.html
uint hi = (off & 0xFF) == 0xFF ? off - 0xFF : off + 1;
uint oldPC = PC;
PC = ReadByte(off) | (ReadByte(hi) << 8);
if ((oldPC & 0xFF00) != (PC & 0xFF00))
{
Cycle += 2;
}
}
else
{
throw new NotImplementedException();
}
}
[OpCodeDef(OpCode = 0xB0, Cycles = 2)]
private void BCS() => Branch(F.Carry);
[OpCodeDef(OpCode = 0x90, Cycles = 2)]
private void BCC() => Branch(!F.Carry);
[OpCodeDef(OpCode = 0xF0, Cycles = 2)]
private void BEQ() => Branch(F.Zero);
[OpCodeDef(OpCode = 0xD0, Cycles = 2)]
private void BNE() => Branch(!F.Zero);
[OpCodeDef(OpCode = 0x70, Cycles = 2)]
private void BVS() => Branch(F.Overflow);
[OpCodeDef(OpCode = 0x50, Cycles = 2)]
private void BVC() => Branch(!F.Overflow);
[OpCodeDef(OpCode = 0x10, Cycles = 2)]
private void BPL() => Branch(!F.Negative);
[OpCodeDef(OpCode = 0x30, Cycles = 2)]
private void BMI() => Branch(F.Negative);
[OpCodeDef(OpCode = 0x81, Mode = IndirectX, Cycles = 6)]
[OpCodeDef(OpCode = 0x91, Mode = IndirectY, Cycles = 6)]
[OpCodeDef(OpCode = 0x95, Mode = ZeroPageX, Cycles = 4)]
[OpCodeDef(OpCode = 0x99, Mode = AbsoluteY, Cycles = 5)]
[OpCodeDef(OpCode = 0x9D, Mode = AbsoluteX, Cycles = 5)]
[OpCodeDef(OpCode = 0x85, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0x8D, Mode = Absolute, Cycles = 4)]
private void STA() => AddressWrite(A);
[OpCodeDef(OpCode = 0x96, Mode = ZeroPageY, Cycles = 4)]
[OpCodeDef(OpCode = 0x86, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0x8E, Mode = Absolute, Cycles = 4)]
private void STX() => AddressWrite(X);
[OpCodeDef(OpCode = 0x94, Mode = ZeroPageX, Cycles = 4)]
[OpCodeDef(OpCode = 0x84, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0x8C, Mode = Absolute, Cycles = 4)]
private void STY() => AddressWrite(Y);
[OpCodeDef(OpCode = 0x18, Cycles = 2)]
private void CLC() => F.Carry = false;
[OpCodeDef(OpCode = 0x38, Cycles = 2)]
private void SEC() => F.Carry = true;
[OpCodeDef(OpCode = 0x58, Cycles = 2)]
private void CLI() => F.InterruptsDisabled = false;
[OpCodeDef(OpCode = 0x78, Cycles = 2)]
private void SEI() => F.InterruptsDisabled = true;
[OpCodeDef(OpCode = 0xB8, Cycles = 2)]
private void CLV() => F.Overflow = false;
[OpCodeDef(OpCode = 0xD8, Cycles = 2)]
private void CLD() => F.DecimalMode = false;
[OpCodeDef(OpCode = 0xF8, Cycles = 2)]
private void SED() => F.DecimalMode = true;
[OpCodeDef(OpCode = 0xEA, Cycles = 2)]
[OpCodeDef(OpCode = 0x1A, Cycles = 2)] // Unofficial
[OpCodeDef(OpCode = 0x3A, Cycles = 2)] // Unofficial
[OpCodeDef(OpCode = 0x5A, Cycles = 2)] // Unofficial
[OpCodeDef(OpCode = 0x7A, Cycles = 2)] // Unofficial
[OpCodeDef(OpCode = 0xDA, Cycles = 2)] // Unofficial
[OpCodeDef(OpCode = 0xFA, Cycles = 2)] // Unofficial
private void NOP() { }
[OpCodeDef(OpCode = 0xA1, Mode = IndirectX, Cycles = 6)]
[OpCodeDef(OpCode = 0xA5, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0xA9, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0xAD, Mode = Absolute, Cycles = 4)]
[OpCodeDef(OpCode = 0xB1, Mode = IndirectY, Cycles = 5, PageBoundary = true)]
[OpCodeDef(OpCode = 0xB5, Mode = ZeroPageX, Cycles = 4)]
[OpCodeDef(OpCode = 0xB9, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)]
[OpCodeDef(OpCode = 0xBD, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)]
private void LDA() => A = AddressRead();
[OpCodeDef(OpCode = 0xA0, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0xA4, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0xAC, Mode = Absolute, Cycles = 4)]
[OpCodeDef(OpCode = 0xB4, Mode = ZeroPageX, Cycles = 4)]
[OpCodeDef(OpCode = 0xBC, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)]
private void LDY() => Y = AddressRead();
[OpCodeDef(OpCode = 0xA2, Mode = Immediate, Cycles = 2, RMW = true)]
[OpCodeDef(OpCode = 0xA6, Mode = ZeroPage, Cycles = 3, RMW = true)]
[OpCodeDef(OpCode = 0xAE, Mode = Absolute, Cycles = 4, RMW = true)]
[OpCodeDef(OpCode = 0xB6, Mode = ZeroPageY, Cycles = 4, RMW = true)]
[OpCodeDef(OpCode = 0xBE, Mode = AbsoluteY, Cycles = 4, PageBoundary = true, RMW = true)]
private void LDX() => X = AddressRead();
[OpCodeDef(OpCode = 0x01, Mode = IndirectX, Cycles = 6)]
[OpCodeDef(OpCode = 0x05, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0x09, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0x0D, Mode = Absolute, Cycles = 4)]
[OpCodeDef(OpCode = 0x11, Mode = IndirectY, Cycles = 5, PageBoundary = true)]
[OpCodeDef(OpCode = 0x15, Mode = ZeroPageX, Cycles = 4)]
[OpCodeDef(OpCode = 0x19, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)]
[OpCodeDef(OpCode = 0x1D, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)]
private void ORA() => A |= AddressRead();
[OpCodeDef(OpCode = 0x21, Mode = IndirectX, Cycles = 6)]
[OpCodeDef(OpCode = 0x25, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0x29, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0x2D, Mode = Absolute, Cycles = 4)]
[OpCodeDef(OpCode = 0x31, Mode = IndirectY, Cycles = 5, PageBoundary = true)]
[OpCodeDef(OpCode = 0x35, Mode = ZeroPageX, Cycles = 4)]
[OpCodeDef(OpCode = 0x39, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)]
[OpCodeDef(OpCode = 0x3D, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)]
private void AND() => A &= AddressRead();
[OpCodeDef(OpCode = 0x41, Mode = IndirectX, Cycles = 6)]
[OpCodeDef(OpCode = 0x45, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0x49, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0x4D, Mode = Absolute, Cycles = 4)]
[OpCodeDef(OpCode = 0x51, Mode = IndirectY, Cycles = 5, PageBoundary = true)]
[OpCodeDef(OpCode = 0x55, Mode = ZeroPageX, Cycles = 4)]
[OpCodeDef(OpCode = 0x59, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)]
[OpCodeDef(OpCode = 0x5D, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)]
private void EOR() => A ^= AddressRead();
[OpCodeDef(OpCode = 0xE1, Mode = IndirectX, Cycles = 6)]
[OpCodeDef(OpCode = 0xE5, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0x69, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0xE9, Mode = Immediate, Cycles = 2)] // Official duplicate of $69
[OpCodeDef(OpCode = 0xEB, Mode = Immediate, Cycles = 2)] // Unofficial duplicate of $69
[OpCodeDef(OpCode = 0xED, Mode = Absolute, Cycles = 4)]
[OpCodeDef(OpCode = 0xF1, Mode = IndirectY, Cycles = 5, PageBoundary = true)]
[OpCodeDef(OpCode = 0xF5, Mode = ZeroPageX, Cycles = 4)]
[OpCodeDef(OpCode = 0xF9, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)]
[OpCodeDef(OpCode = 0xFD, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)]
private void SBC() => ADCImpl((byte)~AddressRead());
[OpCodeDef(OpCode = 0x61, Mode = IndirectX, Cycles = 6)]
[OpCodeDef(OpCode = 0x65, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0x69, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0x6D, Mode = Absolute, Cycles = 4)]
[OpCodeDef(OpCode = 0x71, Mode = IndirectY, Cycles = 5, PageBoundary = true)]
[OpCodeDef(OpCode = 0x75, Mode = ZeroPageX, Cycles = 4)]
[OpCodeDef(OpCode = 0x79, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)]
[OpCodeDef(OpCode = 0x7D, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)]
private void ADC() => ADCImpl(AddressRead());
private void ADCImpl(uint val)
{
int nA = (sbyte)A + (sbyte)val + (sbyte)(F.Carry ? 1 : 0);
F.Overflow = nA < -128 || nA > 127;
F.Carry = (A + val + (F.Carry ? 1 : 0)) > 0xFF;
A = (byte)(nA & 0xFF);
}
[OpCodeDef(OpCode = 0x00, Cycles = 7)]
private void BRK()
{
NextByte();
Push(P | BreakSourceBit);
F.InterruptsDisabled = true;
PC = ReadByte(0xFFFE) | (ReadByte(0xFFFF) << 8);
}
[OpCodeDef(OpCode = 0xC1, Mode = IndirectX, Cycles = 6)]
[OpCodeDef(OpCode = 0xC5, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0xC9, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0xCD, Mode = Absolute, Cycles = 4)]
[OpCodeDef(OpCode = 0xD1, Mode = IndirectY, Cycles = 5, PageBoundary = true)]
[OpCodeDef(OpCode = 0xD5, Mode = ZeroPageX, Cycles = 4)]
[OpCodeDef(OpCode = 0xD9, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)]
[OpCodeDef(OpCode = 0xDD, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)]
private void CMP() => CMPImpl(A);
[OpCodeDef(OpCode = 0xE0, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0xE4, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0xEC, Mode = Absolute, Cycles = 4)]
private void CPX() => CMPImpl(X);
[OpCodeDef(OpCode = 0xC0, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0xC4, Mode = ZeroPage, Cycles = 3)]
[OpCodeDef(OpCode = 0xCC, Mode = Absolute, Cycles = 4)]
private void CPY() => CMPImpl(Y);
private void CMPImpl(uint reg)
{
long d = reg - (int)AddressRead();
F.Negative = (d & 0x80) > 0 && d != 0;
F.Carry = d >= 0;
F.Zero = d == 0;
}
[OpCodeDef(OpCode = 0x46, Mode = ZeroPage, Cycles = 5, RMW = true)]
[OpCodeDef(OpCode = 0x4E, Mode = Absolute, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0x56, Mode = ZeroPageX, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0x5E, Mode = AbsoluteX, Cycles = 7, RMW = true)]
[OpCodeDef(OpCode = 0x4A, Mode = Direct, Cycles = 2, RMW = true)]
private void LSR()
{
uint D = AddressRead();
F.Carry = (D & 0x1) > 0;
D >>= 1;
_F(D);
AddressWrite(D);
}
[OpCodeDef(OpCode = 0x06, Mode = ZeroPage, Cycles = 5, RMW = true)]
[OpCodeDef(OpCode = 0x0E, Mode = Absolute, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0x16, Mode = ZeroPageX, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0x1E, Mode = AbsoluteX, Cycles = 7, RMW = true)]
[OpCodeDef(OpCode = 0x0A, Mode = Direct, Cycles = 2, RMW = true)]
private void ASL()
{
uint D = AddressRead();
F.Carry = (D & 0x80) > 0;
D <<= 1;
_F(D);
AddressWrite(D);
}
[OpCodeDef(OpCode = 0x66, Mode = ZeroPage, Cycles = 5, RMW = true)]
[OpCodeDef(OpCode = 0x6E, Mode = Absolute, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0x76, Mode = ZeroPageX, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0x7E, Mode = AbsoluteX, Cycles = 7, RMW = true)]
[OpCodeDef(OpCode = 0x6A, Mode = Direct, Cycles = 2, RMW = true)]
private void ROR()
{
uint D = AddressRead();
bool c = F.Carry;
F.Carry = (D & 0x1) > 0;
D >>= 1;
if (c) D |= 0x80;
_F(D);
AddressWrite(D);
}
[OpCodeDef(OpCode = 0x26, Mode = ZeroPage, Cycles = 5, RMW = true)]
[OpCodeDef(OpCode = 0x2E, Mode = Absolute, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0x36, Mode = ZeroPageX, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0x3E, Mode = AbsoluteX, Cycles = 7, RMW = true)]
[OpCodeDef(OpCode = 0x2A, Mode = Direct, Cycles = 2, RMW = true)]
private void ROL()
{
uint D = AddressRead();
bool c = F.Carry;
F.Carry = (D & 0x80) > 0;
D <<= 1;
if (c) D |= 0x1;
_F(D);
AddressWrite(D);
}
[OpCodeDef(OpCode = 0xE6, Mode = ZeroPage, Cycles = 5, RMW = true)]
[OpCodeDef(OpCode = 0xEE, Mode = Absolute, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0xF6, Mode = ZeroPageX, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0xFE, Mode = AbsoluteX, Cycles = 7, RMW = true)]
private void INC()
{
byte D = (byte)(AddressRead() + 1);
_F(D);
AddressWrite(D);
}
[OpCodeDef(OpCode = 0xC6, Mode = ZeroPage, Cycles = 5, RMW = true)]
[OpCodeDef(OpCode = 0xCE, Mode = Absolute, Cycles = 3, RMW = true)]
[OpCodeDef(OpCode = 0xD6, Mode = ZeroPageX, Cycles = 6, RMW = true)]
[OpCodeDef(OpCode = 0xDE, Mode = AbsoluteX, Cycles = 7, RMW = true)]
private void DEC()
{
byte D = (byte)(AddressRead() - 1);
_F(D);
AddressWrite(D);
}
#region Unofficial Opcodes
[OpCodeDef(OpCode = 0x80, Cycles = 2)]
[OpCodeDef(OpCode = 0x82, Cycles = 2)]
[OpCodeDef(OpCode = 0x89, Cycles = 2)]
[OpCodeDef(OpCode = 0xC2, Cycles = 2)]
[OpCodeDef(OpCode = 0xE2, Cycles = 2)]
private void SKB() => NextByte(); // Essentially a 2-byte NOP
[OpCodeDef(OpCode = 0x0B, Mode = Immediate, Cycles = 2)]
[OpCodeDef(OpCode = 0x2B, Mode = Immediate, Cycles = 2)]
private void ANC()
{
A &= AddressRead();
F.Carry = F.Negative;
}
[OpCodeDef(OpCode = 0x4B, Mode = Immediate, Cycles = 2)]
private void ALR()
{
A &= AddressRead();
F.Carry = (A & 0x1) > 0;
A >>= 1;
_F(A);
}
[OpCodeDef(OpCode = 0x6B, Mode = Immediate, Cycles = 2)]
private void ARR()
{
A &= AddressRead();
bool c = F.Carry;
F.Carry = (A & 0x1) > 0;
A >>= 1;
if (c) A |= 0x80;
_F(A);
}
[OpCodeDef(OpCode = 0xAB, Mode = Immediate, Cycles = 2)]
private void ATX()
{
// This opcode ORs the A register with #$EE, ANDs the result with an immediate
// value, and then stores the result in both A and X.
A |= ReadByte(0xEE);
A &= AddressRead();
X = A;
}
#endregion
}
}