mirror of https://github.com/stella-emu/stella.git
4822 lines
90 KiB
TeX
4822 lines
90 KiB
TeX
//============================================================================
|
|
//
|
|
// MM MM 6666 555555 0000 2222
|
|
// MMMM MMMM 66 66 55 00 00 22 22
|
|
// MM MMM MM 66 55 00 00 22
|
|
// MM M MM 66666 55555 00 00 22222 -- "A 6502 Microprocessor Emulator"
|
|
// MM MM 66 66 55 00 00 22
|
|
// MM MM 66 66 55 55 00 00 22
|
|
// MM MM 6666 5555 0000 222222
|
|
//
|
|
// Copyright (c) 1995-2019 by Bradford W. Mott, Stephen Anthony
|
|
// and the Stella Team
|
|
//
|
|
// See the file "License.txt" for information on usage and redistribution of
|
|
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
|
//============================================================================
|
|
|
|
/**
|
|
Code and cases to emulate each of the 6502 instructions.
|
|
|
|
Recompile with the following:
|
|
'm4 M6502.m4 > M6502.ins'
|
|
|
|
@author Bradford W. Mott and Stephen Anthony
|
|
*/
|
|
|
|
#ifndef NOTSAMEPAGE
|
|
#define NOTSAMEPAGE(_addr1, _addr2) (((_addr1) ^ (_addr2)) & 0xff00)
|
|
#endif
|
|
|
|
#ifndef SET_LAST_PEEK
|
|
#ifdef DEBUGGER_SUPPORT
|
|
#define SET_LAST_PEEK(_addr1, _addr2) _addr1 = _addr2;
|
|
#else
|
|
#define SET_LAST_PEEK(_addr1, _addr2)
|
|
#endif
|
|
#endif
|
|
|
|
#ifndef CLEAR_LAST_PEEK
|
|
#ifdef DEBUGGER_SUPPORT
|
|
#define CLEAR_LAST_PEEK(_addr) _addr = -1;
|
|
#else
|
|
#define CLEAR_LAST_PEEK(_addr)
|
|
#endif
|
|
#endif
|
|
|
|
#ifndef SET_LAST_POKE
|
|
#ifdef DEBUGGER_SUPPORT
|
|
#define SET_LAST_POKE(_addr) myDataAddressForPoke = _addr;
|
|
#else
|
|
#define SET_LAST_POKE(_addr)
|
|
#endif
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// ADC
|
|
case 0x69:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x65:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x75:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x6d:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x7d:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + X);
|
|
if((low + X) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x79:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x61:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
intermediateAddress = peek(pointer++, DISASM_DATA);
|
|
intermediateAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x71:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// ASR
|
|
case 0x4b:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
A &= operand;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = A & 0x01;
|
|
|
|
A >>= 1;
|
|
|
|
notZ = A;
|
|
N = false;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// ANC
|
|
case 0x0b:
|
|
case 0x2b:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
A &= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
C = N;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// AND
|
|
case 0x29:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
A &= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x25:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
A &= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x35:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
A &= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x2d:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
A &= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x3d:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + X);
|
|
if((low + X) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
A &= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x39:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
A &= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x21:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
intermediateAddress = peek(pointer++, DISASM_DATA);
|
|
intermediateAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
A &= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x31:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
A &= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// ANE
|
|
case 0x8b:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
// NOTE: The implementation of this instruction is based on
|
|
// information from the 64doc.txt file. This instruction is
|
|
// reported to be unstable!
|
|
A = (A | 0xee) & X & operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// ARR
|
|
case 0x6b:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
// NOTE: The implementation of this instruction is based on
|
|
// information from the 64doc.txt file. There are mixed
|
|
// reports on its operation!
|
|
if(!D)
|
|
{
|
|
A &= operand;
|
|
A = ((A >> 1) & 0x7f) | (C ? 0x80 : 0x00);
|
|
|
|
C = A & 0x40;
|
|
V = (A & 0x40) ^ ((A & 0x20) << 1);
|
|
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
else
|
|
{
|
|
uInt8 value = A & operand;
|
|
|
|
A = ((value >> 1) & 0x7f) | (C ? 0x80 : 0x00);
|
|
N = C;
|
|
notZ = A;
|
|
V = (value ^ A) & 0x40;
|
|
|
|
if(((value & 0x0f) + (value & 0x01)) > 0x05)
|
|
{
|
|
A = (A & 0xf0) | ((A + 0x06) & 0x0f);
|
|
}
|
|
|
|
if(((value & 0xf0) + (value & 0x10)) > 0x50)
|
|
{
|
|
A += 0x60;
|
|
C = true;
|
|
}
|
|
else
|
|
{
|
|
C = false;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// ASL
|
|
case 0x0a:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in A
|
|
C = A & 0x80;
|
|
|
|
A <<= 1;
|
|
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x06:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x16:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x0e:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x1e:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// BIT
|
|
case 0x24:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
notZ = (A & operand);
|
|
N = operand & 0x80;
|
|
V = operand & 0x40;
|
|
}
|
|
break;
|
|
|
|
case 0x2C:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
notZ = (A & operand);
|
|
N = operand & 0x80;
|
|
V = operand & 0x40;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// Branches
|
|
case 0x90:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
if(!C)
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
uInt16 address = PC + Int8(operand);
|
|
if(NOTSAMEPAGE(PC, address))
|
|
peek((PC & 0xFF00) | (address & 0x00FF), DISASM_NONE);
|
|
PC = address;
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
case 0xb0:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
if(C)
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
uInt16 address = PC + Int8(operand);
|
|
if(NOTSAMEPAGE(PC, address))
|
|
peek((PC & 0xFF00) | (address & 0x00FF), DISASM_NONE);
|
|
PC = address;
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
case 0xf0:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
if(!notZ)
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
uInt16 address = PC + Int8(operand);
|
|
if(NOTSAMEPAGE(PC, address))
|
|
peek((PC & 0xFF00) | (address & 0x00FF), DISASM_NONE);
|
|
PC = address;
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
case 0x30:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
if(N)
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
uInt16 address = PC + Int8(operand);
|
|
if(NOTSAMEPAGE(PC, address))
|
|
peek((PC & 0xFF00) | (address & 0x00FF), DISASM_NONE);
|
|
PC = address;
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
case 0xD0:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
if(notZ)
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
uInt16 address = PC + Int8(operand);
|
|
if(NOTSAMEPAGE(PC, address))
|
|
peek((PC & 0xFF00) | (address & 0x00FF), DISASM_NONE);
|
|
PC = address;
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
case 0x10:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
if(!N)
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
uInt16 address = PC + Int8(operand);
|
|
if(NOTSAMEPAGE(PC, address))
|
|
peek((PC & 0xFF00) | (address & 0x00FF), DISASM_NONE);
|
|
PC = address;
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
case 0x50:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
if(!V)
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
uInt16 address = PC + Int8(operand);
|
|
if(NOTSAMEPAGE(PC, address))
|
|
peek((PC & 0xFF00) | (address & 0x00FF), DISASM_NONE);
|
|
PC = address;
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
case 0x70:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
if(V)
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
uInt16 address = PC + Int8(operand);
|
|
if(NOTSAMEPAGE(PC, address))
|
|
peek((PC & 0xFF00) | (address & 0x00FF), DISASM_NONE);
|
|
PC = address;
|
|
}
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// BRK
|
|
case 0x00:
|
|
{
|
|
peek(PC++, DISASM_NONE);
|
|
|
|
B = true;
|
|
|
|
poke(0x0100 + SP--, PC >> 8, DISASM_WRITE);
|
|
poke(0x0100 + SP--, PC & 0x00ff, DISASM_WRITE);
|
|
poke(0x0100 + SP--, PS(), DISASM_WRITE);
|
|
|
|
I = true;
|
|
|
|
PC = peek(0xfffe, DISASM_DATA);
|
|
PC |= (uInt16(peek(0xffff, DISASM_DATA)) << 8);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// CLC
|
|
case 0x18:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
C = false;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// CLD
|
|
case 0xd8:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
D = false;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// CLI
|
|
case 0x58:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
I = false;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// CLV
|
|
case 0xb8:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
V = false;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// CMP
|
|
case 0xc9:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(A) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xc5:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(A) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xd5:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(A) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xcd:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(A) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xdd:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + X);
|
|
if((low + X) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
uInt16 value = uInt16(A) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xd9:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
uInt16 value = uInt16(A) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xc1:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
intermediateAddress = peek(pointer++, DISASM_DATA);
|
|
intermediateAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(A) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xd1:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
uInt16 value = uInt16(A) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// CPX
|
|
case 0xe0:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(X) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xe4:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(X) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xec:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(X) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// CPY
|
|
case 0xc0:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(Y) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xc4:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(Y) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xcc:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(Y) - uInt16(operand);
|
|
|
|
notZ = value;
|
|
N = value & 0x0080;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// DCP
|
|
case 0xcf:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
uInt16 value2 = uInt16(A) - uInt16(value);
|
|
notZ = value2;
|
|
N = value2 & 0x0080;
|
|
C = !(value2 & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xdf:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
uInt16 value2 = uInt16(A) - uInt16(value);
|
|
notZ = value2;
|
|
N = value2 & 0x0080;
|
|
C = !(value2 & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xdb:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
uInt16 value2 = uInt16(A) - uInt16(value);
|
|
notZ = value2;
|
|
N = value2 & 0x0080;
|
|
C = !(value2 & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xc7:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
uInt16 value2 = uInt16(A) - uInt16(value);
|
|
notZ = value2;
|
|
N = value2 & 0x0080;
|
|
C = !(value2 & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xd7:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
uInt16 value2 = uInt16(A) - uInt16(value);
|
|
notZ = value2;
|
|
N = value2 & 0x0080;
|
|
C = !(value2 & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xc3:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
operandAddress = peek(pointer++, DISASM_DATA);
|
|
operandAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
uInt16 value2 = uInt16(A) - uInt16(value);
|
|
notZ = value2;
|
|
N = value2 & 0x0080;
|
|
C = !(value2 & 0x0100);
|
|
}
|
|
break;
|
|
|
|
case 0xd3:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
uInt16 value2 = uInt16(A) - uInt16(value);
|
|
notZ = value2;
|
|
N = value2 & 0x0080;
|
|
C = !(value2 & 0x0100);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// DEC
|
|
case 0xc6:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
notZ = value;
|
|
N = value & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xd6:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
notZ = value;
|
|
N = value & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xce:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
notZ = value;
|
|
N = value & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xde:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand - 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
notZ = value;
|
|
N = value & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// DEX
|
|
case 0xca:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
X--;
|
|
|
|
notZ = X;
|
|
N = X & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// DEY
|
|
case 0x88:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
Y--;
|
|
|
|
notZ = Y;
|
|
N = Y & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// EOR
|
|
case 0x49:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x45:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x55:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x4d:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x5d:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + X);
|
|
if((low + X) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x59:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x41:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
intermediateAddress = peek(pointer++, DISASM_DATA);
|
|
intermediateAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x51:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// INC
|
|
case 0xe6:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand + 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
notZ = value;
|
|
N = value & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xf6:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand + 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
notZ = value;
|
|
N = value & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xee:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand + 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
notZ = value;
|
|
N = value & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xfe:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = operand + 1;
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
notZ = value;
|
|
N = value & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// INX
|
|
case 0xe8:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
X++;
|
|
notZ = X;
|
|
N = X & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// INY
|
|
case 0xc8:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
Y++;
|
|
notZ = Y;
|
|
N = Y & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// ISB
|
|
case 0xef:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
operand = operand + 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xff:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
operand = operand + 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xfb:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
operand = operand + 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xe7:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
operand = operand + 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xf7:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
operand = operand + 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xe3:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
operandAddress = peek(pointer++, DISASM_DATA);
|
|
operandAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
operand = operand + 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xf3:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
operand = operand + 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// JMP
|
|
case 0x4c:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
}
|
|
{
|
|
PC = operandAddress;
|
|
}
|
|
break;
|
|
|
|
case 0x6c:
|
|
{
|
|
uInt16 addr = peek(PC++, DISASM_CODE);
|
|
addr |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
|
|
// Simulate the error in the indirect addressing mode!
|
|
uInt16 high = NOTSAMEPAGE(addr, addr + 1) ? (addr & 0xff00) : (addr + 1);
|
|
|
|
operandAddress = peek(addr, DISASM_DATA);
|
|
operandAddress |= (uInt16(peek(high, DISASM_DATA)) << 8);
|
|
}
|
|
{
|
|
PC = operandAddress;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// JSR
|
|
case 0x20:
|
|
{
|
|
uInt8 low = peek(PC++, DISASM_CODE);
|
|
peek(0x0100 + SP, DISASM_NONE);
|
|
|
|
// It seems that the 650x does not push the address of the next instruction
|
|
// on the stack it actually pushes the address of the next instruction
|
|
// minus one. This is compensated for in the RTS instruction
|
|
poke(0x0100 + SP--, PC >> 8, DISASM_WRITE);
|
|
poke(0x0100 + SP--, PC & 0xff, DISASM_WRITE);
|
|
|
|
PC = (low | (uInt16(peek(PC, DISASM_CODE)) << 8));
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// LAS
|
|
case 0xbb:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
A = X = SP = SP & operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// LAX
|
|
case 0xaf:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
SET_LAST_PEEK(myLastSrcAddressX, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
X = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xbf:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
SET_LAST_PEEK(myLastSrcAddressX, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
X = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xa7:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
SET_LAST_PEEK(myLastSrcAddressX, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
X = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xb7:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress) // TODO - check this
|
|
SET_LAST_PEEK(myLastSrcAddressX, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
X = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xa3:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
intermediateAddress = peek(pointer++, DISASM_DATA);
|
|
intermediateAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
SET_LAST_PEEK(myLastSrcAddressX, intermediateAddress) // TODO - check this
|
|
{
|
|
A = operand;
|
|
X = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xb3:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
SET_LAST_PEEK(myLastSrcAddressX, intermediateAddress) // TODO - check this
|
|
{
|
|
A = operand;
|
|
X = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
//////////////////////////////////////////////////
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// LDA
|
|
case 0xa9:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
CLEAR_LAST_PEEK(myLastSrcAddressA)
|
|
{
|
|
A = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xa5:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xb5:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xad:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xbd:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + X);
|
|
if((low + X) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xb9:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xa1:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
intermediateAddress = peek(pointer++, DISASM_DATA);
|
|
intermediateAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xb1:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A = operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
//////////////////////////////////////////////////
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// LDX
|
|
case 0xa2:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
CLEAR_LAST_PEEK(myLastSrcAddressX)
|
|
{
|
|
X = operand;
|
|
notZ = X;
|
|
N = X & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xa6:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressX, intermediateAddress)
|
|
{
|
|
X = operand;
|
|
notZ = X;
|
|
N = X & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xb6:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressX, intermediateAddress)
|
|
{
|
|
X = operand;
|
|
notZ = X;
|
|
N = X & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xae:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressX, intermediateAddress)
|
|
{
|
|
X = operand;
|
|
notZ = X;
|
|
N = X & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xbe:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressX, intermediateAddress)
|
|
{
|
|
X = operand;
|
|
notZ = X;
|
|
N = X & 0x80;
|
|
}
|
|
break;
|
|
//////////////////////////////////////////////////
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// LDY
|
|
case 0xa0:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
CLEAR_LAST_PEEK(myLastSrcAddressY)
|
|
{
|
|
Y = operand;
|
|
notZ = Y;
|
|
N = Y & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xa4:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressY, intermediateAddress)
|
|
{
|
|
Y = operand;
|
|
notZ = Y;
|
|
N = Y & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xb4:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressY, intermediateAddress)
|
|
{
|
|
Y = operand;
|
|
notZ = Y;
|
|
N = Y & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xac:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressY, intermediateAddress)
|
|
{
|
|
Y = operand;
|
|
notZ = Y;
|
|
N = Y & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0xbc:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + X);
|
|
if((low + X) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressY, intermediateAddress)
|
|
{
|
|
Y = operand;
|
|
notZ = Y;
|
|
N = Y & 0x80;
|
|
}
|
|
break;
|
|
//////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////
|
|
// LSR
|
|
case 0x4a:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit
|
|
C = A & 0x01;
|
|
|
|
A >>= 1;
|
|
|
|
notZ = A;
|
|
N = false;
|
|
}
|
|
break;
|
|
|
|
|
|
case 0x46:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand >>= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = false;
|
|
}
|
|
break;
|
|
|
|
case 0x56:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand >>= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = false;
|
|
}
|
|
break;
|
|
|
|
case 0x4e:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand >>= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = false;
|
|
}
|
|
break;
|
|
|
|
case 0x5e:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand >>= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = false;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// LXA
|
|
case 0xab:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
// NOTE: The implementation of this instruction is based on
|
|
// information from the 64doc.txt file. This instruction is
|
|
// reported to be very unstable!
|
|
A = X = (A | 0xee) & operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// NOP
|
|
case 0x1a:
|
|
case 0x3a:
|
|
case 0x5a:
|
|
case 0x7a:
|
|
case 0xda:
|
|
case 0xea:
|
|
case 0xfa:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
}
|
|
break;
|
|
|
|
case 0x80:
|
|
case 0x82:
|
|
case 0x89:
|
|
case 0xc2:
|
|
case 0xe2:
|
|
{
|
|
peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
}
|
|
break;
|
|
|
|
case 0x04:
|
|
case 0x44:
|
|
case 0x64:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
}
|
|
break;
|
|
|
|
case 0x14:
|
|
case 0x34:
|
|
case 0x54:
|
|
case 0x74:
|
|
case 0xd4:
|
|
case 0xf4:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += X;
|
|
peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
}
|
|
break;
|
|
|
|
case 0x0c:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
}
|
|
break;
|
|
|
|
case 0x1c:
|
|
case 0x3c:
|
|
case 0x5c:
|
|
case 0x7c:
|
|
case 0xdc:
|
|
case 0xfc:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + X);
|
|
if((low + X) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + X;
|
|
peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
}
|
|
break;
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// ORA
|
|
case 0x09:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
CLEAR_LAST_PEEK(myLastSrcAddressA)
|
|
{
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x05:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x15:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x0d:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x1d:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + X);
|
|
if((low + X) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x19:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x01:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
intermediateAddress = peek(pointer++, DISASM_DATA);
|
|
intermediateAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x11:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, intermediateAddress)
|
|
{
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
//////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////
|
|
// PHA
|
|
case 0x48:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
SET_LAST_POKE(myLastSrcAddressA)
|
|
{
|
|
poke(0x0100 + SP--, A, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// PHP
|
|
case 0x08:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
// TODO - add tracking for this opcode
|
|
{
|
|
poke(0x0100 + SP--, PS(), DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// PLA
|
|
case 0x68:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
// TODO - add tracking for this opcode
|
|
{
|
|
peek(0x0100 + SP++, DISASM_NONE);
|
|
A = peek(0x0100 + SP, DISASM_DATA);
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// PLP
|
|
case 0x28:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
// TODO - add tracking for this opcode
|
|
{
|
|
peek(0x0100 + SP++, DISASM_NONE);
|
|
PS(peek(0x0100 + SP, DISASM_DATA));
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// RLA
|
|
case 0x2f:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = (operand << 1) | (C ? 1 : 0);
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
A &= value;
|
|
C = operand & 0x80;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x3f:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = (operand << 1) | (C ? 1 : 0);
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
A &= value;
|
|
C = operand & 0x80;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x3b:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = (operand << 1) | (C ? 1 : 0);
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
A &= value;
|
|
C = operand & 0x80;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x27:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = (operand << 1) | (C ? 1 : 0);
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
A &= value;
|
|
C = operand & 0x80;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x37:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = (operand << 1) | (C ? 1 : 0);
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
A &= value;
|
|
C = operand & 0x80;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x23:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
operandAddress = peek(pointer++, DISASM_DATA);
|
|
operandAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = (operand << 1) | (C ? 1 : 0);
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
A &= value;
|
|
C = operand & 0x80;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x33:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
uInt8 value = (operand << 1) | (C ? 1 : 0);
|
|
poke(operandAddress, value, DISASM_WRITE);
|
|
|
|
A &= value;
|
|
C = operand & 0x80;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// ROL
|
|
case 0x2a:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the left-most bit
|
|
C = A & 0x80;
|
|
|
|
A = (A << 1) | (oldC ? 1 : 0);
|
|
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x26:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the left-most bit in operand
|
|
C = operand & 0x80;
|
|
|
|
operand = (operand << 1) | (oldC ? 1 : 0);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x36:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the left-most bit in operand
|
|
C = operand & 0x80;
|
|
|
|
operand = (operand << 1) | (oldC ? 1 : 0);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x2e:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the left-most bit in operand
|
|
C = operand & 0x80;
|
|
|
|
operand = (operand << 1) | (oldC ? 1 : 0);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x3e:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the left-most bit in operand
|
|
C = operand & 0x80;
|
|
|
|
operand = (operand << 1) | (oldC ? 1 : 0);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// ROR
|
|
case 0x6a:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = A & 0x01;
|
|
|
|
A = ((A >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x66:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x76:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x6e:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x7e:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
notZ = operand;
|
|
N = operand & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// RRA
|
|
case 0x6f:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x7f:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x7b:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x67:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x77:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x63:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
operandAddress = peek(pointer++, DISASM_DATA);
|
|
operandAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x73:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
bool oldC = C;
|
|
|
|
// Set carry flag according to the right-most bit
|
|
C = operand & 0x01;
|
|
|
|
operand = ((operand >> 1) & 0x7f) | (oldC ? 0x80 : 0x00);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
if(!D)
|
|
{
|
|
Int32 sum = A + operand + (C ? 1 : 0);
|
|
N = sum & 0x80;
|
|
V = ~(A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
C = sum & 0xff00;
|
|
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) + (operand & 0x0f) + (C ? 1 : 0);
|
|
Int32 hi = (A & 0xf0) + (operand & 0xf0);
|
|
notZ = (lo+hi) & 0xff;
|
|
if(lo > 0x09)
|
|
{
|
|
hi += 0x10;
|
|
lo += 0x06;
|
|
}
|
|
N = hi & 0x80;
|
|
V = ~(A ^ operand) & (A ^ hi) & 0x80;
|
|
if(hi > 0x90)
|
|
hi += 0x60;
|
|
C = hi & 0xff00;
|
|
|
|
A = (lo & 0x0f) + (hi & 0xf0);
|
|
}
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// RTI
|
|
case 0x40:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
peek(0x0100 + SP++, DISASM_NONE);
|
|
PS(peek(0x0100 + SP++, DISASM_NONE));
|
|
PC = peek(0x0100 + SP++, DISASM_NONE);
|
|
PC |= (uInt16(peek(0x0100 + SP, DISASM_NONE)) << 8);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// RTS
|
|
case 0x60:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
peek(0x0100 + SP++, DISASM_NONE);
|
|
PC = peek(0x0100 + SP++, DISASM_NONE);
|
|
PC |= (uInt16(peek(0x0100 + SP, DISASM_NONE)) << 8);
|
|
peek(PC++, DISASM_NONE);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SAX
|
|
case 0x8f:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
}
|
|
{
|
|
poke(operandAddress, A & X, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x87:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
poke(operandAddress, A & X, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x97:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + Y) & 0xFF;
|
|
}
|
|
{
|
|
poke(operandAddress, A & X, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x83:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
operandAddress = peek(pointer++, DISASM_DATA);
|
|
operandAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
}
|
|
{
|
|
poke(operandAddress, A & X, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SBC
|
|
case 0xe9:
|
|
case 0xeb:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xe5:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xf5:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress += X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xed:
|
|
{
|
|
intermediateAddress = peek(PC++, DISASM_CODE);
|
|
intermediateAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xfd:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + X);
|
|
if((low + X) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + X;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xf9:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xe1:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
intermediateAddress = peek(pointer++, DISASM_DATA);
|
|
intermediateAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
{
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
case 0xf1:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
intermediateAddress = high | uInt8(low + Y);
|
|
if((low + Y) > 0xFF)
|
|
{
|
|
peek(intermediateAddress, DISASM_NONE);
|
|
intermediateAddress = (high | low) + Y;
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
else
|
|
{
|
|
operand = peek(intermediateAddress, DISASM_DATA);
|
|
}
|
|
}
|
|
{
|
|
// N, V, Z, C flags are the same in either mode (C calculated at the end)
|
|
Int32 sum = A - operand - (C ? 0 : 1);
|
|
N = sum & 0x80;
|
|
V = (A ^ operand) & (A ^ sum) & 0x80;
|
|
notZ = sum & 0xff;
|
|
|
|
if(!D)
|
|
{
|
|
A = uInt8(sum);
|
|
}
|
|
else
|
|
{
|
|
Int32 lo = (A & 0x0f) - (operand & 0x0f) - (C ? 0 : 1);
|
|
Int32 hi = (A & 0xf0) - (operand & 0xf0);
|
|
if(lo & 0x10)
|
|
{
|
|
lo -= 6;
|
|
hi--;
|
|
}
|
|
if(hi & 0x0100)
|
|
hi -= 0x60;
|
|
|
|
A = (lo & 0x0f) | (hi & 0xf0);
|
|
}
|
|
C = (sum & 0xff00) == 0;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SBX
|
|
case 0xcb:
|
|
{
|
|
operand = peek(PC++, DISASM_CODE);
|
|
}
|
|
{
|
|
uInt16 value = uInt16(X & A) - uInt16(operand);
|
|
X = (value & 0xff);
|
|
|
|
notZ = X;
|
|
N = X & 0x80;
|
|
C = !(value & 0x0100);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SEC
|
|
case 0x38:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
C = true;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SED
|
|
case 0xf8:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
D = true;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SEI
|
|
case 0x78:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
{
|
|
I = true;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SHA
|
|
case 0x9f:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
}
|
|
{
|
|
// NOTE: There are mixed reports on the actual operation
|
|
// of this instruction!
|
|
poke(operandAddress, A & X & (((operandAddress >> 8) & 0xff) + 1), DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x93:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
}
|
|
{
|
|
// NOTE: There are mixed reports on the actual operation
|
|
// of this instruction!
|
|
poke(operandAddress, A & X & (((operandAddress >> 8) & 0xff) + 1), DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SHS
|
|
case 0x9b:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
}
|
|
{
|
|
// NOTE: There are mixed reports on the actual operation
|
|
// of this instruction!
|
|
SP = A & X;
|
|
poke(operandAddress, A & X & (((operandAddress >> 8) & 0xff) + 1), DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SHX
|
|
case 0x9e:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
}
|
|
{
|
|
// NOTE: There are mixed reports on the actual operation
|
|
// of this instruction!
|
|
poke(operandAddress, X & (((operandAddress >> 8) & 0xff) + 1), DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SHY
|
|
case 0x9c:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
}
|
|
{
|
|
// NOTE: There are mixed reports on the actual operation
|
|
// of this instruction!
|
|
poke(operandAddress, Y & (((operandAddress >> 8) & 0xff) + 1), DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SLO
|
|
case 0x0f:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x1f:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x1b:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x07:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x17:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x03:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
operandAddress = peek(pointer++, DISASM_DATA);
|
|
operandAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x13:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the left-most bit in value
|
|
C = operand & 0x80;
|
|
|
|
operand <<= 1;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A |= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
//////////////////////////////////////////////////
|
|
// SRE
|
|
case 0x4f:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand = (operand >> 1) & 0x7f;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x5f:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand = (operand >> 1) & 0x7f;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x5b:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand = (operand >> 1) & 0x7f;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x47:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand = (operand >> 1) & 0x7f;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x57:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand = (operand >> 1) & 0x7f;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x43:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
operandAddress = peek(pointer++, DISASM_DATA);
|
|
operandAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand = (operand >> 1) & 0x7f;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
case 0x53:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
operand = peek(operandAddress, DISASM_DATA);
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
}
|
|
{
|
|
// Set carry flag according to the right-most bit in value
|
|
C = operand & 0x01;
|
|
|
|
operand = (operand >> 1) & 0x7f;
|
|
poke(operandAddress, operand, DISASM_WRITE);
|
|
|
|
A ^= operand;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// STA
|
|
case 0x85:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
}
|
|
SET_LAST_POKE(myLastSrcAddressA)
|
|
{
|
|
poke(operandAddress, A, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x95:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
}
|
|
{
|
|
poke(operandAddress, A, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x8d:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
}
|
|
SET_LAST_POKE(myLastSrcAddressA)
|
|
{
|
|
poke(operandAddress, A, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x9d:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + X), DISASM_NONE);
|
|
operandAddress = (high | low) + X;
|
|
}
|
|
{
|
|
poke(operandAddress, A, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x99:
|
|
{
|
|
uInt16 low = peek(PC++, DISASM_CODE);
|
|
uInt16 high = (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
}
|
|
{
|
|
poke(operandAddress, A, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x81:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
peek(pointer, DISASM_NONE);
|
|
pointer += X;
|
|
operandAddress = peek(pointer++, DISASM_DATA);
|
|
operandAddress |= (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
}
|
|
{
|
|
poke(operandAddress, A, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x91:
|
|
{
|
|
uInt8 pointer = peek(PC++, DISASM_CODE);
|
|
uInt16 low = peek(pointer++, DISASM_DATA);
|
|
uInt16 high = (uInt16(peek(pointer, DISASM_DATA)) << 8);
|
|
peek(high | uInt8(low + Y), DISASM_NONE);
|
|
operandAddress = (high | low) + Y;
|
|
}
|
|
{
|
|
poke(operandAddress, A, DISASM_WRITE);
|
|
}
|
|
break;
|
|
//////////////////////////////////////////////////
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// STX
|
|
case 0x86:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
}
|
|
SET_LAST_POKE(myLastSrcAddressX)
|
|
{
|
|
poke(operandAddress, X, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x96:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + Y) & 0xFF;
|
|
}
|
|
{
|
|
poke(operandAddress, X, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x8e:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
}
|
|
SET_LAST_POKE(myLastSrcAddressX)
|
|
{
|
|
poke(operandAddress, X, DISASM_WRITE);
|
|
}
|
|
break;
|
|
//////////////////////////////////////////////////
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// STY
|
|
case 0x84:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
}
|
|
SET_LAST_POKE(myLastSrcAddressY)
|
|
{
|
|
poke(operandAddress, Y, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x94:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
peek(operandAddress, DISASM_NONE);
|
|
operandAddress = (operandAddress + X) & 0xFF;
|
|
}
|
|
{
|
|
poke(operandAddress, Y, DISASM_WRITE);
|
|
}
|
|
break;
|
|
|
|
case 0x8c:
|
|
{
|
|
operandAddress = peek(PC++, DISASM_CODE);
|
|
operandAddress |= (uInt16(peek(PC++, DISASM_CODE)) << 8);
|
|
}
|
|
SET_LAST_POKE(myLastSrcAddressY)
|
|
{
|
|
poke(operandAddress, Y, DISASM_WRITE);
|
|
}
|
|
break;
|
|
//////////////////////////////////////////////////
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// Remaining MOVE opcodes
|
|
case 0xaa:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressX, myLastSrcAddressA)
|
|
{
|
|
X = A;
|
|
notZ = X;
|
|
N = X & 0x80;
|
|
}
|
|
break;
|
|
|
|
|
|
case 0xa8:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressY, myLastSrcAddressA)
|
|
{
|
|
Y = A;
|
|
notZ = Y;
|
|
N = Y & 0x80;
|
|
}
|
|
break;
|
|
|
|
|
|
case 0xba:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressX, myLastSrcAddressS)
|
|
{
|
|
X = SP;
|
|
notZ = X;
|
|
N = X & 0x80;
|
|
}
|
|
break;
|
|
|
|
|
|
case 0x8a:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, myLastSrcAddressX)
|
|
{
|
|
A = X;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
|
|
|
|
case 0x9a:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressS, myLastSrcAddressX)
|
|
{
|
|
SP = X;
|
|
}
|
|
break;
|
|
|
|
|
|
case 0x98:
|
|
{
|
|
peek(PC, DISASM_NONE);
|
|
}
|
|
SET_LAST_PEEK(myLastSrcAddressA, myLastSrcAddressY)
|
|
{
|
|
A = Y;
|
|
notZ = A;
|
|
N = A & 0x80;
|
|
}
|
|
break;
|
|
//////////////////////////////////////////////////
|