mirror of
https://github.com/AppleWin/AppleWin.git
synced 2024-11-04 15:05:28 +00:00
37ad1ebd63
Also: . support new CpuType independently of Apple2Type . save-state: don't save disk track image if no disk . save-state: re-init AppleWin internals & UI to reflect changed Apple2Type
6570 lines
227 KiB
C++
6570 lines
227 KiB
C++
/*
|
|
* z80.c
|
|
*
|
|
* Written by
|
|
* Andreas Boose <viceteam@t-online.de>
|
|
*
|
|
* This file is part of VICE, the Versatile Commodore Emulator.
|
|
* See README for copyright notice.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
|
|
* 02111-1307 USA.
|
|
*
|
|
*/
|
|
|
|
#include "..\StdAfx.h"
|
|
|
|
#include "..\AppleWin.h"
|
|
#include "..\CPU.h"
|
|
#include "..\Memory.h"
|
|
#include "..\YamlHelper.h"
|
|
|
|
|
|
#undef IN // Defined in windef.h
|
|
#undef OUT // Defined in windef.h
|
|
|
|
//#include "vice.h" // [AppleWin-TC]
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "..\CommonVICE\6510core.h" // [AppleWin-TC]
|
|
#include "..\CommonVICE\alarm.h"
|
|
#include "daa.h"
|
|
//#include "debug.h" // [AppleWin-TC]
|
|
#include "..\CommonVICE\interrupt.h"
|
|
//#include "log.h" // [AppleWin-TC]
|
|
//#include "maincpu.h" // [AppleWin-TC]
|
|
//#include "monitor.h" // [AppleWin-TC]
|
|
#include "..\CommonVICE\types.h"
|
|
#include "z80.h"
|
|
#include "z80mem.h"
|
|
#include "z80regs.h"
|
|
|
|
|
|
/*#define DEBUG_Z80*/
|
|
|
|
static CLOCK maincpu_clk = 0; // [AppleWin-TC]
|
|
|
|
static BYTE reg_a = 0;
|
|
static BYTE reg_b = 0;
|
|
static BYTE reg_c = 0;
|
|
static BYTE reg_d = 0;
|
|
static BYTE reg_e = 0;
|
|
static BYTE reg_f = 0;
|
|
static BYTE reg_h = 0;
|
|
static BYTE reg_l = 0;
|
|
static BYTE reg_ixh = 0;
|
|
static BYTE reg_ixl = 0;
|
|
static BYTE reg_iyh = 0;
|
|
static BYTE reg_iyl = 0;
|
|
static WORD reg_sp = 0;
|
|
static DWORD z80_reg_pc = 0;
|
|
static BYTE reg_i = 0;
|
|
static BYTE reg_r = 0;
|
|
|
|
static BYTE iff1 = 0;
|
|
static BYTE iff2 = 0;
|
|
static BYTE im_mode = 0;
|
|
|
|
static BYTE reg_a2 = 0;
|
|
static BYTE reg_b2 = 0;
|
|
static BYTE reg_c2 = 0;
|
|
static BYTE reg_d2 = 0;
|
|
static BYTE reg_e2 = 0;
|
|
static BYTE reg_f2 = 0;
|
|
static BYTE reg_h2 = 0;
|
|
static BYTE reg_l2 = 0;
|
|
|
|
#if 0 // [AppleWin-TC] Not used
|
|
static int dma_request = 0;
|
|
#endif
|
|
|
|
static BYTE *z80_bank_base;
|
|
static int z80_bank_limit;
|
|
|
|
|
|
#if 0 // [AppleWin-TC] Not used
|
|
void z80_trigger_dma(void)
|
|
{
|
|
dma_request = 1;
|
|
}
|
|
#endif
|
|
|
|
void z80_reset(void)
|
|
{
|
|
z80_reg_pc = 0;
|
|
z80_regs.reg_pc = 0;
|
|
iff1 = 0;
|
|
iff2 = 0;
|
|
im_mode = 0;
|
|
}
|
|
|
|
/*inline*/ static BYTE *z80mem_read_base(int addr) // [AppleWin-TC]
|
|
{
|
|
BYTE *p = _z80mem_read_base_tab_ptr[addr >> 8];
|
|
|
|
if (p == 0)
|
|
return p;
|
|
|
|
return p - (addr & 0xff00);
|
|
}
|
|
|
|
/*inline*/ static int z80mem_read_limit(int addr) // [AppleWin-TC]
|
|
{
|
|
return z80mem_read_limit_tab_ptr[addr >> 8];
|
|
}
|
|
|
|
#define JUMP(addr) \
|
|
do { \
|
|
z80_reg_pc = (addr); \
|
|
z80_bank_base = z80mem_read_base(z80_reg_pc); \
|
|
z80_bank_limit = z80mem_read_limit(z80_reg_pc); \
|
|
z80_old_reg_pc = z80_reg_pc; \
|
|
} while (0)
|
|
|
|
|
|
#define LOAD(addr) \
|
|
(*_z80mem_read_tab_ptr[(addr) >> 8])((WORD)(addr))
|
|
|
|
#define STORE(addr, value) \
|
|
(*_z80mem_write_tab_ptr[(addr) >> 8])((WORD)(addr), (BYTE)(value))
|
|
|
|
#define IN(addr) \
|
|
(io_read_tab[(addr) >> 8])((WORD)(addr))
|
|
|
|
#define OUT(addr, value) \
|
|
(io_write_tab[(addr) >> 8])((WORD)(addr), (BYTE)(value))
|
|
|
|
#define opcode_t DWORD
|
|
#define FETCH_OPCODE(o) ((o) = (LOAD(z80_reg_pc) \
|
|
| (LOAD(z80_reg_pc + 1) << 8) \
|
|
| (LOAD(z80_reg_pc + 2) << 16) \
|
|
| (LOAD(z80_reg_pc + 3) << 24)))
|
|
|
|
#define p0 (opcode & 0xff)
|
|
#define p1 ((opcode >> 8) & 0xff)
|
|
#define p2 ((opcode >> 16) & 0xff)
|
|
#define p3 (opcode >> 24)
|
|
|
|
#define p12 ((opcode >> 8) & 0xffff)
|
|
#define p23 ((opcode >> 16) & 0xffff)
|
|
|
|
#define CLK maincpu_clk
|
|
|
|
#define INC_PC(value) (z80_reg_pc += (value))
|
|
|
|
/* ------------------------------------------------------------------------- */
|
|
|
|
#if 0 // [AppleWin-TC]
|
|
static unsigned int z80_last_opcode_info;
|
|
|
|
#define LAST_OPCODE_INFO z80_last_opcode_info
|
|
|
|
/* Remember the number of the last opcode. By default, the opcode does not
|
|
delay interrupt and does not change the I flag. */
|
|
#define SET_LAST_OPCODE(x) \
|
|
OPINFO_SET(LAST_OPCODE_INFO, (x), 0, 0, 0)
|
|
|
|
/* Remember that the last opcode delayed a pending IRQ or NMI by one cycle. */
|
|
#define OPCODE_DELAYS_INTERRUPT() \
|
|
OPINFO_SET_DELAYS_INTERRUPT(LAST_OPCODE_INFO, 1)
|
|
|
|
/* Remember that the last opcode changed the I flag from 0 to 1, so we have
|
|
to dispatch an IRQ even if the I flag is 0 when we check it. */
|
|
#define OPCODE_DISABLES_IRQ() \
|
|
OPINFO_SET_DISABLES_IRQ(LAST_OPCODE_INFO, 1)
|
|
|
|
/* Remember that the last opcode changed the I flag from 1 to 0, so we must
|
|
not dispatch an IRQ even if the I flag is 1 when we check it. */
|
|
#define OPCODE_ENABLES_IRQ() \
|
|
OPINFO_SET_ENABLES_IRQ(LAST_OPCODE_INFO, 1)
|
|
|
|
#else // [AppleWin-TC]
|
|
|
|
#define SET_LAST_OPCODE(x)
|
|
#define OPCODE_DELAYS_INTERRUPT()
|
|
#define OPCODE_DISABLES_IRQ()
|
|
#define OPCODE_ENABLES_IRQ()
|
|
#endif
|
|
|
|
/* ------------------------------------------------------------------------- */
|
|
|
|
/* Word register manipulation. */
|
|
|
|
#define BC_WORD() ((reg_b << 8) | reg_c)
|
|
#define DE_WORD() ((reg_d << 8) | reg_e)
|
|
#define HL_WORD() ((reg_h << 8) | reg_l)
|
|
#define IX_WORD() ((reg_ixh << 8) | reg_ixl)
|
|
#define IY_WORD() ((reg_iyh << 8) | reg_iyl)
|
|
|
|
#define IX_WORD_OFF(offset) (IX_WORD() + (signed char)(offset))
|
|
#define IY_WORD_OFF(offset) (IY_WORD() + (signed char)(offset))
|
|
|
|
#define DEC_BC_WORD() \
|
|
do { \
|
|
if (!reg_c) \
|
|
reg_b--; \
|
|
reg_c--; \
|
|
} while (0)
|
|
|
|
#define DEC_DE_WORD() \
|
|
do { \
|
|
if (!reg_e) \
|
|
reg_d--; \
|
|
reg_e--; \
|
|
} while (0)
|
|
|
|
#define DEC_HL_WORD() \
|
|
do { \
|
|
if (!reg_l) \
|
|
reg_h--; \
|
|
reg_l--; \
|
|
} while (0)
|
|
|
|
#define DEC_IX_WORD() \
|
|
do { \
|
|
if (!reg_ixl) \
|
|
reg_ixh--; \
|
|
reg_ixl--; \
|
|
} while (0)
|
|
|
|
#define DEC_IY_WORD() \
|
|
do { \
|
|
if (!reg_iyl) \
|
|
reg_iyh--; \
|
|
reg_iyl--; \
|
|
} while (0)
|
|
|
|
#define INC_BC_WORD() \
|
|
do { \
|
|
reg_c++; \
|
|
if (!reg_c) \
|
|
reg_b++; \
|
|
} while (0)
|
|
|
|
#define INC_DE_WORD() \
|
|
do { \
|
|
reg_e++; \
|
|
if (!reg_e) \
|
|
reg_d++; \
|
|
} while (0)
|
|
|
|
#define INC_HL_WORD() \
|
|
do { \
|
|
reg_l++; \
|
|
if (!reg_l) \
|
|
reg_h++; \
|
|
} while (0)
|
|
|
|
#define INC_IX_WORD() \
|
|
do { \
|
|
reg_ixl++; \
|
|
if (!reg_ixl) \
|
|
reg_ixh++; \
|
|
} while (0)
|
|
|
|
#define INC_IY_WORD() \
|
|
do { \
|
|
reg_iyl++; \
|
|
if (!reg_iyl) \
|
|
reg_iyh++; \
|
|
} while (0)
|
|
|
|
/* ------------------------------------------------------------------------- */
|
|
|
|
/* Flags. */
|
|
|
|
#define C_FLAG 0x01
|
|
#define N_FLAG 0x02
|
|
#define P_FLAG 0x04
|
|
#define U3_FLAG 0x08
|
|
#define H_FLAG 0x10
|
|
#define U5_FLAG 0x20
|
|
#define Z_FLAG 0x40
|
|
#define S_FLAG 0x80
|
|
|
|
#define LOCAL_SET_CARRY(val) \
|
|
do { \
|
|
if (val) \
|
|
reg_f |= C_FLAG; \
|
|
else \
|
|
reg_f &= ~C_FLAG; \
|
|
} while (0)
|
|
|
|
#define LOCAL_SET_NADDSUB(val) \
|
|
do { \
|
|
if (val) \
|
|
reg_f |= N_FLAG; \
|
|
else \
|
|
reg_f &= ~N_FLAG; \
|
|
} while (0)
|
|
|
|
#define LOCAL_SET_PARITY(val) \
|
|
do { \
|
|
if (val) \
|
|
reg_f |= P_FLAG; \
|
|
else \
|
|
reg_f &= ~P_FLAG; \
|
|
} while (0)
|
|
|
|
#define LOCAL_SET_HALFCARRY(val) \
|
|
do { \
|
|
if (val) \
|
|
reg_f |= H_FLAG; \
|
|
else \
|
|
reg_f &= ~H_FLAG; \
|
|
} while (0)
|
|
|
|
#define LOCAL_SET_ZERO(val) \
|
|
do { \
|
|
if (val) \
|
|
reg_f |= Z_FLAG; \
|
|
else \
|
|
reg_f &= ~Z_FLAG; \
|
|
} while (0)
|
|
|
|
#define LOCAL_SET_SIGN(val) \
|
|
do { \
|
|
if (val) \
|
|
reg_f |= S_FLAG; \
|
|
else \
|
|
reg_f &= ~S_FLAG; \
|
|
} while (0)
|
|
|
|
#define LOCAL_CARRY() (reg_f & C_FLAG)
|
|
#define LOCAL_NADDSUB() (reg_f & N_FLAG)
|
|
#define LOCAL_PARITY() (reg_f & P_FLAG)
|
|
#define LOCAL_HALFCARRY() (reg_f & H_FLAG)
|
|
#define LOCAL_ZERO() (reg_f & Z_FLAG)
|
|
#define LOCAL_SIGN() (reg_f & S_FLAG)
|
|
|
|
static const BYTE SZP[256] = {
|
|
P_FLAG|Z_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
P_FLAG, 0, 0, P_FLAG,
|
|
0, P_FLAG, P_FLAG, 0,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG, S_FLAG|P_FLAG, S_FLAG|P_FLAG, S_FLAG,
|
|
S_FLAG|P_FLAG, S_FLAG, S_FLAG, S_FLAG|P_FLAG };
|
|
|
|
/* ------------------------------------------------------------------------- */
|
|
|
|
z80_regs_t z80_regs;
|
|
|
|
static void import_registers(void)
|
|
{
|
|
reg_a = z80_regs.reg_af >> 8;
|
|
reg_f = z80_regs.reg_af & 0xff;
|
|
reg_b = z80_regs.reg_bc >> 8;
|
|
reg_c = z80_regs.reg_bc & 0xff;
|
|
reg_d = z80_regs.reg_de >> 8;
|
|
reg_e = z80_regs.reg_de & 0xff;
|
|
reg_h = z80_regs.reg_hl >> 8;
|
|
reg_l = z80_regs.reg_hl & 0xff;
|
|
reg_ixh = z80_regs.reg_ix >> 8;
|
|
reg_ixl = z80_regs.reg_ix & 0xff;
|
|
reg_iyh = z80_regs.reg_iy >> 8;
|
|
reg_iyl = z80_regs.reg_iy & 0xff;
|
|
reg_sp = z80_regs.reg_sp;
|
|
z80_reg_pc = (DWORD)z80_regs.reg_pc;
|
|
reg_i = z80_regs.reg_i;
|
|
reg_r = z80_regs.reg_r;
|
|
reg_a2 = z80_regs.reg_af2 >> 8;
|
|
reg_f2 = z80_regs.reg_af2 & 0xff;
|
|
reg_b2 = z80_regs.reg_bc2 >> 8;
|
|
reg_c2 = z80_regs.reg_bc2 & 0xff;
|
|
reg_d2 = z80_regs.reg_de2 >> 8;
|
|
reg_e2 = z80_regs.reg_de2 & 0xff;
|
|
reg_h2 = z80_regs.reg_hl2 >> 8;
|
|
reg_l2 = z80_regs.reg_hl2 & 0xff;
|
|
}
|
|
|
|
static void export_registers(void)
|
|
{
|
|
z80_regs.reg_af = (reg_a << 8) | reg_f;
|
|
z80_regs.reg_bc = (reg_b << 8) | reg_c;
|
|
z80_regs.reg_de = (reg_d << 8) | reg_e;
|
|
z80_regs.reg_hl = (reg_h << 8) | reg_l;
|
|
z80_regs.reg_ix = (reg_ixh << 8) | reg_ixl;
|
|
z80_regs.reg_iy = (reg_iyh << 8) | reg_iyl;
|
|
z80_regs.reg_sp = reg_sp;
|
|
z80_regs.reg_pc = (WORD)z80_reg_pc;
|
|
z80_regs.reg_i = reg_i;
|
|
z80_regs.reg_r = reg_r;
|
|
z80_regs.reg_af2 = (reg_a2 << 8) | reg_f2;
|
|
z80_regs.reg_bc2 = (reg_b2 << 8) | reg_c2;
|
|
z80_regs.reg_de2 = (reg_d2 << 8) | reg_e2;
|
|
z80_regs.reg_hl2 = (reg_h2 << 8) | reg_l2;
|
|
}
|
|
|
|
/* ------------------------------------------------------------------------- */
|
|
|
|
// [AppleWin-TC] Z80 IRQs not supported
|
|
|
|
#if 0
|
|
/* Interrupt handling. */
|
|
|
|
#define DO_INTERRUPT(int_kind) \
|
|
do { \
|
|
BYTE ik = (int_kind); \
|
|
\
|
|
if (ik & (IK_IRQ | IK_NMI)) { \
|
|
if ((ik & IK_NMI) && 0) { \
|
|
} else if ((ik & IK_IRQ) && iff1 \
|
|
&& !OPINFO_DISABLES_IRQ(LAST_OPCODE_INFO)) { \
|
|
WORD jumpdst; \
|
|
/*TRACE_IRQ();*/ \
|
|
if (monitor_mask[e_comp_space] & (MI_STEP)) { \
|
|
monitor_check_icount_interrupt(); \
|
|
} \
|
|
CLK += 4; \
|
|
--reg_sp; \
|
|
STORE((reg_sp), ((BYTE)(z80_reg_pc >> 8))); \
|
|
CLK += 4; \
|
|
--reg_sp; \
|
|
STORE((reg_sp), ((BYTE)(z80_reg_pc & 0xff))); \
|
|
iff1 = 0; \
|
|
iff2 = 0; \
|
|
if (im_mode == 1) { \
|
|
jumpdst = 0x38; \
|
|
CLK += 4; \
|
|
JUMP(jumpdst); \
|
|
CLK += 3; \
|
|
} else { \
|
|
jumpdst = (LOAD(reg_i << 8) << 8); \
|
|
CLK += 4; \
|
|
jumpdst |= (LOAD((reg_i << 8) + 1)); \
|
|
JUMP(jumpdst); \
|
|
CLK += 3; \
|
|
} \
|
|
} \
|
|
} \
|
|
if (ik & (IK_TRAP | IK_RESET)) { \
|
|
if (ik & IK_TRAP) { \
|
|
export_registers(); \
|
|
interrupt_do_trap(cpu_int_status, (WORD)z80_reg_pc); \
|
|
import_registers(); \
|
|
if (cpu_int_status->global_pending_int & IK_RESET) \
|
|
ik |= IK_RESET; \
|
|
} \
|
|
if (ik & IK_RESET) { \
|
|
interrupt_ack_reset(cpu_int_status); \
|
|
maincpu_reset(); \
|
|
} \
|
|
} \
|
|
if (ik & (IK_MONITOR)) { \
|
|
caller_space = e_comp_space; \
|
|
if (monitor_force_import(e_comp_space)) \
|
|
import_registers(); \
|
|
if (monitor_mask[e_comp_space]) \
|
|
export_registers(); \
|
|
if (monitor_mask[e_comp_space] & (MI_BREAK)) { \
|
|
if (monitor_check_breakpoints(e_comp_space, \
|
|
(WORD)z80_reg_pc)) { \
|
|
monitor_startup(); \
|
|
} \
|
|
} \
|
|
if (monitor_mask[e_comp_space] & (MI_STEP)) { \
|
|
monitor_check_icount((WORD)z80_reg_pc); \
|
|
} \
|
|
if (monitor_mask[e_comp_space] & (MI_WATCH)) { \
|
|
monitor_check_watchpoints((WORD)z80_reg_pc); \
|
|
} \
|
|
} \
|
|
} while (0)
|
|
#endif
|
|
|
|
/* ------------------------------------------------------------------------- */
|
|
|
|
/* Opcodes. */
|
|
|
|
#define ADC(loadval, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
BYTE tmp, carry, value; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
value = (BYTE)(loadval); \
|
|
carry = LOCAL_CARRY(); \
|
|
tmp = reg_a + value + carry; \
|
|
reg_f = SZP[tmp]; \
|
|
LOCAL_SET_CARRY((WORD)((WORD)reg_a + (WORD)value \
|
|
+ (WORD)(carry)) & 0x100); \
|
|
LOCAL_SET_HALFCARRY((reg_a ^ value ^ tmp) & H_FLAG); \
|
|
LOCAL_SET_PARITY((~(reg_a ^ value)) & (reg_a ^ tmp) & 0x80); \
|
|
reg_a = tmp; \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define ADCHLREG(reg_valh, reg_vall) \
|
|
do { \
|
|
DWORD tmp, carry; \
|
|
\
|
|
carry = LOCAL_CARRY(); \
|
|
tmp = (DWORD)((reg_h << 8) + reg_l) \
|
|
+ (DWORD)((reg_valh << 8) + reg_vall) + carry; \
|
|
LOCAL_SET_ZERO(!(tmp & 0xffff)); \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_SIGN(tmp & 0x8000); \
|
|
LOCAL_SET_CARRY(tmp & 0x10000); \
|
|
LOCAL_SET_HALFCARRY(((tmp >> 8) ^ reg_valh ^ reg_h) & H_FLAG); \
|
|
LOCAL_SET_PARITY((~(reg_h ^ reg_valh)) & \
|
|
(reg_valh ^ (tmp >> 8)) & 0x80); \
|
|
reg_h = (BYTE)(tmp >> 8); \
|
|
reg_l = (BYTE)(tmp & 0xff); \
|
|
CLK += 15; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define ADCHLSP() \
|
|
do { \
|
|
DWORD tmp, carry; \
|
|
\
|
|
carry = LOCAL_CARRY(); \
|
|
tmp = (DWORD)((reg_h << 8) + reg_l) + (DWORD)(reg_sp) + carry; \
|
|
LOCAL_SET_ZERO(!(tmp & 0xffff)); \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_SIGN(tmp & 0x8000); \
|
|
LOCAL_SET_CARRY(tmp & 0x10000); \
|
|
LOCAL_SET_HALFCARRY(((tmp >> 8) ^ (reg_sp >> 8) ^ reg_h) & H_FLAG); \
|
|
LOCAL_SET_PARITY((~(reg_h ^ (reg_sp >> 8))) & \
|
|
((reg_sp >> 8) ^ (tmp >> 8)) & 0x80); \
|
|
reg_h = (BYTE)(tmp >> 8); \
|
|
reg_l = (BYTE)(tmp & 0xff); \
|
|
CLK += 15; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define ADD(loadval, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
BYTE tmp, value; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
value = (BYTE)(loadval); \
|
|
tmp = reg_a + value; \
|
|
reg_f = SZP[tmp]; \
|
|
LOCAL_SET_CARRY((WORD)((WORD)reg_a + (WORD)value) & 0x100); \
|
|
LOCAL_SET_HALFCARRY((reg_a ^ value ^ tmp) & H_FLAG); \
|
|
LOCAL_SET_PARITY((~(reg_a ^ value)) & (reg_a ^ tmp) & 0x80); \
|
|
reg_a = tmp; \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define ADDXXREG(reg_dsth, reg_dstl, reg_valh, reg_vall, clk_inc, pc_inc) \
|
|
do { \
|
|
DWORD tmp; \
|
|
\
|
|
tmp = (DWORD)((reg_dsth << 8) + reg_dstl) \
|
|
+ (DWORD)((reg_valh << 8) + reg_vall); \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_CARRY(tmp & 0x10000); \
|
|
LOCAL_SET_HALFCARRY(((tmp >> 8) ^ reg_valh ^ reg_dsth) & H_FLAG); \
|
|
reg_dsth = (BYTE)(tmp >> 8); \
|
|
reg_dstl = (BYTE)(tmp & 0xff); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define ADDXXSP(reg_dsth, reg_dstl, clk_inc, pc_inc) \
|
|
do { \
|
|
DWORD tmp; \
|
|
\
|
|
tmp = (DWORD)((reg_dsth << 8) + reg_dstl) + (DWORD)(reg_sp); \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_CARRY(tmp & 0x10000); \
|
|
LOCAL_SET_HALFCARRY(((tmp >> 8) ^ (reg_sp >> 8) ^ reg_dsth) & H_FLAG); \
|
|
reg_dsth = (BYTE)(tmp >> 8); \
|
|
reg_dstl = (BYTE)(tmp & 0xff); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define AND(value, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
reg_a &= (value); \
|
|
reg_f = SZP[reg_a]; \
|
|
LOCAL_SET_HALFCARRY(1); \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define BIT(reg_val, value, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_HALFCARRY(1); \
|
|
LOCAL_SET_ZERO(!((reg_val) & (1 << value))); \
|
|
/***LOCAL_SET_PARITY(LOCAL_ZERO());***/ \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define BRANCH(cond, value, pc_inc) \
|
|
do { \
|
|
if (cond) { \
|
|
unsigned int dest_addr; \
|
|
\
|
|
dest_addr = z80_reg_pc + pc_inc + (signed char)(value); \
|
|
z80_reg_pc = dest_addr & 0xffff; \
|
|
CLK += 7; \
|
|
} else { \
|
|
CLK += 7; \
|
|
INC_PC(pc_inc); \
|
|
} \
|
|
} while (0)
|
|
|
|
#define CALL(reg_val, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
INC_PC(pc_inc); \
|
|
CLK += clk_inc1; \
|
|
--reg_sp; \
|
|
STORE((reg_sp), ((BYTE)(z80_reg_pc >> 8))); \
|
|
CLK += clk_inc2; \
|
|
--reg_sp; \
|
|
STORE((reg_sp), ((BYTE)(z80_reg_pc & 0xff))); \
|
|
JUMP(reg_val); \
|
|
CLK += clk_inc3; \
|
|
} while (0)
|
|
|
|
#define CALL_COND(reg_value, cond, clk_inc1, clk_inc2, clk_inc3, \
|
|
clk_inc4, pc_inc) \
|
|
do { \
|
|
if (cond) { \
|
|
CALL(reg_value, clk_inc1, clk_inc2, clk_inc3, pc_inc); \
|
|
} else { \
|
|
CLK += clk_inc4; \
|
|
INC_PC(3); \
|
|
} \
|
|
} while (0)
|
|
|
|
#define CCF(clk_inc, pc_inc) \
|
|
do { \
|
|
LOCAL_SET_HALFCARRY((LOCAL_CARRY())); \
|
|
LOCAL_SET_CARRY(!(LOCAL_CARRY())); \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define CP(loadval, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
BYTE tmp, value; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
value = (BYTE)(loadval); \
|
|
tmp = reg_a - value; \
|
|
reg_f = N_FLAG | SZP[tmp]; \
|
|
LOCAL_SET_CARRY(value > reg_a); \
|
|
LOCAL_SET_HALFCARRY((reg_a ^ value ^ tmp) & H_FLAG); \
|
|
LOCAL_SET_PARITY((reg_a ^ value) & (reg_a ^ tmp) & 0x80); \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define CPDI(HL_FUNC) \
|
|
do { \
|
|
BYTE val, tmp; \
|
|
\
|
|
CLK += 4; \
|
|
val = LOAD(HL_WORD()); \
|
|
tmp = reg_a - val; \
|
|
HL_FUNC; \
|
|
DEC_BC_WORD(); \
|
|
reg_f = N_FLAG | SZP[tmp] | LOCAL_CARRY(); \
|
|
/***LOCAL_SET_CARRY(val > reg_a);***/ \
|
|
LOCAL_SET_HALFCARRY((reg_a ^ val ^ tmp) & H_FLAG); \
|
|
LOCAL_SET_PARITY(reg_b | reg_c); \
|
|
CLK += 1; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define CPDIR(HL_FUNC) \
|
|
do { \
|
|
BYTE val, tmp; \
|
|
\
|
|
CLK += 4; \
|
|
val = LOAD(HL_WORD()); \
|
|
tmp = reg_a - val; \
|
|
HL_FUNC; \
|
|
DEC_BC_WORD(); \
|
|
CLK += 17; \
|
|
if (!(BC_WORD() && tmp)) { \
|
|
reg_f = N_FLAG | SZP[tmp] | LOCAL_CARRY(); \
|
|
/***LOCAL_SET_CARRY(val > reg_a);***/ \
|
|
LOCAL_SET_HALFCARRY((reg_a ^ val ^ tmp) & H_FLAG); \
|
|
LOCAL_SET_PARITY(reg_b | reg_c); \
|
|
CLK += 5; \
|
|
INC_PC(2); \
|
|
} \
|
|
} while (0)
|
|
|
|
#define CPL(clk_inc, pc_inc) \
|
|
do { \
|
|
reg_a ^= 0xff; \
|
|
LOCAL_SET_NADDSUB(1); \
|
|
LOCAL_SET_HALFCARRY(1); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define DAA(clk_inc, pc_inc) \
|
|
do { \
|
|
WORD tmp; \
|
|
\
|
|
tmp = reg_a | (LOCAL_CARRY() ? 0x100 : 0) \
|
|
| (LOCAL_HALFCARRY() ? 0x200 : 0) \
|
|
| (LOCAL_NADDSUB() ? 0x400 : 0); \
|
|
reg_a = daa_reg_a[tmp]; \
|
|
reg_f = daa_reg_f[tmp]; \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define DECXXIND(reg_val, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((reg_val)); \
|
|
tmp--; \
|
|
CLK += clk_inc2; \
|
|
STORE((reg_val), tmp); \
|
|
reg_f = N_FLAG | SZP[tmp] | LOCAL_CARRY(); \
|
|
LOCAL_SET_PARITY((tmp == 0x7f)); \
|
|
LOCAL_SET_HALFCARRY(((tmp & 0x0f) == 0x0f)); \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define DECINC(FUNC, clk_inc, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc; \
|
|
FUNC; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define DECREG(reg_val, clk_inc, pc_inc) \
|
|
do { \
|
|
reg_val--; \
|
|
reg_f = N_FLAG | SZP[reg_val] | LOCAL_CARRY(); \
|
|
LOCAL_SET_PARITY((reg_val == 0x7f)); \
|
|
LOCAL_SET_HALFCARRY(((reg_val & 0x0f) == 0x0f)); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define DJNZ(value, pc_inc) \
|
|
do { \
|
|
reg_b--; \
|
|
/***LOCAL_SET_NADDSUB(1);***/ \
|
|
BRANCH(reg_b, value, pc_inc); \
|
|
} while (0)
|
|
|
|
#define DI(clk_inc, pc_inc) \
|
|
do { \
|
|
iff1 = 0; \
|
|
iff2 = 0; \
|
|
OPCODE_DISABLES_IRQ(); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define EI(clk_inc, pc_inc) \
|
|
do { \
|
|
iff1 = 1; \
|
|
iff2 = 1; \
|
|
OPCODE_DISABLES_IRQ(); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define EXAFAF(clk_inc, pc_inc) \
|
|
do { \
|
|
BYTE tmpl, tmph; \
|
|
\
|
|
tmph = reg_a; \
|
|
tmpl = reg_f; \
|
|
reg_a = reg_a2; \
|
|
reg_f = reg_f2; \
|
|
reg_a2 = tmph; \
|
|
reg_f2 = tmpl; \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define EXX(clk_inc, pc_inc) \
|
|
do { \
|
|
BYTE tmpl, tmph; \
|
|
\
|
|
tmph = reg_b; \
|
|
tmpl = reg_c; \
|
|
reg_b = reg_b2; \
|
|
reg_c = reg_c2; \
|
|
reg_b2 = tmph; \
|
|
reg_c2 = tmpl; \
|
|
tmph = reg_d; \
|
|
tmpl = reg_e; \
|
|
reg_d = reg_d2; \
|
|
reg_e = reg_e2; \
|
|
reg_d2 = tmph; \
|
|
reg_e2 = tmpl; \
|
|
tmph = reg_h; \
|
|
tmpl = reg_l; \
|
|
reg_h = reg_h2; \
|
|
reg_l = reg_l2; \
|
|
reg_h2 = tmph; \
|
|
reg_l2 = tmpl; \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define EXDEHL(clk_inc, pc_inc) \
|
|
do { \
|
|
BYTE tmpl, tmph; \
|
|
\
|
|
tmph = reg_d; \
|
|
tmpl = reg_e; \
|
|
reg_d = reg_h; \
|
|
reg_e = reg_l; \
|
|
reg_h = tmph; \
|
|
reg_l = tmpl; \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define EXXXSP(reg_valh, reg_vall, clk_inc1, clk_inc2, clk_inc3, \
|
|
clk_inc4, clk_inc5, pc_inc) \
|
|
do { \
|
|
BYTE tmpl, tmph; \
|
|
\
|
|
tmph = reg_valh; \
|
|
tmpl = reg_vall; \
|
|
CLK += clk_inc1; \
|
|
reg_valh = LOAD(reg_sp + 1); \
|
|
CLK += clk_inc2; \
|
|
reg_vall = LOAD(reg_sp); \
|
|
CLK += clk_inc3; \
|
|
STORE((reg_sp + 1), tmph); \
|
|
CLK += clk_inc4; \
|
|
STORE(reg_sp, tmpl); \
|
|
CLK += clk_inc5; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
/* FIXME: Continue if INT occurs. */
|
|
#define HALT() \
|
|
do { \
|
|
CLK += 4; \
|
|
} while (0)
|
|
|
|
#define IM(value) \
|
|
do { \
|
|
im_mode = value; \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define INA(value, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
/***reg_a = IN(value);***/ \
|
|
reg_a = IN((reg_a << 8) | value); \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define INBC(reg_val, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
reg_val = IN(BC_WORD()); \
|
|
reg_f = SZP[reg_val & 0xff] | LOCAL_CARRY(); \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define INBC0(clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
BYTE tmp; \
|
|
CLK += clk_inc1; \
|
|
tmp = IN(BC_WORD()); \
|
|
reg_f = SZP[tmp] | LOCAL_CARRY(); \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define INCXXIND(reg_val, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((reg_val)); \
|
|
tmp++; \
|
|
CLK += clk_inc2; \
|
|
STORE((reg_val), tmp); \
|
|
reg_f = SZP[tmp] | LOCAL_CARRY(); \
|
|
LOCAL_SET_PARITY((tmp == 0x80)); \
|
|
LOCAL_SET_HALFCARRY(!(tmp & 0x0f)); \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define INCREG(reg_val, clk_inc, pc_inc) \
|
|
do { \
|
|
reg_val++; \
|
|
reg_f = SZP[reg_val] | LOCAL_CARRY(); \
|
|
LOCAL_SET_PARITY((reg_val == 0x80)); \
|
|
LOCAL_SET_HALFCARRY(!(reg_val & 0x0f)); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define INDI(HL_FUNC) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += 4; \
|
|
tmp = IN(BC_WORD()); \
|
|
CLK += 4; \
|
|
STORE(HL_WORD(), tmp); \
|
|
HL_FUNC; \
|
|
reg_b--; \
|
|
reg_f = N_FLAG; \
|
|
LOCAL_SET_ZERO(!reg_b); \
|
|
CLK += 4; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define INDIR(HL_FUNC) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += 4; \
|
|
tmp = IN(BC_WORD()); \
|
|
CLK += 4; \
|
|
STORE(HL_WORD(), tmp); \
|
|
HL_FUNC; \
|
|
reg_b--; \
|
|
if (!reg_b) { \
|
|
CLK += 4; \
|
|
reg_f = N_FLAG | Z_FLAG; \
|
|
INC_PC(2); \
|
|
} else { \
|
|
reg_f = N_FLAG; \
|
|
} \
|
|
CLK += 4; \
|
|
} while (0)
|
|
|
|
#define JMP(addr, clk_inc) \
|
|
do { \
|
|
CLK += clk_inc; \
|
|
JUMP(addr); \
|
|
} while (0)
|
|
|
|
#define JMP_COND(addr, cond, clk_inc1, clk_inc2) \
|
|
do { \
|
|
if (cond) { \
|
|
JMP(addr, clk_inc1); \
|
|
} else { \
|
|
CLK += clk_inc2; \
|
|
INC_PC(3); \
|
|
} \
|
|
} while (0)
|
|
|
|
#define LDAIR(reg_val) \
|
|
do { \
|
|
CLK += 6; \
|
|
reg_a = reg_val; \
|
|
reg_f = SZP[reg_a] | LOCAL_CARRY(); \
|
|
LOCAL_SET_PARITY(iff2); \
|
|
CLK += 3; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define LDDI(DE_FUNC, HL_FUNC) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += 4; \
|
|
tmp = LOAD(HL_WORD()); \
|
|
CLK += 4; \
|
|
STORE(DE_WORD(), tmp); \
|
|
DEC_BC_WORD(); \
|
|
DE_FUNC; \
|
|
HL_FUNC; \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_PARITY(reg_b | reg_c); \
|
|
LOCAL_SET_HALFCARRY(0); \
|
|
CLK += 12; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define LDDIR(DE_FUNC, HL_FUNC) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += 4; \
|
|
tmp = LOAD(HL_WORD()); \
|
|
CLK += 4; \
|
|
STORE(DE_WORD(), tmp); \
|
|
DEC_BC_WORD(); \
|
|
DE_FUNC; \
|
|
HL_FUNC; \
|
|
CLK += 13; \
|
|
if (!(BC_WORD())) { \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_PARITY(0); \
|
|
LOCAL_SET_HALFCARRY(0); \
|
|
CLK += 5; \
|
|
INC_PC(2); \
|
|
} \
|
|
} while (0)
|
|
|
|
#define LDIND(val, reg_valh, reg_vall, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
reg_vall = LOAD((val)); \
|
|
CLK += clk_inc2; \
|
|
reg_valh = LOAD((val) + 1); \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define LDSP(value, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
reg_sp = (WORD)(value); \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define LDSPIND(value, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
reg_sp = LOAD(value); \
|
|
CLK += clk_inc2; \
|
|
reg_sp |= LOAD(value + 1) << 8; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define LDREG(reg_dest, value, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = (BYTE)(value); \
|
|
reg_dest = tmp; \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define LDW(value, reg_valh, reg_vall, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
reg_vall = (BYTE)((value) & 0xff); \
|
|
reg_valh = (BYTE)((value) >> 8); \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define NEG() \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
tmp = 0 - reg_a; \
|
|
reg_f = N_FLAG | SZP[tmp]; \
|
|
LOCAL_SET_HALFCARRY((reg_a ^ tmp) & H_FLAG); \
|
|
LOCAL_SET_PARITY(reg_a & tmp & 0x80); \
|
|
LOCAL_SET_CARRY(reg_a > 0); \
|
|
reg_a = tmp; \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define NOP(clk_inc, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define OR(reg_val, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
reg_a |= reg_val; \
|
|
reg_f = SZP[reg_a]; \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define OUTA(value, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
/***OUT(value, reg_a);***/ \
|
|
OUT((reg_a << 8) | value, reg_a); \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define OUTBC(value, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
OUT(BC_WORD(), value); \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define OUTDI(HL_FUNC) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += 4; \
|
|
tmp = LOAD(HL_WORD()); \
|
|
CLK += 4; \
|
|
OUT(BC_WORD(), tmp); \
|
|
HL_FUNC; \
|
|
reg_b--; \
|
|
reg_f = N_FLAG; \
|
|
LOCAL_SET_ZERO(!reg_b); \
|
|
CLK += 4; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define OTDIR(HL_FUNC) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += 4; \
|
|
tmp = LOAD(HL_WORD()); \
|
|
CLK += 4; \
|
|
OUT(BC_WORD(), tmp); \
|
|
HL_FUNC; \
|
|
reg_b--; \
|
|
if (!reg_b) { \
|
|
CLK += 4; \
|
|
reg_f = N_FLAG | Z_FLAG; \
|
|
INC_PC(2); \
|
|
} else { \
|
|
reg_f = N_FLAG; \
|
|
} \
|
|
CLK += 4; \
|
|
} while (0)
|
|
|
|
#define POP(reg_valh, reg_vall, pc_inc) \
|
|
do { \
|
|
CLK += 4; \
|
|
reg_vall = LOAD(reg_sp); \
|
|
++reg_sp; \
|
|
CLK += 4; \
|
|
reg_valh = LOAD(reg_sp); \
|
|
++reg_sp; \
|
|
CLK += 2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define PUSH(reg_valh, reg_vall, pc_inc) \
|
|
do { \
|
|
CLK += 4; \
|
|
--reg_sp; \
|
|
STORE((reg_sp), (reg_valh)); \
|
|
CLK += 4; \
|
|
--reg_sp; \
|
|
STORE((reg_sp), (reg_vall)); \
|
|
CLK += 3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RES(reg_val, value) \
|
|
do { \
|
|
reg_val &= (~(1 << value)); \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define RESXX(value, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
tmp &= (~(1 << value)); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RESXXREG(value, reg_val, addr, clk_inc1, clk_inc2, \
|
|
clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
tmp &= (~(1 << value)); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_val = tmp; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RET(clk_inc1, clk_inc2, clk_inc3) \
|
|
do { \
|
|
WORD tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD(reg_sp); \
|
|
CLK += clk_inc2; \
|
|
tmp |= LOAD((reg_sp + 1)) << 8; \
|
|
reg_sp += 2; \
|
|
JUMP(tmp); \
|
|
CLK += clk_inc3; \
|
|
} while (0)
|
|
|
|
#define RET_COND(cond, clk_inc1, clk_inc2, clk_inc3, clk_inc4, pc_inc) \
|
|
do { \
|
|
if (cond) { \
|
|
RET(clk_inc1, clk_inc2, clk_inc3); \
|
|
} else { \
|
|
CLK += clk_inc4; \
|
|
INC_PC(pc_inc); \
|
|
} \
|
|
} while (0)
|
|
|
|
#define RETNI() \
|
|
do { \
|
|
WORD tmp; \
|
|
\
|
|
CLK += 4; \
|
|
tmp = LOAD(reg_sp); \
|
|
CLK += 4; \
|
|
tmp |= LOAD((reg_sp + 1)) << 8; \
|
|
reg_sp += 2; \
|
|
iff1 = iff2; \
|
|
JUMP(tmp); \
|
|
CLK += 2; \
|
|
} while (0)
|
|
|
|
#define RL(reg_val) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = (reg_val & 0x80) ? C_FLAG : 0; \
|
|
reg_val = (reg_val << 1) | LOCAL_CARRY(); \
|
|
reg_f = rot | SZP[reg_val]; \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define RLA(clk_inc, pc_inc) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = (reg_a & 0x80) ? C_FLAG : 0; \
|
|
reg_a = (reg_a << 1) | LOCAL_CARRY(); \
|
|
LOCAL_SET_CARRY(rot); \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_HALFCARRY(0); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RLC(reg_val) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = (reg_val & 0x80) ? C_FLAG : 0; \
|
|
reg_val = (reg_val << 1) | rot; \
|
|
reg_f = rot | SZP[reg_val]; \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define RLCA(clk_inc, pc_inc) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = (reg_a & 0x80) ? C_FLAG : 0; \
|
|
reg_a = (reg_a << 1) | rot; \
|
|
LOCAL_SET_CARRY(rot); \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_HALFCARRY(0); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RLCXX(addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = (tmp & 0x80) ? C_FLAG : 0; \
|
|
tmp = (tmp << 1) | rot; \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RLCXXREG(reg_val, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = (tmp & 0x80) ? C_FLAG : 0; \
|
|
tmp = (tmp << 1) | rot; \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_val = tmp; \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RLD() \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
tmp = LOAD(HL_WORD()); \
|
|
CLK += 8; \
|
|
STORE(HL_WORD(), (tmp << 4) | (reg_a & 0x0f)); \
|
|
reg_a = (tmp >> 4) | (reg_a & 0xf0); \
|
|
reg_f = SZP[reg_a] | LOCAL_CARRY(); \
|
|
CLK += 10; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define RLXX(addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = (tmp & 0x80) ? C_FLAG : 0; \
|
|
tmp = (tmp << 1) | LOCAL_CARRY(); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RLXXREG(reg_val, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = (tmp & 0x80) ? C_FLAG : 0; \
|
|
tmp = (tmp << 1) | LOCAL_CARRY(); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_val = tmp; \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RR(reg_val) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = reg_val & C_FLAG; \
|
|
reg_val = (reg_val >> 1) | (LOCAL_CARRY() ? 0x80 : 0); \
|
|
reg_f = rot | SZP[reg_val]; \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define RRA(clk_inc, pc_inc) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = reg_a & C_FLAG; \
|
|
reg_a = (reg_a >> 1) | (LOCAL_CARRY() ? 0x80 : 0); \
|
|
LOCAL_SET_CARRY(rot); \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_HALFCARRY(0); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RRC(reg_val) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = reg_val & C_FLAG; \
|
|
reg_val = (reg_val >> 1) | ((rot) ? 0x80 : 0); \
|
|
reg_f = rot | SZP[reg_val]; \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define RRCA(clk_inc, pc_inc) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = reg_a & C_FLAG; \
|
|
reg_a = (reg_a >> 1) | ((rot) ? 0x80 : 0); \
|
|
LOCAL_SET_CARRY(rot); \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
LOCAL_SET_HALFCARRY(0); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RRCXX(addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = tmp & C_FLAG; \
|
|
tmp = (tmp >> 1) | ((rot) ? 0x80 : 0); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RRCXXREG(reg_val, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = tmp & C_FLAG; \
|
|
tmp = (tmp >> 1) | ((rot) ? 0x80 : 0); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_val = tmp; \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RRD() \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
tmp = LOAD(HL_WORD()); \
|
|
CLK += 8; \
|
|
STORE(HL_WORD(), (tmp >> 4) | (reg_a << 4)); \
|
|
reg_a = (tmp & 0x0f) | (reg_a & 0xf0); \
|
|
reg_f = SZP[reg_a] | LOCAL_CARRY(); \
|
|
CLK += 10; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define RRXX(addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = tmp & C_FLAG; \
|
|
tmp = (tmp >> 1) | (LOCAL_CARRY() ? 0x80 : 0); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define RRXXREG(reg_val, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = tmp & C_FLAG; \
|
|
tmp = (tmp >> 1) | (LOCAL_CARRY() ? 0x80 : 0); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_val = tmp; \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SBCHLREG(reg_valh, reg_vall) \
|
|
do { \
|
|
DWORD tmp; \
|
|
BYTE carry; \
|
|
\
|
|
carry = LOCAL_CARRY(); \
|
|
tmp = (DWORD)(HL_WORD()) - (DWORD)((reg_valh << 8) + reg_vall) \
|
|
- (DWORD)(carry); \
|
|
reg_f = N_FLAG; \
|
|
LOCAL_SET_CARRY(tmp & 0x10000); \
|
|
LOCAL_SET_HALFCARRY((reg_h ^ reg_valh ^ (tmp >> 8)) & H_FLAG); \
|
|
LOCAL_SET_PARITY(((reg_h ^ (tmp >> 8)) & (reg_h ^ reg_valh)) & 0x80); \
|
|
LOCAL_SET_ZERO(!(tmp & 0xffff)); \
|
|
LOCAL_SET_SIGN(tmp & 0x8000); \
|
|
reg_h = (BYTE)(tmp >> 8); \
|
|
reg_l = (BYTE)(tmp & 0xff); \
|
|
CLK += 15; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define SBCHLSP() \
|
|
do { \
|
|
DWORD tmp; \
|
|
BYTE carry; \
|
|
\
|
|
carry = LOCAL_CARRY(); \
|
|
tmp = (DWORD)(HL_WORD()) - (DWORD)reg_sp - (DWORD)(carry); \
|
|
reg_f = N_FLAG; \
|
|
LOCAL_SET_CARRY(tmp & 0x10000); \
|
|
LOCAL_SET_HALFCARRY((reg_h ^ (reg_sp >> 8) ^ (tmp >> 8)) & H_FLAG); \
|
|
LOCAL_SET_PARITY(((reg_h ^ (tmp >> 8)) \
|
|
& (reg_h ^ (reg_sp >> 8))) & 0x80); \
|
|
LOCAL_SET_ZERO(!(tmp & 0xffff)); \
|
|
LOCAL_SET_SIGN(tmp & 0x8000); \
|
|
reg_h = (BYTE)(tmp >> 8); \
|
|
reg_l = (BYTE)(tmp & 0xff); \
|
|
CLK += 15; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define SBC(loadval, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
BYTE tmp, carry, value; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
value = (BYTE)(loadval); \
|
|
carry = LOCAL_CARRY(); \
|
|
tmp = reg_a - value - carry; \
|
|
reg_f = N_FLAG | SZP[tmp]; \
|
|
LOCAL_SET_HALFCARRY((reg_a ^ value ^ tmp) & H_FLAG); \
|
|
LOCAL_SET_PARITY((reg_a ^ value) & (reg_a ^ tmp) & 0x80); \
|
|
LOCAL_SET_CARRY((WORD)((WORD)value \
|
|
+ (WORD)(carry)) > reg_a); \
|
|
reg_a = tmp; \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SCF(clk_inc, pc_inc) \
|
|
do { \
|
|
LOCAL_SET_CARRY(1); \
|
|
LOCAL_SET_HALFCARRY(0); \
|
|
LOCAL_SET_NADDSUB(0); \
|
|
CLK += clk_inc; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SET(reg_val, value) \
|
|
do { \
|
|
reg_val |= (1 << value); \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define SETXX(value, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
tmp |= (1 << value); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SETXXREG(value, reg_val, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
tmp |= (1 << value); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_val = tmp; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SLA(reg_val) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = (reg_val & 0x80) ? C_FLAG : 0; \
|
|
reg_val <<= 1; \
|
|
reg_f = rot | SZP[reg_val]; \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define SLAXX(addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = (tmp & 0x80) ? C_FLAG : 0; \
|
|
tmp <<= 1; \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SLAXXREG(reg_val, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = (tmp & 0x80) ? C_FLAG : 0; \
|
|
tmp <<= 1; \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_val = tmp; \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SLL(reg_val) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = (reg_val & 0x80) ? C_FLAG : 0; \
|
|
reg_val = (reg_val << 1) | 1; \
|
|
reg_f = rot | SZP[reg_val]; \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define SLLXX(addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = (tmp & 0x80) ? C_FLAG : 0; \
|
|
tmp = (tmp << 1) | 1; \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SLLXXREG(reg_val, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = (tmp & 0x80) ? C_FLAG : 0; \
|
|
tmp = (tmp << 1) | 1; \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_val = tmp; \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SRA(reg_val) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = reg_val & C_FLAG; \
|
|
reg_val = (reg_val >> 1) | (reg_val & 0x80); \
|
|
reg_f = rot | SZP[reg_val]; \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define SRAXX(addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = tmp & C_FLAG; \
|
|
tmp = (tmp >> 1) | (tmp & 0x80); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SRAXXREG(reg_val, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = tmp & C_FLAG; \
|
|
tmp = (tmp >> 1) | (tmp & 0x80); \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_val = tmp; \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SRL(reg_val) \
|
|
do { \
|
|
BYTE rot; \
|
|
\
|
|
rot = reg_val & C_FLAG; \
|
|
reg_val >>= 1; \
|
|
reg_f = rot | SZP[reg_val]; \
|
|
CLK += 8; \
|
|
INC_PC(2); \
|
|
} while (0)
|
|
|
|
#define SRLXX(addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = tmp & C_FLAG; \
|
|
tmp >>= 1; \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SRLXXREG(reg_val, addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
BYTE rot, tmp; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
tmp = LOAD((addr)); \
|
|
rot = tmp & C_FLAG; \
|
|
tmp >>= 1; \
|
|
CLK += clk_inc2; \
|
|
STORE((addr), tmp); \
|
|
reg_val = tmp; \
|
|
reg_f = rot | SZP[tmp]; \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define STW(addr, reg_valh, reg_vall, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
STORE((WORD)(addr), reg_vall); \
|
|
CLK += clk_inc2; \
|
|
STORE((WORD)(addr + 1), reg_valh); \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define STSPW(addr, clk_inc1, clk_inc2, clk_inc3, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
STORE((WORD)(addr), (reg_sp & 0xff)); \
|
|
CLK += clk_inc2; \
|
|
STORE((WORD)(addr + 1), (reg_sp >> 8)); \
|
|
CLK += clk_inc3; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define STREG(addr, reg_val, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
STORE(addr, reg_val); \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define SUB(loadval, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
BYTE tmp, value; \
|
|
\
|
|
CLK += clk_inc1; \
|
|
value = (BYTE)(loadval); \
|
|
tmp = reg_a - value; \
|
|
reg_f = N_FLAG | SZP[tmp]; \
|
|
LOCAL_SET_HALFCARRY((reg_a ^ value ^ tmp) & H_FLAG); \
|
|
LOCAL_SET_PARITY((reg_a ^ value) & (reg_a ^ tmp) & 0x80); \
|
|
LOCAL_SET_CARRY(value > reg_a); \
|
|
reg_a = tmp; \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
#define XOR(value, clk_inc1, clk_inc2, pc_inc) \
|
|
do { \
|
|
CLK += clk_inc1; \
|
|
reg_a ^= value; \
|
|
reg_f = SZP[reg_a]; \
|
|
CLK += clk_inc2; \
|
|
INC_PC(pc_inc); \
|
|
} while (0)
|
|
|
|
|
|
/* ------------------------------------------------------------------------- */
|
|
|
|
/* Extented opcodes. */
|
|
|
|
static void opcode_cb(BYTE ip1, BYTE ip2, BYTE ip3, WORD ip12, WORD ip23)
|
|
{
|
|
switch (ip1) {
|
|
case 0x00: /* RLC B */
|
|
RLC(reg_b);
|
|
break;
|
|
case 0x01: /* RLC C */
|
|
RLC(reg_c);
|
|
break;
|
|
case 0x02: /* RLC D */
|
|
RLC(reg_d);
|
|
break;
|
|
case 0x03: /* RLC E */
|
|
RLC(reg_e);
|
|
break;
|
|
case 0x04: /* RLC H */
|
|
RLC(reg_h);
|
|
break;
|
|
case 0x05: /* RLC L */
|
|
RLC(reg_l);
|
|
break;
|
|
case 0x06: /* RLC (HL) */
|
|
RLCXX(HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x07: /* RLC A */
|
|
RLC(reg_a);
|
|
break;
|
|
case 0x08: /* RRC B */
|
|
RRC(reg_b);
|
|
break;
|
|
case 0x09: /* RRC C */
|
|
RRC(reg_c);
|
|
break;
|
|
case 0x0a: /* RRC D */
|
|
RRC(reg_d);
|
|
break;
|
|
case 0x0b: /* RRC E */
|
|
RRC(reg_e);
|
|
break;
|
|
case 0x0c: /* RRC H */
|
|
RRC(reg_h);
|
|
break;
|
|
case 0x0d: /* RRC L */
|
|
RRC(reg_l);
|
|
break;
|
|
case 0x0e: /* RRC (HL) */
|
|
RRCXX(HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x0f: /* RRC A */
|
|
RRC(reg_a);
|
|
break;
|
|
case 0x10: /* RL B */
|
|
RL(reg_b);
|
|
break;
|
|
case 0x11: /* RL C */
|
|
RL(reg_c);
|
|
break;
|
|
case 0x12: /* RL D */
|
|
RL(reg_d);
|
|
break;
|
|
case 0x13: /* RL E */
|
|
RL(reg_e);
|
|
break;
|
|
case 0x14: /* RL H */
|
|
RL(reg_h);
|
|
break;
|
|
case 0x15: /* RL L */
|
|
RL(reg_l);
|
|
break;
|
|
case 0x16: /* RL (HL) */
|
|
RLXX(HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x17: /* RL A */
|
|
RL(reg_a);
|
|
break;
|
|
case 0x18: /* RR B */
|
|
RR(reg_b);
|
|
break;
|
|
case 0x19: /* RR C */
|
|
RR(reg_c);
|
|
break;
|
|
case 0x1a: /* RR D */
|
|
RR(reg_d);
|
|
break;
|
|
case 0x1b: /* RR E */
|
|
RR(reg_e);
|
|
break;
|
|
case 0x1c: /* RR H */
|
|
RR(reg_h);
|
|
break;
|
|
case 0x1d: /* RR L */
|
|
RR(reg_l);
|
|
break;
|
|
case 0x1e: /* RR (HL) */
|
|
RRXX(HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x1f: /* RR A */
|
|
RR(reg_a);
|
|
break;
|
|
case 0x20: /* SLA B */
|
|
SLA(reg_b);
|
|
break;
|
|
case 0x21: /* SLA C */
|
|
SLA(reg_c);
|
|
break;
|
|
case 0x22: /* SLA D */
|
|
SLA(reg_d);
|
|
break;
|
|
case 0x23: /* SLA E */
|
|
SLA(reg_e);
|
|
break;
|
|
case 0x24: /* SLA H */
|
|
SLA(reg_h);
|
|
break;
|
|
case 0x25: /* SLA L */
|
|
SLA(reg_l);
|
|
break;
|
|
case 0x26: /* SLA (HL) */
|
|
SLAXX(HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x27: /* SLA A */
|
|
SLA(reg_a);
|
|
break;
|
|
case 0x28: /* SRA B */
|
|
SRA(reg_b);
|
|
break;
|
|
case 0x29: /* SRA C */
|
|
SRA(reg_c);
|
|
break;
|
|
case 0x2a: /* SRA D */
|
|
SRA(reg_d);
|
|
break;
|
|
case 0x2b: /* SRA E */
|
|
SRA(reg_e);
|
|
break;
|
|
case 0x2c: /* SRA H */
|
|
SRA(reg_h);
|
|
break;
|
|
case 0x2d: /* SRA L */
|
|
SRA(reg_l);
|
|
break;
|
|
case 0x2e: /* SRA (HL) */
|
|
SRAXX(HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x2f: /* SRA A */
|
|
SRA(reg_a);
|
|
break;
|
|
case 0x30: /* SLL B */
|
|
SLL(reg_b);
|
|
break;
|
|
case 0x31: /* SLL C */
|
|
SLL(reg_c);
|
|
break;
|
|
case 0x32: /* SLL D */
|
|
SLL(reg_d);
|
|
break;
|
|
case 0x33: /* SLL E */
|
|
SLL(reg_e);
|
|
break;
|
|
case 0x34: /* SLL H */
|
|
SLL(reg_h);
|
|
break;
|
|
case 0x35: /* SLL L */
|
|
SLL(reg_l);
|
|
break;
|
|
case 0x36: /* SLL (HL) */
|
|
SLLXX(HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x37: /* SLL A */
|
|
SLL(reg_a);
|
|
break;
|
|
case 0x38: /* SRL B */
|
|
SRL(reg_b);
|
|
break;
|
|
case 0x39: /* SRL C */
|
|
SRL(reg_c);
|
|
break;
|
|
case 0x3a: /* SRL D */
|
|
SRL(reg_d);
|
|
break;
|
|
case 0x3b: /* SRL E */
|
|
SRL(reg_e);
|
|
break;
|
|
case 0x3c: /* SRL H */
|
|
SRL(reg_h);
|
|
break;
|
|
case 0x3d: /* SRL L */
|
|
SRL(reg_l);
|
|
break;
|
|
case 0x3e: /* SRL (HL) */
|
|
SRLXX(HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x3f: /* SRL A */
|
|
SRL(reg_a);
|
|
break;
|
|
case 0x40: /* BIT B 0 */
|
|
BIT(reg_b, 0, 0, 8, 2);
|
|
break;
|
|
case 0x41: /* BIT C 0 */
|
|
BIT(reg_c, 0, 0, 8, 2);
|
|
break;
|
|
case 0x42: /* BIT D 0 */
|
|
BIT(reg_d, 0, 0, 8, 2);
|
|
break;
|
|
case 0x43: /* BIT E 0 */
|
|
BIT(reg_e, 0, 0, 8, 2);
|
|
break;
|
|
case 0x44: /* BIT H 0 */
|
|
BIT(reg_h, 0, 0, 8, 2);
|
|
break;
|
|
case 0x45: /* BIT L 0 */
|
|
BIT(reg_l, 0, 0, 8, 2);
|
|
break;
|
|
case 0x46: /* BIT (HL) 0 */
|
|
BIT(LOAD(HL_WORD()), 0, 4, 8, 2);
|
|
break;
|
|
case 0x47: /* BIT A 0 */
|
|
BIT(reg_a, 0, 0, 8, 2);
|
|
break;
|
|
case 0x48: /* BIT B 1 */
|
|
BIT(reg_b, 1, 0, 8, 2);
|
|
break;
|
|
case 0x49: /* BIT C 1 */
|
|
BIT(reg_c, 1, 0, 8, 2);
|
|
break;
|
|
case 0x4a: /* BIT D 1 */
|
|
BIT(reg_d, 1, 0, 8, 2);
|
|
break;
|
|
case 0x4b: /* BIT E 1 */
|
|
BIT(reg_e, 1, 0, 8, 2);
|
|
break;
|
|
case 0x4c: /* BIT H 1 */
|
|
BIT(reg_h, 1, 0, 8, 2);
|
|
break;
|
|
case 0x4d: /* BIT L 1 */
|
|
BIT(reg_l, 1, 0, 8, 2);
|
|
break;
|
|
case 0x4e: /* BIT (HL) 1 */
|
|
BIT(LOAD(HL_WORD()), 1, 4, 8, 2);
|
|
break;
|
|
case 0x4f: /* BIT A 1 */
|
|
BIT(reg_a, 1, 0, 8, 2);
|
|
break;
|
|
case 0x50: /* BIT B 2 */
|
|
BIT(reg_b, 2, 0, 8, 2);
|
|
break;
|
|
case 0x51: /* BIT C 2 */
|
|
BIT(reg_c, 2, 0, 8, 2);
|
|
break;
|
|
case 0x52: /* BIT D 2 */
|
|
BIT(reg_d, 2, 0, 8, 2);
|
|
break;
|
|
case 0x53: /* BIT E 2 */
|
|
BIT(reg_e, 2, 0, 8, 2);
|
|
break;
|
|
case 0x54: /* BIT H 2 */
|
|
BIT(reg_h, 2, 0, 8, 2);
|
|
break;
|
|
case 0x55: /* BIT L 2 */
|
|
BIT(reg_l, 2, 0, 8, 2);
|
|
break;
|
|
case 0x56: /* BIT (HL) 2 */
|
|
BIT(LOAD(HL_WORD()), 2, 4, 8, 2);
|
|
break;
|
|
case 0x57: /* BIT A 2 */
|
|
BIT(reg_a, 2, 0, 8, 2);
|
|
break;
|
|
case 0x58: /* BIT B 3 */
|
|
BIT(reg_b, 3, 0, 8, 2);
|
|
break;
|
|
case 0x59: /* BIT C 3 */
|
|
BIT(reg_c, 3, 0, 8, 2);
|
|
break;
|
|
case 0x5a: /* BIT D 3 */
|
|
BIT(reg_d, 3, 0, 8, 2);
|
|
break;
|
|
case 0x5b: /* BIT E 3 */
|
|
BIT(reg_e, 3, 0, 8, 2);
|
|
break;
|
|
case 0x5c: /* BIT H 3 */
|
|
BIT(reg_h, 3, 0, 8, 2);
|
|
break;
|
|
case 0x5d: /* BIT L 3 */
|
|
BIT(reg_l, 3, 0, 8, 2);
|
|
break;
|
|
case 0x5e: /* BIT (HL) 3 */
|
|
BIT(LOAD(HL_WORD()), 3, 4, 8, 2);
|
|
break;
|
|
case 0x5f: /* BIT A 3 */
|
|
BIT(reg_a, 3, 0, 8, 2);
|
|
break;
|
|
case 0x60: /* BIT B 4 */
|
|
BIT(reg_b, 4, 0, 8, 2);
|
|
break;
|
|
case 0x61: /* BIT C 4 */
|
|
BIT(reg_c, 4, 0, 8, 2);
|
|
break;
|
|
case 0x62: /* BIT D 4 */
|
|
BIT(reg_d, 4, 0, 8, 2);
|
|
break;
|
|
case 0x63: /* BIT E 4 */
|
|
BIT(reg_e, 4, 0, 8, 2);
|
|
break;
|
|
case 0x64: /* BIT H 4 */
|
|
BIT(reg_h, 4, 0, 8, 2);
|
|
break;
|
|
case 0x65: /* BIT L 4 */
|
|
BIT(reg_l, 4, 0, 8, 2);
|
|
break;
|
|
case 0x66: /* BIT (HL) 4 */
|
|
BIT(LOAD(HL_WORD()), 4, 4, 8, 2);
|
|
break;
|
|
case 0x67: /* BIT A 4 */
|
|
BIT(reg_a, 4, 0, 8, 2);
|
|
break;
|
|
case 0x68: /* BIT B 5 */
|
|
BIT(reg_b, 5, 0, 8, 2);
|
|
break;
|
|
case 0x69: /* BIT C 5 */
|
|
BIT(reg_c, 5, 0, 8, 2);
|
|
break;
|
|
case 0x6a: /* BIT D 5 */
|
|
BIT(reg_d, 5, 0, 8, 2);
|
|
break;
|
|
case 0x6b: /* BIT E 5 */
|
|
BIT(reg_e, 5, 0, 8, 2);
|
|
break;
|
|
case 0x6c: /* BIT H 5 */
|
|
BIT(reg_h, 5, 0, 8, 2);
|
|
break;
|
|
case 0x6d: /* BIT L 5 */
|
|
BIT(reg_l, 5, 0, 8, 2);
|
|
break;
|
|
case 0x6e: /* BIT (HL) 5 */
|
|
BIT(LOAD(HL_WORD()), 5, 4, 8, 2);
|
|
break;
|
|
case 0x6f: /* BIT A 5 */
|
|
BIT(reg_a, 5, 0, 8, 2);
|
|
break;
|
|
case 0x70: /* BIT B 6 */
|
|
BIT(reg_b, 6, 0, 8, 2);
|
|
break;
|
|
case 0x71: /* BIT C 6 */
|
|
BIT(reg_c, 6, 0, 8, 2);
|
|
break;
|
|
case 0x72: /* BIT D 6 */
|
|
BIT(reg_d, 6, 0, 8, 2);
|
|
break;
|
|
case 0x73: /* BIT E 6 */
|
|
BIT(reg_e, 6, 0, 8, 2);
|
|
break;
|
|
case 0x74: /* BIT H 6 */
|
|
BIT(reg_h, 6, 0, 8, 2);
|
|
break;
|
|
case 0x75: /* BIT L 6 */
|
|
BIT(reg_l, 6, 0, 8, 2);
|
|
break;
|
|
case 0x76: /* BIT (HL) 6 */
|
|
BIT(LOAD(HL_WORD()), 6, 4, 8, 2);
|
|
break;
|
|
case 0x77: /* BIT A 6 */
|
|
BIT(reg_a, 6, 0, 8, 2);
|
|
break;
|
|
case 0x78: /* BIT B 7 */
|
|
BIT(reg_b, 7, 0, 8, 2);
|
|
break;
|
|
case 0x79: /* BIT C 7 */
|
|
BIT(reg_c, 7, 0, 8, 2);
|
|
break;
|
|
case 0x7a: /* BIT D 7 */
|
|
BIT(reg_d, 7, 0, 8, 2);
|
|
break;
|
|
case 0x7b: /* BIT E 7 */
|
|
BIT(reg_e, 7, 0, 8, 2);
|
|
break;
|
|
case 0x7c: /* BIT H 7 */
|
|
BIT(reg_h, 7, 0, 8, 2);
|
|
break;
|
|
case 0x7d: /* BIT L 7 */
|
|
BIT(reg_l, 7, 0, 8, 2);
|
|
break;
|
|
case 0x7e: /* BIT (HL) 7 */
|
|
BIT(LOAD(HL_WORD()), 7, 4, 8, 2);
|
|
break;
|
|
case 0x7f: /* BIT A 7 */
|
|
BIT(reg_a, 7, 0, 8, 2);
|
|
break;
|
|
case 0x80: /* RES B 0 */
|
|
RES(reg_b, 0);
|
|
break;
|
|
case 0x81: /* RES C 0 */
|
|
RES(reg_c, 0);
|
|
break;
|
|
case 0x82: /* RES D 0 */
|
|
RES(reg_d, 0);
|
|
break;
|
|
case 0x83: /* RES E 0 */
|
|
RES(reg_e, 0);
|
|
break;
|
|
case 0x84: /* RES H 0 */
|
|
RES(reg_h, 0);
|
|
break;
|
|
case 0x85: /* RES L 0 */
|
|
RES(reg_l, 0);
|
|
break;
|
|
case 0x86: /* RES (HL) 0 */
|
|
RESXX(0, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x87: /* RES A 0 */
|
|
RES(reg_a, 0);
|
|
break;
|
|
case 0x88: /* RES B 1 */
|
|
RES(reg_b, 1);
|
|
break;
|
|
case 0x89: /* RES C 1 */
|
|
RES(reg_c, 1);
|
|
break;
|
|
case 0x8a: /* RES D 1 */
|
|
RES(reg_d, 1);
|
|
break;
|
|
case 0x8b: /* RES E 1 */
|
|
RES(reg_e, 1);
|
|
break;
|
|
case 0x8c: /* RES H 1 */
|
|
RES(reg_h, 1);
|
|
break;
|
|
case 0x8d: /* RES L 1 */
|
|
RES(reg_l, 1);
|
|
break;
|
|
case 0x8e: /* RES (HL) 1 */
|
|
RESXX(1, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x8f: /* RES A 1 */
|
|
RES(reg_a, 1);
|
|
break;
|
|
case 0x90: /* RES B 2 */
|
|
RES(reg_b, 2);
|
|
break;
|
|
case 0x91: /* RES C 2 */
|
|
RES(reg_c, 2);
|
|
break;
|
|
case 0x92: /* RES D 2 */
|
|
RES(reg_d, 2);
|
|
break;
|
|
case 0x93: /* RES E 2 */
|
|
RES(reg_e, 2);
|
|
break;
|
|
case 0x94: /* RES H 2 */
|
|
RES(reg_h, 2);
|
|
break;
|
|
case 0x95: /* RES L 2 */
|
|
RES(reg_l, 2);
|
|
break;
|
|
case 0x96: /* RES (HL) 2 */
|
|
RESXX(2, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x97: /* RES A 2 */
|
|
RES(reg_a, 2);
|
|
break;
|
|
case 0x98: /* RES B 3 */
|
|
RES(reg_b, 3);
|
|
break;
|
|
case 0x99: /* RES C 3 */
|
|
RES(reg_c, 3);
|
|
break;
|
|
case 0x9a: /* RES D 3 */
|
|
RES(reg_d, 3);
|
|
break;
|
|
case 0x9b: /* RES E 3 */
|
|
RES(reg_e, 3);
|
|
break;
|
|
case 0x9c: /* RES H 3 */
|
|
RES(reg_h, 3);
|
|
break;
|
|
case 0x9d: /* RES L 3 */
|
|
RES(reg_l, 3);
|
|
break;
|
|
case 0x9e: /* RES (HL) 3 */
|
|
RESXX(3, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0x9f: /* RES A 3 */
|
|
RES(reg_a, 3);
|
|
break;
|
|
case 0xa0: /* RES B 4 */
|
|
RES(reg_b, 4);
|
|
break;
|
|
case 0xa1: /* RES C 4 */
|
|
RES(reg_c, 4);
|
|
break;
|
|
case 0xa2: /* RES D 4 */
|
|
RES(reg_d, 4);
|
|
break;
|
|
case 0xa3: /* RES E 4 */
|
|
RES(reg_e, 4);
|
|
break;
|
|
case 0xa4: /* RES H 4 */
|
|
RES(reg_h, 4);
|
|
break;
|
|
case 0xa5: /* RES L 4 */
|
|
RES(reg_l, 4);
|
|
break;
|
|
case 0xa6: /* RES (HL) 4 */
|
|
RESXX(4, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xa7: /* RES A 4 */
|
|
RES(reg_a, 4);
|
|
break;
|
|
case 0xa8: /* RES B 5 */
|
|
RES(reg_b, 5);
|
|
break;
|
|
case 0xa9: /* RES C 5 */
|
|
RES(reg_c, 5);
|
|
break;
|
|
case 0xaa: /* RES D 5 */
|
|
RES(reg_d, 5);
|
|
break;
|
|
case 0xab: /* RES E 5 */
|
|
RES(reg_e, 5);
|
|
break;
|
|
case 0xac: /* RES H 5 */
|
|
RES(reg_h, 5);
|
|
break;
|
|
case 0xad: /* RES L 5 */
|
|
RES(reg_l, 5);
|
|
break;
|
|
case 0xae: /* RES (HL) 5 */
|
|
RESXX(5, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xaf: /* RES A 5 */
|
|
RES(reg_a, 5);
|
|
break;
|
|
case 0xb0: /* RES B 6 */
|
|
RES(reg_b, 6);
|
|
break;
|
|
case 0xb1: /* RES C 6 */
|
|
RES(reg_c, 6);
|
|
break;
|
|
case 0xb2: /* RES D 6 */
|
|
RES(reg_d, 6);
|
|
break;
|
|
case 0xb3: /* RES E 6 */
|
|
RES(reg_e, 6);
|
|
break;
|
|
case 0xb4: /* RES H 6 */
|
|
RES(reg_h, 6);
|
|
break;
|
|
case 0xb5: /* RES L 6 */
|
|
RES(reg_l, 6);
|
|
break;
|
|
case 0xb6: /* RES (HL) 6 */
|
|
RESXX(6, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xb7: /* RES A 6 */
|
|
RES(reg_a, 6);
|
|
break;
|
|
case 0xb8: /* RES B 7 */
|
|
RES(reg_b, 7);
|
|
break;
|
|
case 0xb9: /* RES C 7 */
|
|
RES(reg_c, 7);
|
|
break;
|
|
case 0xba: /* RES D 7 */
|
|
RES(reg_d, 7);
|
|
break;
|
|
case 0xbb: /* RES E 7 */
|
|
RES(reg_e, 7);
|
|
break;
|
|
case 0xbc: /* RES H 7 */
|
|
RES(reg_h, 7);
|
|
break;
|
|
case 0xbd: /* RES L 7 */
|
|
RES(reg_l, 7);
|
|
break;
|
|
case 0xbe: /* RES (HL) 7 */
|
|
RESXX(7, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xbf: /* RES A 7 */
|
|
RES(reg_a, 7);
|
|
break;
|
|
case 0xc0: /* SET B 0 */
|
|
SET(reg_b, 0);
|
|
break;
|
|
case 0xc1: /* SET C 0 */
|
|
SET(reg_c, 0);
|
|
break;
|
|
case 0xc2: /* SET D 0 */
|
|
SET(reg_d, 0);
|
|
break;
|
|
case 0xc3: /* SET E 0 */
|
|
SET(reg_e, 0);
|
|
break;
|
|
case 0xc4: /* SET H 0 */
|
|
SET(reg_h, 0);
|
|
break;
|
|
case 0xc5: /* SET L 0 */
|
|
SET(reg_l, 0);
|
|
break;
|
|
case 0xc6: /* SET (HL) 0 */
|
|
SETXX(0, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xc7: /* SET A 0 */
|
|
SET(reg_a, 0);
|
|
break;
|
|
case 0xc8: /* SET B 1 */
|
|
SET(reg_b, 1);
|
|
break;
|
|
case 0xc9: /* SET C 1 */
|
|
SET(reg_c, 1);
|
|
break;
|
|
case 0xca: /* SET D 1 */
|
|
SET(reg_d, 1);
|
|
break;
|
|
case 0xcb: /* SET E 1 */
|
|
SET(reg_e, 1);
|
|
break;
|
|
case 0xcc: /* SET H 1 */
|
|
SET(reg_h, 1);
|
|
break;
|
|
case 0xcd: /* SET L 1 */
|
|
SET(reg_l, 1);
|
|
break;
|
|
case 0xce: /* SET (HL) 1 */
|
|
SETXX(1, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xcf: /* SET A 1 */
|
|
SET(reg_a, 1);
|
|
break;
|
|
case 0xd0: /* SET B 2 */
|
|
SET(reg_b, 2);
|
|
break;
|
|
case 0xd1: /* SET C 2 */
|
|
SET(reg_c, 2);
|
|
break;
|
|
case 0xd2: /* SET D 2 */
|
|
SET(reg_d, 2);
|
|
break;
|
|
case 0xd3: /* SET E 2 */
|
|
SET(reg_e, 2);
|
|
break;
|
|
case 0xd4: /* SET H 2 */
|
|
SET(reg_h, 2);
|
|
break;
|
|
case 0xd5: /* SET L 2 */
|
|
SET(reg_l, 2);
|
|
break;
|
|
case 0xd6: /* SET (HL) 2 */
|
|
SETXX(2, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xd7: /* SET A 2 */
|
|
SET(reg_a, 2);
|
|
break;
|
|
case 0xd8: /* SET B 3 */
|
|
SET(reg_b, 3);
|
|
break;
|
|
case 0xd9: /* SET C 3 */
|
|
SET(reg_c, 3);
|
|
break;
|
|
case 0xda: /* SET D 3 */
|
|
SET(reg_d, 3);
|
|
break;
|
|
case 0xdb: /* SET E 3 */
|
|
SET(reg_e, 3);
|
|
break;
|
|
case 0xdc: /* SET H 3 */
|
|
SET(reg_h, 3);
|
|
break;
|
|
case 0xdd: /* SET L 3 */
|
|
SET(reg_l, 3);
|
|
break;
|
|
case 0xde: /* SET (HL) 3 */
|
|
SETXX(3, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xdf: /* SET A 3 */
|
|
SET(reg_a, 3);
|
|
break;
|
|
case 0xe0: /* SET B 4 */
|
|
SET(reg_b, 4);
|
|
break;
|
|
case 0xe1: /* SET C 4 */
|
|
SET(reg_c, 4);
|
|
break;
|
|
case 0xe2: /* SET D 4 */
|
|
SET(reg_d, 4);
|
|
break;
|
|
case 0xe3: /* SET E 4 */
|
|
SET(reg_e, 4);
|
|
break;
|
|
case 0xe4: /* SET H 4 */
|
|
SET(reg_h, 4);
|
|
break;
|
|
case 0xe5: /* SET L 4 */
|
|
SET(reg_l, 4);
|
|
break;
|
|
case 0xe6: /* SET (HL) 4 */
|
|
SETXX(4, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xe7: /* SET A 4 */
|
|
SET(reg_a, 4);
|
|
break;
|
|
case 0xe8: /* SET B 5 */
|
|
SET(reg_b, 5);
|
|
break;
|
|
case 0xe9: /* SET C 5 */
|
|
SET(reg_c, 5);
|
|
break;
|
|
case 0xea: /* SET D 5 */
|
|
SET(reg_d, 5);
|
|
break;
|
|
case 0xeb: /* SET E 5 */
|
|
SET(reg_e, 5);
|
|
break;
|
|
case 0xec: /* SET H 5 */
|
|
SET(reg_h, 5);
|
|
break;
|
|
case 0xed: /* SET L 5 */
|
|
SET(reg_l, 5);
|
|
break;
|
|
case 0xee: /* SET (HL) 5 */
|
|
SETXX(5, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xef: /* SET A 5 */
|
|
SET(reg_a, 5);
|
|
break;
|
|
case 0xf0: /* SET B 6 */
|
|
SET(reg_b, 6);
|
|
break;
|
|
case 0xf1: /* SET C 6 */
|
|
SET(reg_c, 6);
|
|
break;
|
|
case 0xf2: /* SET D 6 */
|
|
SET(reg_d, 6);
|
|
break;
|
|
case 0xf3: /* SET E 6 */
|
|
SET(reg_e, 6);
|
|
break;
|
|
case 0xf4: /* SET H 6 */
|
|
SET(reg_h, 6);
|
|
break;
|
|
case 0xf5: /* SET L 6 */
|
|
SET(reg_l, 6);
|
|
break;
|
|
case 0xf6: /* SET (HL) 6 */
|
|
SETXX(6, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xf7: /* SET A 6 */
|
|
SET(reg_a, 6);
|
|
break;
|
|
case 0xf8: /* SET B 7 */
|
|
SET(reg_b, 7);
|
|
break;
|
|
case 0xf9: /* SET C 7 */
|
|
SET(reg_c, 7);
|
|
break;
|
|
case 0xfa: /* SET D 7 */
|
|
SET(reg_d, 7);
|
|
break;
|
|
case 0xfb: /* SET E 7 */
|
|
SET(reg_e, 7);
|
|
break;
|
|
case 0xfc: /* SET H 7 */
|
|
SET(reg_h, 7);
|
|
break;
|
|
case 0xfd: /* SET L 7 */
|
|
SET(reg_l, 7);
|
|
break;
|
|
case 0xfe: /* SET (HL) 7 */
|
|
SETXX(7, HL_WORD(), 4, 4, 7, 2);
|
|
break;
|
|
case 0xff: /* SET A 7 */
|
|
SET(reg_a, 7);
|
|
break;
|
|
default:
|
|
INC_PC(2);
|
|
}
|
|
}
|
|
|
|
static void opcode_dd_cb(BYTE iip2, BYTE iip3, WORD iip23)
|
|
{
|
|
switch (iip3) {
|
|
case 0x00: /* RLC (IX+d),B */
|
|
RLCXXREG(reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x01: /* RLC (IX+d),C */
|
|
RLCXXREG(reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x02: /* RLC (IX+d),D */
|
|
RLCXXREG(reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x03: /* RLC (IX+d),E */
|
|
RLCXXREG(reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x04: /* RLC (IX+d),H */
|
|
RLCXXREG(reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x05: /* RLC (IX+d),L */
|
|
RLCXXREG(reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x06: /* RLC (IX+d) */
|
|
RLCXX(IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x07: /* RLC (IX+d),A */
|
|
RLCXXREG(reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x08: /* RRC (IX+d),B */
|
|
RRCXXREG(reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x09: /* RRC (IX+d),C */
|
|
RRCXXREG(reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0a: /* RRC (IX+d),D */
|
|
RRCXXREG(reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0b: /* RRC (IX+d),E */
|
|
RRCXXREG(reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0c: /* RRC (IX+d),H */
|
|
RRCXXREG(reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0d: /* RRC (IX+d),L */
|
|
RRCXXREG(reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0e: /* RRC (IX+d) */
|
|
RRCXX(IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0f: /* RRC (IX+d),A */
|
|
RRCXXREG(reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x10: /* RL (IX+d),B */
|
|
RLXXREG(reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x11: /* RL (IX+d),C */
|
|
RLXXREG(reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x12: /* RL (IX+d),D */
|
|
RLXXREG(reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x13: /* RL (IX+d),E */
|
|
RLXXREG(reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x14: /* RL (IX+d),H */
|
|
RLXXREG(reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x15: /* RL (IX+d),L */
|
|
RLXXREG(reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x16: /* RL (IX+d) */
|
|
RLXX(IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x17: /* RL (IX+d),A */
|
|
RLXXREG(reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x18: /* RR (IX+d),B */
|
|
RRXXREG(reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x19: /* RR (IX+d),C */
|
|
RRXXREG(reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1a: /* RR (IX+d),D */
|
|
RRXXREG(reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1b: /* RR (IX+d),E */
|
|
RRXXREG(reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1c: /* RR (IX+d),H */
|
|
RRXXREG(reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1d: /* RR (IX+d),L */
|
|
RRXXREG(reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1e: /* RR (IX+d) */
|
|
RRXX(IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1f: /* RR (IX+d),A */
|
|
RRXXREG(reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x20: /* SLA (IX+d),B */
|
|
SLAXXREG(reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x21: /* SLA (IX+d),C */
|
|
SLAXXREG(reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x22: /* SLA (IX+d),D */
|
|
SLAXXREG(reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x23: /* SLA (IX+d),E */
|
|
SLAXXREG(reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x24: /* SLA (IX+d),H */
|
|
SLAXXREG(reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x25: /* SLA (IX+d),L */
|
|
SLAXXREG(reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x26: /* SLA (IX+d) */
|
|
SLAXX(IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x27: /* SLA (IX+d),A */
|
|
SLAXXREG(reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x28: /* SRA (IX+d),B */
|
|
SRAXXREG(reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x29: /* SRA (IX+d),C */
|
|
SRAXXREG(reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2a: /* SRA (IX+d),D */
|
|
SRAXXREG(reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2b: /* SRA (IX+d),E */
|
|
SRAXXREG(reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2c: /* SRA (IX+d),H */
|
|
SRAXXREG(reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2d: /* SRA (IX+d),L */
|
|
SRAXXREG(reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2e: /* SRA (IX+d) */
|
|
SRAXX(IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2f: /* SRA (IX+d),A */
|
|
SRAXXREG(reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x30: /* SLL (IX+d),B */
|
|
SLLXXREG(reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x31: /* SLL (IX+d),C */
|
|
SLLXXREG(reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x32: /* SLL (IX+d),D */
|
|
SLLXXREG(reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x33: /* SLL (IX+d),E */
|
|
SLLXXREG(reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x34: /* SLL (IX+d),H */
|
|
SLLXXREG(reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x35: /* SLL (IX+d),L */
|
|
SLLXXREG(reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x36: /* SLL (IX+d) */
|
|
SLLXX(IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x37: /* SLL (IX+d),A */
|
|
SLLXXREG(reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x38: /* SRL (IX+d),B */
|
|
SRLXXREG(reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x39: /* SRL (IX+d),C */
|
|
SRLXXREG(reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3a: /* SRL (IX+d),D */
|
|
SRLXXREG(reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3b: /* SRL (IX+d),E */
|
|
SRLXXREG(reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3c: /* SRL (IX+d),H */
|
|
SRLXXREG(reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3d: /* SRL (IX+d),L */
|
|
SRLXXREG(reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3e: /* SRL (IX+d) */
|
|
SRLXX(IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3f: /* SRL (IX+d),A */
|
|
SRLXXREG(reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x40: /* BIT (IX+d) 0 */
|
|
case 0x41:
|
|
case 0x42:
|
|
case 0x43:
|
|
case 0x44:
|
|
case 0x45:
|
|
case 0x46:
|
|
case 0x47:
|
|
BIT(LOAD(IX_WORD_OFF(iip2)), 0, 8, 12, 4);
|
|
break;
|
|
case 0x48: /* BIT (IX+d) 1 */
|
|
case 0x49:
|
|
case 0x4a:
|
|
case 0x4b:
|
|
case 0x4c:
|
|
case 0x4d:
|
|
case 0x4e:
|
|
case 0x4f:
|
|
BIT(LOAD(IX_WORD_OFF(iip2)), 1, 8, 12, 4);
|
|
break;
|
|
case 0x50: /* BIT (IX+d) 2 */
|
|
case 0x51:
|
|
case 0x52:
|
|
case 0x53:
|
|
case 0x54:
|
|
case 0x55:
|
|
case 0x56:
|
|
case 0x57:
|
|
BIT(LOAD(IX_WORD_OFF(iip2)), 2, 8, 12, 4);
|
|
break;
|
|
case 0x58: /* BIT (IX+d) 3 */
|
|
case 0x59:
|
|
case 0x5a:
|
|
case 0x5b:
|
|
case 0x5c:
|
|
case 0x5d:
|
|
case 0x5e:
|
|
case 0x5f:
|
|
BIT(LOAD(IX_WORD_OFF(iip2)), 3, 8, 12, 4);
|
|
break;
|
|
case 0x60: /* BIT (IX+d) 4 */
|
|
case 0x61:
|
|
case 0x62:
|
|
case 0x63:
|
|
case 0x64:
|
|
case 0x65:
|
|
case 0x66:
|
|
case 0x67:
|
|
BIT(LOAD(IX_WORD_OFF(iip2)), 4, 8, 12, 4);
|
|
break;
|
|
case 0x68: /* BIT (IX+d) 5 */
|
|
case 0x69:
|
|
case 0x6a:
|
|
case 0x6b:
|
|
case 0x6c:
|
|
case 0x6d:
|
|
case 0x6e:
|
|
case 0x6f:
|
|
BIT(LOAD(IX_WORD_OFF(iip2)), 5, 8, 12, 4);
|
|
break;
|
|
case 0x70: /* BIT (IX+d) 6 */
|
|
case 0x71:
|
|
case 0x72:
|
|
case 0x73:
|
|
case 0x74:
|
|
case 0x75:
|
|
case 0x76:
|
|
case 0x77:
|
|
BIT(LOAD(IX_WORD_OFF(iip2)), 6, 8, 12, 4);
|
|
break;
|
|
case 0x78: /* BIT (IX+d) 7 */
|
|
case 0x79:
|
|
case 0x7a:
|
|
case 0x7b:
|
|
case 0x7c:
|
|
case 0x7d:
|
|
case 0x7e:
|
|
case 0x7f:
|
|
BIT(LOAD(IX_WORD_OFF(iip2)), 7, 8, 12, 4);
|
|
break;
|
|
case 0x80: /* RES (IX+d),B 0 */
|
|
RESXXREG(0, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x81: /* RES (IX+d),C 0 */
|
|
RESXXREG(0, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x82: /* RES (IX+d),D 0 */
|
|
RESXXREG(0, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x83: /* RES (IX+d),E 0 */
|
|
RESXXREG(0, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x84: /* RES (IX+d),H 0 */
|
|
RESXXREG(0, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x85: /* RES (IX+d),L 0 */
|
|
RESXXREG(0, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x86: /* RES (IX+d) 0 */
|
|
RESXX(0, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x87: /* RES (IX+d),A 0 */
|
|
RESXXREG(0, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x88: /* RES (IX+d),B 1 */
|
|
RESXXREG(1, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x89: /* RES (IX+d),C 1 */
|
|
RESXXREG(1, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8a: /* RES (IX+d),D 1 */
|
|
RESXXREG(1, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8b: /* RES (IX+d),E 1 */
|
|
RESXXREG(1, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8c: /* RES (IX+d),H 1 */
|
|
RESXXREG(1, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8d: /* RES (IX+d),L 1 */
|
|
RESXXREG(1, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8e: /* RES (IX+d) 1 */
|
|
RESXX(1, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8f: /* RES (IX+d),A 1 */
|
|
RESXXREG(1, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x90: /* RES (IX+d),B 2 */
|
|
RESXXREG(2, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x91: /* RES (IX+d),C 2 */
|
|
RESXXREG(2, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x92: /* RES (IX+d),D 2 */
|
|
RESXXREG(2, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x93: /* RES (IX+d),E 2 */
|
|
RESXXREG(2, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x94: /* RES (IX+d),H 2 */
|
|
RESXXREG(2, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x95: /* RES (IX+d),L 2 */
|
|
RESXXREG(2, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x96: /* RES (IX+d) 2 */
|
|
RESXX(2, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x97: /* RES (IX+d),A 2 */
|
|
RESXXREG(2, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x98: /* RES (IX+d),B 3 */
|
|
RESXXREG(3, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x99: /* RES (IX+d),C 3 */
|
|
RESXXREG(3, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9a: /* RES (IX+d),D 3 */
|
|
RESXXREG(3, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9b: /* RES (IX+d),E 3 */
|
|
RESXXREG(3, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9c: /* RES (IX+d),H 3 */
|
|
RESXXREG(3, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9d: /* RES (IX+d),L 3 */
|
|
RESXXREG(3, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9e: /* RES (IX+d) 3 */
|
|
RESXX(3, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9f: /* RES (IX+d),A 3 */
|
|
RESXXREG(3, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa0: /* RES (IX+d),B 4 */
|
|
RESXXREG(4, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa1: /* RES (IX+d),C 4 */
|
|
RESXXREG(4, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa2: /* RES (IX+d),D 4 */
|
|
RESXXREG(4, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa3: /* RES (IX+d),E 4 */
|
|
RESXXREG(4, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa4: /* RES (IX+d),H 4 */
|
|
RESXXREG(4, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa5: /* RES (IX+d),L 4 */
|
|
RESXXREG(4, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa6: /* RES (IX+d) 4 */
|
|
RESXX(4, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa7: /* RES (IX+d),A 4 */
|
|
RESXXREG(4, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa8: /* RES (IX+d),B 5 */
|
|
RESXXREG(5, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa9: /* RES (IX+d),C 5 */
|
|
RESXXREG(5, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xaa: /* RES (IX+d),D 5 */
|
|
RESXXREG(5, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xab: /* RES (IX+d),E 5 */
|
|
RESXXREG(5, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xac: /* RES (IX+d),H 5 */
|
|
RESXXREG(5, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xad: /* RES (IX+d),L 5 */
|
|
RESXXREG(5, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xae: /* RES (IX+d) 5 */
|
|
RESXX(5, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xaf: /* RES (IX+d),A 5 */
|
|
RESXXREG(5, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb0: /* RES (IX+d),B 6 */
|
|
RESXXREG(6, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb1: /* RES (IX+d),C 6 */
|
|
RESXXREG(6, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb2: /* RES (IX+d),D 6 */
|
|
RESXXREG(6, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb3: /* RES (IX+d),E 6 */
|
|
RESXXREG(6, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb4: /* RES (IX+d),H 6 */
|
|
RESXXREG(6, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb5: /* RES (IX+d),L 6 */
|
|
RESXXREG(6, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb6: /* RES (IX+d) 6 */
|
|
RESXX(6, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb7: /* RES (IX+d),A 6 */
|
|
RESXXREG(6, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb8: /* RES (IX+d),B 7 */
|
|
RESXXREG(7, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb9: /* RES (IX+d),C 7 */
|
|
RESXXREG(7, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xba: /* RES (IX+d),D 7 */
|
|
RESXXREG(7, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xbb: /* RES (IX+d),E 7 */
|
|
RESXXREG(7, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xbc: /* RES (IX+d),H 7 */
|
|
RESXXREG(7, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xbd: /* RES (IX+d),L 7 */
|
|
RESXXREG(7, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xbe: /* RES (IX+d) 7 */
|
|
RESXX(7, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xbf: /* RES (IX+d),A 7 */
|
|
RESXXREG(7, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc0: /* SET (IX+d),B 0 */
|
|
SETXXREG(0, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc1: /* SET (IX+d),C 0 */
|
|
SETXXREG(0, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc2: /* SET (IX+d),D 0 */
|
|
SETXXREG(0, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc3: /* SET (IX+d),E 0 */
|
|
SETXXREG(0, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc4: /* SET (IX+d),H 0 */
|
|
SETXXREG(0, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc5: /* SET (IX+d),L 0 */
|
|
SETXXREG(0, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc6: /* SET (IX+d) 0 */
|
|
SETXX(0, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc7: /* SET (IX+d),A 0 */
|
|
SETXXREG(0, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc8: /* SET (IX+d),B 1 */
|
|
SETXXREG(1, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc9: /* SET (IX+d),C 1 */
|
|
SETXXREG(1, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xca: /* SET (IX+d),D 1 */
|
|
SETXXREG(1, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xcb: /* SET (IX+d),E 1 */
|
|
SETXXREG(1, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xcc: /* SET (IX+d),H 1 */
|
|
SETXXREG(1, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xcd: /* SET (IX+d),L 1 */
|
|
SETXXREG(1, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xce: /* SET (IX+d) 1 */
|
|
SETXX(1, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xcf: /* SET (IX+d),A 1 */
|
|
SETXXREG(1, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd0: /* SET (IX+d),B 2 */
|
|
SETXXREG(2, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd1: /* SET (IX+d),C 2 */
|
|
SETXXREG(2, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd2: /* SET (IX+d),D 2 */
|
|
SETXXREG(2, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd3: /* SET (IX+d),E 2 */
|
|
SETXXREG(2, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd4: /* SET (IX+d),H 2 */
|
|
SETXXREG(2, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd5: /* SET (IX+d),L 2 */
|
|
SETXXREG(2, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd6: /* SET (IX+d) 2 */
|
|
SETXX(2, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd7: /* SET (IX+d),A 2 */
|
|
SETXXREG(2, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd8: /* SET (IX+d),B 3 */
|
|
SETXXREG(3, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd9: /* SET (IX+d),C 3 */
|
|
SETXXREG(3, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xda: /* SET (IX+d),D 3 */
|
|
SETXXREG(3, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xdb: /* SET (IX+d),E 3 */
|
|
SETXXREG(3, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xdc: /* SET (IX+d),H 3 */
|
|
SETXXREG(3, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xdd: /* SET (IX+d),L 3 */
|
|
SETXXREG(3, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xde: /* SET (IX+d) 3 */
|
|
SETXX(3, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xdf: /* SET (IX+d),A 3 */
|
|
SETXXREG(3, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe0: /* SET (IX+d),B 4 */
|
|
SETXXREG(4, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe1: /* SET (IX+d),C 4 */
|
|
SETXXREG(4, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe2: /* SET (IX+d),D 4 */
|
|
SETXXREG(4, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe3: /* SET (IX+d),E 4 */
|
|
SETXXREG(4, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe4: /* SET (IX+d),H 4 */
|
|
SETXXREG(4, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe5: /* SET (IX+d),L 4 */
|
|
SETXXREG(4, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe6: /* SET (IX+d) 4 */
|
|
SETXX(4, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe7: /* SET (IX+d),A 4 */
|
|
SETXXREG(4, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe8: /* SET (IX+d),B 5 */
|
|
SETXXREG(5, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe9: /* SET (IX+d),C 5 */
|
|
SETXXREG(5, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xea: /* SET (IX+d),D 5 */
|
|
SETXXREG(5, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xeb: /* SET (IX+d),E 5 */
|
|
SETXXREG(5, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xec: /* SET (IX+d),H 5 */
|
|
SETXXREG(5, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xed: /* SET (IX+d),L 5 */
|
|
SETXXREG(5, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xee: /* SET (IX+d) 5 */
|
|
SETXX(5, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xef: /* SET (IX+d),A 5 */
|
|
SETXXREG(5, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf0: /* SET (IX+d),B 6 */
|
|
SETXXREG(6, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf1: /* SET (IX+d),C 6 */
|
|
SETXXREG(6, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf2: /* SET (IX+d),D 6 */
|
|
SETXXREG(6, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf3: /* SET (IX+d),E 6 */
|
|
SETXXREG(6, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf4: /* SET (IX+d),H 6 */
|
|
SETXXREG(6, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf5: /* SET (IX+d),L 6 */
|
|
SETXXREG(6, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf6: /* SET (IX+d) 6 */
|
|
SETXX(6, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf7: /* SET (IX+d),A 6 */
|
|
SETXXREG(6, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf8: /* SET (IX+d),B 7 */
|
|
SETXXREG(7, reg_b, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf9: /* SET (IX+d),C 7 */
|
|
SETXXREG(7, reg_c, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xfa: /* SET (IX+d),D 7 */
|
|
SETXXREG(7, reg_d, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xfb: /* SET (IX+d),E 7 */
|
|
SETXXREG(7, reg_e, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xfc: /* SET (IX+d),H 7 */
|
|
SETXXREG(7, reg_h, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xfd: /* SET (IX+d),L 7 */
|
|
SETXXREG(7, reg_l, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xfe: /* SET (IX+d) 7 */
|
|
SETXX(7, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xff: /* SET (IX+d),A 7 */
|
|
SETXXREG(7, reg_a, IX_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
default:
|
|
INC_PC(4);
|
|
}
|
|
}
|
|
|
|
static void opcode_dd(BYTE ip1, BYTE ip2, BYTE ip3, WORD ip12, WORD ip23)
|
|
{
|
|
switch (ip1) {
|
|
case 0x00: /* NOP */
|
|
NOP(8, 2);
|
|
break;
|
|
case 0x01: /* LD BC # */
|
|
LDW(ip23, reg_b, reg_c, 10, 0, 4);
|
|
break;
|
|
case 0x02: /* LD (BC) A */
|
|
STREG(BC_WORD(), reg_a, 8, 3, 2);
|
|
break;
|
|
case 0x03: /* INC BC */
|
|
DECINC(INC_BC_WORD(), 10, 2);
|
|
break;
|
|
case 0x04: /* INC B */
|
|
INCREG(reg_b, 7, 2);
|
|
break;
|
|
case 0x05: /* DEC B */
|
|
DECREG(reg_b, 7, 2);
|
|
break;
|
|
case 0x06: /* LD B # */
|
|
LDREG(reg_b, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x07: /* RLCA */
|
|
RLCA(8, 2);
|
|
break;
|
|
case 0x08: /* EX AF AF' */
|
|
EXAFAF(12, 2);
|
|
break;
|
|
case 0x09: /* ADD IX BC */
|
|
ADDXXREG(reg_ixh, reg_ixl, reg_b, reg_c, 15, 2);
|
|
break;
|
|
case 0x0a: /* LD A (BC) */
|
|
LDREG(reg_a, LOAD(BC_WORD()), 8, 3, 2);
|
|
break;
|
|
case 0x0b: /* DEC BC */
|
|
DECINC(DEC_BC_WORD(), 10, 2);
|
|
break;
|
|
case 0x0c: /* INC C */
|
|
INCREG(reg_c, 7, 2);
|
|
break;
|
|
case 0x0d: /* DEC C */
|
|
DECREG(reg_c, 7, 2);
|
|
break;
|
|
case 0x0e: /* LD C # */
|
|
LDREG(reg_c, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x0f: /* RRCA */
|
|
RRCA(8, 2);
|
|
break;
|
|
case 0x10: /* DJNZ */
|
|
DJNZ(ip2, 3);
|
|
break;
|
|
case 0x11: /* LD DE # */
|
|
LDW(ip23, reg_d, reg_e, 10, 0, 4);
|
|
break;
|
|
case 0x12: /* LD (DE) A */
|
|
STREG(DE_WORD(), reg_a, 8, 3, 2);
|
|
break;
|
|
case 0x13: /* INC DE */
|
|
DECINC(INC_DE_WORD(), 10, 2);
|
|
break;
|
|
case 0x14: /* INC D */
|
|
INCREG(reg_d, 7, 2);
|
|
break;
|
|
case 0x15: /* DEC D */
|
|
DECREG(reg_d, 7, 2);
|
|
break;
|
|
case 0x16: /* LD D # */
|
|
LDREG(reg_d, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x17: /* RLA */
|
|
RLA(8, 2);
|
|
break;
|
|
case 0x19: /* ADD IX DE */
|
|
ADDXXREG(reg_ixh, reg_ixl, reg_d, reg_e, 15, 2);
|
|
break;
|
|
case 0x1a: /* LD A DE */
|
|
LDREG(reg_a, LOAD(DE_WORD()), 8, 3, 2);
|
|
break;
|
|
case 0x1b: /* DEC DE */
|
|
DECINC(DEC_DE_WORD(), 10, 2);
|
|
break;
|
|
case 0x1c: /* INC E */
|
|
INCREG(reg_e, 7, 2);
|
|
break;
|
|
case 0x1d: /* DEC E */
|
|
DECREG(reg_e, 7, 2);
|
|
break;
|
|
case 0x1e: /* LD E # */
|
|
LDREG(reg_e, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x1f: /* RRA */
|
|
RRA(8, 2);
|
|
break;
|
|
case 0x20: /* JR NZ */
|
|
BRANCH(!LOCAL_ZERO(), ip2, 3);
|
|
break;
|
|
case 0x21: /* LD IX # */
|
|
LDW(ip23, reg_ixh, reg_ixl, 10, 4, 4);
|
|
break;
|
|
case 0x22: /* LD (WORD) IX */
|
|
STW(ip23, reg_ixh, reg_ixl, 4, 9, 7, 4);
|
|
break;
|
|
case 0x23: /* INC IX */
|
|
DECINC(INC_IX_WORD(), 10, 2);
|
|
break;
|
|
case 0x24: /* INC IXH */
|
|
INCREG(reg_ixh, 7, 2);
|
|
break;
|
|
case 0x25: /* DEC IXH */
|
|
DECREG(reg_ixh, 7, 2);
|
|
break;
|
|
case 0x26: /* LD IXH # */
|
|
LDREG(reg_ixh, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x27: /* DAA */
|
|
DAA(8, 2);
|
|
break;
|
|
case 0x29: /* ADD IX IX */
|
|
ADDXXREG(reg_ixh, reg_ixl, reg_ixh, reg_ixl, 15, 2);
|
|
break;
|
|
case 0x28: /* JR Z */
|
|
BRANCH(LOCAL_ZERO(), ip2, 3);
|
|
break;
|
|
case 0x2a: /* LD IX (WORD) */
|
|
LDIND(ip23, reg_ixh, reg_ixl, 4, 4, 12, 4);
|
|
break;
|
|
case 0x2b: /* DEC IX */
|
|
DECINC(DEC_IX_WORD(), 10, 2);
|
|
break;
|
|
case 0x2c: /* INC IXL */
|
|
INCREG(reg_ixl, 7, 2);
|
|
break;
|
|
case 0x2d: /* DEC IXL */
|
|
DECREG(reg_ixl, 7, 2);
|
|
break;
|
|
case 0x2e: /* LD IXL # */
|
|
LDREG(reg_ixl, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x2f: /* CPL */
|
|
CPL(8, 2);
|
|
break;
|
|
case 0x30: /* JR NC */
|
|
BRANCH(!LOCAL_CARRY(), ip2, 3);
|
|
break;
|
|
case 0x31: /* LD SP # */
|
|
LDSP(ip23, 10, 0, 4);
|
|
break;
|
|
case 0x32: /* LD (WORD) A */
|
|
STREG(ip23, reg_a, 10, 7, 4);
|
|
break;
|
|
case 0x33: /* INC SP */
|
|
DECINC(reg_sp++, 10, 2);
|
|
break;
|
|
case 0x34: /* INC (IX+d) */
|
|
INCXXIND(IX_WORD_OFF(ip2), 4, 7, 12, 3);
|
|
break;
|
|
case 0x35: /* DEC (IX+d) */
|
|
DECXXIND(IX_WORD_OFF(ip2), 4, 7, 12, 3);
|
|
break;
|
|
case 0x36: /* LD (IX+d) # */
|
|
STREG(IX_WORD_OFF(ip2), ip3, 8, 11, 4);
|
|
break;
|
|
case 0x37: /* SCF */
|
|
SCF(8, 2);
|
|
break;
|
|
case 0x38: /* JR C */
|
|
BRANCH(LOCAL_CARRY(), ip2, 3);
|
|
break;
|
|
case 0x39: /* ADD IX SP */
|
|
ADDXXSP(reg_ixh, reg_ixl, 15, 2);
|
|
break;
|
|
case 0x3a: /* LD A (WORD) */
|
|
LDREG(reg_a, LOAD(ip23), 10, 7, 4);
|
|
break;
|
|
case 0x3b: /* DEC SP */
|
|
DECINC(reg_sp--, 10, 2);
|
|
break;
|
|
case 0x3c: /* INC A */
|
|
INCREG(reg_a, 7, 2);
|
|
break;
|
|
case 0x3d: /* DEC A */
|
|
DECREG(reg_a, 7, 2);
|
|
break;
|
|
case 0x3e: /* LD A # */
|
|
LDREG(reg_a, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x3f: /* CCF */
|
|
CCF(8, 2);
|
|
break;
|
|
case 0x40: /* LD B B */
|
|
LDREG(reg_b, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x41: /* LD B C */
|
|
LDREG(reg_b, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x42: /* LD B D */
|
|
LDREG(reg_b, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x43: /* LD B E */
|
|
LDREG(reg_b, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x44: /* LD B IXH */
|
|
LDREG(reg_b, reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x45: /* LD B IXL */
|
|
LDREG(reg_b, reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x46: /* LD B (IX+d) */
|
|
LDREG(reg_b, LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x47: /* LD B A */
|
|
LDREG(reg_b, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x48: /* LD C B */
|
|
LDREG(reg_c, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x49: /* LD C C */
|
|
LDREG(reg_c, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x4a: /* LD C D */
|
|
LDREG(reg_c, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x4b: /* LD C E */
|
|
LDREG(reg_c, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x4c: /* LD C IXH */
|
|
LDREG(reg_c, reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x4d: /* LD C IXL */
|
|
LDREG(reg_c, reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x4e: /* LD C (IX+d) */
|
|
LDREG(reg_c, LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x4f: /* LD C A */
|
|
LDREG(reg_c, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x50: /* LD D B */
|
|
LDREG(reg_d, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x51: /* LD D C */
|
|
LDREG(reg_d, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x52: /* LD D D */
|
|
LDREG(reg_d, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x53: /* LD D E */
|
|
LDREG(reg_d, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x54: /* LD D IXH */
|
|
LDREG(reg_d, reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x55: /* LD D L */
|
|
LDREG(reg_d, reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x56: /* LD D (IX+d) */
|
|
LDREG(reg_d, LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x57: /* LD D A */
|
|
LDREG(reg_d, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x58: /* LD E B */
|
|
LDREG(reg_e, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x59: /* LD E C */
|
|
LDREG(reg_e, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x5a: /* LD E D */
|
|
LDREG(reg_e, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x5b: /* LD E E */
|
|
LDREG(reg_e, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x5c: /* LD E IXH */
|
|
LDREG(reg_e, reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x5d: /* LD E IXL */
|
|
LDREG(reg_e, reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x5e: /* LD E (IX+d) */
|
|
LDREG(reg_e, LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x5f: /* LD E A */
|
|
LDREG(reg_e, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x60: /* LD IXH B */
|
|
LDREG(reg_ixh, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x61: /* LD IXH C */
|
|
LDREG(reg_ixh, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x62: /* LD IXH D */
|
|
LDREG(reg_ixh, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x63: /* LD IXH E */
|
|
LDREG(reg_ixh, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x64: /* LD IXH IXH */
|
|
LDREG(reg_ixh, reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x65: /* LD IXH IXL */
|
|
LDREG(reg_ixh, reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x66: /* LD H (IX+d) */
|
|
LDREG(reg_h, LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x67: /* LD IXH A */
|
|
LDREG(reg_ixh, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x68: /* LD IXL B */
|
|
LDREG(reg_ixl, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x69: /* LD IXL C */
|
|
LDREG(reg_ixl, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x6a: /* LD IXL D */
|
|
LDREG(reg_ixl, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x6b: /* LD IXL E */
|
|
LDREG(reg_ixl, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x6c: /* LD IXL IXH */
|
|
LDREG(reg_ixl, reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x6d: /* LD IXL IXL */
|
|
LDREG(reg_ixl, reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x6e: /* LD L (IX+d) */
|
|
LDREG(reg_l, LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x6f: /* LD IXL A */
|
|
LDREG(reg_ixl, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x70: /* LD (IX+d) B */
|
|
STREG(IX_WORD_OFF(ip2), reg_b, 8, 11, 3);
|
|
break;
|
|
case 0x71: /* LD (IX+d) C */
|
|
STREG(IX_WORD_OFF(ip2), reg_c, 8, 11, 3);
|
|
break;
|
|
case 0x72: /* LD (IX+d) D */
|
|
STREG(IX_WORD_OFF(ip2), reg_d, 8, 11, 3);
|
|
break;
|
|
case 0x73: /* LD (IX+d) E */
|
|
STREG(IX_WORD_OFF(ip2), reg_e, 8, 11, 3);
|
|
break;
|
|
case 0x74: /* LD (IX+d) H */
|
|
STREG(IX_WORD_OFF(ip2), reg_h, 8, 11, 3);
|
|
break;
|
|
case 0x75: /* LD (IX+d) L */
|
|
STREG(IX_WORD_OFF(ip2), reg_l, 8, 11, 3);
|
|
break;
|
|
case 0x76: /* HALT */
|
|
HALT();
|
|
break;
|
|
case 0x77: /* LD (IX+d) A */
|
|
STREG(IX_WORD_OFF(ip2), reg_a, 8, 11, 3);
|
|
break;
|
|
case 0x78: /* LD A B */
|
|
LDREG(reg_a, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x79: /* LD A C */
|
|
LDREG(reg_a, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x7a: /* LD A D */
|
|
LDREG(reg_a, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x7b: /* LD A E */
|
|
LDREG(reg_a, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x7c: /* LD A IXH */
|
|
LDREG(reg_a, reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x7d: /* LD A IXL */
|
|
LDREG(reg_a, reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x7e: /* LD A (IX+d) */
|
|
LDREG(reg_a, LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x7f: /* LD A A */
|
|
LDREG(reg_a, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x80: /* ADD B */
|
|
ADD(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x81: /* ADD C */
|
|
ADD(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x82: /* ADD D */
|
|
ADD(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x83: /* ADD E */
|
|
ADD(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x84: /* ADD IXH */
|
|
ADD(reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x85: /* ADD IXL */
|
|
ADD(reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x86: /* ADD (IX+d) */
|
|
ADD(LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x87: /* ADD A */
|
|
ADD(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x88: /* ADC B */
|
|
ADC(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x89: /* ADC C */
|
|
ADC(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x8a: /* ADC D */
|
|
ADC(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x8b: /* ADC E */
|
|
ADC(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x8c: /* ADC IXH */
|
|
ADC(reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x8d: /* ADC IXL */
|
|
ADC(reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x8e: /* ADC (IX+d) */
|
|
ADC(LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x8f: /* ADC A */
|
|
ADC(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x90: /* SUB B */
|
|
SUB(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x91: /* SUB C */
|
|
SUB(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x92: /* SUB D */
|
|
SUB(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x93: /* SUB E */
|
|
SUB(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x94: /* SUB IXH */
|
|
SUB(reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x95: /* SUB IXL */
|
|
SUB(reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x96: /* SUB (IX+d) */
|
|
SUB(LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x97: /* SUB A */
|
|
SUB(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x98: /* SBC B */
|
|
SBC(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x99: /* SBC C */
|
|
SBC(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x9a: /* SBC D */
|
|
SBC(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x9b: /* SBC E */
|
|
SBC(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x9c: /* SBC IXH */
|
|
SBC(reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0x9d: /* SBC IXL */
|
|
SBC(reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0x9e: /* SBC (IX+d) */
|
|
SBC(LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x9f: /* SBC A */
|
|
SBC(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0xa0: /* AND B */
|
|
AND(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0xa1: /* AND C */
|
|
AND(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0xa2: /* AND D */
|
|
AND(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0xa3: /* AND E */
|
|
AND(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0xa4: /* AND IXH */
|
|
AND(reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0xa5: /* AND IXL */
|
|
AND(reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0xa6: /* AND (IX+d) */
|
|
AND(LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0xa7: /* AND A */
|
|
AND(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0xa8: /* XOR B */
|
|
XOR(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0xa9: /* XOR C */
|
|
XOR(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0xaa: /* XOR D */
|
|
XOR(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0xab: /* XOR E */
|
|
XOR(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0xac: /* XOR IXH */
|
|
XOR(reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0xad: /* XOR IXL */
|
|
XOR(reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0xae: /* XOR (IX+d) */
|
|
XOR(LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0xaf: /* XOR A */
|
|
XOR(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0xb0: /* OR B */
|
|
OR(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0xb1: /* OR C */
|
|
OR(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0xb2: /* OR D */
|
|
OR(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0xb3: /* OR E */
|
|
OR(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0xb4: /* OR IXH */
|
|
OR(reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0xb5: /* OR IXL */
|
|
OR(reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0xb6: /* OR (IX+d) */
|
|
OR(LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0xb7: /* OR A */
|
|
OR(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0xb8: /* CP B */
|
|
CP(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0xb9: /* CP C */
|
|
CP(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0xba: /* CP D */
|
|
CP(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0xbb: /* CP E */
|
|
CP(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0xbc: /* CP IXH */
|
|
CP(reg_ixh, 0, 4, 2);
|
|
break;
|
|
case 0xbd: /* CP IXL */
|
|
CP(reg_ixl, 0, 4, 2);
|
|
break;
|
|
case 0xbe: /* CP (IX+d) */
|
|
CP(LOAD(IX_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0xbf: /* CP A */
|
|
CP(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0xc1: /* POP BC */
|
|
POP(reg_b, reg_c, 2);
|
|
break;
|
|
case 0xc5: /* PUSH BC */
|
|
PUSH(reg_b, reg_c, 2);
|
|
break;
|
|
case 0xcb: /* OPCODE DD CB */
|
|
opcode_dd_cb((BYTE)ip2, (BYTE)ip3, (WORD)ip23);
|
|
break;
|
|
case 0xd1: /* POP DE */
|
|
POP(reg_d, reg_e, 2);
|
|
break;
|
|
case 0xd3: /* OUT A */
|
|
OUTA(ip2, 8, 7, 3);
|
|
break;
|
|
case 0xd5: /* PUSH DE */
|
|
PUSH(reg_d, reg_e, 2);
|
|
break;
|
|
case 0xd9: /* EXX */
|
|
EXX(12, 2);
|
|
break;
|
|
case 0xdb: /* IN A */
|
|
INA(ip2, 8, 7, 3);
|
|
break;
|
|
case 0xdd: /* Skip DD */
|
|
NOP(4, 1);
|
|
break;
|
|
case 0xe1: /* POP IX */
|
|
POP(reg_ixh, reg_ixl, 2);
|
|
break;
|
|
case 0xe3: /* EX IX (SP) */
|
|
EXXXSP(reg_ixh, reg_ixl, 4, 4, 4, 4, 7, 2);
|
|
break;
|
|
case 0xe5: /* PUSH IX */
|
|
PUSH(reg_ixh, reg_ixl, 2);
|
|
break;
|
|
case 0xe9: /* LD PC IX */
|
|
JMP((IX_WORD()), 8);
|
|
break;
|
|
case 0xeb:
|
|
EXDEHL(8, 2);
|
|
break;
|
|
case 0xed: /* Skip DD */
|
|
NOP(4, 1);
|
|
break;
|
|
case 0xf1: /* POP AF */
|
|
POP(reg_a, reg_f, 2);
|
|
break;
|
|
case 0xf3: /* DI */
|
|
DI(8, 2);
|
|
break;
|
|
case 0xf5: /* PUSH AF */
|
|
PUSH(reg_a, reg_f, 2);
|
|
break;
|
|
case 0xf9: /* LD SP IX */
|
|
LDSP(IX_WORD(), 4, 6, 2);
|
|
break;
|
|
case 0xfb: /* EI */
|
|
EI(8, 2);
|
|
break;
|
|
case 0xfd: /* Skip DD */
|
|
NOP(4, 1);
|
|
break;
|
|
default:
|
|
#ifdef DEBUG_Z80
|
|
log_message(LOG_DEFAULT,
|
|
"%i PC %04x A%02x F%02x B%02x C%02x D%02x E%02x "
|
|
"H%02x L%02x SP%04x OP DD %02x %02x %02x.",
|
|
(int)(CLK), (unsigned int)(z80_reg_pc),
|
|
reg_a, reg_f, reg_b, reg_c, reg_d, reg_e,
|
|
reg_h, reg_l, reg_sp, ip1, ip2, ip3);
|
|
#endif
|
|
INC_PC(2);
|
|
}
|
|
}
|
|
|
|
static void opcode_ed(BYTE ip1, BYTE ip2, BYTE ip3, WORD ip12, WORD ip23)
|
|
{
|
|
switch (ip1) {
|
|
case 0x40: /* IN B BC */
|
|
INBC(reg_b, 4, 8, 2);
|
|
break;
|
|
case 0x41: /* OUT BC B */
|
|
OUTBC(reg_b, 4, 8, 2);
|
|
break;
|
|
case 0x42: /* SBC HL BC */
|
|
SBCHLREG(reg_b, reg_c);
|
|
break;
|
|
case 0x43: /* LD (WORD) BC */
|
|
STW(ip23, reg_b, reg_c, 4, 13, 3, 4);
|
|
break;
|
|
case 0x44: /* NEG */
|
|
NEG();
|
|
break;
|
|
case 0x45: /* RETN */
|
|
RETNI();
|
|
break;
|
|
case 0x46: /* IM0 */
|
|
IM(0);
|
|
break;
|
|
case 0x47: /* LD I A */
|
|
LDREG(reg_i, reg_a, 6, 3, 2);
|
|
break;
|
|
case 0x48: /* IN C BC */
|
|
INBC(reg_c, 4, 8, 2);
|
|
break;
|
|
case 0x49: /* OUT BC C */
|
|
OUTBC(reg_c, 4, 8, 2);
|
|
break;
|
|
case 0x4a: /* ADC HL BC */
|
|
ADCHLREG(reg_b, reg_c);
|
|
break;
|
|
case 0x4b: /* LD BC (WORD) */
|
|
LDIND(ip23, reg_b, reg_c, 4, 4, 12, 4);
|
|
break;
|
|
case 0x4d: /* RETI */
|
|
RETNI();
|
|
break;
|
|
case 0x4f: /* LD R A FIXME: Not emulated. */
|
|
NOP(8, 2);
|
|
break;
|
|
case 0x50: /* IN D BC */
|
|
INBC(reg_d, 4, 8, 2);
|
|
break;
|
|
case 0x51: /* OUT BC D */
|
|
OUTBC(reg_d, 4, 8, 2);
|
|
break;
|
|
case 0x52: /* SBC HL DE */
|
|
SBCHLREG(reg_d, reg_e);
|
|
break;
|
|
case 0x53: /* LD (WORD) DE */
|
|
STW(ip23, reg_d, reg_e, 4, 13, 3, 4);
|
|
break;
|
|
case 0x56: /* IM1 */
|
|
IM(1);
|
|
break;
|
|
case 0x57: /* LD A I */
|
|
LDAIR(reg_i);
|
|
break;
|
|
case 0x58: /* IN E BC */
|
|
INBC(reg_e, 4, 8, 2);
|
|
break;
|
|
case 0x59: /* OUT BC E */
|
|
OUTBC(reg_e, 4, 8, 2);
|
|
break;
|
|
case 0x5a: /* ADC HL DE */
|
|
ADCHLREG(reg_d, reg_e);
|
|
break;
|
|
case 0x5b: /* LD DE (WORD) */
|
|
LDIND(ip23, reg_d, reg_e, 4, 4, 12, 4);
|
|
break;
|
|
case 0x5e: /* IM2 */
|
|
IM(2);
|
|
break;
|
|
case 0x5f: /* LD A R */
|
|
LDAIR((BYTE)(CLK & 0xff));
|
|
break;
|
|
case 0x60: /* IN H BC */
|
|
INBC(reg_h, 4, 8, 2);
|
|
break;
|
|
case 0x61: /* OUT BC H */
|
|
OUTBC(reg_h, 4, 8, 2);
|
|
break;
|
|
case 0x62: /* SBC HL HL */
|
|
SBCHLREG(reg_h, reg_l);
|
|
break;
|
|
case 0x63: /* LD (WORD) HL */
|
|
STW(ip23, reg_h, reg_l, 4, 13, 3, 4);
|
|
break;
|
|
case 0x67: /* RRD */
|
|
RRD();
|
|
break;
|
|
case 0x68: /* IN L BC */
|
|
INBC(reg_l, 4, 8, 2);
|
|
break;
|
|
case 0x69: /* OUT BC L */
|
|
OUTBC(reg_l, 4, 8, 2);
|
|
break;
|
|
case 0x6a: /* ADC HL HL */
|
|
ADCHLREG(reg_h, reg_l);
|
|
break;
|
|
case 0x6b: /* LD HL (WORD) */
|
|
LDIND(ip23, reg_h, reg_l, 4, 4, 12, 4);
|
|
break;
|
|
case 0x6f: /* RLD */
|
|
RLD();
|
|
break;
|
|
case 0x70: /* IN F BC */
|
|
INBC0(4, 8, 2);
|
|
break;
|
|
case 0x71: /* OUT BC #0 */
|
|
OUTBC(0, 4, 8, 2);
|
|
break;
|
|
case 0x72: /* SBC HL SP */
|
|
SBCHLSP();
|
|
break;
|
|
case 0x73: /* LD (WORD) SP */
|
|
STSPW(ip23, 4, 13, 3, 4);
|
|
break;
|
|
case 0x78: /* IN A BC */
|
|
INBC(reg_a, 4, 8, 2);
|
|
break;
|
|
case 0x79: /* OUT BC A */
|
|
OUTBC(reg_a, 4, 8, 2);
|
|
break;
|
|
case 0x7a: /* ADC HL SP */
|
|
ADCHLSP();
|
|
break;
|
|
case 0x7b: /* LD SP (WORD) */
|
|
LDSPIND(ip23, 4, 4, 12, 4);
|
|
break;
|
|
case 0xa0: /* LDI */
|
|
LDDI(INC_DE_WORD(), INC_HL_WORD());
|
|
break;
|
|
case 0xa1: /* CPI */
|
|
CPDI(INC_HL_WORD());
|
|
break;
|
|
case 0xa2: /* INI */
|
|
INDI(INC_HL_WORD());
|
|
break;
|
|
case 0xa3: /* OUTI */
|
|
OUTDI(INC_HL_WORD());
|
|
break;
|
|
case 0xa8: /* LDD */
|
|
LDDI(DEC_DE_WORD(), DEC_HL_WORD());
|
|
break;
|
|
case 0xa9: /* CPD */
|
|
CPDI(DEC_HL_WORD());
|
|
break;
|
|
case 0xaa: /* IND */
|
|
INDI(DEC_HL_WORD());
|
|
break;
|
|
case 0xab: /* OUTD */
|
|
OUTDI(DEC_HL_WORD());
|
|
break;
|
|
case 0xb0: /* LDIR */
|
|
LDDIR(INC_DE_WORD(), INC_HL_WORD());
|
|
break;
|
|
case 0xb1: /* CPIR */
|
|
CPDIR(INC_HL_WORD());
|
|
break;
|
|
case 0xb2: /* INIR */
|
|
INDIR(INC_HL_WORD());
|
|
break;
|
|
case 0xb3: /* OTIR */
|
|
OTDIR(INC_HL_WORD());
|
|
break;
|
|
case 0xb8: /* LDDR */
|
|
LDDIR(DEC_DE_WORD(), DEC_HL_WORD());
|
|
break;
|
|
case 0xb9: /* CPDR */
|
|
CPDIR(DEC_HL_WORD());
|
|
break;
|
|
case 0xba: /* INDR */
|
|
INDIR(DEC_HL_WORD());
|
|
break;
|
|
case 0xbb: /* OTDR */
|
|
OTDIR(DEC_HL_WORD());
|
|
break;
|
|
case 0xcb: /* NOP */
|
|
NOP(8, 2);
|
|
break;
|
|
case 0xdd: /* NOP */
|
|
NOP(8, 2);
|
|
break;
|
|
case 0xed: /* NOP */
|
|
NOP(8, 2);
|
|
break;
|
|
case 0xfd: /* NOP */
|
|
NOP(8, 2);
|
|
break;
|
|
default:
|
|
#ifdef DEBUG_Z80
|
|
log_message(LOG_DEFAULT,
|
|
"%i PC %04x A%02x F%02x B%02x C%02x D%02x E%02x "
|
|
"H%02x L%02x SP%04x OP ED %02x %02x %02x.",
|
|
(int)(CLK), (unsigned int)(z80_reg_pc),
|
|
reg_a, reg_f, reg_b, reg_c, reg_d, reg_e,
|
|
reg_h, reg_l, reg_sp, ip1, ip2, ip3);
|
|
#endif
|
|
INC_PC(2);
|
|
}
|
|
}
|
|
|
|
static void opcode_fd_cb(BYTE iip2, BYTE iip3, WORD iip23)
|
|
{
|
|
switch (iip3) {
|
|
case 0x00: /* RLC (IY+d),B */
|
|
RLCXXREG(reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x01: /* RLC (IY+d),C */
|
|
RLCXXREG(reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x02: /* RLC (IY+d),D */
|
|
RLCXXREG(reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x03: /* RLC (IY+d),E */
|
|
RLCXXREG(reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x04: /* RLC (IY+d),H */
|
|
RLCXXREG(reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x05: /* RLC (IY+d),L */
|
|
RLCXXREG(reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x06: /* RLC (IY+d) */
|
|
RLCXX(IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x07: /* RLC (IY+d),A */
|
|
RLCXXREG(reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x08: /* RRC (IY+d),B */
|
|
RRCXXREG(reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x09: /* RRC (IY+d),C */
|
|
RRCXXREG(reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0a: /* RRC (IY+d),D */
|
|
RRCXXREG(reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0b: /* RRC (IY+d),E */
|
|
RRCXXREG(reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0c: /* RRC (IY+d),H */
|
|
RRCXXREG(reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0d: /* RRC (IY+d),L */
|
|
RRCXXREG(reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0e: /* RRC (IY+d) */
|
|
RRCXX(IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x0f: /* RRC (IY+d),A */
|
|
RRCXXREG(reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x10: /* RL (IY+d),B */
|
|
RLXXREG(reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x11: /* RL (IY+d),C */
|
|
RLXXREG(reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x12: /* RL (IY+d),D */
|
|
RLXXREG(reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x13: /* RL (IY+d),E */
|
|
RLXXREG(reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x14: /* RL (IY+d),H */
|
|
RLXXREG(reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x15: /* RL (IY+d),L */
|
|
RLXXREG(reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x16: /* RL (IY+d) */
|
|
RLXX(IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x17: /* RL (IY+d),A */
|
|
RLXXREG(reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x18: /* RR (IY+d),B */
|
|
RRXXREG(reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x19: /* RR (IY+d),C */
|
|
RRXXREG(reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1a: /* RR (IY+d),D */
|
|
RRXXREG(reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1b: /* RR (IY+d),E */
|
|
RRXXREG(reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1c: /* RR (IY+d),H */
|
|
RRXXREG(reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1d: /* RR (IY+d),L */
|
|
RRXXREG(reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1e: /* RR (IY+d) */
|
|
RRXX(IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x1f: /* RR (IY+d),A */
|
|
RRXXREG(reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x20: /* SLA (IY+d),B */
|
|
SLAXXREG(reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x21: /* SLA (IY+d),C */
|
|
SLAXXREG(reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x22: /* SLA (IY+d),D */
|
|
SLAXXREG(reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x23: /* SLA (IY+d),E */
|
|
SLAXXREG(reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x24: /* SLA (IY+d),H */
|
|
SLAXXREG(reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x25: /* SLA (IY+d),L */
|
|
SLAXXREG(reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x26: /* SLA (IY+d) */
|
|
SLAXX(IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x27: /* SLA (IY+d),A */
|
|
SLAXXREG(reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x28: /* SRA (IY+d),B */
|
|
SRAXXREG(reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x29: /* SRA (IY+d),C */
|
|
SRAXXREG(reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2a: /* SRA (IY+d),D */
|
|
SRAXXREG(reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2b: /* SRA (IY+d),E */
|
|
SRAXXREG(reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2c: /* SRA (IY+d),H */
|
|
SRAXXREG(reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2d: /* SRA (IY+d),L */
|
|
SRAXXREG(reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2e: /* SRA (IY+d) */
|
|
SRAXX(IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x2f: /* SRA (IY+d),A */
|
|
SRAXXREG(reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x30: /* SLL (IY+d),B */
|
|
SLLXXREG(reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x31: /* SLL (IY+d),C */
|
|
SLLXXREG(reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x32: /* SLL (IY+d),D */
|
|
SLLXXREG(reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x33: /* SLL (IY+d),E */
|
|
SLLXXREG(reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x34: /* SLL (IY+d),H */
|
|
SLLXXREG(reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x35: /* SLL (IY+d),L */
|
|
SLLXXREG(reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x36: /* SLL (IY+d) */
|
|
SLLXX(IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x37: /* SLL (IY+d),A */
|
|
SLLXXREG(reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x38: /* SRL (IY+d),B */
|
|
SRLXXREG(reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x39: /* SRL (IY+d),C */
|
|
SRLXXREG(reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3a: /* SRL (IY+d),D */
|
|
SRLXXREG(reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3b: /* SRL (IY+d),E */
|
|
SRLXXREG(reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3c: /* SRL (IY+d),H */
|
|
SRLXXREG(reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3d: /* SRL (IY+d),L */
|
|
SRLXXREG(reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3e: /* SRL (IY+d) */
|
|
SRLXX(IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x3f: /* SRL (IY+d),A */
|
|
SRLXXREG(reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x40: /* BIT (IY+d) 0 */
|
|
case 0x41:
|
|
case 0x42:
|
|
case 0x43:
|
|
case 0x44:
|
|
case 0x45:
|
|
case 0x46:
|
|
case 0x47:
|
|
BIT(LOAD(IY_WORD_OFF(iip2)), 0, 8, 12, 4);
|
|
break;
|
|
case 0x48: /* BIT (IY+d) 1 */
|
|
case 0x49:
|
|
case 0x4a:
|
|
case 0x4b:
|
|
case 0x4c:
|
|
case 0x4d:
|
|
case 0x4e:
|
|
case 0x4f:
|
|
BIT(LOAD(IY_WORD_OFF(iip2)), 1, 8, 12, 4);
|
|
break;
|
|
case 0x50: /* BIT (IY+d) 2 */
|
|
case 0x51:
|
|
case 0x52:
|
|
case 0x53:
|
|
case 0x54:
|
|
case 0x55:
|
|
case 0x56:
|
|
case 0x57:
|
|
BIT(LOAD(IY_WORD_OFF(iip2)), 2, 8, 12, 4);
|
|
break;
|
|
case 0x58: /* BIT (IY+d) 3 */
|
|
case 0x59:
|
|
case 0x5a:
|
|
case 0x5b:
|
|
case 0x5c:
|
|
case 0x5d:
|
|
case 0x5e:
|
|
case 0x5f:
|
|
BIT(LOAD(IY_WORD_OFF(iip2)), 3, 8, 12, 4);
|
|
break;
|
|
case 0x60: /* BIT (IY+d) 4 */
|
|
case 0x61:
|
|
case 0x62:
|
|
case 0x63:
|
|
case 0x64:
|
|
case 0x65:
|
|
case 0x66:
|
|
case 0x67:
|
|
BIT(LOAD(IY_WORD_OFF(iip2)), 4, 8, 12, 4);
|
|
break;
|
|
case 0x68: /* BIT (IY+d) 5 */
|
|
case 0x69:
|
|
case 0x6a:
|
|
case 0x6b:
|
|
case 0x6c:
|
|
case 0x6d:
|
|
case 0x6e:
|
|
case 0x6f:
|
|
BIT(LOAD(IY_WORD_OFF(iip2)), 5, 8, 12, 4);
|
|
break;
|
|
case 0x70: /* BIT (IY+d) 6 */
|
|
case 0x71:
|
|
case 0x72:
|
|
case 0x73:
|
|
case 0x74:
|
|
case 0x75:
|
|
case 0x76:
|
|
case 0x77:
|
|
BIT(LOAD(IY_WORD_OFF(iip2)), 6, 8, 12, 4);
|
|
break;
|
|
case 0x78: /* BIT (IY+d) 7 */
|
|
case 0x79:
|
|
case 0x7a:
|
|
case 0x7b:
|
|
case 0x7c:
|
|
case 0x7d:
|
|
case 0x7e:
|
|
case 0x7f:
|
|
BIT(LOAD(IY_WORD_OFF(iip2)), 7, 8, 12, 4);
|
|
break;
|
|
case 0x80: /* RES (IY+d),B 0 */
|
|
RESXXREG(0, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x81: /* RES (IY+d),C 0 */
|
|
RESXXREG(0, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x82: /* RES (IY+d),D 0 */
|
|
RESXXREG(0, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x83: /* RES (IY+d),E 0 */
|
|
RESXXREG(0, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x84: /* RES (IY+d),H 0 */
|
|
RESXXREG(0, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x85: /* RES (IY+d),L 0 */
|
|
RESXXREG(0, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x86: /* RES (IY+d) 0 */
|
|
RESXX(0, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x87: /* RES (IY+d),A 0 */
|
|
RESXXREG(0, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x88: /* RES (IY+d),B 1 */
|
|
RESXXREG(1, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x89: /* RES (IY+d),C 1 */
|
|
RESXXREG(1, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8a: /* RES (IY+d),D 1 */
|
|
RESXXREG(1, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8b: /* RES (IY+d),E 1 */
|
|
RESXXREG(1, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8c: /* RES (IY+d),H 1 */
|
|
RESXXREG(1, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8d: /* RES (IY+d),L 1 */
|
|
RESXXREG(1, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8e: /* RES (IY+d) 1 */
|
|
RESXX(1, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x8f: /* RES (IY+d),A 1 */
|
|
RESXXREG(1, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x90: /* RES (IY+d),B 2 */
|
|
RESXXREG(2, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x91: /* RES (IY+d),C 2 */
|
|
RESXXREG(2, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x92: /* RES (IY+d),D 2 */
|
|
RESXXREG(2, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x93: /* RES (IY+d),E 2 */
|
|
RESXXREG(2, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x94: /* RES (IY+d),H 2 */
|
|
RESXXREG(2, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x95: /* RES (IY+d),L 2 */
|
|
RESXXREG(2, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x96: /* RES (IY+d) 2 */
|
|
RESXX(2, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x97: /* RES (IY+d),A 2 */
|
|
RESXXREG(2, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x98: /* RES (IY+d),B 3 */
|
|
RESXXREG(3, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x99: /* RES (IY+d),C 3 */
|
|
RESXXREG(3, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9a: /* RES (IY+d),D 3 */
|
|
RESXXREG(3, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9b: /* RES (IY+d),E 3 */
|
|
RESXXREG(3, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9c: /* RES (IY+d),H 3 */
|
|
RESXXREG(3, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9d: /* RES (IY+d),L 3 */
|
|
RESXXREG(3, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9e: /* RES (IY+d) 3 */
|
|
RESXX(3, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0x9f: /* RES (IY+d),A 3 */
|
|
RESXXREG(3, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa0: /* RES (IY+d),B 4 */
|
|
RESXXREG(4, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa1: /* RES (IY+d),C 4 */
|
|
RESXXREG(4, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa2: /* RES (IY+d),D 4 */
|
|
RESXXREG(4, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa3: /* RES (IY+d),E 4 */
|
|
RESXXREG(4, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa4: /* RES (IY+d),H 4 */
|
|
RESXXREG(4, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa5: /* RES (IY+d),L 4 */
|
|
RESXXREG(4, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa6: /* RES (IY+d) 4 */
|
|
RESXX(4, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa7: /* RES (IY+d),A 4 */
|
|
RESXXREG(4, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa8: /* RES (IY+d),B 5 */
|
|
RESXXREG(5, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xa9: /* RES (IY+d),C 5 */
|
|
RESXXREG(5, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xaa: /* RES (IY+d),D 5 */
|
|
RESXXREG(5, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xab: /* RES (IY+d),E 5 */
|
|
RESXXREG(5, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xac: /* RES (IY+d),H 5 */
|
|
RESXXREG(5, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xad: /* RES (IY+d),L 5 */
|
|
RESXXREG(5, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xae: /* RES (IY+d) 5 */
|
|
RESXX(5, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xaf: /* RES (IY+d),A 5 */
|
|
RESXXREG(5, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb0: /* RES (IY+d),B 6 */
|
|
RESXXREG(6, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb1: /* RES (IY+d),C 6 */
|
|
RESXXREG(6, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb2: /* RES (IY+d),D 6 */
|
|
RESXXREG(6, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb3: /* RES (IY+d),E 6 */
|
|
RESXXREG(6, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb4: /* RES (IY+d),H 6 */
|
|
RESXXREG(6, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb5: /* RES (IY+d),L 6 */
|
|
RESXXREG(6, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb6: /* RES (IY+d) 6 */
|
|
RESXX(6, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb7: /* RES (IY+d),A 6 */
|
|
RESXXREG(6, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb8: /* RES (IY+d),B 7 */
|
|
RESXXREG(7, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xb9: /* RES (IY+d),C 7 */
|
|
RESXXREG(7, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xba: /* RES (IY+d),D 7 */
|
|
RESXXREG(7, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xbb: /* RES (IY+d),E 7 */
|
|
RESXXREG(7, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xbc: /* RES (IY+d),H 7 */
|
|
RESXXREG(7, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xbd: /* RES (IY+d),L 7 */
|
|
RESXXREG(7, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xbe: /* RES (IY+d) 7 */
|
|
RESXX(7, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xbf: /* RES (IY+d),A 7 */
|
|
RESXXREG(7, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc0: /* SET (IY+d),B 0 */
|
|
SETXXREG(0, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc1: /* SET (IY+d),C 0 */
|
|
SETXXREG(0, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc2: /* SET (IY+d),D 0 */
|
|
SETXXREG(0, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc3: /* SET (IY+d),E 0 */
|
|
SETXXREG(0, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc4: /* SET (IY+d),H 0 */
|
|
SETXXREG(0, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc5: /* SET (IY+d),L 0 */
|
|
SETXXREG(0, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc6: /* SET (IY+d) 0 */
|
|
SETXX(0, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc7: /* SET (IY+d),A 0 */
|
|
SETXXREG(0, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc8: /* SET (IY+d),B 1 */
|
|
SETXXREG(1, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xc9: /* SET (IY+d),C 1 */
|
|
SETXXREG(1, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xca: /* SET (IY+d),D 1 */
|
|
SETXXREG(1, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xcb: /* SET (IY+d),E 1 */
|
|
SETXXREG(1, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xcc: /* SET (IY+d),H 1 */
|
|
SETXXREG(1, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xcd: /* SET (IY+d),L 1 */
|
|
SETXXREG(1, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xce: /* SET (IY+d) 1 */
|
|
SETXX(1, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xcf: /* SET (IY+d),A 1 */
|
|
SETXXREG(1, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd0: /* SET (IY+d),B 2 */
|
|
SETXXREG(2, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd1: /* SET (IY+d),C 2 */
|
|
SETXXREG(2, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd2: /* SET (IY+d),D 2 */
|
|
SETXXREG(2, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd3: /* SET (IY+d),E 2 */
|
|
SETXXREG(2, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd4: /* SET (IY+d),H 2 */
|
|
SETXXREG(2, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd5: /* SET (IY+d),L 2 */
|
|
SETXXREG(2, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd6: /* SET (IY+d) 2 */
|
|
SETXX(2, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd7: /* SET (IY+d),A 2 */
|
|
SETXXREG(2, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd8: /* SET (IY+d),B 3 */
|
|
SETXXREG(3, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xd9: /* SET (IY+d),C 3 */
|
|
SETXXREG(3, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xda: /* SET (IY+d),D 3 */
|
|
SETXXREG(3, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xdb: /* SET (IY+d),E 3 */
|
|
SETXXREG(3, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xdc: /* SET (IY+d),H 3 */
|
|
SETXXREG(3, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xdd: /* SET (IY+d),L 3 */
|
|
SETXXREG(3, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xde: /* SET (IY+d) 3 */
|
|
SETXX(3, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xdf: /* SET (IY+d),A 3 */
|
|
SETXXREG(3, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe0: /* SET (IY+d),B 4 */
|
|
SETXXREG(4, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe1: /* SET (IY+d),C 4 */
|
|
SETXXREG(4, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe2: /* SET (IY+d),D 4 */
|
|
SETXXREG(4, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe3: /* SET (IY+d),E 4 */
|
|
SETXXREG(4, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe4: /* SET (IY+d),H 4 */
|
|
SETXXREG(4, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe5: /* SET (IY+d),L 4 */
|
|
SETXXREG(4, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe6: /* SET (IY+d) 4 */
|
|
SETXX(4, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe7: /* SET (IY+d),A 4 */
|
|
SETXXREG(4, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe8: /* SET (IY+d),B 5 */
|
|
SETXXREG(5, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xe9: /* SET (IY+d),C 5 */
|
|
SETXXREG(5, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xea: /* SET (IY+d),D 5 */
|
|
SETXXREG(5, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xeb: /* SET (IY+d),E 5 */
|
|
SETXXREG(5, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xec: /* SET (IY+d),H 5 */
|
|
SETXXREG(5, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xed: /* SET (IY+d),L 5 */
|
|
SETXXREG(5, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xee: /* SET (IY+d) 5 */
|
|
SETXX(5, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xef: /* SET (IY+d),A 5 */
|
|
SETXXREG(5, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf0: /* SET (IY+d),B 6 */
|
|
SETXXREG(6, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf1: /* SET (IY+d),C 6 */
|
|
SETXXREG(6, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf2: /* SET (IY+d),D 6 */
|
|
SETXXREG(6, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf3: /* SET (IY+d),E 6 */
|
|
SETXXREG(6, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf4: /* SET (IY+d),H 6 */
|
|
SETXXREG(6, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf5: /* SET (IY+d),L 6 */
|
|
SETXXREG(6, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf6: /* SET (IY+d) 6 */
|
|
SETXX(6, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf7: /* SET (IY+d),A 6 */
|
|
SETXXREG(6, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf8: /* SET (IY+d),B 7 */
|
|
SETXXREG(7, reg_b, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xf9: /* SET (IY+d),C 7 */
|
|
SETXXREG(7, reg_c, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xfa: /* SET (IY+d),D 7 */
|
|
SETXXREG(7, reg_d, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xfb: /* SET (IY+d),E 7 */
|
|
SETXXREG(7, reg_e, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xfc: /* SET (IY+d),H 7 */
|
|
SETXXREG(7, reg_h, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xfd: /* SET (IY+d),L 7 */
|
|
SETXXREG(7, reg_l, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xfe: /* SET (IY+d) 7 */
|
|
SETXX(7, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
case 0xff: /* SET (IY+d),A 7 */
|
|
SETXXREG(7, reg_a, IY_WORD_OFF(iip2), 4, 4, 15, 4);
|
|
break;
|
|
default:
|
|
INC_PC(4);
|
|
}
|
|
}
|
|
|
|
|
|
static void opcode_fd(BYTE ip1, BYTE ip2, BYTE ip3, WORD ip12, WORD ip23)
|
|
{
|
|
switch (ip1) {
|
|
case 0x00: /* NOP */
|
|
NOP(8, 2);
|
|
break;
|
|
case 0x01: /* LD BC # */
|
|
LDW(ip23, reg_b, reg_c, 10, 0, 4);
|
|
break;
|
|
case 0x02: /* LD (BC) A */
|
|
STREG(BC_WORD(), reg_a, 8, 3, 2);
|
|
break;
|
|
case 0x03: /* INC BC */
|
|
DECINC(INC_BC_WORD(), 10, 2);
|
|
break;
|
|
case 0x04: /* INC B */
|
|
INCREG(reg_b, 7, 2);
|
|
break;
|
|
case 0x05: /* DEC B */
|
|
DECREG(reg_b, 7, 2);
|
|
break;
|
|
case 0x06: /* LD B # */
|
|
LDREG(reg_b, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x07: /* RLCA */
|
|
RLCA(8, 2);
|
|
break;
|
|
case 0x08: /* EX AF AF' */
|
|
EXAFAF(12, 2);
|
|
break;
|
|
case 0x09: /* ADD IY BC */
|
|
ADDXXREG(reg_iyh, reg_iyl, reg_b, reg_c, 15, 2);
|
|
break;
|
|
case 0x0a: /* LD A (BC) */
|
|
LDREG(reg_a, LOAD(BC_WORD()), 8, 3, 2);
|
|
break;
|
|
case 0x0b: /* DEC BC */
|
|
DECINC(DEC_BC_WORD(), 10, 2);
|
|
break;
|
|
case 0x0c: /* INC C */
|
|
INCREG(reg_c, 7, 2);
|
|
break;
|
|
case 0x0d: /* DEC C */
|
|
DECREG(reg_c, 7, 2);
|
|
break;
|
|
case 0x0e: /* LD C # */
|
|
LDREG(reg_c, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x0f: /* RRCA */
|
|
RRCA(8, 2);
|
|
break;
|
|
case 0x10: /* DJNZ */
|
|
DJNZ(ip2, 3);
|
|
break;
|
|
case 0x11: /* LD DE # */
|
|
LDW(ip23, reg_d, reg_e, 10, 0, 4);
|
|
break;
|
|
case 0x12: /* LD (DE) A */
|
|
STREG(DE_WORD(), reg_a, 8, 3, 2);
|
|
break;
|
|
case 0x13: /* INC DE */
|
|
DECINC(INC_DE_WORD(), 10, 2);
|
|
break;
|
|
case 0x14: /* INC D */
|
|
INCREG(reg_d, 7, 2);
|
|
break;
|
|
case 0x15: /* DEC D */
|
|
DECREG(reg_d, 7, 2);
|
|
break;
|
|
case 0x16: /* LD D # */
|
|
LDREG(reg_d, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x17: /* RLA */
|
|
RLA(8, 2);
|
|
break;
|
|
case 0x19: /* ADD IY DE */
|
|
ADDXXREG(reg_iyh, reg_iyl, reg_d, reg_e, 15, 2);
|
|
break;
|
|
case 0x1a: /* LD A (DE) */
|
|
LDREG(reg_a, LOAD(DE_WORD()), 8, 3, 2);
|
|
break;
|
|
case 0x1b: /* DEC DE */
|
|
DECINC(DEC_DE_WORD(), 10, 2);
|
|
break;
|
|
case 0x1c: /* INC E */
|
|
INCREG(reg_e, 7, 2);
|
|
break;
|
|
case 0x1d: /* DEC E */
|
|
DECREG(reg_e, 7, 2);
|
|
break;
|
|
case 0x1e: /* LD E # */
|
|
LDREG(reg_e, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x1f: /* RRA */
|
|
RRA(8, 2);
|
|
break;
|
|
case 0x20: /* JR NZ */
|
|
BRANCH(!LOCAL_ZERO(), ip2, 3);
|
|
break;
|
|
case 0x21: /* LD IY # */
|
|
LDW(ip23, reg_iyh, reg_iyl, 10, 4, 4);
|
|
break;
|
|
case 0x22: /* LD (WORD) IY */
|
|
STW(ip23, reg_iyh, reg_iyl, 4, 9, 7, 4);
|
|
break;
|
|
case 0x23: /* INC IY */
|
|
DECINC(INC_IY_WORD(), 10, 2);
|
|
break;
|
|
case 0x24: /* INC IYH */
|
|
INCREG(reg_iyh, 7, 2);
|
|
break;
|
|
case 0x25: /* DEC IYH */
|
|
DECREG(reg_iyh, 7, 2);
|
|
break;
|
|
case 0x26: /* LD IYH # */
|
|
LDREG(reg_iyh, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x27: /* DAA */
|
|
DAA(8, 2);
|
|
break;
|
|
case 0x28: /* JR Z */
|
|
BRANCH(LOCAL_ZERO(), ip2, 3);
|
|
break;
|
|
case 0x29: /* ADD IY IY */
|
|
ADDXXREG(reg_iyh, reg_iyl, reg_iyh, reg_iyl, 15, 2);
|
|
break;
|
|
case 0x2a: /* LD IY (WORD) */
|
|
LDIND(ip23, reg_iyh, reg_iyl, 4, 4, 12, 4);
|
|
break;
|
|
case 0x2b: /* DEC IY */
|
|
DECINC(DEC_IY_WORD(), 10, 2);
|
|
break;
|
|
case 0x2c: /* INC IYL */
|
|
INCREG(reg_iyl, 7, 2);
|
|
break;
|
|
case 0x2d: /* DEC IYL */
|
|
DECREG(reg_iyl, 7, 2);
|
|
break;
|
|
case 0x2e: /* LD IYL # */
|
|
LDREG(reg_iyl, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x2f: /* CPL */
|
|
CPL(8, 2);
|
|
break;
|
|
case 0x30: /* JR NC */
|
|
BRANCH(!LOCAL_CARRY(), ip2, 3);
|
|
break;
|
|
case 0x31: /* LD SP # */
|
|
LDSP(ip23, 10, 0, 4);
|
|
break;
|
|
case 0x32: /* LD (WORD) A */
|
|
STREG(ip23, reg_a, 10, 7, 4);
|
|
break;
|
|
case 0x33: /* INC SP */
|
|
DECINC(reg_sp++, 10, 2);
|
|
break;
|
|
case 0x34: /* INC (IY+d) */
|
|
INCXXIND(IY_WORD_OFF(ip2), 4, 7, 12, 3);
|
|
break;
|
|
case 0x35: /* DEC (IY+d) */
|
|
DECXXIND(IY_WORD_OFF(ip2), 4, 7, 12, 3);
|
|
break;
|
|
case 0x36: /* LD (IY+d) # */
|
|
STREG(IY_WORD_OFF(ip2), ip3, 8, 11, 4);
|
|
break;
|
|
case 0x37: /* SCF */
|
|
SCF(8, 2);
|
|
break;
|
|
case 0x38: /* JR C */
|
|
BRANCH(LOCAL_CARRY(), ip2, 3);
|
|
break;
|
|
case 0x39: /* ADD IY SP */
|
|
ADDXXSP(reg_iyh, reg_iyl, 15, 2);
|
|
break;
|
|
case 0x3a: /* LD A (WORD) */
|
|
LDREG(reg_a, LOAD(ip23), 10, 7, 4);
|
|
break;
|
|
case 0x3b: /* DEC SP */
|
|
DECINC(reg_sp--, 10, 2);
|
|
break;
|
|
case 0x3c: /* INC A */
|
|
INCREG(reg_a, 7, 2);
|
|
break;
|
|
case 0x3d: /* DEC A */
|
|
DECREG(reg_a, 7, 2);
|
|
break;
|
|
case 0x3e: /* LD A # */
|
|
LDREG(reg_a, ip2, 4, 5, 3);
|
|
break;
|
|
case 0x3f: /* CCF */
|
|
CCF(8, 2);
|
|
break;
|
|
case 0x40: /* LD B B */
|
|
LDREG(reg_b, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x41: /* LD B C */
|
|
LDREG(reg_b, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x42: /* LD B D */
|
|
LDREG(reg_b, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x43: /* LD B E */
|
|
LDREG(reg_b, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x44: /* LD B IYH */
|
|
LDREG(reg_b, reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x45: /* LD B IYL */
|
|
LDREG(reg_b, reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x46: /* LD B (IY+d) */
|
|
LDREG(reg_b, LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x47: /* LD B A */
|
|
LDREG(reg_b, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x48: /* LD C B */
|
|
LDREG(reg_c, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x49: /* LD C C */
|
|
LDREG(reg_c, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x4a: /* LD C D */
|
|
LDREG(reg_c, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x4b: /* LD C E */
|
|
LDREG(reg_c, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x4c: /* LD C IYH */
|
|
LDREG(reg_c, reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x4d: /* LD C IYL */
|
|
LDREG(reg_c, reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x4e: /* LD C (IY+d) */
|
|
LDREG(reg_c, LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x4f: /* LD C A */
|
|
LDREG(reg_c, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x50: /* LD D B */
|
|
LDREG(reg_d, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x51: /* LD D C */
|
|
LDREG(reg_d, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x52: /* LD D D */
|
|
LDREG(reg_d, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x53: /* LD D E */
|
|
LDREG(reg_d, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x54: /* LD D IYH */
|
|
LDREG(reg_d, reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x55: /* LD D IYL */
|
|
LDREG(reg_d, reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x56: /* LD D (IY+d) */
|
|
LDREG(reg_d, LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x57: /* LD D A */
|
|
LDREG(reg_d, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x58: /* LD E B */
|
|
LDREG(reg_e, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x59: /* LD E C */
|
|
LDREG(reg_e, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x5a: /* LD E D */
|
|
LDREG(reg_e, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x5b: /* LD E E */
|
|
LDREG(reg_e, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x5c: /* LD E IYH */
|
|
LDREG(reg_e, reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x5d: /* LD E IYL */
|
|
LDREG(reg_e, reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x5e: /* LD E (IY+d) */
|
|
LDREG(reg_e, LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x5f: /* LD E A */
|
|
LDREG(reg_e, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x60: /* LD IYH B */
|
|
LDREG(reg_iyh, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x61: /* LD IYH C */
|
|
LDREG(reg_iyh, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x62: /* LD IYH D */
|
|
LDREG(reg_iyh, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x63: /* LD IYH E */
|
|
LDREG(reg_iyh, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x64: /* LD IYH IYH */
|
|
LDREG(reg_iyh, reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x65: /* LD IYH IYL */
|
|
LDREG(reg_iyh, reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x66: /* LD H (IY+d) */
|
|
LDREG(reg_h, LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x67: /* LD IYH A */
|
|
LDREG(reg_iyh, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x68: /* LD IYL B */
|
|
LDREG(reg_iyl, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x69: /* LD IYL C */
|
|
LDREG(reg_iyl, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x6a: /* LD IYL D */
|
|
LDREG(reg_iyl, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x6b: /* LD IYL E */
|
|
LDREG(reg_iyl, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x6c: /* LD IYL IYH */
|
|
LDREG(reg_iyl, reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x6d: /* LD IYL IYL */
|
|
LDREG(reg_iyl, reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x6e: /* LD L (IY+d) */
|
|
LDREG(reg_l, LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x6f: /* LD IYL A */
|
|
LDREG(reg_iyl, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x70: /* LD (IY+d) B */
|
|
STREG(IY_WORD_OFF(ip2), reg_b, 8, 11, 3);
|
|
break;
|
|
case 0x71: /* LD (IY+d) C */
|
|
STREG(IY_WORD_OFF(ip2), reg_c, 8, 11, 3);
|
|
break;
|
|
case 0x72: /* LD (IY+d) D */
|
|
STREG(IY_WORD_OFF(ip2), reg_d, 8, 11, 3);
|
|
break;
|
|
case 0x73: /* LD (IY+d) E */
|
|
STREG(IY_WORD_OFF(ip2), reg_e, 8, 11, 3);
|
|
break;
|
|
case 0x74: /* LD (IY+d) H */
|
|
STREG(IY_WORD_OFF(ip2), reg_h, 8, 11, 3);
|
|
break;
|
|
case 0x75: /* LD (IY+d) L */
|
|
STREG(IY_WORD_OFF(ip2), reg_l, 8, 11, 3);
|
|
break;
|
|
case 0x76: /* HALT */
|
|
HALT();
|
|
break;
|
|
case 0x77: /* LD (IY+d) A */
|
|
STREG(IY_WORD_OFF(ip2), reg_a, 8, 11, 3);
|
|
break;
|
|
case 0x78: /* LD A B */
|
|
LDREG(reg_a, reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x79: /* LD A C */
|
|
LDREG(reg_a, reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x7a: /* LD A D */
|
|
LDREG(reg_a, reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x7b: /* LD A E */
|
|
LDREG(reg_a, reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x7c: /* LD A IYH */
|
|
LDREG(reg_a, reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x7d: /* LD A IYL */
|
|
LDREG(reg_a, reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x7e: /* LD A (IY+d) */
|
|
LDREG(reg_a, LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x7f: /* LD A A */
|
|
LDREG(reg_a, reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x80: /* ADD B */
|
|
ADD(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x81: /* ADD C */
|
|
ADD(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x82: /* ADD D */
|
|
ADD(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x83: /* ADD E */
|
|
ADD(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x84: /* ADD IYH */
|
|
ADD(reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x85: /* ADD IYL */
|
|
ADD(reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x86: /* ADD (IY+d) */
|
|
ADD(LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x87: /* ADD A */
|
|
ADD(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x88: /* ADC B */
|
|
ADC(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x89: /* ADC C */
|
|
ADC(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x8a: /* ADC D */
|
|
ADC(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x8b: /* ADC E */
|
|
ADC(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x8c: /* ADC IYH */
|
|
ADC(reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x8d: /* ADC IYL */
|
|
ADC(reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x8e: /* ADC (IY+d) */
|
|
ADC(LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x8f: /* ADC A */
|
|
ADC(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x90: /* SUB B */
|
|
SUB(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x91: /* SUB C */
|
|
SUB(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x92: /* SUB D */
|
|
SUB(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x93: /* SUB E */
|
|
SUB(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x94: /* SUB IYH */
|
|
SUB(reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x95: /* SUB IYL */
|
|
SUB(reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x96: /* SUB (IY+d) */
|
|
SUB(LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x97: /* SUB A */
|
|
SUB(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0x98: /* SBC B */
|
|
SBC(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0x99: /* SBC C */
|
|
SBC(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0x9a: /* SBC D */
|
|
SBC(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0x9b: /* SBC E */
|
|
SBC(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0x9c: /* SBC IYH */
|
|
SBC(reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0x9d: /* SBC IYL */
|
|
SBC(reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0x9e: /* SBC (IY+d) */
|
|
SBC(LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0x9f: /* SBC A */
|
|
SBC(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0xa0: /* AND B */
|
|
AND(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0xa1: /* AND C */
|
|
AND(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0xa2: /* AND D */
|
|
AND(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0xa3: /* AND E */
|
|
AND(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0xa4: /* AND IYH */
|
|
AND(reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0xa5: /* AND IYL */
|
|
AND(reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0xa6: /* AND (IY+d) */
|
|
AND(LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0xa7: /* AND A */
|
|
AND(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0xa8: /* XOR B */
|
|
XOR(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0xa9: /* XOR C */
|
|
XOR(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0xaa: /* XOR D */
|
|
XOR(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0xab: /* XOR E */
|
|
XOR(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0xac: /* XOR IYH */
|
|
XOR(reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0xad: /* XOR IYL */
|
|
XOR(reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0xae: /* XOR (IY+d) */
|
|
XOR(LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0xaf: /* XOR A */
|
|
XOR(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0xb0: /* OR B */
|
|
OR(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0xb1: /* OR C */
|
|
OR(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0xb2: /* OR D */
|
|
OR(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0xb3: /* OR E */
|
|
OR(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0xb4: /* OR IYH */
|
|
OR(reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0xb5: /* OR IYL */
|
|
OR(reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0xb6: /* OR (IY+d) */
|
|
OR(LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0xb7: /* OR A */
|
|
OR(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0xb8: /* CP B */
|
|
CP(reg_b, 0, 4, 2);
|
|
break;
|
|
case 0xb9: /* CP C */
|
|
CP(reg_c, 0, 4, 2);
|
|
break;
|
|
case 0xba: /* CP D */
|
|
CP(reg_d, 0, 4, 2);
|
|
break;
|
|
case 0xbb: /* CP E */
|
|
CP(reg_e, 0, 4, 2);
|
|
break;
|
|
case 0xbc: /* CP IYH */
|
|
CP(reg_iyh, 0, 4, 2);
|
|
break;
|
|
case 0xbd: /* CP IYL */
|
|
CP(reg_iyl, 0, 4, 2);
|
|
break;
|
|
case 0xbe: /* CP (IY+d) */
|
|
CP(LOAD(IY_WORD_OFF(ip2)), 8, 11, 3);
|
|
break;
|
|
case 0xbf: /* CP A */
|
|
CP(reg_a, 0, 4, 2);
|
|
break;
|
|
case 0xc1: /* POP BC */
|
|
POP(reg_b, reg_c, 2);
|
|
break;
|
|
case 0xc5: /* PUSH BC */
|
|
PUSH(reg_b, reg_c, 2);
|
|
break;
|
|
case 0xcb: /* OPCODE FD CB */
|
|
opcode_fd_cb((BYTE)ip2, (BYTE)ip3, (WORD)ip23);
|
|
break;
|
|
case 0xd1: /* POP DE */
|
|
POP(reg_d, reg_e, 2);
|
|
break;
|
|
case 0xd3: /* OUT A */
|
|
OUTA(ip2, 8, 7, 3);
|
|
break;
|
|
case 0xd5: /* PUSH DE */
|
|
PUSH(reg_d, reg_e, 2);
|
|
break;
|
|
case 0xd9: /* EXX */
|
|
EXX(12, 2);
|
|
break;
|
|
case 0xdb: /* IN A */
|
|
INA(ip2, 8, 7, 3);
|
|
break;
|
|
case 0xdd: /* Skip FD */
|
|
NOP(4, 1);
|
|
break;
|
|
case 0xe1: /* POP IY */
|
|
POP(reg_iyh, reg_iyl, 2);
|
|
break;
|
|
case 0xe3: /* EX IY (SP) */
|
|
EXXXSP(reg_iyh, reg_iyl, 4, 4, 4, 4, 7, 2);
|
|
break;
|
|
case 0xe5: /* PUSH IY */
|
|
PUSH(reg_iyh, reg_iyl, 2);
|
|
break;
|
|
case 0xe9: /* LD PC IY */
|
|
JMP((IY_WORD()), 8);
|
|
break;
|
|
case 0xeb:
|
|
EXDEHL(8, 2);
|
|
break;
|
|
case 0xed: /* Skip FD */
|
|
NOP(4, 1);
|
|
break;
|
|
case 0xf1: /* POP AF */
|
|
POP(reg_a, reg_f, 2);
|
|
break;
|
|
case 0xf3: /* DI */
|
|
DI(8, 2);
|
|
break;
|
|
case 0xf5: /* PUSH AF */
|
|
PUSH(reg_a, reg_f, 2);
|
|
break;
|
|
case 0xf9: /* LD SP IY */
|
|
LDSP(IY_WORD(), 4, 6, 2);
|
|
break;
|
|
case 0xfb: /* EI */
|
|
EI(8, 2);
|
|
break;
|
|
case 0xfd: /* Skip FD */
|
|
NOP(4, 1);
|
|
break;
|
|
default:
|
|
#ifdef DEBUG_Z80
|
|
log_message(LOG_DEFAULT,
|
|
"%i PC %04x A%02x F%02x B%02x C%02x D%02x E%02x "
|
|
"H%02x L%02x SP%04x OP FD %02x %02x %02x.",
|
|
(int)(CLK), (unsigned int)z80_reg_pc,
|
|
reg_a, reg_f, reg_b, reg_c, reg_d, reg_e,
|
|
reg_h, reg_l, reg_sp, ip1, ip2, ip3);
|
|
#endif
|
|
INC_PC(2);
|
|
}
|
|
}
|
|
|
|
/* ------------------------------------------------------------------------- */
|
|
|
|
/* Z80 mainloop. */
|
|
|
|
static const double uZ80ClockMultiplier = CLK_Z80 / CLK_6502;
|
|
inline static ULONG ConvertZ80TStatesTo6502Cycles(UINT uTStates)
|
|
{
|
|
return (uTStates < 0) ? 0 : (ULONG) ((double)uTStates / uZ80ClockMultiplier);
|
|
}
|
|
|
|
//void z80_mainloop(interrupt_cpu_status_t *cpu_int_status,
|
|
// alarm_context_t *cpu_alarm_context)
|
|
|
|
DWORD z80_mainloop(ULONG uTotalCycles, ULONG uExecutedCycles)
|
|
{
|
|
opcode_t opcode;
|
|
|
|
import_registers();
|
|
|
|
//z80mem_set_bank_pointer(&z80_bank_base, &z80_bank_limit); // [AppleWin-TC] Not used
|
|
|
|
//dma_request = 0; // [AppleWin-TC] Not used
|
|
|
|
uTotalCycles = (ULONG) ((double)uTotalCycles * uZ80ClockMultiplier);
|
|
uExecutedCycles = (ULONG) ((double)uExecutedCycles * uZ80ClockMultiplier);
|
|
maincpu_clk = uExecutedCycles; // Must be signed int, as cycles can go -ve
|
|
|
|
do {
|
|
|
|
// [AppleWin-TC] Z80 IRQs not supported
|
|
//while (CLK >= alarm_context_next_pending_clk(cpu_alarm_context))
|
|
// alarm_context_dispatch(cpu_alarm_context, CLK);
|
|
|
|
//{
|
|
// enum cpu_int pending_interrupt;
|
|
|
|
// pending_interrupt = cpu_int_status->global_pending_int;
|
|
// if (pending_interrupt != IK_NONE) {
|
|
// DO_INTERRUPT(pending_interrupt);
|
|
// while (CLK >= alarm_context_next_pending_clk(cpu_alarm_context))
|
|
// alarm_context_dispatch(cpu_alarm_context, CLK);
|
|
// }
|
|
//}
|
|
|
|
FETCH_OPCODE(opcode);
|
|
|
|
#ifdef DEBUG
|
|
if (debug.maincpu_traceflg)
|
|
log_message(LOG_DEFAULT,
|
|
".%04x %i %-25s A%02x F%02x B%02x C%02x D%02x E%02x "
|
|
"H%02x L%02x S%04x",
|
|
(unsigned int)z80_reg_pc, /*CLK*/ 0,
|
|
mon_disassemble_to_string(e_comp_space, z80_reg_pc,
|
|
p0, p1, p2, p3, 1, "z80"),
|
|
reg_a, reg_f, reg_b, reg_c,
|
|
reg_d, reg_e, reg_h, reg_l, reg_sp);
|
|
#endif
|
|
|
|
SET_LAST_OPCODE(p0);
|
|
|
|
switch (p0) {
|
|
case 0x00: /* NOP */
|
|
NOP(4, 1);
|
|
break;
|
|
case 0x01: /* LD BC # */
|
|
LDW(p12, reg_b, reg_c, 10, 0, 3);
|
|
break;
|
|
case 0x02: /* LD (BC) A */
|
|
STREG(BC_WORD(), reg_a, 4, 3, 1);
|
|
break;
|
|
case 0x03: /* INC BC */
|
|
DECINC(INC_BC_WORD(), 6, 1);
|
|
break;
|
|
case 0x04: /* INC B */
|
|
INCREG(reg_b, 4, 1);
|
|
break;
|
|
case 0x05: /* DEC B */
|
|
DECREG(reg_b, 4, 1);
|
|
break;
|
|
case 0x06: /* LD B # */
|
|
LDREG(reg_b, p1, 4, 3, 2);
|
|
break;
|
|
case 0x07: /* RLCA */
|
|
RLCA(4, 1);
|
|
break;
|
|
case 0x08: /* EX AF AF' */
|
|
EXAFAF(8, 1);
|
|
break;
|
|
case 0x09: /* ADD HL BC */
|
|
ADDXXREG(reg_h, reg_l, reg_b, reg_c, 11, 1);
|
|
break;
|
|
case 0x0a: /* LD A (BC) */
|
|
LDREG(reg_a, LOAD(BC_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x0b: /* DEC BC */
|
|
DECINC(DEC_BC_WORD(), 6, 1);
|
|
break;
|
|
case 0x0c: /* INC C */
|
|
INCREG(reg_c, 4, 1);
|
|
break;
|
|
case 0x0d: /* DEC C */
|
|
DECREG(reg_c, 4, 1);
|
|
break;
|
|
case 0x0e: /* LD C # */
|
|
LDREG(reg_c, p1, 4, 3, 2);
|
|
break;
|
|
case 0x0f: /* RRCA */
|
|
RRCA(4, 1);
|
|
break;
|
|
case 0x10: /* DJNZ */
|
|
DJNZ(p1, 2);
|
|
break;
|
|
case 0x11: /* LD DE # */
|
|
LDW(p12, reg_d, reg_e, 10, 0, 3);
|
|
break;
|
|
case 0x12: /* LD (DE) A */
|
|
STREG(DE_WORD(), reg_a, 4, 3, 1);
|
|
break;
|
|
case 0x13: /* INC DE */
|
|
DECINC(INC_DE_WORD(), 6, 1);
|
|
break;
|
|
case 0x14: /* INC D */
|
|
INCREG(reg_d, 4, 1);
|
|
break;
|
|
case 0x15: /* DEC D */
|
|
DECREG(reg_d, 4, 1);
|
|
break;
|
|
case 0x16: /* LD D # */
|
|
LDREG(reg_d, p1, 4, 3, 2);
|
|
break;
|
|
case 0x17: /* RLA */
|
|
RLA(4, 1);
|
|
break;
|
|
case 0x18: /* JR */
|
|
BRANCH(1, p1, 2);
|
|
break;
|
|
case 0x19: /* ADD HL DE */
|
|
ADDXXREG(reg_h, reg_l, reg_d, reg_e, 11, 1);
|
|
break;
|
|
case 0x1a: /* LD A (DE) */
|
|
LDREG(reg_a, LOAD(DE_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x1b: /* DEC DE */
|
|
DECINC(DEC_DE_WORD(), 6, 1);
|
|
break;
|
|
case 0x1c: /* INC E */
|
|
INCREG(reg_e, 4, 1);
|
|
break;
|
|
case 0x1d: /* DEC E */
|
|
DECREG(reg_e, 4, 1);
|
|
break;
|
|
case 0x1e: /* LD E # */
|
|
LDREG(reg_e, p1, 4, 3, 2);
|
|
break;
|
|
case 0x1f: /* RRA */
|
|
RRA(4, 1);
|
|
break;
|
|
case 0x20: /* JR NZ */
|
|
BRANCH(!LOCAL_ZERO(), p1, 2);
|
|
break;
|
|
case 0x21: /* LD HL # */
|
|
LDW(p12, reg_h, reg_l, 10, 0, 3);
|
|
break;
|
|
case 0x22: /* LD (WORD) HL */
|
|
STW(p12, reg_h, reg_l, 4, 9, 3, 3);
|
|
break;
|
|
case 0x23: /* INC HL */
|
|
DECINC(INC_HL_WORD(), 6, 1);
|
|
break;
|
|
case 0x24: /* INC H */
|
|
INCREG(reg_h, 4, 1);
|
|
break;
|
|
case 0x25: /* DEC H */
|
|
DECREG(reg_h, 4, 1);
|
|
break;
|
|
case 0x26: /* LD H # */
|
|
LDREG(reg_h, p1, 4, 3, 2);
|
|
break;
|
|
case 0x27: /* DAA */
|
|
DAA(4, 1);
|
|
break;
|
|
case 0x28: /* JR Z */
|
|
BRANCH(LOCAL_ZERO(), p1, 2);
|
|
break;
|
|
case 0x29: /* ADD HL HL */
|
|
ADDXXREG(reg_h, reg_l, reg_h, reg_l, 11, 1);
|
|
break;
|
|
case 0x2a: /* LD HL (WORD) */
|
|
LDIND(p12, reg_h, reg_l, 4, 4, 8, 3);
|
|
break;
|
|
case 0x2b: /* DEC HL */
|
|
DECINC(DEC_HL_WORD(), 6, 1);
|
|
break;
|
|
case 0x2c: /* INC L */
|
|
INCREG(reg_l, 4, 1);
|
|
break;
|
|
case 0x2d: /* DEC L */
|
|
DECREG(reg_l, 4, 1);
|
|
break;
|
|
case 0x2e: /* LD L # */
|
|
LDREG(reg_l, p1, 4, 3, 2);
|
|
break;
|
|
case 0x2f: /* CPL */
|
|
CPL(4, 1);
|
|
break;
|
|
case 0x30: /* JR NC */
|
|
BRANCH(!LOCAL_CARRY(), p1, 2);
|
|
break;
|
|
case 0x31: /* LD SP # */
|
|
LDSP(p12, 10, 0, 3);
|
|
break;
|
|
case 0x32: /* LD (WORD) A */
|
|
STREG(p12, reg_a, 10, 3, 3);
|
|
break;
|
|
case 0x33: /* INC SP */
|
|
DECINC(reg_sp++, 6, 1);
|
|
break;
|
|
case 0x34: /* INC (HL) */
|
|
INCXXIND(HL_WORD(), 4, 4, 3, 1);
|
|
break;
|
|
case 0x35: /* DEC (HL) */
|
|
DECXXIND(HL_WORD(), 4, 4, 3, 1);
|
|
break;
|
|
case 0x36: /* LD (HL) # */
|
|
STREG(HL_WORD(), p1, 8, 2, 2);
|
|
break;
|
|
case 0x37: /* SCF */
|
|
SCF(4, 1);
|
|
break;
|
|
case 0x38: /* JR C */
|
|
BRANCH(LOCAL_CARRY(), p1, 2);
|
|
break;
|
|
case 0x39: /* ADD HL SP */
|
|
ADDXXSP(reg_h, reg_l, 11, 1);
|
|
break;
|
|
case 0x3a: /* LD A (WORD) */
|
|
LDREG(reg_a, LOAD(p12), 10, 3, 3);
|
|
break;
|
|
case 0x3b: /* DEC SP */
|
|
DECINC(reg_sp--, 6, 1);
|
|
break;
|
|
case 0x3c: /* INC A */
|
|
INCREG(reg_a, 4, 1);
|
|
break;
|
|
case 0x3d: /* DEC A */
|
|
DECREG(reg_a, 4, 1);
|
|
break;
|
|
case 0x3e: /* LD A # */
|
|
LDREG(reg_a, p1, 4, 3, 2);
|
|
break;
|
|
case 0x3f: /* CCF */
|
|
CCF(4, 1);
|
|
break;
|
|
case 0x40: /* LD B B */
|
|
LDREG(reg_b, reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x41: /* LD B C */
|
|
LDREG(reg_b, reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x42: /* LD B D */
|
|
LDREG(reg_b, reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x43: /* LD B E */
|
|
LDREG(reg_b, reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x44: /* LD B H */
|
|
LDREG(reg_b, reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x45: /* LD B L */
|
|
LDREG(reg_b, reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x46: /* LD B (HL) */
|
|
LDREG(reg_b, LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x47: /* LD B A */
|
|
LDREG(reg_b, reg_a, 0, 4, 1);
|
|
break;
|
|
case 0x48: /* LD C B */
|
|
LDREG(reg_c, reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x49: /* LD C C */
|
|
LDREG(reg_c, reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x4a: /* LD C D */
|
|
LDREG(reg_c, reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x4b: /* LD C E */
|
|
LDREG(reg_c, reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x4c: /* LD C H */
|
|
LDREG(reg_c, reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x4d: /* LD C L */
|
|
LDREG(reg_c, reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x4e: /* LD C (HL) */
|
|
LDREG(reg_c, LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x4f: /* LD C A */
|
|
LDREG(reg_c, reg_a, 0, 4, 1);
|
|
break;
|
|
case 0x50: /* LD D B */
|
|
LDREG(reg_d, reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x51: /* LD D C */
|
|
LDREG(reg_d, reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x52: /* LD D D */
|
|
LDREG(reg_d, reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x53: /* LD D E */
|
|
LDREG(reg_d, reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x54: /* LD D H */
|
|
LDREG(reg_d, reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x55: /* LD D L */
|
|
LDREG(reg_d, reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x56: /* LD D (HL) */
|
|
LDREG(reg_d, LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x57: /* LD D A */
|
|
LDREG(reg_d, reg_a, 0, 4, 1);
|
|
break;
|
|
case 0x58: /* LD E B */
|
|
LDREG(reg_e, reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x59: /* LD E C */
|
|
LDREG(reg_e, reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x5a: /* LD E D */
|
|
LDREG(reg_e, reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x5b: /* LD E E */
|
|
LDREG(reg_e, reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x5c: /* LD E H */
|
|
LDREG(reg_e, reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x5d: /* LD E L */
|
|
LDREG(reg_e, reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x5e: /* LD E (HL) */
|
|
LDREG(reg_e, LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x5f: /* LD E A */
|
|
LDREG(reg_e, reg_a, 0, 4, 1);
|
|
break;
|
|
case 0x60: /* LD H B */
|
|
LDREG(reg_h, reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x61: /* LD H C */
|
|
LDREG(reg_h, reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x62: /* LD H D */
|
|
LDREG(reg_h, reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x63: /* LD H E */
|
|
LDREG(reg_h, reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x64: /* LD H H */
|
|
LDREG(reg_h, reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x65: /* LD H L */
|
|
LDREG(reg_h, reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x66: /* LD H (HL) */
|
|
LDREG(reg_h, LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x67: /* LD H A */
|
|
LDREG(reg_h, reg_a, 0, 4, 1);
|
|
break;
|
|
case 0x68: /* LD L B */
|
|
LDREG(reg_l, reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x69: /* LD L C */
|
|
LDREG(reg_l, reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x6a: /* LD L D */
|
|
LDREG(reg_l, reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x6b: /* LD L E */
|
|
LDREG(reg_l, reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x6c: /* LD L H */
|
|
LDREG(reg_l, reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x6d: /* LD L L */
|
|
LDREG(reg_l, reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x6e: /* LD L (HL) */
|
|
LDREG(reg_l, LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x6f: /* LD L A */
|
|
LDREG(reg_l, reg_a, 0, 4, 1);
|
|
break;
|
|
case 0x70: /* LD (HL) B */
|
|
STREG(HL_WORD(), reg_b, 4, 3, 1);
|
|
break;
|
|
case 0x71: /* LD (HL) C */
|
|
STREG(HL_WORD(), reg_c, 4, 3, 1);
|
|
break;
|
|
case 0x72: /* LD (HL) D */
|
|
STREG(HL_WORD(), reg_d, 4, 3, 1);
|
|
break;
|
|
case 0x73: /* LD (HL) E */
|
|
STREG(HL_WORD(), reg_e, 4, 3, 1);
|
|
break;
|
|
case 0x74: /* LD (HL) H */
|
|
STREG(HL_WORD(), reg_h, 4, 3, 1);
|
|
break;
|
|
case 0x75: /* LD (HL) L */
|
|
STREG(HL_WORD(), reg_l, 4, 3, 1);
|
|
break;
|
|
case 0x76: /* HALT */
|
|
HALT();
|
|
break;
|
|
case 0x77: /* LD (HL) A */
|
|
STREG(HL_WORD(), reg_a, 4, 3, 1);
|
|
break;
|
|
case 0x78: /* LD A B */
|
|
LDREG(reg_a, reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x79: /* LD A C */
|
|
LDREG(reg_a, reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x7a: /* LD A D */
|
|
LDREG(reg_a, reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x7b: /* LD A E */
|
|
LDREG(reg_a, reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x7c: /* LD A H */
|
|
LDREG(reg_a, reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x7d: /* LD A L */
|
|
LDREG(reg_a, reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x7e: /* LD A (HL) */
|
|
LDREG(reg_a, LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x7f: /* LD A A */
|
|
LDREG(reg_a, reg_a, 0, 4, 1);
|
|
break;
|
|
case 0x80: /* ADD B */
|
|
ADD(reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x81: /* ADD C */
|
|
ADD(reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x82: /* ADD D */
|
|
ADD(reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x83: /* ADD E */
|
|
ADD(reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x84: /* ADD H */
|
|
ADD(reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x85: /* ADD L */
|
|
ADD(reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x86: /* ADD (HL) */
|
|
ADD(LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x87: /* ADD A */
|
|
ADD(reg_a, 0, 4, 1);
|
|
break;
|
|
case 0x88: /* ADC B */
|
|
ADC(reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x89: /* ADC C */
|
|
ADC(reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x8a: /* ADC D */
|
|
ADC(reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x8b: /* ADC E */
|
|
ADC(reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x8c: /* ADC H */
|
|
ADC(reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x8d: /* ADC L */
|
|
ADC(reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x8e: /* ADC (HL) */
|
|
ADC(LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x8f: /* ADC A */
|
|
ADC(reg_a, 0, 4, 1);
|
|
break;
|
|
case 0x90: /* SUB B */
|
|
SUB(reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x91: /* SUB C */
|
|
SUB(reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x92: /* SUB D */
|
|
SUB(reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x93: /* SUB E */
|
|
SUB(reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x94: /* SUB H */
|
|
SUB(reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x95: /* SUB L */
|
|
SUB(reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x96: /* SUB (HL) */
|
|
SUB(LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x97: /* SUB A */
|
|
SUB(reg_a, 0, 4, 1);
|
|
break;
|
|
case 0x98: /* SBC B */
|
|
SBC(reg_b, 0, 4, 1);
|
|
break;
|
|
case 0x99: /* SBC C */
|
|
SBC(reg_c, 0, 4, 1);
|
|
break;
|
|
case 0x9a: /* SBC D */
|
|
SBC(reg_d, 0, 4, 1);
|
|
break;
|
|
case 0x9b: /* SBC E */
|
|
SBC(reg_e, 0, 4, 1);
|
|
break;
|
|
case 0x9c: /* SBC H */
|
|
SBC(reg_h, 0, 4, 1);
|
|
break;
|
|
case 0x9d: /* SBC L */
|
|
SBC(reg_l, 0, 4, 1);
|
|
break;
|
|
case 0x9e: /* SBC (HL) */
|
|
SBC(LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0x9f: /* SBC A */
|
|
SBC(reg_a, 0, 4, 1);
|
|
break;
|
|
case 0xa0: /* AND B */
|
|
AND(reg_b, 0, 4, 1);
|
|
break;
|
|
case 0xa1: /* AND C */
|
|
AND(reg_c, 0, 4, 1);
|
|
break;
|
|
case 0xa2: /* AND D */
|
|
AND(reg_d, 0, 4, 1);
|
|
break;
|
|
case 0xa3: /* AND E */
|
|
AND(reg_e, 0, 4, 1);
|
|
break;
|
|
case 0xa4: /* AND H */
|
|
AND(reg_h, 0, 4, 1);
|
|
break;
|
|
case 0xa5: /* AND L */
|
|
AND(reg_l, 0, 4, 1);
|
|
break;
|
|
case 0xa6: /* AND (HL) */
|
|
AND(LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0xa7: /* AND A */
|
|
AND(reg_a, 0, 4, 1);
|
|
break;
|
|
case 0xa8: /* XOR B */
|
|
XOR(reg_b, 0, 4, 1);
|
|
break;
|
|
case 0xa9: /* XOR C */
|
|
XOR(reg_c, 0, 4, 1);
|
|
break;
|
|
case 0xaa: /* XOR D */
|
|
XOR(reg_d, 0, 4, 1);
|
|
break;
|
|
case 0xab: /* XOR E */
|
|
XOR(reg_e, 0, 4, 1);
|
|
break;
|
|
case 0xac: /* XOR H */
|
|
XOR(reg_h, 0, 4, 1);
|
|
break;
|
|
case 0xad: /* XOR L */
|
|
XOR(reg_l, 0, 4, 1);
|
|
break;
|
|
case 0xae: /* XOR (HL) */
|
|
XOR(LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0xaf: /* XOR A */
|
|
XOR(reg_a, 0, 4, 1);
|
|
break;
|
|
case 0xb0: /* OR B */
|
|
OR(reg_b, 0, 4, 1);
|
|
break;
|
|
case 0xb1: /* OR C */
|
|
OR(reg_c, 0, 4, 1);
|
|
break;
|
|
case 0xb2: /* OR D */
|
|
OR(reg_d, 0, 4, 1);
|
|
break;
|
|
case 0xb3: /* OR E */
|
|
OR(reg_e, 0, 4, 1);
|
|
break;
|
|
case 0xb4: /* OR H */
|
|
OR(reg_h, 0, 4, 1);
|
|
break;
|
|
case 0xb5: /* OR L */
|
|
OR(reg_l, 0, 4, 1);
|
|
break;
|
|
case 0xb6: /* OR (HL) */
|
|
OR(LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0xb7: /* OR A */
|
|
OR(reg_a, 0, 4, 1);
|
|
break;
|
|
case 0xb8: /* CP B */
|
|
CP(reg_b, 0, 4, 1);
|
|
break;
|
|
case 0xb9: /* CP C */
|
|
CP(reg_c, 0, 4, 1);
|
|
break;
|
|
case 0xba: /* CP D */
|
|
CP(reg_d, 0, 4, 1);
|
|
break;
|
|
case 0xbb: /* CP E */
|
|
CP(reg_e, 0, 4, 1);
|
|
break;
|
|
case 0xbc: /* CP H */
|
|
CP(reg_h, 0, 4, 1);
|
|
break;
|
|
case 0xbd: /* CP L */
|
|
CP(reg_l, 0, 4, 1);
|
|
break;
|
|
case 0xbe: /* CP (HL) */
|
|
CP(LOAD(HL_WORD()), 4, 3, 1);
|
|
break;
|
|
case 0xbf: /* CP A */
|
|
CP(reg_a, 0, 4, 1);
|
|
break;
|
|
case 0xc0: /* RET NZ */
|
|
RET_COND(!LOCAL_ZERO(), 4, 4, 2, 5, 1);
|
|
break;
|
|
case 0xc1: /* POP BC */
|
|
POP(reg_b, reg_c, 1);
|
|
break;
|
|
case 0xc2: /* JP NZ */
|
|
JMP_COND(p12, !LOCAL_ZERO(), 10, 10);
|
|
break;
|
|
case 0xc3: /* JP */
|
|
JMP(p12, 10);
|
|
break;
|
|
case 0xc4: /* CALL NZ */
|
|
CALL_COND(p12, !LOCAL_ZERO(), 3, 3, 4, 10, 3);
|
|
break;
|
|
case 0xc5: /* PUSH BC */
|
|
PUSH(reg_b, reg_c, 1);
|
|
break;
|
|
case 0xc6: /* ADD # */
|
|
ADD(p1, 4, 3, 2);
|
|
break;
|
|
case 0xc7: /* RST 00 */
|
|
CALL(0x00, 3, 3, 5, 1);
|
|
break;
|
|
case 0xc8: /* RET Z */
|
|
RET_COND(LOCAL_ZERO(), 4, 4, 2, 5, 1);
|
|
break;
|
|
case 0xc9: /* RET */
|
|
RET(4, 4, 2);
|
|
break;
|
|
case 0xca: /* JP Z */
|
|
JMP_COND(p12, LOCAL_ZERO(), 10, 10);
|
|
break;
|
|
case 0xcb: /* OPCODE CB */
|
|
opcode_cb((BYTE)p1, (BYTE)p2, (BYTE)p3, (WORD)p12, (WORD)p23);
|
|
break;
|
|
case 0xcc: /* CALL Z */
|
|
CALL_COND(p12, LOCAL_ZERO(), 3, 3, 4, 10, 3);
|
|
break;
|
|
case 0xcd: /* CALL */
|
|
CALL(p12, 3, 3, 11, 3);
|
|
break;
|
|
case 0xce: /* ADC # */
|
|
ADC(p1, 4, 3, 2);
|
|
break;
|
|
case 0xcf: /* RST 08 */
|
|
CALL(0x08, 3, 3, 5, 1);
|
|
break;
|
|
case 0xd0: /* RET NC */
|
|
RET_COND(!LOCAL_CARRY(), 4, 4, 2, 5, 1);
|
|
break;
|
|
case 0xd1: /* POP DE */
|
|
POP(reg_d, reg_e, 1);
|
|
break;
|
|
case 0xd2: /* JP NC */
|
|
JMP_COND(p12, !LOCAL_CARRY(), 10, 10);
|
|
break;
|
|
case 0xd3: /* OUT A */
|
|
OUTA(p1, 4, 7, 2);
|
|
break;
|
|
case 0xd4: /* CALL NC */
|
|
CALL_COND(p12, !LOCAL_CARRY(), 3, 3, 4, 10, 3);
|
|
break;
|
|
case 0xd5: /* PUSH DE */
|
|
PUSH(reg_d, reg_e, 1);
|
|
break;
|
|
case 0xd6: /* SUB # */
|
|
SUB(p1, 4, 3, 2);
|
|
break;
|
|
case 0xd7: /* RST 10 */
|
|
CALL(0x10, 3, 3, 5, 1);
|
|
break;
|
|
case 0xd8: /* RET C */
|
|
RET_COND(LOCAL_CARRY(), 4, 4, 2, 5, 1);
|
|
break;
|
|
case 0xd9: /* EXX */
|
|
EXX(8, 1);
|
|
break;
|
|
case 0xda: /* JP C */
|
|
JMP_COND(p12, LOCAL_CARRY(), 10, 10);
|
|
break;
|
|
case 0xdb: /* IN A */
|
|
INA(p1, 4, 7, 2);
|
|
break;
|
|
case 0xdc: /* CALL C */
|
|
CALL_COND(p12, LOCAL_CARRY(), 3, 3, 4, 10, 3);
|
|
break;
|
|
case 0xdd: /* OPCODE DD */
|
|
opcode_dd((BYTE)p1, (BYTE)p2, (BYTE)p3, (WORD)p12, (WORD)p23);
|
|
break;
|
|
case 0xde: /* SBC # */
|
|
SBC(p1, 4, 3, 2);
|
|
break;
|
|
case 0xdf: /* RST 18 */
|
|
CALL(0x18, 3, 3, 5, 1);
|
|
break;
|
|
case 0xe0: /* RET PO */
|
|
RET_COND(!LOCAL_PARITY(), 4, 4, 2, 5, 1);
|
|
break;
|
|
case 0xe1: /* POP HL */
|
|
POP(reg_h, reg_l, 1);
|
|
break;
|
|
case 0xe2: /* JP PO */
|
|
JMP_COND(p12, !LOCAL_PARITY(), 10, 10);
|
|
break;
|
|
case 0xe3: /* EX HL (SP) */
|
|
EXXXSP(reg_h, reg_l, 4, 4, 4, 4, 3, 1);
|
|
break;
|
|
case 0xe4: /* CALL PO */
|
|
CALL_COND(p12, !LOCAL_PARITY(), 3, 3, 4, 10, 3);
|
|
break;
|
|
case 0xe5: /* PUSH HL */
|
|
PUSH(reg_h, reg_l, 1);
|
|
break;
|
|
case 0xe6: /* AND # */
|
|
AND(p1, 4, 3, 2);
|
|
break;
|
|
case 0xe7: /* RST 20 */
|
|
CALL(0x20, 3, 3, 5, 1);
|
|
break;
|
|
case 0xe8: /* RET PE */
|
|
RET_COND(LOCAL_PARITY(), 4, 4, 2, 5, 1);
|
|
break;
|
|
case 0xe9: /* LD PC HL */
|
|
JMP((HL_WORD()), 4);
|
|
break;
|
|
case 0xea: /* JP PE */
|
|
JMP_COND(p12, LOCAL_PARITY(), 10, 10);
|
|
break;
|
|
case 0xeb: /* EX DE HL */
|
|
EXDEHL(4, 1);
|
|
break;
|
|
case 0xec: /* CALL PE */
|
|
CALL_COND(p12, LOCAL_PARITY(), 3, 3, 4, 10, 3);
|
|
break;
|
|
case 0xed: /* OPCODE ED */
|
|
opcode_ed((BYTE)p1, (BYTE)p2, (BYTE)p3, (WORD)p12, (WORD)p23);
|
|
break;
|
|
case 0xee: /* XOR # */
|
|
XOR(p1, 4, 3, 2);
|
|
break;
|
|
case 0xef: /* RST 28 */
|
|
CALL(0x28, 3, 3, 5, 1);
|
|
break;
|
|
case 0xf0: /* RET P */
|
|
RET_COND(!LOCAL_SIGN(), 4, 4, 2, 5, 1);
|
|
break;
|
|
case 0xf1: /* POP AF */
|
|
POP(reg_a, reg_f, 1);
|
|
break;
|
|
case 0xf2: /* JP P */
|
|
JMP_COND(p12, !LOCAL_SIGN(), 10, 10);
|
|
break;
|
|
case 0xf3: /* DI */
|
|
DI(4, 1);
|
|
break;
|
|
case 0xf4: /* CALL P */
|
|
CALL_COND(p12, !LOCAL_SIGN(), 3, 3, 4, 10, 3);
|
|
break;
|
|
case 0xf5: /* PUSH AF */
|
|
PUSH(reg_a, reg_f, 1);
|
|
break;
|
|
case 0xf6: /* OR # */
|
|
OR(p1, 4, 3, 2);
|
|
break;
|
|
case 0xf7: /* RST 30 */
|
|
CALL(0x30, 3, 3, 5, 1);
|
|
break;
|
|
case 0xf8: /* RET M */
|
|
RET_COND(LOCAL_SIGN(), 4, 4, 2, 5, 1);
|
|
break;
|
|
case 0xf9: /* LD SP HL */
|
|
LDSP(HL_WORD(), 4, 2, 1);
|
|
break;
|
|
case 0xfa: /* JP M */
|
|
JMP_COND(p12, LOCAL_SIGN(), 10, 10);
|
|
break;
|
|
case 0xfb: /* EI */
|
|
EI(4, 1);
|
|
break;
|
|
case 0xfc: /* CALL M */
|
|
CALL_COND(p12, LOCAL_SIGN(), 3, 3, 4, 10, 3);
|
|
break;
|
|
case 0xfd: /* OPCODE FD */
|
|
opcode_fd((BYTE)p1, (BYTE)p2, (BYTE)p3, (WORD)p12, (WORD)p23);
|
|
break;
|
|
case 0xfe: /* CP # */
|
|
CP(p1, 4, 3, 2);
|
|
break;
|
|
case 0xff: /* RST 38 */
|
|
CALL(0x38, 3, 3, 5, 1);
|
|
break;
|
|
}
|
|
|
|
//cpu_int_status->num_dma_per_opcode = 0; // [AppleWin-TC] Not used
|
|
|
|
if (GetActiveCpu() != CPU_Z80) // [AppleWin-TC]
|
|
break;
|
|
|
|
//} while (!dma_request);
|
|
} while (maincpu_clk < uTotalCycles); // [AppleWin-TC]
|
|
|
|
export_registers();
|
|
|
|
return ConvertZ80TStatesTo6502Cycles(maincpu_clk - uExecutedCycles);
|
|
}
|
|
|
|
/****************************************************************************/
|
|
/* Read a byte from given memory location */
|
|
/****************************************************************************/
|
|
BYTE z80_RDMEM(WORD Addr)
|
|
{
|
|
WORD addr;
|
|
|
|
switch (Addr / 0x1000)
|
|
{
|
|
case 0x0:
|
|
case 0x1:
|
|
case 0x2:
|
|
case 0x3:
|
|
case 0x4:
|
|
case 0x5:
|
|
case 0x6:
|
|
case 0x7:
|
|
case 0x8:
|
|
case 0x9:
|
|
case 0xA:
|
|
addr = (WORD)Addr + 0x1000;
|
|
return CpuRead( addr, ConvertZ80TStatesTo6502Cycles(maincpu_clk) );
|
|
break;
|
|
|
|
case 0xB:
|
|
case 0xC:
|
|
case 0xD:
|
|
addr = (WORD)Addr + 0x2000;
|
|
return CpuRead( addr, ConvertZ80TStatesTo6502Cycles(maincpu_clk) );
|
|
break;
|
|
|
|
case 0xE:
|
|
addr = (WORD)Addr - 0x2000;
|
|
if ((addr & 0xF000) == 0xC000)
|
|
{
|
|
return IORead[(addr>>4) & 0xFF]( regs.pc, addr, 0, 0, ConvertZ80TStatesTo6502Cycles(maincpu_clk) );
|
|
}
|
|
else
|
|
{
|
|
return *(mem+addr);
|
|
}
|
|
break;
|
|
|
|
case 0xF:
|
|
addr = (WORD)Addr - 0xF000;
|
|
return CpuRead( addr, ConvertZ80TStatesTo6502Cycles(maincpu_clk) );
|
|
break;
|
|
}
|
|
return 255;
|
|
}
|
|
|
|
/****************************************************************************/
|
|
/* Write a byte to given memory location */
|
|
/****************************************************************************/
|
|
void z80_WRMEM(WORD Addr, BYTE Value)
|
|
{
|
|
unsigned int laddr;
|
|
WORD addr;
|
|
|
|
laddr = Addr & 0x0FFF;
|
|
switch (Addr & 0xF000)
|
|
{
|
|
case 0x0000: addr = laddr+0x1000; break;
|
|
case 0x1000: addr = laddr+0x2000; break;
|
|
case 0x2000: addr = laddr+0x3000; break;
|
|
case 0x3000: addr = laddr+0x4000; break;
|
|
case 0x4000: addr = laddr+0x5000; break;
|
|
case 0x5000: addr = laddr+0x6000; break;
|
|
case 0x6000: addr = laddr+0x7000; break;
|
|
case 0x7000: addr = laddr+0x8000; break;
|
|
case 0x8000: addr = laddr+0x9000; break;
|
|
case 0x9000: addr = laddr+0xA000; break;
|
|
case 0xA000: addr = laddr+0xB000; break;
|
|
case 0xB000: addr = laddr+0xD000; break;
|
|
case 0xC000: addr = laddr+0xE000; break;
|
|
case 0xD000: addr = laddr+0xF000; break;
|
|
case 0xE000: addr = laddr+0xC000; break;
|
|
case 0xF000: addr = laddr+0x0000; break;
|
|
}
|
|
CpuWrite( addr, Value, ConvertZ80TStatesTo6502Cycles(maincpu_clk) );
|
|
}
|
|
|
|
//===========================================================================
|
|
|
|
#define SS_YAML_VALUE_CARD_Z80 "Z80"
|
|
|
|
#define SS_YAML_KEY_REGA "A"
|
|
#define SS_YAML_KEY_REGB "B"
|
|
#define SS_YAML_KEY_REGC "C"
|
|
#define SS_YAML_KEY_REGD "D"
|
|
#define SS_YAML_KEY_REGE "E"
|
|
#define SS_YAML_KEY_REGF "F"
|
|
#define SS_YAML_KEY_REGH "H"
|
|
#define SS_YAML_KEY_REGL "L"
|
|
#define SS_YAML_KEY_REGIX "IX"
|
|
#define SS_YAML_KEY_REGIY "IY"
|
|
#define SS_YAML_KEY_REGSP "SP"
|
|
#define SS_YAML_KEY_REGPC "PC"
|
|
#define SS_YAML_KEY_REGI "I"
|
|
#define SS_YAML_KEY_REGR "R"
|
|
#define SS_YAML_KEY_IFF1 "IFF1"
|
|
#define SS_YAML_KEY_IFF2 "IFF2"
|
|
#define SS_YAML_KEY_IM_MODE "IM Mode"
|
|
#define SS_YAML_KEY_REGA2 "A'"
|
|
#define SS_YAML_KEY_REGB2 "B'"
|
|
#define SS_YAML_KEY_REGC2 "C'"
|
|
#define SS_YAML_KEY_REGD2 "D'"
|
|
#define SS_YAML_KEY_REGE2 "E'"
|
|
#define SS_YAML_KEY_REGF2 "F'"
|
|
#define SS_YAML_KEY_REGH2 "H'"
|
|
#define SS_YAML_KEY_REGL2 "L'"
|
|
#define SS_YAML_KEY_ACTIVE "Active"
|
|
|
|
std::string Z80_GetSnapshotCardName(void)
|
|
{
|
|
static const std::string name(SS_YAML_VALUE_CARD_Z80);
|
|
return name;
|
|
}
|
|
|
|
void Z80_SaveSnapshot(class YamlSaveHelper& yamlSaveHelper, const UINT uSlot)
|
|
{
|
|
YamlSaveHelper::Slot slot(yamlSaveHelper, Z80_GetSnapshotCardName(), uSlot, 1); // fixme: object should know its slot
|
|
|
|
YamlSaveHelper::Label state(yamlSaveHelper, "%s:\n", SS_YAML_KEY_STATE);
|
|
|
|
// SoftCard SW & HW details: http://apple2info.net/images/f/f0/SC-SWHW.pdf
|
|
// . SoftCard uses the Apple II's DMA circuit to pause the 6502 (no CLK to 6502)
|
|
// . But: "In Apple II DMA, the 6502 CPU will die after approximately 15 clocks because it depends on the clock to refresh its internal registers."
|
|
// ref: Apple Tech Note: https://archive.org/stream/IIe_2523004_RDY_Line/IIe_2523004_RDY_Line_djvu.txt
|
|
// NB. Not for 65C02 which is a static processor.
|
|
// . SoftCard controls the 6502's RDY line to periodically allow only 1 memory fetch by 6502 (ie. the opcode fetch)
|
|
//
|
|
// So save ActiveCPU to SS_CARD_Z80 (so RDY is like IRQ & NMI signals, ie. saved struct of the producer's card)
|
|
//
|
|
// NB. Save-state only occurs when message pump runs:
|
|
// . ie. at end of 1ms emulation burst
|
|
// Either 6502 or Z80 could be active.
|
|
//
|
|
|
|
yamlSaveHelper.Save("%s: %d\n", SS_YAML_KEY_ACTIVE, GetActiveCpu() == CPU_Z80 ? 1 : 0);
|
|
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGA, reg_a);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGB, reg_b);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGC, reg_c);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGD, reg_d);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGE, reg_e);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGF, reg_f);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGH, reg_h);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGL, reg_l);
|
|
yamlSaveHelper.Save("%s: 0x%04X\n", SS_YAML_KEY_REGIX, ((USHORT)reg_ixh<<8)|(USHORT)reg_ixl);
|
|
yamlSaveHelper.Save("%s: 0x%04X\n", SS_YAML_KEY_REGIY, ((USHORT)reg_iyh<<8)|(USHORT)reg_iyl);
|
|
yamlSaveHelper.Save("%s: 0x%04X\n", SS_YAML_KEY_REGSP, reg_sp);
|
|
yamlSaveHelper.Save("%s: 0x%04X\n", SS_YAML_KEY_REGPC, (USHORT)z80_reg_pc);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGI, reg_i);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGR, reg_r);
|
|
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_IFF1, iff1);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_IFF2, iff2);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_IM_MODE, im_mode);
|
|
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGA2, reg_a2);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGB2, reg_b2);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGC2, reg_c2);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGD2, reg_d2);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGE2, reg_e2);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGF2, reg_f2);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGH2, reg_h2);
|
|
yamlSaveHelper.Save("%s: 0x%02X\n", SS_YAML_KEY_REGL2, reg_l2);
|
|
}
|
|
|
|
bool Z80_LoadSnapshot(class YamlLoadHelper& yamlLoadHelper, UINT uSlot, UINT version)
|
|
{
|
|
if (uSlot != 4 && uSlot != 5) // fixme
|
|
throw std::string("Card: wrong slot");
|
|
|
|
if (version != 1)
|
|
throw std::string("Card: wrong version");
|
|
|
|
reg_a = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGA);
|
|
reg_b = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGB);
|
|
reg_c = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGC);
|
|
reg_d = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGD);
|
|
reg_e = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGE);
|
|
reg_f = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGF);
|
|
reg_h = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGH);
|
|
reg_l = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGL);
|
|
USHORT IX = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGIX);
|
|
reg_ixh = IX >> 8;
|
|
reg_ixl = IX & 0xFF;
|
|
USHORT IY = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGIY);
|
|
reg_iyh = IY >> 8;
|
|
reg_iyl = IY & 0xFF;
|
|
reg_sp = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGSP);
|
|
z80_reg_pc = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGPC);
|
|
reg_i = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGI);
|
|
reg_r = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGR);
|
|
|
|
iff1 = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_IFF1);
|
|
iff2 = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_IFF2);
|
|
im_mode = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_IM_MODE);
|
|
|
|
reg_a2 = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGA2);
|
|
reg_b2 = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGB2);
|
|
reg_c2 = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGC2);
|
|
reg_d2 = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGD2);
|
|
reg_e2 = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGE2);
|
|
reg_f2 = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGF2);
|
|
reg_h2 = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGH2);
|
|
reg_l2 = yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_REGL2);
|
|
|
|
export_registers();
|
|
|
|
if ( yamlLoadHelper.GetMapValueUINT(SS_YAML_KEY_ACTIVE) )
|
|
SetActiveCpu(CPU_Z80); // Support MS SoftCard in multiple slots (only one Z80 can be active at any one time)
|
|
|
|
return true;
|
|
}
|