1851 lines
52 KiB
C
1851 lines
52 KiB
C
/**
|
|
* CoreCPU - The Quick6502 Project
|
|
* corecpu.c
|
|
*
|
|
* Created by Manoel Trapier on 24/02/08
|
|
* Copyright 2008 986 Corp. All rights reserved.
|
|
*
|
|
* $LastChangedDate$
|
|
* $Author$
|
|
* $HeadURL$
|
|
* $Revision$
|
|
*
|
|
*/
|
|
|
|
/* Depending on the OS, one of these provide the malloc function */
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <os_dependent.h>
|
|
|
|
/*******************************************************************************
|
|
/!\ WARNING this debug tool slow down a lot the emulator! /!\
|
|
/!\ Use it only if you really need it ! /!\
|
|
*******************************************************************************/
|
|
//#define TRACE_INSTRUCTIONS
|
|
|
|
#ifdef TRACE_INSTRUCTIONS
|
|
#define TRACEi(trace) do { console_printf(Console_Debug, "$%04X - ", cpu->reg_PC - 1); console_printf_d trace ; console_printf(Console_Debug, "\n"); } while(0)
|
|
#define TRACEiE(trace) do { console_printf(Console_Debug, "$%04X - ", cpu->reg_PC - 1); console_printf_d trace ; console_printf(Console_Debug, "\n"); } while(0)
|
|
#else
|
|
#define TRACEi(trace) { }
|
|
//#define TRACEiE(trace) { }
|
|
#define TRACEiE(trace) do { console_printf(Console_Debug, "$%04X - ", cpu->reg_PC - 1); console_printf_d trace ; console_printf(Console_Debug, "\n"); } while(0)
|
|
#endif
|
|
|
|
#define _INTERNAL_QUICK6502_CORECPU_
|
|
#include "corecpu.h"
|
|
|
|
|
|
/*** Instructions useful macros ***/
|
|
#define INSTRUCTION(s) static inline void I_##s (quick6502_cpu *cpu)
|
|
|
|
#define NZ_FLAG_UPDATE(value) cpu->reg_P = (cpu->reg_P & ~(Q6502_N_FLAG | Q6502_Z_FLAG)) | \
|
|
(value & 0x80) | ((value)?0:Q6502_Z_FLAG)
|
|
|
|
#define CROSS_CYCLE_UPDATE(value) if ((value) & 0x0F00) cpu->page_crossed = 1
|
|
|
|
#define MEMORY_READ_ZP() cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC++))
|
|
|
|
#define MEMORY_READ_IX() cpu->memory_read( (cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC ) + cpu->reg_X ) & 0xFF) |\
|
|
(cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC++) + cpu->reg_X + 1) << 8) )
|
|
#define MEMORY_READ_IY() cpu->memory_read( ( cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC ) ) |\
|
|
(cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC++) + 1) << 8) ) + cpu->reg_Y )
|
|
|
|
#define MEMORY_READ_ZX() cpu->memory_page0_read( (cpu->memory_opcode_read(cpu->reg_PC++) + cpu->reg_X) )
|
|
#define MEMORY_READ_ZY() cpu->memory_page0_read( (cpu->memory_opcode_read(cpu->reg_PC++) + cpu->reg_Y) )
|
|
|
|
#define MEMORY_READ_AB() cpu->memory_read( ((cpu->memory_opcode_read(cpu->reg_PC++) ) |\
|
|
(cpu->memory_opcode_read(cpu->reg_PC++) << 8) ))
|
|
|
|
#define MEMORY_READ_AX() cpu->memory_read( ((op1 ) |\
|
|
(op2 << 8) ) + cpu->reg_X)
|
|
#define MEMORY_READ_AY() cpu->memory_read( ((op1 ) |\
|
|
(op2 << 8) ) + cpu->reg_Y )
|
|
|
|
|
|
#define MEMORY_WRITE_ZP(val) cpu->memory_page0_write(cpu->memory_opcode_read(cpu->reg_PC++), val)
|
|
|
|
#define MEMORY_WRITE_IX(val) cpu->memory_write( (cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC ) + cpu->reg_X ) & 0xFF) |\
|
|
(cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC++) + cpu->reg_X + 1) << 8) , val)
|
|
#define MEMORY_WRITE_IY(val) cpu->memory_write( ( cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC ) ) |\
|
|
(cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC++) + 1) << 8) ) + cpu->reg_Y , val)
|
|
|
|
#define MEMORY_WRITE_ZX(val) cpu->memory_page0_write( (cpu->memory_opcode_read(cpu->reg_PC++) + cpu->reg_X), val)
|
|
#define MEMORY_WRITE_ZY(val) cpu->memory_page0_write( (cpu->memory_opcode_read(cpu->reg_PC++) + cpu->reg_Y), val)
|
|
|
|
#define MEMORY_WRITE_AB(val) cpu->memory_write( ((cpu->memory_opcode_read(cpu->reg_PC++) ) |\
|
|
(cpu->memory_opcode_read(cpu->reg_PC++) << 8) ), val)
|
|
|
|
#define MEMORY_WRITE_AX(val) cpu->memory_write( ((cpu->memory_opcode_read(cpu->reg_PC++) ) |\
|
|
(cpu->memory_opcode_read(cpu->reg_PC++) << 8) ) + cpu->reg_X, val)
|
|
#define MEMORY_WRITE_AY(val) cpu->memory_write( ((cpu->memory_opcode_read(cpu->reg_PC++) ) |\
|
|
(cpu->memory_opcode_read(cpu->reg_PC++) << 8) ) + cpu->reg_Y, val)
|
|
|
|
|
|
#define PUSH_S(value) cpu->memory_stack_write(0x100 | (cpu->reg_S--), value)
|
|
#define POP_S() (cpu->memory_stack_read (0x100 | (++cpu->reg_S) ))
|
|
|
|
#ifdef NO_DECIMAL
|
|
|
|
#define ADC_OPERATION(read) do {\
|
|
unsigned short tmp = 0; unsigned char v = read; \
|
|
tmp = cpu->reg_A + v + (cpu->reg_P & Q6502_C_FLAG); \
|
|
cpu->reg_P = (cpu->reg_P & ~(Q6502_C_FLAG | Q6502_N_FLAG | Q6502_Z_FLAG | Q6502_V_FLAG)) | \
|
|
(tmp & 0x80) | ((tmp&0xFF)?0:Q6502_Z_FLAG) | \
|
|
((tmp & 0xFF00)?Q6502_C_FLAG:0) | \
|
|
( ( ~(cpu->reg_A^v)&(cpu->reg_A^tmp) )&0x80?Q6502_V_FLAG:0 ); \
|
|
cpu->reg_A = tmp & 0xFF; \
|
|
} while(0)
|
|
|
|
#define SBC_OPERATION(read) do {\
|
|
unsigned short tmp = 0; unsigned char v = read; \
|
|
tmp = cpu->reg_A - v - (~cpu->reg_P & Q6502_C_FLAG); \
|
|
cpu->reg_P = (cpu->reg_P & ~(Q6502_C_FLAG | Q6502_N_FLAG | Q6502_Z_FLAG | Q6502_V_FLAG)) | \
|
|
(tmp & Q6502_N_FLAG) | ((tmp&0xFF)?0:Q6502_Z_FLAG) | \
|
|
((tmp & 0xFF00)?0:Q6502_C_FLAG) | \
|
|
( ( (cpu->reg_A^v)&(cpu->reg_A^tmp) )&0x80?Q6502_V_FLAG:0 ); \
|
|
cpu->reg_A = tmp & 0xFF; \
|
|
} while(0)
|
|
|
|
#else
|
|
#error Quick6502 doesn't actually support DECIMAL mode
|
|
#endif
|
|
|
|
|
|
#define AND_OPERATION(read) cpu->reg_A &= read; NZ_FLAG_UPDATE(cpu->reg_A)
|
|
|
|
/* CMP is like SBC but without storing the result value */
|
|
#define CMP_OPERATION(register, read) do { \
|
|
unsigned short tmp = 0; \
|
|
tmp = register - read; \
|
|
cpu->reg_P = (cpu->reg_P & ~(Q6502_C_FLAG | Q6502_N_FLAG | Q6502_Z_FLAG)) | \
|
|
(tmp & Q6502_N_FLAG) | ((tmp&0xFF)?0:Q6502_Z_FLAG) | \
|
|
((tmp & 0xFF00)?0:Q6502_C_FLAG); \
|
|
} while(0)
|
|
|
|
#define EOR_OPERATION(read) cpu->reg_A ^= read; NZ_FLAG_UPDATE(cpu->reg_A)
|
|
#define ORA_OPERATION(read) cpu->reg_A |= read; NZ_FLAG_UPDATE(cpu->reg_A)
|
|
|
|
#define BIT_OPERATION(read) do { \
|
|
byte tmp = read; \
|
|
cpu->reg_P = (cpu->reg_P & ~(Q6502_V_FLAG | Q6502_N_FLAG | Q6502_Z_FLAG)) | \
|
|
(tmp & Q6502_N_FLAG) | (tmp & Q6502_V_FLAG) | \
|
|
((tmp & cpu->reg_A)?0:Q6502_Z_FLAG); \
|
|
} while(0)
|
|
|
|
#define ASL_OPERATION(val) cpu->reg_P = (cpu->reg_P & ~(Q6502_C_FLAG | Q6502_N_FLAG | Q6502_Z_FLAG)) | \
|
|
((val&0x40)?Q6502_N_FLAG:0) | \
|
|
((val&0x80)?Q6502_C_FLAG:0) | \
|
|
((val&0x7F)?0:Q6502_Z_FLAG); \
|
|
val = val << 1
|
|
|
|
#define LSR_OPERATION(val) cpu->reg_P = (cpu->reg_P & ~(Q6502_C_FLAG | Q6502_N_FLAG | Q6502_Z_FLAG)) | \
|
|
(val & Q6502_C_FLAG) | \
|
|
((val&0xFE)?0:Q6502_Z_FLAG); \
|
|
val = val >> 1
|
|
|
|
#define ROR_OPERATION(val) do {\
|
|
unsigned short tmp = val | (cpu->reg_P & Q6502_C_FLAG) << 8; \
|
|
cpu->reg_P = (cpu->reg_P & ~(Q6502_C_FLAG | Q6502_N_FLAG | Q6502_Z_FLAG)) | \
|
|
( tmp&Q6502_C_FLAG) | /* Set the C flag */ \
|
|
((tmp&0x100) >> 1) | /* Set the N flag */ \
|
|
((tmp&0x1FE)?0:Q6502_Z_FLAG); /* 0x1FE will be the new 8bit value */ \
|
|
val = (tmp>>1) & 0xFF; \
|
|
} while(0)
|
|
|
|
#define ROL_OPERATION(val) do {\
|
|
unsigned short tmp = (val << 1) | (cpu->reg_P & Q6502_C_FLAG); \
|
|
cpu->reg_P = (cpu->reg_P & ~(Q6502_C_FLAG | Q6502_N_FLAG | Q6502_Z_FLAG)) | \
|
|
((tmp&0x100)?Q6502_C_FLAG:0) | /* Set the C flag */ \
|
|
((tmp&0x80)) | /* Set the N flag */ \
|
|
((tmp&0xFF)?0:Q6502_Z_FLAG); /* 0x1FE will be the new 8bit value */ \
|
|
val = tmp & 0xFF; \
|
|
} while(0)
|
|
|
|
|
|
/** Function used for execution of instruction */
|
|
static inline int quick6502_exec_one(quick6502_cpu *cpu);
|
|
|
|
/**
|
|
* Initialise the CPU
|
|
*
|
|
* Inputs:
|
|
*
|
|
* - CPU Init structure:
|
|
* - Memory Read function pointer
|
|
* - Memory Write function pointer
|
|
* - Fast memory read function pointer (for opcodes read)
|
|
* - Fast page 0 function Read/Write
|
|
* - Fast page 1 function Read/Write
|
|
*
|
|
* Output:
|
|
*
|
|
* (void *): An opaque pointer to the internal structure of the CPU.
|
|
* NULL if an error occured !
|
|
*/
|
|
quick6502_cpu *quick6502_init(quick6502_cpuconfig *config)
|
|
{
|
|
quick6502_cpu *cpu;
|
|
|
|
/* Alloc structure */
|
|
cpu = (quick6502_cpu *) malloc (sizeof (quick6502_cpu));
|
|
if (!cpu)
|
|
return NULL;
|
|
|
|
/* Initialise other variables */
|
|
cpu->running = 0; /* CPU is currently NOT running */
|
|
|
|
cpu->cycle_done = 0;
|
|
cpu->int_pending = 0;
|
|
|
|
cpu->page_crossed = 0;
|
|
|
|
/* Initialise registers */
|
|
cpu->reg_A = 0;
|
|
cpu->reg_X = 0;
|
|
cpu->reg_Y = 0;
|
|
cpu->reg_S = 0xFF;
|
|
|
|
cpu->reg_P = Q6502_D_FLAG | Q6502_I_FLAG;
|
|
|
|
if (config->memory_read != NULL)
|
|
cpu->memory_read = config->memory_read;
|
|
else
|
|
goto init_error;
|
|
|
|
if (config->memory_write != NULL)
|
|
cpu->memory_write = config->memory_write;
|
|
else
|
|
goto init_error;
|
|
|
|
if (config->memory_opcode_read != NULL)
|
|
cpu->memory_opcode_read = config->memory_opcode_read;
|
|
else
|
|
cpu->memory_opcode_read = config->memory_read;
|
|
|
|
|
|
if (config->memory_page0_read != NULL)
|
|
cpu->memory_page0_read = config->memory_page0_read;
|
|
else
|
|
cpu->memory_page0_read = config->memory_read;
|
|
|
|
if (config->memory_page0_write != NULL)
|
|
cpu->memory_page0_write = config->memory_page0_write;
|
|
else
|
|
cpu->memory_page0_write = config->memory_write;
|
|
|
|
if (config->memory_stack_read != NULL)
|
|
cpu->memory_stack_read = config->memory_stack_read;
|
|
else
|
|
cpu->memory_stack_read = config->memory_read;
|
|
|
|
if (config->memory_stack_write != NULL)
|
|
cpu->memory_stack_write = config->memory_stack_write;
|
|
else
|
|
cpu->memory_stack_write = config->memory_write;
|
|
|
|
return cpu;
|
|
|
|
init_error:
|
|
if (cpu)
|
|
free (cpu);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/** Reset the CPU (must be done after init) */
|
|
void quick6502_reset(quick6502_cpu *cpu)
|
|
{
|
|
/* Initialise registers */
|
|
/*cpu->reg_A = 0;
|
|
cpu->reg_X = 0;
|
|
cpu->reg_Y = 0;
|
|
cpu->reg_S = 0xFF;*/
|
|
|
|
//cpu->reg_P = Q6502_D_FLAG | Q6502_I_FLAG | 0x20 | Q6502_B_FLAG;
|
|
|
|
/* Set the PC to the RESET vector */
|
|
cpu->reg_PC = ( cpu->memory_read(Q6502_RESET_HIGH) << 8)
|
|
| cpu->memory_read(Q6502_RESET_LOW);
|
|
|
|
cpu->exit_loop = 0;
|
|
}
|
|
|
|
/**
|
|
* Run cpu for at least X cycles
|
|
*
|
|
* Output:
|
|
*
|
|
* int: (Number of cycle really done) - (Number of cycle asked)
|
|
*/
|
|
int quick6502_run(quick6502_cpu *cpu, int cycles)
|
|
{
|
|
cpu->running = !0;
|
|
|
|
while(cpu->cycle_done < cycles)
|
|
{
|
|
quick6502_exec_one(cpu);
|
|
}
|
|
cpu->cycle_done -= cycles;
|
|
|
|
cpu->running = 0;
|
|
|
|
return cycles + cpu->cycle_done;
|
|
}
|
|
|
|
/** Loop CPU until explicit quit */
|
|
void quick6502_loop(quick6502_cpu *cpu)
|
|
{
|
|
cpu->running = !0;
|
|
while(cpu->exit_loop)
|
|
{
|
|
quick6502_exec_one(cpu);
|
|
}
|
|
cpu->running = 0;
|
|
}
|
|
|
|
/** Run CPU for one instruction */
|
|
void quick6502_exec(quick6502_cpu *cpu)
|
|
{
|
|
cpu->running = !0;
|
|
quick6502_exec_one(cpu);
|
|
cpu->running = 0;
|
|
}
|
|
|
|
/** Send IRQ/NMI/EXITLOOP signal to CPU */
|
|
void quick6502_int(quick6502_cpu *cpu, quick6502_signal signal)
|
|
{
|
|
switch(signal)
|
|
{
|
|
default:
|
|
break;
|
|
|
|
case Q6502_IRQ_SIGNAL:
|
|
if (! (cpu->reg_P & Q6502_I_FLAG) )
|
|
{
|
|
TRACEi(("IRQ Triggered !"));
|
|
PUSH_S((cpu->reg_PC >> 8) & 0xFF );
|
|
PUSH_S((cpu->reg_PC ) & 0xFF );
|
|
PUSH_S( cpu->reg_P & ~Q6502_B_FLAG );
|
|
cpu->reg_P = cpu->reg_P | Q6502_I_FLAG;
|
|
|
|
cpu->reg_PC = (cpu->memory_read(Q6502_IRQ_LOW)) | (cpu->memory_read(Q6502_IRQ_HIGH)<<8);
|
|
|
|
cpu->cycle_done += 7;
|
|
}
|
|
else
|
|
cpu->int_pending = 1;
|
|
|
|
break;
|
|
|
|
case Q6502_NMI_SIGNAL:
|
|
TRACEi(("NMI Triggered !"));
|
|
PUSH_S((cpu->reg_PC >> 8) & 0xFF );
|
|
PUSH_S((cpu->reg_PC ) & 0xFF );
|
|
PUSH_S( cpu->reg_P );
|
|
cpu->reg_P = cpu->reg_P | Q6502_I_FLAG & ~Q6502_B_FLAG;
|
|
|
|
cpu->reg_PC = (cpu->memory_read(Q6502_NMI_LOW)) | (cpu->memory_read(Q6502_NMI_HIGH)<<8);
|
|
|
|
cpu->cycle_done += 7;
|
|
break;
|
|
|
|
case Q6502_STOPLOOP_SIGNAL:
|
|
cpu->exit_loop = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/** Dump CPU State to the given file */
|
|
void quick6502_dump(quick6502_cpu *cpu, FILE * fp)
|
|
{
|
|
short i;
|
|
char instr[20];
|
|
/* Display registers */
|
|
fprintf(fp,
|
|
"Quick6502: PC:$%04X A:$%02X X:$%02X Y:$%02X S:$%02X P:$%02X P:[%c%c%c%c%c%c%c%c]\n",
|
|
cpu->reg_PC, cpu->reg_A, cpu->reg_X, cpu->reg_Y, cpu->reg_S, cpu->reg_P,
|
|
cpu->reg_P&Q6502_N_FLAG ? 'N':'.',
|
|
cpu->reg_P&Q6502_V_FLAG ? 'V':'.',
|
|
'.', /* No real flag here */
|
|
cpu->reg_P&Q6502_B_FLAG ? 'B':'.',
|
|
cpu->reg_P&Q6502_D_FLAG ? 'D':'.',
|
|
cpu->reg_P&Q6502_I_FLAG ? 'I':'.',
|
|
cpu->reg_P&Q6502_Z_FLAG ? 'Z':'.',
|
|
cpu->reg_P&Q6502_C_FLAG ? 'C':'.'
|
|
);
|
|
|
|
/* Display stack */
|
|
fprintf(fp, "Quick6502: Stack: [ ");
|
|
for (i = cpu->reg_S+1; i < 0x100; i++)
|
|
{
|
|
fprintf(fp, "$%02X ", cpu->memory_opcode_read(0x100 | i));
|
|
}
|
|
fprintf(fp, "] Run:%c Cycle:%ld\n", cpu->running?'Y':'N', cpu->cycle_done);
|
|
|
|
quick6502_getinstruction(cpu, cpu->reg_PC, instr);
|
|
fprintf(fp, "Quick6502: Instruction at PC: %s\n", instr);
|
|
}
|
|
|
|
/** Get current instruction name at specified address and put it into buffer */
|
|
void quick6502_getinstruction(quick6502_cpu *cpu, unsigned short addr, char *buffer)
|
|
{
|
|
buffer[0] = 0;
|
|
}
|
|
|
|
/**
|
|
* Free the CPU
|
|
*
|
|
* This function will free the CPU only if it's not currently used, it will
|
|
* return !0 if everything goes well and 0 if the free is impossible
|
|
*/
|
|
int quick6502_free(quick6502_cpu *cpu)
|
|
{
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
*** Here start real CPU logic ***
|
|
*******************************************************************************/
|
|
static byte CycleTable[256] =
|
|
{
|
|
/* 00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F */
|
|
/* 00 */ 7, 6, 0, 8, 3, 3, 5, 5, 3, 2, 2, 2, 4, 4, 6, 6,
|
|
/* 10 */ 2, 5, 0, 8, 4, 4, 6, 6, 2, 4, 2, 7, 4, 4, 7, 7,
|
|
/* 20 */ 6, 6, 0, 8, 3, 3, 5, 5, 4, 2, 2, 2, 4, 4, 6, 6,
|
|
/* 30 */ 2, 5, 0, 8, 4, 4, 6, 6, 2, 4, 2, 7, 4, 4, 7, 7,
|
|
/* 40 */ 6, 6, 0, 8, 3, 3, 5, 5, 3, 2, 2, 2, 3, 4, 6, 6,
|
|
/* 50 */ 2, 5, 0, 8, 4, 4, 6, 6, 2, 4, 2, 7, 4, 4, 7, 7,
|
|
/* 60 */ 6, 6, 0, 8, 3, 3, 5, 5, 4, 2, 2, 2, 5, 4, 6, 6,
|
|
/* 70 */ 2, 5, 0, 8, 4, 4, 6, 6, 2, 4, 2, 7, 4, 4, 7, 7,
|
|
/* 80 */ 2, 6, 2, 6, 3, 3, 3, 3, 2, 2, 2, 2, 4, 4, 4, 4,
|
|
/* 90 */ 2, 6, 0, 6, 4, 4, 4, 4, 2, 5, 2, 5, 5, 5, 5, 5,
|
|
/* A0 */ 2, 6, 2, 6, 3, 3, 3, 3, 2, 2, 2, 2, 4, 4, 4, 4,
|
|
/* B0 */ 2, 5, 0, 5, 4, 4, 4, 4, 2, 4, 2, 4, 4, 4, 4, 4,
|
|
/* C0 */ 2, 6, 2, 8, 3, 3, 5, 5, 2, 2, 2, 2, 4, 4, 6, 6,
|
|
/* D0 */ 2, 5, 0, 8, 4, 4, 6, 6, 2, 4, 2, 7, 4, 4, 7, 7,
|
|
/* E0 */ 2, 6, 2, 8, 3, 3, 5, 5, 2, 2, 2, 2, 4, 4, 6, 6,
|
|
/* F0 */ 2, 5, 0, 8, 4, 4, 6, 6, 2, 4, 2, 7, 4, 4, 7, 7
|
|
/* 00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F */
|
|
};
|
|
|
|
typedef void (*InstructionFunction)(quick6502_cpu *cpu);
|
|
|
|
/*******************************************************************************
|
|
* Instruction implementations
|
|
*******************************************************************************/
|
|
|
|
/**** Other instructions ****/
|
|
INSTRUCTION(ILLEG)
|
|
{
|
|
TRACEiE(("Illegal instruction $%02X", cpu->memory_opcode_read(cpu->reg_PC-1)));
|
|
//exit(-1);
|
|
}
|
|
|
|
/** 58 : CLI - CLear Interrupt **/
|
|
INSTRUCTION(CLIiM)
|
|
{
|
|
TRACEi(("CLC"));
|
|
cpu->reg_P &= ~Q6502_I_FLAG;
|
|
}
|
|
/** 78 : SEI - SEt Interrupt **/
|
|
INSTRUCTION(SEIiM)
|
|
{
|
|
TRACEi(("SEI"));
|
|
cpu->reg_P |= Q6502_I_FLAG;
|
|
}
|
|
|
|
/** 18 : CLC - CLear Carry **/
|
|
INSTRUCTION(CLCiM)
|
|
{
|
|
TRACEi(("CLC"));
|
|
cpu->reg_P &= ~Q6502_C_FLAG;
|
|
}
|
|
/** 38 : SEC - SEt Carry **/
|
|
INSTRUCTION(SECiM)
|
|
{
|
|
TRACEi(("SEC"));
|
|
cpu->reg_P |= Q6502_C_FLAG;
|
|
}
|
|
|
|
/** D8 : CLD - CLear Decimal **/
|
|
INSTRUCTION(CLDiM)
|
|
{
|
|
TRACEi(("CLD"));
|
|
cpu->reg_P &= ~Q6502_D_FLAG;
|
|
}
|
|
/** F8 : SED - SEt Decimal **/
|
|
INSTRUCTION(SEDiM)
|
|
{
|
|
TRACEi(("SED"));
|
|
cpu->reg_P |= Q6502_D_FLAG;
|
|
}
|
|
/** B8 : CLV - CLear oVerflo **/
|
|
INSTRUCTION(CLViM)
|
|
{
|
|
TRACEi(("CLV"));
|
|
cpu->reg_P &= ~Q6502_V_FLAG;
|
|
}
|
|
|
|
/** EA : NOP - NO oPeration **/
|
|
INSTRUCTION(NOPiM)
|
|
{
|
|
TRACEi(("NOP"));
|
|
}
|
|
|
|
/**** Load/Store functions ****/
|
|
|
|
/** A9 : LDA #$xx - LoaD an immediate value into A */
|
|
INSTRUCTION(LDAiM)
|
|
{
|
|
TRACEi(("LDA #$%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_A = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
}
|
|
|
|
/** A2 : LDX #$xx - LoaD an immediate value into X */
|
|
INSTRUCTION(LDXiM)
|
|
{
|
|
TRACEi(("LDX #$%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_X = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
NZ_FLAG_UPDATE(cpu->reg_X);
|
|
}
|
|
|
|
/** A0 : LDY #$xx - LoaD an immediate value into Y */
|
|
INSTRUCTION(LDYiM)
|
|
{
|
|
TRACEi(("LDY #$%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_Y = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
NZ_FLAG_UPDATE(cpu->reg_Y);
|
|
}
|
|
|
|
/** A5: LDA $xx - LoaD to A from zero page **/
|
|
INSTRUCTION(LDAzP)
|
|
{
|
|
TRACEi(("LDA $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_A = MEMORY_READ_ZP();
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
}
|
|
|
|
/** B5: LDA $xx,X - LoaD to A **/
|
|
INSTRUCTION(LDAzX)
|
|
{
|
|
TRACEi(("LDA $%02X,X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_A = MEMORY_READ_ZX();
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
}
|
|
|
|
/** A1: LDA ($xx,X) - LoaD to A **/
|
|
INSTRUCTION(LDAiX)
|
|
{
|
|
TRACEi(("LDA ($%02X,X)", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_A = MEMORY_READ_IX();
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
}
|
|
|
|
/** B1: LDA ($xx),Y - LoaD to A **/
|
|
INSTRUCTION(LDAiY)
|
|
{
|
|
TRACEi(("LDA ($%02X),Y", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_A = MEMORY_READ_IY();
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
CROSS_CYCLE_UPDATE(cpu->memory_opcode_read(cpu->reg_PC-1) + cpu->reg_Y);
|
|
}
|
|
|
|
/** AD: LDA $xxxx - LoaD to A **/
|
|
INSTRUCTION(LDAaB)
|
|
{
|
|
TRACEi(("LDA $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_A = MEMORY_READ_AB();
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
}
|
|
|
|
/** DD: LDA $xxxx,X - LoaD to A **/
|
|
INSTRUCTION(LDAaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("LDA $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_A = MEMORY_READ_AX();
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_X);
|
|
}
|
|
|
|
/** D9: LDA $xxxx,Y - LoaD to A **/
|
|
INSTRUCTION(LDAaY)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
cpu->reg_A = MEMORY_READ_AY();
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_Y);
|
|
}
|
|
|
|
/** 85: STA $xx - STore A to zero page **/
|
|
INSTRUCTION(STAzP)
|
|
{
|
|
TRACEi(("STA $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->memory_page0_write(cpu->memory_opcode_read(cpu->reg_PC++), cpu->reg_A);
|
|
}
|
|
|
|
/** 95: STA $xx,X - STore A **/
|
|
INSTRUCTION(STAzX)
|
|
{
|
|
TRACEi(("STA $%02X,X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
MEMORY_WRITE_ZX(cpu->reg_A);
|
|
}
|
|
|
|
/** 81: STA ($xx,X) - STore A **/
|
|
INSTRUCTION(STAiX)
|
|
{
|
|
TRACEi(("STA ($%02X,X)", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
MEMORY_WRITE_IX(cpu->reg_A);
|
|
}
|
|
|
|
/** 91: STA ($xx),Y - STore A **/
|
|
INSTRUCTION(STAiY)
|
|
{
|
|
TRACEi(("STA ($%02X),Y", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
MEMORY_WRITE_IY(cpu->reg_A);
|
|
}
|
|
|
|
/** 8D: STA $xxxx - STore A **/
|
|
INSTRUCTION(STAaB)
|
|
{
|
|
TRACEi(("STA $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
MEMORY_WRITE_AB(cpu->reg_A);
|
|
}
|
|
|
|
/** 9D: STA $xxxx,X - STore A **/
|
|
INSTRUCTION(STAaX)
|
|
{
|
|
TRACEi(("STA $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
MEMORY_WRITE_AX(cpu->reg_A);
|
|
}
|
|
|
|
/** 99: STA $xxxx,Y - STore A **/
|
|
INSTRUCTION(STAaY)
|
|
{
|
|
TRACEi(("STA $%02X%02X,Y", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
MEMORY_WRITE_AY(cpu->reg_A);
|
|
}
|
|
|
|
/** A6: LDX $xx - LoaD to X from zero page **/
|
|
INSTRUCTION(LDXzP)
|
|
{
|
|
TRACEi(("LDX $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_X = MEMORY_READ_ZP();
|
|
NZ_FLAG_UPDATE(cpu->reg_X);
|
|
}
|
|
|
|
/** B6: LDX $xx,Y - LoaD to X **/
|
|
INSTRUCTION(LDXzY)
|
|
{
|
|
TRACEi(("LDX $%02X,Y", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_X = MEMORY_READ_ZY();
|
|
NZ_FLAG_UPDATE(cpu->reg_X);
|
|
}
|
|
|
|
/** AE: LDX $xxxx - LoaD to X **/
|
|
INSTRUCTION(LDXaB)
|
|
{
|
|
TRACEi(("LDX $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_X = MEMORY_READ_AB();
|
|
NZ_FLAG_UPDATE(cpu->reg_X);
|
|
}
|
|
|
|
/** BE: LDX $xxxx,Y - LoaD to X **/
|
|
INSTRUCTION(LDXaY)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
TRACEi(("LDX $%02X%02X,Y", op2, op1));
|
|
cpu->reg_X = MEMORY_READ_AY();
|
|
NZ_FLAG_UPDATE(cpu->reg_X);
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_Y);
|
|
}
|
|
|
|
/** B6: STX $xx - STore X to zero page **/
|
|
INSTRUCTION(STXzP)
|
|
{
|
|
TRACEi(("STX $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->memory_page0_write(cpu->memory_opcode_read(cpu->reg_PC++), cpu->reg_X);
|
|
}
|
|
|
|
/** 96: STX $xx,Y - STore X **/
|
|
INSTRUCTION(STXzY)
|
|
{
|
|
TRACEi(("STX $%02X,Y", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
MEMORY_WRITE_ZY(cpu->reg_X);
|
|
}
|
|
|
|
/** 8E: STX $xxxx - STore X **/
|
|
INSTRUCTION(STXaB)
|
|
{
|
|
TRACEi(("STX $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
MEMORY_WRITE_AB(cpu->reg_X);
|
|
}
|
|
|
|
/** A4: LDY $xx - LoaD to Y from zero page **/
|
|
INSTRUCTION(LDYzP)
|
|
{
|
|
TRACEi(("LDY $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_Y = MEMORY_READ_ZP();
|
|
NZ_FLAG_UPDATE(cpu->reg_Y);
|
|
}
|
|
|
|
/** B4: LDY $xx,X - LoaD to Y **/
|
|
INSTRUCTION(LDYzX)
|
|
{
|
|
TRACEi(("LDY $%02X,X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_Y = MEMORY_READ_ZX();
|
|
NZ_FLAG_UPDATE(cpu->reg_Y);
|
|
}
|
|
|
|
/** AC: LDY $xxxx - LoaD to Y **/
|
|
INSTRUCTION(LDYaB)
|
|
{
|
|
TRACEi(("LDY $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_Y = MEMORY_READ_AB();
|
|
NZ_FLAG_UPDATE(cpu->reg_Y);
|
|
}
|
|
|
|
/** BC: LDY $xxxx,X - LoaD to Y **/
|
|
INSTRUCTION(LDYaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("LDY $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_Y = MEMORY_READ_AX();
|
|
NZ_FLAG_UPDATE(cpu->reg_Y);
|
|
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_X);
|
|
}
|
|
|
|
/** 84: STY $xx - STore Y to zero page **/
|
|
INSTRUCTION(STYzP)
|
|
{
|
|
TRACEi(("STY $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->memory_page0_write(cpu->memory_opcode_read(cpu->reg_PC++), cpu->reg_Y);
|
|
}
|
|
|
|
/** 94: STY $xx,X - STore Y **/
|
|
INSTRUCTION(STYzX)
|
|
{
|
|
TRACEi(("STY $%02X,X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
MEMORY_WRITE_ZX(cpu->reg_Y);
|
|
}
|
|
|
|
/** 8C: STY $xxxx - STore Y **/
|
|
INSTRUCTION(STYaB)
|
|
{
|
|
TRACEi(("STY $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
MEMORY_WRITE_AB(cpu->reg_Y);
|
|
}
|
|
|
|
/**** Register functions ****/
|
|
|
|
/** AA : TAX - Transfer A to X **/
|
|
INSTRUCTION(TAXiM)
|
|
{
|
|
TRACEi(("TAX"));
|
|
cpu->reg_X = cpu->reg_A;
|
|
NZ_FLAG_UPDATE(cpu->reg_X);
|
|
}
|
|
|
|
/** 8A : TXA - Transfer X to A **/
|
|
INSTRUCTION(TXAiM)
|
|
{
|
|
TRACEi(("TXA"));
|
|
cpu->reg_A = cpu->reg_X;
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
}
|
|
|
|
/** A8 : TAY - Transfer A to Y **/
|
|
INSTRUCTION(TAYiM)
|
|
{
|
|
TRACEi(("TAY"));
|
|
cpu->reg_Y = cpu->reg_A;
|
|
NZ_FLAG_UPDATE(cpu->reg_Y);
|
|
}
|
|
|
|
/** 98 : TYA - Transfer Y to A **/
|
|
INSTRUCTION(TYAiM)
|
|
{
|
|
TRACEi(("TYA"));
|
|
cpu->reg_A = cpu->reg_Y;
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
}
|
|
|
|
/* BA : TSX - Transfer S to X **/
|
|
INSTRUCTION(TSXiM)
|
|
{
|
|
TRACEi(("TSX"));
|
|
cpu->reg_X = cpu->reg_S;
|
|
NZ_FLAG_UPDATE(cpu->reg_X);
|
|
}
|
|
|
|
/** 9A : TXS - Transfer X to S **/
|
|
INSTRUCTION(TXSiM)
|
|
{
|
|
TRACEi(("TXS"));
|
|
cpu->reg_S = cpu->reg_X;
|
|
}
|
|
|
|
/**** Simple register operation instructions ****/
|
|
|
|
/** CA : DEX - DEcrement X **/
|
|
INSTRUCTION(DEXiM)
|
|
{
|
|
TRACEi(("DEX"));
|
|
cpu->reg_X --;
|
|
NZ_FLAG_UPDATE(cpu->reg_X);
|
|
}
|
|
|
|
/** 88 : DEY - DEcrement Y **/
|
|
INSTRUCTION(DEYiM)
|
|
{
|
|
TRACEi(("DEY"));
|
|
cpu->reg_Y --;
|
|
NZ_FLAG_UPDATE(cpu->reg_Y);
|
|
}
|
|
|
|
/** E8 : INX - INcrement X **/
|
|
INSTRUCTION(INXiM)
|
|
{
|
|
TRACEi(("INX"));
|
|
cpu->reg_X ++;
|
|
NZ_FLAG_UPDATE(cpu->reg_X);
|
|
}
|
|
|
|
/** C8 : INY - INcrement Y **/
|
|
INSTRUCTION(INYiM)
|
|
{
|
|
TRACEi(("INY"));
|
|
cpu->reg_Y ++;
|
|
NZ_FLAG_UPDATE(cpu->reg_Y);
|
|
}
|
|
|
|
/**** Stack related instructions ****/
|
|
|
|
/** 48 : PHA - PusH A */
|
|
INSTRUCTION(PHAiM)
|
|
{
|
|
TRACEi(("PHA"));
|
|
PUSH_S(cpu->reg_A);
|
|
}
|
|
|
|
/** 68 : PLA - PuLl A */
|
|
INSTRUCTION(PLAiM)
|
|
{
|
|
TRACEi(("PLA"));
|
|
cpu->reg_A = POP_S();
|
|
NZ_FLAG_UPDATE(cpu->reg_A);
|
|
}
|
|
|
|
/** 08 : PHP - PusH P */
|
|
INSTRUCTION(PHPiM)
|
|
{
|
|
TRACEi(("PHP"));
|
|
PUSH_S((cpu->reg_P | Q6502_R_FLAG | Q6502_B_FLAG));
|
|
}
|
|
|
|
/** 28 : PLP - PuLl P */
|
|
INSTRUCTION(PLPiM)
|
|
{
|
|
TRACEi(("PLP"));
|
|
cpu->reg_P = POP_S() & ~(Q6502_R_FLAG | Q6502_B_FLAG);
|
|
if (cpu->int_pending != 0)
|
|
{
|
|
quick6502_int(cpu, Q6502_IRQ_SIGNAL);
|
|
}
|
|
}
|
|
|
|
/**** Branch instructions ****/
|
|
|
|
/** 20 : JSR $xxxx - Jump to SubRoutine */
|
|
INSTRUCTION(JSRaB)
|
|
{
|
|
TRACEi(("JSR $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_PC++;
|
|
PUSH_S((cpu->reg_PC >> 8) & 0xFF);
|
|
PUSH_S((cpu->reg_PC ) & 0xFF);
|
|
cpu->reg_PC = ((cpu->memory_opcode_read(cpu->reg_PC-1) ) | (cpu->memory_opcode_read(cpu->reg_PC) << 8));
|
|
}
|
|
|
|
/** 60 : RTS - ReTurn from Subrutine */
|
|
INSTRUCTION(RTSiM)
|
|
{
|
|
TRACEi(("RTS"));
|
|
cpu->reg_PC = POP_S() | (POP_S() << 8);
|
|
cpu->reg_PC ++;
|
|
}
|
|
|
|
/** 4C : JMP $xxxx - JuMP inconditionaly to $xxxx **/
|
|
INSTRUCTION(JMPaB)
|
|
{
|
|
TRACEi(("JMP $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_PC = cpu->memory_opcode_read(cpu->reg_PC++) | (cpu->memory_opcode_read(cpu->reg_PC) << 8);
|
|
}
|
|
|
|
/** 6C : JMP ($xxxx) - JuMP inconditionaly to ($xxxx) **/
|
|
INSTRUCTION(JMPiD)
|
|
{
|
|
TRACEi(("JMP ($%02X%02X)", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
cpu->reg_PC = cpu->memory_opcode_read(cpu->reg_PC) | (cpu->memory_opcode_read(cpu->reg_PC+1) << 8);
|
|
cpu->reg_PC = cpu->memory_read(cpu->reg_PC) |
|
|
(cpu->memory_read((cpu->reg_PC & 0xFF00) | ((cpu->reg_PC+1) & 0x00FF)) << 8);
|
|
}
|
|
|
|
/** 00 : BRK - BReaK **/
|
|
INSTRUCTION(BRKiM)
|
|
{
|
|
TRACEi(("BRK"));
|
|
cpu->reg_PC++;
|
|
PUSH_S((cpu->reg_PC >> 8) & 0xFF);
|
|
PUSH_S((cpu->reg_PC ) & 0xFF);
|
|
PUSH_S( cpu->reg_P );
|
|
cpu->reg_P = cpu->reg_P | Q6502_I_FLAG | Q6502_B_FLAG;
|
|
|
|
cpu->reg_PC = (cpu->memory_read(Q6502_IRQ_LOW)) | (cpu->memory_read(Q6502_IRQ_HIGH)<<8);
|
|
}
|
|
|
|
/** 40 : RTI - ReTurn from Interruption **/
|
|
INSTRUCTION(RTIiM)
|
|
{
|
|
TRACEi(("RTI"));
|
|
cpu->reg_P = POP_S();
|
|
cpu->reg_PC = POP_S() | (POP_S() << 8);
|
|
|
|
if (cpu->int_pending != 0)
|
|
{
|
|
quick6502_int(cpu, Q6502_IRQ_SIGNAL);
|
|
}
|
|
}
|
|
|
|
/** 90 : BCC - Branch if Carry Clear **/
|
|
INSTRUCTION(BCCrE)
|
|
{
|
|
TRACEi(("BCC $%04X", cpu->reg_PC + (signed char) cpu->memory_opcode_read(cpu->reg_PC) + 1));
|
|
if (!(cpu->reg_P & Q6502_C_FLAG))
|
|
{
|
|
cpu->reg_PC += (signed char) cpu->memory_opcode_read(cpu->reg_PC++) + 1;
|
|
/* Need to set timing */
|
|
|
|
/* +1 is same page */
|
|
cpu->cycle_done += 1;
|
|
/* +2 is another */
|
|
}
|
|
else
|
|
cpu->reg_PC ++;
|
|
}
|
|
|
|
/** B0 : BCS - Branch if Carry Set**/
|
|
INSTRUCTION(BCSrE)
|
|
{
|
|
TRACEi(("BCS $%04X", cpu->reg_PC + (signed char) cpu->memory_opcode_read(cpu->reg_PC) + 1));
|
|
if (cpu->reg_P & Q6502_C_FLAG)
|
|
{
|
|
cpu->reg_PC += (signed char) cpu->memory_opcode_read(cpu->reg_PC ++) + 1;
|
|
/* Need to set timing */
|
|
|
|
/* +1 is same page */
|
|
cpu->cycle_done += 1;
|
|
/* +2 is another */
|
|
}
|
|
else
|
|
cpu->reg_PC ++;
|
|
}
|
|
|
|
/** F0 : BEQ - Branch if Equal**/
|
|
INSTRUCTION(BEQrE)
|
|
{
|
|
TRACEi(("BEQ $%04X", cpu->reg_PC + (signed char) cpu->memory_opcode_read(cpu->reg_PC) + 1));
|
|
if (cpu->reg_P & Q6502_Z_FLAG)
|
|
{
|
|
cpu->reg_PC += (signed char) cpu->memory_opcode_read(cpu->reg_PC ++) + 1;
|
|
/* Need to set timing */
|
|
|
|
/* +1 is same page */
|
|
cpu->cycle_done += 1;
|
|
/* +2 is another */
|
|
}
|
|
else
|
|
cpu->reg_PC ++;
|
|
}
|
|
|
|
/** 30 : BMI - Branch if MInus**/
|
|
INSTRUCTION(BMIrE)
|
|
{
|
|
TRACEi(("BMI $%04X", cpu->reg_PC + (signed char) cpu->memory_opcode_read(cpu->reg_PC) + 1));
|
|
if (cpu->reg_P & Q6502_N_FLAG)
|
|
{
|
|
cpu->reg_PC += (signed char) cpu->memory_opcode_read(cpu->reg_PC ++) + 1;
|
|
/* Need to set timing */
|
|
|
|
/* +1 is same page */
|
|
cpu->cycle_done += 1;
|
|
/* +2 is another */
|
|
}
|
|
else
|
|
cpu->reg_PC ++;
|
|
}
|
|
|
|
/** D0 : Bxx - Branch if Not Equal**/
|
|
INSTRUCTION(BNErE)
|
|
{
|
|
TRACEi(("BNE $%04X", cpu->reg_PC + (signed char) cpu->memory_opcode_read(cpu->reg_PC) + 1));
|
|
if (!(cpu->reg_P & Q6502_Z_FLAG))
|
|
{
|
|
cpu->reg_PC += (signed char) cpu->memory_opcode_read(cpu->reg_PC ++) + 1;
|
|
/* Need to set timing */
|
|
|
|
/* +1 is same page */
|
|
cpu->cycle_done += 1;
|
|
/* +2 is another */
|
|
}
|
|
else
|
|
cpu->reg_PC ++;
|
|
}
|
|
|
|
/** 10 : BPL - Branch if PLus **/
|
|
INSTRUCTION(BPLrE)
|
|
{
|
|
TRACEi(("BPL $%04X", cpu->reg_PC + (signed char) cpu->memory_opcode_read(cpu->reg_PC) + 1));
|
|
if (!(cpu->reg_P & Q6502_N_FLAG))
|
|
{
|
|
cpu->reg_PC += (signed char) cpu->memory_opcode_read(cpu->reg_PC ++) + 1;
|
|
/* Need to set timing */
|
|
|
|
/* +1 is same page */
|
|
cpu->cycle_done += 1;
|
|
/* +2 is another */
|
|
}
|
|
else
|
|
cpu->reg_PC ++;
|
|
}
|
|
|
|
/** 50 : BVC - Branch if oVerflow Clear**/
|
|
INSTRUCTION(BVCrE)
|
|
{
|
|
TRACEi(("BVC $%04X", cpu->reg_PC + (signed char) cpu->memory_opcode_read(cpu->reg_PC) + 1));
|
|
if (!(cpu->reg_P & Q6502_V_FLAG))
|
|
{
|
|
cpu->reg_PC += (signed char) cpu->memory_opcode_read(cpu->reg_PC ++) + 1;
|
|
/* Need to set timing */
|
|
|
|
/* +1 is same page */
|
|
cpu->cycle_done += 1;
|
|
/* +2 is another */
|
|
}
|
|
else
|
|
cpu->reg_PC ++;
|
|
}
|
|
|
|
/** 70 : BVS - Branch if oVerflow Set**/
|
|
INSTRUCTION(BVSrE)
|
|
{
|
|
TRACEi(("BVS $%04X", cpu->reg_PC + (signed char) cpu->memory_opcode_read(cpu->reg_PC) + 1));
|
|
if (cpu->reg_P & Q6502_V_FLAG)
|
|
{
|
|
cpu->reg_PC += (signed char) cpu->memory_opcode_read(cpu->reg_PC ++) + 1;
|
|
/* Need to set timing */
|
|
|
|
/* +1 is same page */
|
|
cpu->cycle_done += 1;
|
|
/* +2 is another */
|
|
}
|
|
else
|
|
cpu->reg_PC ++;
|
|
}
|
|
|
|
/*** Mathematical functions ***/
|
|
|
|
/** 69 : ADC - ADd with Carry **/
|
|
INSTRUCTION(ADCiM)
|
|
{
|
|
TRACEi(("ADC #$%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
ADC_OPERATION(cpu->memory_opcode_read(cpu->reg_PC++));
|
|
}
|
|
|
|
/** 65 : ADC - ADd with Carry **/
|
|
INSTRUCTION(ADCzP)
|
|
{
|
|
TRACEi(("ADC $%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
ADC_OPERATION(MEMORY_READ_ZP());
|
|
}
|
|
|
|
/** 75 : ADC - ADd with Carry **/
|
|
INSTRUCTION(ADCzX)
|
|
{
|
|
TRACEi(("ADC $%02X,X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
ADC_OPERATION(MEMORY_READ_ZX());
|
|
}
|
|
|
|
/** 61 : ADC - ADd with Carry **/
|
|
INSTRUCTION(ADCiX)
|
|
{
|
|
TRACEi(("ADC ($%02X,X)", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
ADC_OPERATION(MEMORY_READ_IX());
|
|
}
|
|
|
|
/** 71 : ADC - ADd with Carry **/
|
|
INSTRUCTION(ADCiY)
|
|
{
|
|
TRACEi(("ADC ($%02X),Y", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
ADC_OPERATION(MEMORY_READ_IY());
|
|
CROSS_CYCLE_UPDATE(cpu->memory_opcode_read(cpu->reg_PC-1) + cpu->reg_Y);
|
|
}
|
|
|
|
/** 6D : ADC - ADd with Carry **/
|
|
INSTRUCTION(ADCaB)
|
|
{
|
|
TRACEi(("ADC $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
ADC_OPERATION(MEMORY_READ_AB());
|
|
}
|
|
|
|
/** 7D : ADC - ADd with Carry **/
|
|
INSTRUCTION(ADCaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("ADC $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
ADC_OPERATION(MEMORY_READ_AX());
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_X);
|
|
}
|
|
|
|
/** 79 : ADC - ADd with Carry **/
|
|
INSTRUCTION(ADCaY)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
TRACEi(("ADC $%02X%02X,Y", op2, op1));
|
|
ADC_OPERATION(MEMORY_READ_AY());
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_Y);
|
|
}
|
|
|
|
/** E9 : SBC - SuBstract with Carry **/
|
|
INSTRUCTION(SBCiM)
|
|
{
|
|
TRACEi(("SBC #$%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
SBC_OPERATION(cpu->memory_opcode_read(cpu->reg_PC++));
|
|
}
|
|
|
|
/** E5 : SBC - SuBstract with Carry **/
|
|
INSTRUCTION(SBCzP)
|
|
{
|
|
TRACEi(("SBC $%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
SBC_OPERATION(MEMORY_READ_ZP());
|
|
}
|
|
|
|
/** F5 : SBC - SuBstract with Carry **/
|
|
INSTRUCTION(SBCzX)
|
|
{
|
|
TRACEi(("SBC $%02X,X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
SBC_OPERATION(MEMORY_READ_ZX());
|
|
}
|
|
|
|
/** E1 : SBC - SuBstract with Carry **/
|
|
INSTRUCTION(SBCiX)
|
|
{
|
|
TRACEi(("SBC ($%02X,X)", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
SBC_OPERATION(MEMORY_READ_IX());
|
|
}
|
|
|
|
/** F1 : SBC - SuBstract with Carry **/
|
|
INSTRUCTION(SBCiY)
|
|
{
|
|
TRACEi(("SBC ($%02X),Y", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
SBC_OPERATION(MEMORY_READ_IY());
|
|
CROSS_CYCLE_UPDATE(cpu->memory_opcode_read(cpu->reg_PC-1) + cpu->reg_Y);
|
|
}
|
|
|
|
/** ED : SBC - SuBstract with Carry **/
|
|
INSTRUCTION(SBCaB)
|
|
{
|
|
TRACEi(("SBC $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
SBC_OPERATION(MEMORY_READ_AB());
|
|
}
|
|
|
|
/** FD : SBC - SuBstract with Carry **/
|
|
INSTRUCTION(SBCaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("SBC $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
SBC_OPERATION(MEMORY_READ_AX());
|
|
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_X);
|
|
}
|
|
|
|
/** F9 : SBC - SuBstract with Carry **/
|
|
INSTRUCTION(SBCaY)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
TRACEi(("SBC $%02X%02X,Y", op2, op1));
|
|
SBC_OPERATION(MEMORY_READ_AY());
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_Y);
|
|
}
|
|
|
|
/** C9 : CMP - CoMPare **/
|
|
INSTRUCTION(CMPiM)
|
|
{
|
|
TRACEi(("CMP #$%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
CMP_OPERATION(cpu->reg_A, cpu->memory_opcode_read(cpu->reg_PC++));
|
|
}
|
|
|
|
/** C5 : CMP - CoMPare **/
|
|
INSTRUCTION(CMPzP)
|
|
{
|
|
TRACEi(("CMP $%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
CMP_OPERATION(cpu->reg_A, MEMORY_READ_ZP());
|
|
}
|
|
|
|
/** D5 : CMP - CoMPare **/
|
|
INSTRUCTION(CMPzX)
|
|
{
|
|
TRACEi(("CMP $%02X,X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
CMP_OPERATION(cpu->reg_A, MEMORY_READ_ZX());
|
|
}
|
|
|
|
/** C1 : CMP - CoMPare **/
|
|
INSTRUCTION(CMPiX)
|
|
{
|
|
TRACEi(("CMP ($%02X,X)", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
CMP_OPERATION(cpu->reg_A, MEMORY_READ_IX());
|
|
}
|
|
|
|
/** D1 : CMP - CoMPare **/
|
|
INSTRUCTION(CMPiY)
|
|
{
|
|
TRACEi(("CMP ($%02X),Y", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
CMP_OPERATION(cpu->reg_A, MEMORY_READ_IY());
|
|
CROSS_CYCLE_UPDATE(cpu->memory_opcode_read(cpu->reg_PC-1) + cpu->reg_Y);
|
|
}
|
|
|
|
/** CD : CMP - CoMPare **/
|
|
INSTRUCTION(CMPaB)
|
|
{
|
|
TRACEi(("CMP $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
CMP_OPERATION(cpu->reg_A, MEMORY_READ_AB());
|
|
}
|
|
|
|
/** DD : CMP - CoMPare **/
|
|
INSTRUCTION(CMPaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("CMP $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
CMP_OPERATION(cpu->reg_A, MEMORY_READ_AX());
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_X);
|
|
}
|
|
|
|
/** D9 : CMP - CoMPare **/
|
|
INSTRUCTION(CMPaY)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
TRACEi(("CMP $%02X%02X,Y", op2, op1));
|
|
CMP_OPERATION(cpu->reg_A, MEMORY_READ_AY());
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_Y);
|
|
}
|
|
|
|
/** E0 : CPX - ComPare with Y **/
|
|
INSTRUCTION(CPXiM)
|
|
{
|
|
TRACEi(("CPX #$%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
CMP_OPERATION(cpu->reg_X, cpu->memory_opcode_read(cpu->reg_PC++));
|
|
}
|
|
|
|
/** E4 : CPX - ComPare with X **/
|
|
INSTRUCTION(CPXzP)
|
|
{
|
|
TRACEi(("CPX $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
CMP_OPERATION(cpu->reg_X, MEMORY_READ_ZP());
|
|
}
|
|
|
|
/** EC : CPX - ComPare with X **/
|
|
INSTRUCTION(CPXaB)
|
|
{
|
|
TRACEi(("CPX $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
CMP_OPERATION(cpu->reg_X, MEMORY_READ_AB());
|
|
}
|
|
|
|
/** C0 : CPY - ComPare with Y **/
|
|
INSTRUCTION(CPYiM)
|
|
{
|
|
TRACEi(("CPY #$%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
CMP_OPERATION(cpu->reg_Y, cpu->memory_opcode_read(cpu->reg_PC++));
|
|
}
|
|
|
|
/** C4 : CPY - ComPare with Y **/
|
|
INSTRUCTION(CPYzP)
|
|
{
|
|
TRACEi(("CPY $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
CMP_OPERATION(cpu->reg_Y, MEMORY_READ_ZP());
|
|
}
|
|
|
|
/** CC : CPY - ComPare with Y **/
|
|
INSTRUCTION(CPYaB)
|
|
{
|
|
TRACEi(("CPY $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
CMP_OPERATION(cpu->reg_Y, MEMORY_READ_AB());
|
|
}
|
|
|
|
/** 09 : ORA - OR with A **/
|
|
INSTRUCTION(ORAiM)
|
|
{
|
|
TRACEi(("ORA #$%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
ORA_OPERATION(cpu->memory_opcode_read(cpu->reg_PC++));
|
|
}
|
|
|
|
/** 405 : ORA - OR with A **/
|
|
INSTRUCTION(ORAzP)
|
|
{
|
|
TRACEi(("ORA $%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
ORA_OPERATION(cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC++)));
|
|
}
|
|
|
|
/** 15 : ORA - OR with A **/
|
|
INSTRUCTION(ORAzX)
|
|
{
|
|
TRACEi(("ORA $%02X,X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
ORA_OPERATION(MEMORY_READ_ZX());
|
|
}
|
|
|
|
/** 01 : ORA - OR with A **/
|
|
INSTRUCTION(ORAiX)
|
|
{
|
|
TRACEi(("ORA ($%02X,X)", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
ORA_OPERATION(MEMORY_READ_IX());
|
|
}
|
|
|
|
/** 11 : ORA - OR with A **/
|
|
INSTRUCTION(ORAiY)
|
|
{
|
|
TRACEi(("ORA ($%02X),Y", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
ORA_OPERATION(MEMORY_READ_IY());
|
|
CROSS_CYCLE_UPDATE(cpu->memory_opcode_read(cpu->reg_PC-1) + cpu->reg_Y);
|
|
}
|
|
|
|
/** 0D : ORA - OR with A **/
|
|
INSTRUCTION(ORAaB)
|
|
{
|
|
TRACEi(("ORA $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
ORA_OPERATION(MEMORY_READ_AB());
|
|
}
|
|
|
|
/** 1D : ORA - OR with A **/
|
|
INSTRUCTION(ORAaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("ORA $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
ORA_OPERATION(MEMORY_READ_AX());
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_X);
|
|
}
|
|
|
|
/** 19 : ORA - OR with A **/
|
|
INSTRUCTION(ORAaY)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("ORA $%02X%02X,Y", op2, op1));
|
|
ORA_OPERATION(MEMORY_READ_AY());
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_Y);
|
|
}
|
|
|
|
/** 49 : EOR - Exclusive OR **/
|
|
INSTRUCTION(EORiM)
|
|
{
|
|
TRACEi(("EOR #$%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
EOR_OPERATION(cpu->memory_opcode_read(cpu->reg_PC++));
|
|
}
|
|
|
|
/** 45 : EOR - Exclusive OR **/
|
|
INSTRUCTION(EORzP)
|
|
{
|
|
TRACEi(("EOR $%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
EOR_OPERATION(cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC++)));
|
|
}
|
|
|
|
/** 55 : EOR - Exclusive OR **/
|
|
INSTRUCTION(EORzX)
|
|
{
|
|
TRACEi(("EOR $%02X,X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
EOR_OPERATION(MEMORY_READ_ZX());
|
|
}
|
|
|
|
/** 41 : EOR - Exclusive OR **/
|
|
INSTRUCTION(EORiX)
|
|
{
|
|
TRACEi(("EOR ($%02X,X)", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
EOR_OPERATION(MEMORY_READ_IX());
|
|
}
|
|
|
|
/** 51 : EOR - Exclusive OR **/
|
|
INSTRUCTION(EORiY)
|
|
{
|
|
TRACEi(("EOR ($%02X),Y", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
EOR_OPERATION(MEMORY_READ_IY());
|
|
|
|
CROSS_CYCLE_UPDATE(cpu->memory_opcode_read(cpu->reg_PC-1) + cpu->reg_Y);
|
|
}
|
|
|
|
/** 4D : EOR - Exclusive OR **/
|
|
INSTRUCTION(EORaB)
|
|
{
|
|
TRACEi(("EOR $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
EOR_OPERATION(MEMORY_READ_AB());
|
|
}
|
|
|
|
/** 5D : EOR - Exclusive OR **/
|
|
INSTRUCTION(EORaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("EOR $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
EOR_OPERATION(MEMORY_READ_AX());
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_X);
|
|
}
|
|
|
|
/** 59 : EOR - Exclusive OR **/
|
|
INSTRUCTION(EORaY)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
TRACEi(("EOR $%02X%02X,Y", op2, op1));
|
|
EOR_OPERATION(MEMORY_READ_AY());
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_Y);
|
|
}
|
|
|
|
/** 29 : AND - Logical AND **/
|
|
INSTRUCTION(ANDiM)
|
|
{
|
|
TRACEi(("AND #$%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
AND_OPERATION(cpu->memory_opcode_read(cpu->reg_PC++));
|
|
}
|
|
|
|
/** 25 : AND - Logical AND **/
|
|
INSTRUCTION(ANDzP)
|
|
{
|
|
TRACEi(("AND $%02X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
AND_OPERATION(cpu->memory_page0_read(cpu->memory_opcode_read(cpu->reg_PC++)));
|
|
}
|
|
|
|
/** 35 : AND - Logical AND **/
|
|
INSTRUCTION(ANDzX)
|
|
{
|
|
TRACEi(("AND $%02X,X", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
AND_OPERATION(MEMORY_READ_ZX());
|
|
}
|
|
|
|
/** 21 : AND - Logical AND **/
|
|
INSTRUCTION(ANDiX)
|
|
{
|
|
TRACEi(("AND ($%02X,X)", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
AND_OPERATION(MEMORY_READ_IX());
|
|
}
|
|
|
|
/** 31 : AND - Logical AND **/
|
|
INSTRUCTION(ANDiY)
|
|
{
|
|
TRACEi(("AND ($%02X),Y", cpu->memory_opcode_read(cpu->reg_PC) ));
|
|
AND_OPERATION(MEMORY_READ_IY());
|
|
|
|
CROSS_CYCLE_UPDATE(cpu->memory_opcode_read(cpu->reg_PC-1) + cpu->reg_Y);
|
|
}
|
|
|
|
/** 2D : AND - Logical AND **/
|
|
INSTRUCTION(ANDaB)
|
|
{
|
|
TRACEi(("AND $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
AND_OPERATION(MEMORY_READ_AB());
|
|
}
|
|
|
|
/** 3D : AND - Logical AND **/
|
|
INSTRUCTION(ANDaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("AND $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
AND_OPERATION(MEMORY_READ_AX());
|
|
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_X);
|
|
}
|
|
|
|
/** 39 : AND - Logical AND **/
|
|
INSTRUCTION(ANDaY)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
TRACEi(("AND $%02X%02X,Y", op2, op1));
|
|
AND_OPERATION(MEMORY_READ_AY());
|
|
CROSS_CYCLE_UPDATE(op1 + cpu->reg_Y);
|
|
}
|
|
|
|
/*** Misc instructions ***/
|
|
|
|
/** 24 : BIT **/
|
|
INSTRUCTION(BITzP)
|
|
{
|
|
TRACEi(("BIT $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
BIT_OPERATION(MEMORY_READ_ZP());
|
|
}
|
|
|
|
/** 2C : BIT **/
|
|
INSTRUCTION(BITaB)
|
|
{
|
|
TRACEi(("BIT $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
BIT_OPERATION(MEMORY_READ_AB());
|
|
}
|
|
|
|
/** 2A : ROL A **/
|
|
INSTRUCTION(ROLiM)
|
|
{
|
|
TRACEi(("ROL A"));
|
|
ROL_OPERATION(cpu->reg_A);
|
|
}
|
|
|
|
/** 6A : ROR A **/
|
|
INSTRUCTION(RORiM)
|
|
{
|
|
TRACEi(("ROR A"));
|
|
ROR_OPERATION(cpu->reg_A);
|
|
}
|
|
|
|
/** 0A : ASL A **/
|
|
INSTRUCTION(ASLiM)
|
|
{
|
|
TRACEi(("ASL A"));
|
|
ASL_OPERATION(cpu->reg_A);
|
|
}
|
|
|
|
/** 4A : LSR A **/
|
|
INSTRUCTION(LSRiM)
|
|
{
|
|
TRACEi(("LSR A"));
|
|
LSR_OPERATION(cpu->reg_A);
|
|
}
|
|
|
|
/** 2E : ROL **/
|
|
INSTRUCTION(ROLaB)
|
|
{
|
|
TRACEi(("ROL $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AB();
|
|
cpu->reg_PC -= 2;
|
|
ROL_OPERATION(val);
|
|
MEMORY_WRITE_AB(val);
|
|
}
|
|
|
|
/** 26 : ROL **/
|
|
INSTRUCTION(ROLzP)
|
|
{
|
|
TRACEi(("ROL $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZP();
|
|
cpu->reg_PC -= 1;
|
|
ROL_OPERATION(val);
|
|
MEMORY_WRITE_ZP(val);
|
|
}
|
|
|
|
/** 3E : ROL **/
|
|
INSTRUCTION(ROLaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("ROL $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AX();
|
|
cpu->reg_PC -= 2;
|
|
ROL_OPERATION(val);
|
|
MEMORY_WRITE_AX(val);
|
|
}
|
|
|
|
/** 36 : ROL **/
|
|
INSTRUCTION(ROLzX)
|
|
{
|
|
TRACEi(("ROL $%02X,X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZX();
|
|
cpu->reg_PC -= 1;
|
|
ROL_OPERATION(val);
|
|
MEMORY_WRITE_ZX(val);
|
|
}
|
|
|
|
/** 6E : ROR **/
|
|
INSTRUCTION(RORaB)
|
|
{
|
|
TRACEi(("ROR $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AB();
|
|
cpu->reg_PC -= 2;
|
|
ROR_OPERATION(val);
|
|
MEMORY_WRITE_AB(val);
|
|
}
|
|
|
|
/** 66 : ROR **/
|
|
INSTRUCTION(RORzP)
|
|
{
|
|
TRACEi(("ROR $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZP();
|
|
cpu->reg_PC -= 1;
|
|
ROR_OPERATION(val);
|
|
MEMORY_WRITE_ZP(val);
|
|
}
|
|
|
|
/** 7E : ROR **/
|
|
INSTRUCTION(RORaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("ROR $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AX();
|
|
cpu->reg_PC -= 2;
|
|
ROR_OPERATION(val);
|
|
MEMORY_WRITE_AX(val);
|
|
}
|
|
|
|
/** 76 : ROR **/
|
|
INSTRUCTION(RORzX)
|
|
{
|
|
TRACEi(("ROR $%02X,X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZX();
|
|
cpu->reg_PC -= 1;
|
|
ROR_OPERATION(val);
|
|
MEMORY_WRITE_ZX(val);
|
|
}
|
|
|
|
/** 0E : ASL **/
|
|
INSTRUCTION(ASLaB)
|
|
{
|
|
TRACEi(("ASL $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AB();
|
|
cpu->reg_PC -= 2;
|
|
ASL_OPERATION(val);
|
|
MEMORY_WRITE_AB(val);
|
|
}
|
|
|
|
/** 06 : ASL **/
|
|
INSTRUCTION(ASLzP)
|
|
{
|
|
TRACEi(("ASL $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZP();
|
|
cpu->reg_PC -= 1;
|
|
ASL_OPERATION(val);
|
|
MEMORY_WRITE_ZP(val);
|
|
}
|
|
|
|
/** 1E : ASL **/
|
|
INSTRUCTION(ASLaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("ASL $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AX();
|
|
cpu->reg_PC -= 2;
|
|
ASL_OPERATION(val);
|
|
MEMORY_WRITE_AX(val);
|
|
}
|
|
|
|
/** 16 : ASL **/
|
|
INSTRUCTION(ASLzX)
|
|
{
|
|
TRACEi(("ASL $%02X,X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZX();
|
|
cpu->reg_PC -= 1;
|
|
ASL_OPERATION(val);
|
|
MEMORY_WRITE_ZX(val);
|
|
}
|
|
|
|
/** 4E : LSR **/
|
|
INSTRUCTION(LSRaB)
|
|
{
|
|
TRACEi(("LSR $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AB();
|
|
cpu->reg_PC -= 2;
|
|
LSR_OPERATION(val);
|
|
MEMORY_WRITE_AB(val);
|
|
}
|
|
|
|
/** 46 : LSR **/
|
|
INSTRUCTION(LSRzP)
|
|
{
|
|
TRACEi(("LSR $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZP();
|
|
cpu->reg_PC -= 1;
|
|
LSR_OPERATION(val);
|
|
MEMORY_WRITE_ZP(val);
|
|
}
|
|
|
|
/** 5E : LSR **/
|
|
INSTRUCTION(LSRaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("LSR $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AX();
|
|
cpu->reg_PC -= 2;
|
|
LSR_OPERATION(val);
|
|
MEMORY_WRITE_AX(val);
|
|
}
|
|
|
|
/** 56 : LSR **/
|
|
INSTRUCTION(LSRzX)
|
|
{
|
|
TRACEi(("LSR $%02X,X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZX();
|
|
cpu->reg_PC -= 1;
|
|
LSR_OPERATION(val);
|
|
MEMORY_WRITE_ZX(val);
|
|
}
|
|
|
|
/** CE : DEC **/
|
|
INSTRUCTION(DECaB)
|
|
{
|
|
TRACEi(("DEC $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AB();
|
|
cpu->reg_PC -= 2;
|
|
MEMORY_WRITE_AB(--val);
|
|
NZ_FLAG_UPDATE(val);
|
|
}
|
|
|
|
/** C6 : DEC **/
|
|
INSTRUCTION(DECzP)
|
|
{
|
|
TRACEi(("DEC $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZP();
|
|
cpu->reg_PC -= 1;
|
|
MEMORY_WRITE_ZP(--val);
|
|
NZ_FLAG_UPDATE(val);
|
|
}
|
|
|
|
/** DE : DEC **/
|
|
INSTRUCTION(DECaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("DEC $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AX();
|
|
cpu->reg_PC -= 2;
|
|
MEMORY_WRITE_AX(--val);
|
|
NZ_FLAG_UPDATE(val);
|
|
}
|
|
|
|
/** D6 : DEC **/
|
|
INSTRUCTION(DECzX)
|
|
{
|
|
TRACEi(("DEC $%02X,X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZX();
|
|
cpu->reg_PC -= 1;
|
|
MEMORY_WRITE_ZX(--val);
|
|
NZ_FLAG_UPDATE(val);
|
|
}
|
|
|
|
/** EE : INC **/
|
|
INSTRUCTION(INCaB)
|
|
{
|
|
TRACEi(("INC $%02X%02X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AB();
|
|
cpu->reg_PC -= 2;
|
|
MEMORY_WRITE_AB(++val);
|
|
NZ_FLAG_UPDATE(val);
|
|
}
|
|
|
|
/** E6 : INC **/
|
|
INSTRUCTION(INCzP)
|
|
{
|
|
TRACEi(("INC $%02X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZP();
|
|
cpu->reg_PC -= 1;
|
|
MEMORY_WRITE_ZP(++val);
|
|
NZ_FLAG_UPDATE(val);
|
|
}
|
|
|
|
/** FE : INC **/
|
|
INSTRUCTION(INCaX)
|
|
{
|
|
register byte op1 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
register byte op2 = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("INC $%02X%02X,X", cpu->memory_opcode_read(cpu->reg_PC+1), cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_AX();
|
|
cpu->reg_PC -= 2;
|
|
MEMORY_WRITE_AX(++val);
|
|
NZ_FLAG_UPDATE(val);
|
|
}
|
|
|
|
/** F6 : INC **/
|
|
INSTRUCTION(INCzX)
|
|
{
|
|
TRACEi(("INC $%02X,X", cpu->memory_opcode_read(cpu->reg_PC)));
|
|
byte val = MEMORY_READ_ZX();
|
|
cpu->reg_PC -= 1;
|
|
MEMORY_WRITE_ZX(++val);
|
|
NZ_FLAG_UPDATE(val);
|
|
}
|
|
|
|
/* */
|
|
static InstructionFunction InstructionTable[256] =
|
|
{
|
|
/* 00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F */
|
|
/* 00 */ I_BRKiM, I_ORAiX, I_ILLEG, I_ILLEG, I_ILLEG, I_ORAzP, I_ASLzP, I_ILLEG, I_PHPiM, I_ORAiM, I_ASLiM, I_ILLEG, I_ILLEG, I_ORAaB, I_ASLaB, I_ILLEG,
|
|
/* 10 */ I_BPLrE, I_ORAiY, I_ILLEG, I_ILLEG, I_ILLEG, I_ORAzX, I_ASLzX, I_ILLEG, I_CLCiM, I_ORAaY, I_ILLEG, I_ILLEG, I_ILLEG, I_ORAaX, I_ASLaX, I_ILLEG,
|
|
/* 20 */ I_JSRaB, I_ANDiX, I_ILLEG, I_ILLEG, I_BITzP, I_ANDzP, I_ROLzP, I_ILLEG, I_PLPiM, I_ANDiM, I_ROLiM, I_ILLEG, I_BITaB, I_ANDaB, I_ROLaB, I_ILLEG,
|
|
/* 30 */ I_BMIrE, I_ANDiY, I_ILLEG, I_ILLEG, I_ILLEG, I_ANDzX, I_ROLzX, I_ILLEG, I_SECiM, I_ANDaY, I_ILLEG, I_ILLEG, I_ILLEG, I_ANDaX, I_ROLaX, I_ILLEG,
|
|
/* 40 */ I_RTIiM, I_EORiX, I_ILLEG, I_ILLEG, I_ILLEG, I_EORzP, I_LSRzP, I_ILLEG, I_PHAiM, I_EORiM, I_LSRiM, I_ILLEG, I_JMPaB, I_EORaB, I_LSRaB, I_ILLEG,
|
|
/* 50 */ I_BVCrE, I_EORiY, I_ILLEG, I_ILLEG, I_ILLEG, I_EORzX, I_LSRzX, I_ILLEG, I_CLIiM, I_EORaY, I_ILLEG, I_ILLEG, I_ILLEG, I_EORaX, I_LSRaX, I_ILLEG,
|
|
/* 60 */ I_RTSiM, I_ADCiX, I_ILLEG, I_ILLEG, I_ILLEG, I_ADCzP, I_RORzP, I_ILLEG, I_PLAiM, I_ADCiM, I_RORiM, I_ILLEG, I_JMPiD, I_ADCaB, I_RORaB, I_ILLEG,
|
|
/* 70 */ I_BVSrE, I_ADCiY, I_ILLEG, I_ILLEG, I_ILLEG, I_ADCzX, I_RORzX, I_ILLEG, I_SEIiM, I_ADCaY, I_ILLEG, I_ILLEG, I_ILLEG, I_ADCaX, I_RORaX, I_ILLEG,
|
|
/* 80 */ I_ILLEG, I_STAiX, I_ILLEG, I_ILLEG, I_STYzP, I_STAzP, I_STXzP, I_ILLEG, I_DEYiM, I_ILLEG, I_TXAiM, I_ILLEG, I_STYaB, I_STAaB, I_STXaB, I_ILLEG,
|
|
/* 90 */ I_BCCrE, I_STAiY, I_ILLEG, I_ILLEG, I_STYzX, I_STAzX, I_STXzY, I_ILLEG, I_TYAiM, I_STAaY, I_TXSiM, I_ILLEG, I_ILLEG, I_STAaX, I_ILLEG, I_ILLEG,
|
|
/* A0 */ I_LDYiM, I_LDAiX, I_LDXiM, I_ILLEG, I_LDYzP, I_LDAzP, I_LDXzP, I_ILLEG, I_TAYiM, I_LDAiM, I_TAXiM, I_ILLEG, I_LDYaB, I_LDAaB, I_LDXaB, I_ILLEG,
|
|
/* B0 */ I_BCSrE, I_LDAiY, I_ILLEG, I_ILLEG, I_LDYzX, I_LDAzX, I_LDXzY, I_ILLEG, I_CLViM, I_LDAaY, I_TSXiM, I_ILLEG, I_LDYaX, I_LDAaX, I_LDXaY, I_ILLEG,
|
|
/* C0 */ I_CPYiM, I_CMPiX, I_ILLEG, I_ILLEG, I_CPYzP, I_CMPzP, I_DECzP, I_ILLEG, I_INYiM, I_CMPiM, I_DEXiM, I_ILLEG, I_CPYaB, I_CMPaB, I_DECaB, I_ILLEG,
|
|
/* D0 */ I_BNErE, I_CMPiY, I_ILLEG, I_ILLEG, I_ILLEG, I_CMPzX, I_DECzX, I_ILLEG, I_CLDiM, I_CMPaY, I_ILLEG, I_ILLEG, I_ILLEG, I_CMPaX, I_DECaX, I_ILLEG,
|
|
/* E0 */ I_CPXiM, I_SBCiX, I_ILLEG, I_ILLEG, I_CPXzP, I_SBCzP, I_INCzP, I_ILLEG, I_INXiM, I_SBCiM, I_NOPiM, I_ILLEG, I_CPXaB, I_SBCaB, I_INCaB, I_ILLEG,
|
|
/* F0 */ I_BEQrE, I_SBCiY, I_ILLEG, I_ILLEG, I_ILLEG, I_SBCzX, I_INCzX, I_ILLEG, I_SEDiM, I_SBCaY, I_ILLEG, I_ILLEG, I_ILLEG, I_SBCaX, I_INCaX, I_ILLEG
|
|
/* 00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F */
|
|
};
|
|
|
|
|
|
static inline int quick6502_exec_one(quick6502_cpu *cpu)
|
|
{
|
|
register byte opcode = cpu->memory_opcode_read(cpu->reg_PC++);
|
|
|
|
TRACEi(("Quick6502: PC:$%04X A:$%02X X:$%02X Y:$%02X S:$%02X P:$%02X P:[%c%c%c%c%c%c%c%c]",
|
|
cpu->reg_PC, cpu->reg_A, cpu->reg_X, cpu->reg_Y, cpu->reg_S, cpu->reg_P,
|
|
cpu->reg_P&Q6502_N_FLAG ? 'N':'.',
|
|
cpu->reg_P&Q6502_V_FLAG ? 'V':'.',
|
|
cpu->reg_P&Q6502_R_FLAG ? 'R':'.',
|
|
cpu->reg_P&Q6502_B_FLAG ? 'B':'.',
|
|
cpu->reg_P&Q6502_D_FLAG ? 'D':'.',
|
|
cpu->reg_P&Q6502_I_FLAG ? 'I':'.',
|
|
cpu->reg_P&Q6502_Z_FLAG ? 'Z':'.',
|
|
cpu->reg_P&Q6502_C_FLAG ? 'C':'.'));
|
|
|
|
InstructionTable[opcode](cpu);
|
|
cpu->cycle_done += CycleTable[opcode];
|
|
if (cpu->page_crossed) { cpu->cycle_done++; cpu->page_crossed = 0; }
|
|
if (cpu->int_pending != 0)
|
|
{
|
|
quick6502_int(cpu, Q6502_IRQ_SIGNAL);
|
|
}
|
|
return 0;
|
|
}
|