snes9x/fxinst.cpp

4438 lines
71 KiB
C++

/***********************************************************************************
Snes9x - Portable Super Nintendo Entertainment System (TM) emulator.
(c) Copyright 1996 - 2002 Gary Henderson (gary.henderson@ntlworld.com),
Jerremy Koot (jkoot@snes9x.com)
(c) Copyright 2002 - 2004 Matthew Kendora
(c) Copyright 2002 - 2005 Peter Bortas (peter@bortas.org)
(c) Copyright 2004 - 2005 Joel Yliluoma (http://iki.fi/bisqwit/)
(c) Copyright 2001 - 2006 John Weidman (jweidman@slip.net)
(c) Copyright 2002 - 2006 funkyass (funkyass@spam.shaw.ca),
Kris Bleakley (codeviolation@hotmail.com)
(c) Copyright 2002 - 2010 Brad Jorsch (anomie@users.sourceforge.net),
Nach (n-a-c-h@users.sourceforge.net),
(c) Copyright 2002 - 2011 zones (kasumitokoduck@yahoo.com)
(c) Copyright 2006 - 2007 nitsuja
(c) Copyright 2009 - 2016 BearOso,
OV2
(c) Copyright 2011 - 2016 Hans-Kristian Arntzen,
Daniel De Matteis
(Under no circumstances will commercial rights be given)
BS-X C emulator code
(c) Copyright 2005 - 2006 Dreamer Nom,
zones
C4 x86 assembler and some C emulation code
(c) Copyright 2000 - 2003 _Demo_ (_demo_@zsnes.com),
Nach,
zsKnight (zsknight@zsnes.com)
C4 C++ code
(c) Copyright 2003 - 2006 Brad Jorsch,
Nach
DSP-1 emulator code
(c) Copyright 1998 - 2006 _Demo_,
Andreas Naive (andreasnaive@gmail.com),
Gary Henderson,
Ivar (ivar@snes9x.com),
John Weidman,
Kris Bleakley,
Matthew Kendora,
Nach,
neviksti (neviksti@hotmail.com)
DSP-2 emulator code
(c) Copyright 2003 John Weidman,
Kris Bleakley,
Lord Nightmare (lord_nightmare@users.sourceforge.net),
Matthew Kendora,
neviksti
DSP-3 emulator code
(c) Copyright 2003 - 2006 John Weidman,
Kris Bleakley,
Lancer,
z80 gaiden
DSP-4 emulator code
(c) Copyright 2004 - 2006 Dreamer Nom,
John Weidman,
Kris Bleakley,
Nach,
z80 gaiden
OBC1 emulator code
(c) Copyright 2001 - 2004 zsKnight,
pagefault (pagefault@zsnes.com),
Kris Bleakley
Ported from x86 assembler to C by sanmaiwashi
SPC7110 and RTC C++ emulator code used in 1.39-1.51
(c) Copyright 2002 Matthew Kendora with research by
zsKnight,
John Weidman,
Dark Force
SPC7110 and RTC C++ emulator code used in 1.52+
(c) Copyright 2009 byuu,
neviksti
S-DD1 C emulator code
(c) Copyright 2003 Brad Jorsch with research by
Andreas Naive,
John Weidman
S-RTC C emulator code
(c) Copyright 2001 - 2006 byuu,
John Weidman
ST010 C++ emulator code
(c) Copyright 2003 Feather,
John Weidman,
Kris Bleakley,
Matthew Kendora
Super FX x86 assembler emulator code
(c) Copyright 1998 - 2003 _Demo_,
pagefault,
zsKnight
Super FX C emulator code
(c) Copyright 1997 - 1999 Ivar,
Gary Henderson,
John Weidman
Sound emulator code used in 1.5-1.51
(c) Copyright 1998 - 2003 Brad Martin
(c) Copyright 1998 - 2006 Charles Bilyue'
Sound emulator code used in 1.52+
(c) Copyright 2004 - 2007 Shay Green (gblargg@gmail.com)
S-SMP emulator code used in 1.54+
(c) Copyright 2016 byuu
SH assembler code partly based on x86 assembler code
(c) Copyright 2002 - 2004 Marcus Comstedt (marcus@mc.pp.se)
2xSaI filter
(c) Copyright 1999 - 2001 Derek Liauw Kie Fa
HQ2x, HQ3x, HQ4x filters
(c) Copyright 2003 Maxim Stepin (maxim@hiend3d.com)
NTSC filter
(c) Copyright 2006 - 2007 Shay Green
GTK+ GUI code
(c) Copyright 2004 - 2016 BearOso
Win32 GUI code
(c) Copyright 2003 - 2006 blip,
funkyass,
Matthew Kendora,
Nach,
nitsuja
(c) Copyright 2009 - 2016 OV2
Mac OS GUI code
(c) Copyright 1998 - 2001 John Stiles
(c) Copyright 2001 - 2011 zones
Libretro port
(c) Copyright 2011 - 2016 Hans-Kristian Arntzen,
Daniel De Matteis
(Under no circumstances will commercial rights be given)
Specific ports contains the works of other authors. See headers in
individual files.
Snes9x homepage: http://www.snes9x.com/
Permission to use, copy, modify and/or distribute Snes9x in both binary
and source form, for non-commercial purposes, is hereby granted without
fee, providing that this license information and copyright notice appear
with all copies and any derived work.
This software is provided 'as-is', without any express or implied
warranty. In no event shall the authors be held liable for any damages
arising from the use of this software or it's derivatives.
Snes9x is freeware for PERSONAL USE only. Commercial users should
seek permission of the copyright holders first. Commercial use includes,
but is not limited to, charging money for Snes9x or software derived from
Snes9x, including Snes9x or derivatives in commercial game bundles, and/or
using Snes9x as a promotion for your commercial product.
The copyright holders request that bug fixes and improvements to the code
should be forwarded to them so everyone can benefit from the modifications
in future versions.
Super NES and Super Nintendo Entertainment System are trademarks of
Nintendo Co., Limited and its subsidiary companies.
***********************************************************************************/
#include "snes9x.h"
#include "fxinst.h"
#include "fxemu.h"
// Set this define if you wish the plot instruction to check for y-pos limits (I don't think it's nessecary)
#define CHECK_LIMITS
/*
Codes used:
rn = a GSU register (r0 - r15)
#n = 4 bit immediate value
#pp = 8 bit immediate value
(yy) = 8 bit word address (0x0000 - 0x01fe)
#xx = 16 bit immediate value
(xx) = 16 bit address (0x0000 - 0xffff)
*/
// 00 - stop - stop GSU execution (and maybe generate an IRQ)
static void fx_stop (void)
{
CF(G);
GSU.vCounter = 0;
GSU.vInstCount = GSU.vCounter;
// Check if we need to generate an IRQ
if (!(GSU.pvRegisters[GSU_CFGR] & 0x80))
SF(IRQ);
GSU.vPlotOptionReg = 0;
GSU.vPipe = 1;
CLRFLAGS;
R15++;
}
// 01 - nop - no operation
static void fx_nop (void)
{
CLRFLAGS;
R15++;
}
// 02 - cache - reintialize GSU cache
static void fx_cache (void)
{
uint32 c = R15 & 0xfff0;
if (GSU.vCacheBaseReg != c || !GSU.bCacheActive)
{
fx_flushCache();
GSU.vCacheBaseReg = c;
GSU.bCacheActive = TRUE;
#if 0
if (c < (0x10000 - 512))
{
const uint8 *t = &ROM(c);
memcpy(GSU.pvCache, t, 512);
}
else
{
const uint8 *t1, t2;
uint32 i = 0x10000 - c;
t1 = &ROM(c);
t2 = &ROM(0);
memcpy(GSU.pvCache, t1, i);
memcpy(&GSU.pvCache[i], t2, 512 - i);
}
#endif
}
CLRFLAGS;
R15++;
}
// 03 - lsr - logic shift right
static void fx_lsr (void)
{
uint32 v;
GSU.vCarry = SREG & 1;
v = USEX16(SREG) >> 1;
R15++;
DREG = v;
GSU.vSign = v;
GSU.vZero = v;
TESTR14;
CLRFLAGS;
}
// 04 - rol - rotate left
static void fx_rol (void)
{
uint32 v = USEX16((SREG << 1) + GSU.vCarry);
GSU.vCarry = (SREG >> 15) & 1;
R15++;
DREG = v;
GSU.vSign = v;
GSU.vZero = v;
TESTR14;
CLRFLAGS;
}
// 05 - bra - branch always
static void fx_bra (void)
{
uint8 v = PIPE;
R15++;
FETCHPIPE;
R15 += SEX8(v);
}
// Branch on condition
#define BRA_COND(cond) \
uint8 v = PIPE; \
R15++; \
FETCHPIPE; \
if (cond) \
R15 += SEX8(v); \
else \
R15++
#define TEST_S (GSU.vSign & 0x8000)
#define TEST_Z (USEX16(GSU.vZero) == 0)
#define TEST_OV (GSU.vOverflow >= 0x8000 || GSU.vOverflow < -0x8000)
#define TEST_CY (GSU.vCarry & 1)
// 06 - blt - branch on less than
static void fx_blt (void)
{
BRA_COND((TEST_S != 0) != (TEST_OV != 0));
}
// 07 - bge - branch on greater or equals
static void fx_bge (void)
{
BRA_COND((TEST_S != 0) == (TEST_OV != 0));
}
// 08 - bne - branch on not equal
static void fx_bne (void)
{
BRA_COND(!TEST_Z);
}
// 09 - beq - branch on equal
static void fx_beq (void)
{
BRA_COND(TEST_Z);
}
// 0a - bpl - branch on plus
static void fx_bpl (void)
{
BRA_COND(!TEST_S);
}
// 0b - bmi - branch on minus
static void fx_bmi (void)
{
BRA_COND(TEST_S);
}
// 0c - bcc - branch on carry clear
static void fx_bcc (void)
{
BRA_COND(!TEST_CY);
}
// 0d - bcs - branch on carry set
static void fx_bcs (void)
{
BRA_COND(TEST_CY);
}
// 0e - bvc - branch on overflow clear
static void fx_bvc (void)
{
BRA_COND(!TEST_OV);
}
// 0f - bvs - branch on overflow set
static void fx_bvs (void)
{
BRA_COND(TEST_OV);
}
// 10-1f - to rn - set register n as destination register
// 10-1f (B) - move rn - move one register to another (if B flag is set)
#define FX_TO(reg) \
if (TF(B)) \
{ \
GSU.avReg[(reg)] = SREG; \
CLRFLAGS; \
} \
else \
GSU.pvDreg = &GSU.avReg[reg]; \
R15++
#define FX_TO_R14(reg) \
if (TF(B)) \
{ \
GSU.avReg[(reg)] = SREG; \
CLRFLAGS; \
READR14; \
} \
else \
GSU.pvDreg = &GSU.avReg[reg]; \
R15++
#define FX_TO_R15(reg) \
if (TF(B)) \
{ \
GSU.avReg[(reg)] = SREG; \
CLRFLAGS; \
} \
else \
{ \
GSU.pvDreg = &GSU.avReg[reg]; \
R15++; \
}
static void fx_to_r0 (void)
{
FX_TO(0);
}
static void fx_to_r1 (void)
{
FX_TO(1);
}
static void fx_to_r2 (void)
{
FX_TO(2);
}
static void fx_to_r3 (void)
{
FX_TO(3);
}
static void fx_to_r4 (void)
{
FX_TO(4);
}
static void fx_to_r5 (void)
{
FX_TO(5);
}
static void fx_to_r6 (void)
{
FX_TO(6);
}
static void fx_to_r7 (void)
{
FX_TO(7);
}
static void fx_to_r8 (void)
{
FX_TO(8);
}
static void fx_to_r9 (void)
{
FX_TO(9);
}
static void fx_to_r10 (void)
{
FX_TO(10);
}
static void fx_to_r11 (void)
{
FX_TO(11);
}
static void fx_to_r12 (void)
{
FX_TO(12);
}
static void fx_to_r13 (void)
{
FX_TO(13);
}
static void fx_to_r14 (void)
{
FX_TO_R14(14);
}
static void fx_to_r15 (void)
{
FX_TO_R15(15);
}
// 20-2f - to rn - set register n as source and destination register
#define FX_WITH(reg) \
SF(B); \
GSU.pvSreg = GSU.pvDreg = &GSU.avReg[reg]; \
R15++
static void fx_with_r0 (void)
{
FX_WITH(0);
}
static void fx_with_r1 (void)
{
FX_WITH(1);
}
static void fx_with_r2 (void)
{
FX_WITH(2);
}
static void fx_with_r3 (void)
{
FX_WITH(3);
}
static void fx_with_r4 (void)
{
FX_WITH(4);
}
static void fx_with_r5 (void)
{
FX_WITH(5);
}
static void fx_with_r6 (void)
{
FX_WITH(6);
}
static void fx_with_r7 (void)
{
FX_WITH(7);
}
static void fx_with_r8 (void)
{
FX_WITH(8);
}
static void fx_with_r9 (void)
{
FX_WITH(9);
}
static void fx_with_r10 (void)
{
FX_WITH(10);
}
static void fx_with_r11 (void)
{
FX_WITH(11);
}
static void fx_with_r12 (void)
{
FX_WITH(12);
}
static void fx_with_r13 (void)
{
FX_WITH(13);
}
static void fx_with_r14 (void)
{
FX_WITH(14);
}
static void fx_with_r15 (void)
{
FX_WITH(15);
}
// 30-3b - stw (rn) - store word
#define FX_STW(reg) \
GSU.vLastRamAdr = GSU.avReg[reg]; \
RAM(GSU.avReg[reg]) = (uint8) SREG; \
RAM(GSU.avReg[reg] ^ 1) = (uint8) (SREG >> 8); \
CLRFLAGS; \
R15++
static void fx_stw_r0 (void)
{
FX_STW(0);
}
static void fx_stw_r1 (void)
{
FX_STW(1);
}
static void fx_stw_r2 (void)
{
FX_STW(2);
}
static void fx_stw_r3 (void)
{
FX_STW(3);
}
static void fx_stw_r4 (void)
{
FX_STW(4);
}
static void fx_stw_r5 (void)
{
FX_STW(5);
}
static void fx_stw_r6 (void)
{
FX_STW(6);
}
static void fx_stw_r7 (void)
{
FX_STW(7);
}
static void fx_stw_r8 (void)
{
FX_STW(8);
}
static void fx_stw_r9 (void)
{
FX_STW(9);
}
static void fx_stw_r10 (void)
{
FX_STW(10);
}
static void fx_stw_r11 (void)
{
FX_STW(11);
}
// 30-3b (ALT1) - stb (rn) - store byte
#define FX_STB(reg) \
GSU.vLastRamAdr = GSU.avReg[reg]; \
RAM(GSU.avReg[reg]) = (uint8) SREG; \
CLRFLAGS; \
R15++
static void fx_stb_r0 (void)
{
FX_STB(0);
}
static void fx_stb_r1 (void)
{
FX_STB(1);
}
static void fx_stb_r2 (void)
{
FX_STB(2);
}
static void fx_stb_r3 (void)
{
FX_STB(3);
}
static void fx_stb_r4 (void)
{
FX_STB(4);
}
static void fx_stb_r5 (void)
{
FX_STB(5);
}
static void fx_stb_r6 (void)
{
FX_STB(6);
}
static void fx_stb_r7 (void)
{
FX_STB(7);
}
static void fx_stb_r8 (void)
{
FX_STB(8);
}
static void fx_stb_r9 (void)
{
FX_STB(9);
}
static void fx_stb_r10 (void)
{
FX_STB(10);
}
static void fx_stb_r11 (void)
{
FX_STB(11);
}
// 3c - loop - decrement loop counter, and branch on not zero
static void fx_loop (void)
{
GSU.vSign = GSU.vZero = --R12;
if ((uint16) R12 != 0)
R15 = R13;
else
R15++;
CLRFLAGS;
}
// 3d - alt1 - set alt1 mode
static void fx_alt1 (void)
{
SF(ALT1);
CF(B);
R15++;
}
// 3e - alt2 - set alt2 mode
static void fx_alt2 (void)
{
SF(ALT2);
CF(B);
R15++;
}
// 3f - alt3 - set alt3 mode
static void fx_alt3 (void)
{
SF(ALT1);
SF(ALT2);
CF(B);
R15++;
}
// 40-4b - ldw (rn) - load word from RAM
#define FX_LDW(reg) \
uint32 v; \
GSU.vLastRamAdr = GSU.avReg[reg]; \
v = (uint32) RAM(GSU.avReg[reg]); \
v |= ((uint32) RAM(GSU.avReg[reg] ^ 1)) << 8; \
R15++; \
DREG = v; \
TESTR14; \
CLRFLAGS
static void fx_ldw_r0 (void)
{
FX_LDW(0);
}
static void fx_ldw_r1 (void)
{
FX_LDW(1);
}
static void fx_ldw_r2 (void)
{
FX_LDW(2);
}
static void fx_ldw_r3 (void)
{
FX_LDW(3);
}
static void fx_ldw_r4 (void)
{
FX_LDW(4);
}
static void fx_ldw_r5 (void)
{
FX_LDW(5);
}
static void fx_ldw_r6 (void)
{
FX_LDW(6);
}
static void fx_ldw_r7 (void)
{
FX_LDW(7);
}
static void fx_ldw_r8 (void)
{
FX_LDW(8);
}
static void fx_ldw_r9 (void)
{
FX_LDW(9);
}
static void fx_ldw_r10 (void)
{
FX_LDW(10);
}
static void fx_ldw_r11 (void)
{
FX_LDW(11);
}
// 40-4b (ALT1) - ldb (rn) - load byte
#define FX_LDB(reg) \
uint32 v; \
GSU.vLastRamAdr = GSU.avReg[reg]; \
v = (uint32) RAM(GSU.avReg[reg]); \
R15++; \
DREG = v; \
TESTR14; \
CLRFLAGS
static void fx_ldb_r0 (void)
{
FX_LDB(0);
}
static void fx_ldb_r1 (void)
{
FX_LDB(1);
}
static void fx_ldb_r2 (void)
{
FX_LDB(2);
}
static void fx_ldb_r3 (void)
{
FX_LDB(3);
}
static void fx_ldb_r4 (void)
{
FX_LDB(4);
}
static void fx_ldb_r5 (void)
{
FX_LDB(5);
}
static void fx_ldb_r6 (void)
{
FX_LDB(6);
}
static void fx_ldb_r7 (void)
{
FX_LDB(7);
}
static void fx_ldb_r8 (void)
{
FX_LDB(8);
}
static void fx_ldb_r9 (void)
{
FX_LDB(9);
}
static void fx_ldb_r10 (void)
{
FX_LDB(10);
}
static void fx_ldb_r11 (void)
{
FX_LDB(11);
}
// 4c - plot - plot pixel with R1, R2 as x, y and the color register as the color
static void fx_plot_2bit (void)
{
uint32 x = USEX8(R1);
uint32 y = USEX8(R2);
uint8 *a;
uint8 v, c;
R15++;
CLRFLAGS;
R1++;
#ifdef CHECK_LIMITS
if (y >= GSU.vScreenHeight)
return;
#endif
if (GSU.vPlotOptionReg & 0x02)
c = ((x ^ y) & 1) ? (uint8) (GSU.vColorReg >> 4) : (uint8) GSU.vColorReg;
else
c = (uint8) GSU.vColorReg;
if (!(GSU.vPlotOptionReg & 0x01) && !(c & 0xf))
return;
a = GSU.apvScreen[y >> 3] + GSU.x[x >> 3] + ((y & 7) << 1);
v = 128 >> (x & 7);
if (c & 0x01)
a[0] |= v;
else
a[0] &= ~v;
if (c & 0x02)
a[1] |= v;
else
a[1] &= ~v;
}
// 4c (ALT1) - rpix - read color of the pixel with R1, R2 as x, y
static void fx_rpix_2bit (void)
{
uint32 x = USEX8(R1);
uint32 y = USEX8(R2);
uint8 *a;
uint8 v;
R15++;
CLRFLAGS;
#ifdef CHECK_LIMITS
if (y >= GSU.vScreenHeight)
return;
#endif
a = GSU.apvScreen[y >> 3] + GSU.x[x >> 3] + ((y & 7) << 1);
v = 128 >> (x & 7);
DREG = 0;
DREG |= ((uint32) ((a[0] & v) != 0)) << 0;
DREG |= ((uint32) ((a[1] & v) != 0)) << 1;
TESTR14;
}
// 4c - plot - plot pixel with R1, R2 as x, y and the color register as the color
static void fx_plot_4bit (void)
{
uint32 x = USEX8(R1);
uint32 y = USEX8(R2);
uint8 *a;
uint8 v, c;
R15++;
CLRFLAGS;
R1++;
#ifdef CHECK_LIMITS
if (y >= GSU.vScreenHeight)
return;
#endif
if (GSU.vPlotOptionReg & 0x02)
c = ((x ^ y) & 1) ? (uint8) (GSU.vColorReg >> 4) : (uint8) GSU.vColorReg;
else
c = (uint8) GSU.vColorReg;
if (!(GSU.vPlotOptionReg & 0x01) && !(c & 0xf))
return;
a = GSU.apvScreen[y >> 3] + GSU.x[x >> 3] + ((y & 7) << 1);
v = 128 >> (x & 7);
if (c & 0x01)
a[0x00] |= v;
else
a[0x00] &= ~v;
if (c & 0x02)
a[0x01] |= v;
else
a[0x01] &= ~v;
if (c & 0x04)
a[0x10] |= v;
else
a[0x10] &= ~v;
if (c & 0x08)
a[0x11] |= v;
else
a[0x11] &= ~v;
}
// 4c (ALT1) - rpix - read color of the pixel with R1, R2 as x, y
static void fx_rpix_4bit (void)
{
uint32 x = USEX8(R1);
uint32 y = USEX8(R2);
uint8 *a;
uint8 v;
R15++;
CLRFLAGS;
#ifdef CHECK_LIMITS
if (y >= GSU.vScreenHeight)
return;
#endif
a = GSU.apvScreen[y >> 3] + GSU.x[x >> 3] + ((y & 7) << 1);
v = 128 >> (x & 7);
DREG = 0;
DREG |= ((uint32) ((a[0x00] & v) != 0)) << 0;
DREG |= ((uint32) ((a[0x01] & v) != 0)) << 1;
DREG |= ((uint32) ((a[0x10] & v) != 0)) << 2;
DREG |= ((uint32) ((a[0x11] & v) != 0)) << 3;
TESTR14;
}
// 4c - plot - plot pixel with R1, R2 as x, y and the color register as the color
static void fx_plot_8bit (void)
{
uint32 x = USEX8(R1);
uint32 y = USEX8(R2);
uint8 *a;
uint8 v, c;
R15++;
CLRFLAGS;
R1++;
#ifdef CHECK_LIMITS
if (y >= GSU.vScreenHeight)
return;
#endif
c = (uint8) GSU.vColorReg;
if (!(GSU.vPlotOptionReg & 0x10))
{
if (!(GSU.vPlotOptionReg & 0x01) && !(c & 0xf))
return;
}
else
if (!(GSU.vPlotOptionReg & 0x01) && !c)
return;
a = GSU.apvScreen[y >> 3] + GSU.x[x >> 3] + ((y & 7) << 1);
v = 128 >> (x & 7);
if (c & 0x01)
a[0x00] |= v;
else
a[0x00] &= ~v;
if (c & 0x02)
a[0x01] |= v;
else
a[0x01] &= ~v;
if (c & 0x04)
a[0x10] |= v;
else
a[0x10] &= ~v;
if (c & 0x08)
a[0x11] |= v;
else
a[0x11] &= ~v;
if (c & 0x10)
a[0x20] |= v;
else
a[0x20] &= ~v;
if (c & 0x20)
a[0x21] |= v;
else
a[0x21] &= ~v;
if (c & 0x40)
a[0x30] |= v;
else
a[0x30] &= ~v;
if (c & 0x80)
a[0x31] |= v;
else
a[0x31] &= ~v;
}
// 4c (ALT1) - rpix - read color of the pixel with R1, R2 as x, y
static void fx_rpix_8bit (void)
{
uint32 x = USEX8(R1);
uint32 y = USEX8(R2);
uint8 *a;
uint8 v;
R15++;
CLRFLAGS;
#ifdef CHECK_LIMITS
if (y >= GSU.vScreenHeight)
return;
#endif
a = GSU.apvScreen[y >> 3] + GSU.x[x >> 3] + ((y & 7) << 1);
v = 128 >> (x & 7);
DREG = 0;
DREG |= ((uint32) ((a[0x00] & v) != 0)) << 0;
DREG |= ((uint32) ((a[0x01] & v) != 0)) << 1;
DREG |= ((uint32) ((a[0x10] & v) != 0)) << 2;
DREG |= ((uint32) ((a[0x11] & v) != 0)) << 3;
DREG |= ((uint32) ((a[0x20] & v) != 0)) << 4;
DREG |= ((uint32) ((a[0x21] & v) != 0)) << 5;
DREG |= ((uint32) ((a[0x30] & v) != 0)) << 6;
DREG |= ((uint32) ((a[0x31] & v) != 0)) << 7;
GSU.vZero = DREG;
TESTR14;
}
// 4c - plot - plot pixel with R1, R2 as x, y and the color register as the color
static void fx_plot_obj (void)
{
#ifdef DEBUGGER
fprintf(stderr, "ERROR fx_plot_obj called\n");
#endif
}
// 4c (ALT1) - rpix - read color of the pixel with R1, R2 as x, y
static void fx_rpix_obj (void)
{
#ifdef DEBUGGER
fprintf(stderr, "ERROR fx_rpix_obj called\n");
#endif
}
// 4d - swap - swap upper and lower byte of a register
static void fx_swap (void)
{
uint8 c = (uint8) SREG;
uint8 d = (uint8) (SREG >> 8);
uint32 v = (((uint32) c) << 8) | ((uint32) d);
R15++;
DREG = v;
GSU.vSign = v;
GSU.vZero = v;
TESTR14;
CLRFLAGS;
}
// 4e - color - copy source register to color register
static void fx_color (void)
{
uint8 c = (uint8) SREG;
if (GSU.vPlotOptionReg & 0x04)
c = (c & 0xf0) | (c >> 4);
if (GSU.vPlotOptionReg & 0x08)
{
GSU.vColorReg &= 0xf0;
GSU.vColorReg |= c & 0x0f;
}
else
GSU.vColorReg = USEX8(c);
CLRFLAGS;
R15++;
}
// 4e (ALT1) - cmode - set plot option register
static void fx_cmode (void)
{
GSU.vPlotOptionReg = SREG;
if (GSU.vPlotOptionReg & 0x10)
GSU.vScreenHeight = 256; // OBJ Mode (for drawing into sprites)
else
GSU.vScreenHeight = GSU.vScreenRealHeight;
fx_computeScreenPointers();
CLRFLAGS;
R15++;
}
// 4f - not - perform exclusive exor with 1 on all bits
static void fx_not (void)
{
uint32 v = ~SREG;
R15++;
DREG = v;
GSU.vSign = v;
GSU.vZero = v;
TESTR14;
CLRFLAGS;
}
// 50-5f - add rn - add, register + register
#define FX_ADD(reg) \
int32 s = SUSEX16(SREG) + SUSEX16(GSU.avReg[reg]); \
GSU.vCarry = s >= 0x10000; \
GSU.vOverflow = ~(SREG ^ GSU.avReg[reg]) & (GSU.avReg[reg] ^ s) & 0x8000; \
GSU.vSign = s; \
GSU.vZero = s; \
R15++; \
DREG = s; \
TESTR14; \
CLRFLAGS
static void fx_add_r0 (void)
{
FX_ADD(0);
}
static void fx_add_r1 (void)
{
FX_ADD(1);
}
static void fx_add_r2 (void)
{
FX_ADD(2);
}
static void fx_add_r3 (void)
{
FX_ADD(3);
}
static void fx_add_r4 (void)
{
FX_ADD(4);
}
static void fx_add_r5 (void)
{
FX_ADD(5);
}
static void fx_add_r6 (void)
{
FX_ADD(6);
}
static void fx_add_r7 (void)
{
FX_ADD(7);
}
static void fx_add_r8 (void)
{
FX_ADD(8);
}
static void fx_add_r9 (void)
{
FX_ADD(9);
}
static void fx_add_r10 (void)
{
FX_ADD(10);
}
static void fx_add_r11 (void)
{
FX_ADD(11);
}
static void fx_add_r12 (void)
{
FX_ADD(12);
}
static void fx_add_r13 (void)
{
FX_ADD(13);
}
static void fx_add_r14 (void)
{
FX_ADD(14);
}
static void fx_add_r15 (void)
{
FX_ADD(15);
}
// 50-5f (ALT1) - adc rn - add with carry, register + register
#define FX_ADC(reg) \
int32 s = SUSEX16(SREG) + SUSEX16(GSU.avReg[reg]) + SEX16(GSU.vCarry); \
GSU.vCarry = s >= 0x10000; \
GSU.vOverflow = ~(SREG ^ GSU.avReg[reg]) & (GSU.avReg[reg] ^ s) & 0x8000; \
GSU.vSign = s; \
GSU.vZero = s; \
R15++; \
DREG = s; \
TESTR14; \
CLRFLAGS
static void fx_adc_r0 (void)
{
FX_ADC(0);
}
static void fx_adc_r1 (void)
{
FX_ADC(1);
}
static void fx_adc_r2 (void)
{
FX_ADC(2);
}
static void fx_adc_r3 (void)
{
FX_ADC(3);
}
static void fx_adc_r4 (void)
{
FX_ADC(4);
}
static void fx_adc_r5 (void)
{
FX_ADC(5);
}
static void fx_adc_r6 (void)
{
FX_ADC(6);
}
static void fx_adc_r7 (void)
{
FX_ADC(7);
}
static void fx_adc_r8 (void)
{
FX_ADC(8);
}
static void fx_adc_r9 (void)
{
FX_ADC(9);
}
static void fx_adc_r10 (void)
{
FX_ADC(10);
}
static void fx_adc_r11 (void)
{
FX_ADC(11);
}
static void fx_adc_r12 (void)
{
FX_ADC(12);
}
static void fx_adc_r13 (void)
{
FX_ADC(13);
}
static void fx_adc_r14 (void)
{
FX_ADC(14);
}
static void fx_adc_r15 (void)
{
FX_ADC(15);
}
// 50-5f (ALT2) - add #n - add, register + immediate
#define FX_ADD_I(imm) \
int32 s = SUSEX16(SREG) + imm; \
GSU.vCarry = s >= 0x10000; \
GSU.vOverflow = ~(SREG ^ imm) & (imm ^ s) & 0x8000; \
GSU.vSign = s; \
GSU.vZero = s; \
R15++; \
DREG = s; \
TESTR14; \
CLRFLAGS
static void fx_add_i0 (void)
{
FX_ADD_I(0);
}
static void fx_add_i1 (void)
{
FX_ADD_I(1);
}
static void fx_add_i2 (void)
{
FX_ADD_I(2);
}
static void fx_add_i3 (void)
{
FX_ADD_I(3);
}
static void fx_add_i4 (void)
{
FX_ADD_I(4);
}
static void fx_add_i5 (void)
{
FX_ADD_I(5);
}
static void fx_add_i6 (void)
{
FX_ADD_I(6);
}
static void fx_add_i7 (void)
{
FX_ADD_I(7);
}
static void fx_add_i8 (void)
{
FX_ADD_I(8);
}
static void fx_add_i9 (void)
{
FX_ADD_I(9);
}
static void fx_add_i10 (void)
{
FX_ADD_I(10);
}
static void fx_add_i11 (void)
{
FX_ADD_I(11);
}
static void fx_add_i12 (void)
{
FX_ADD_I(12);
}
static void fx_add_i13 (void)
{
FX_ADD_I(13);
}
static void fx_add_i14 (void)
{
FX_ADD_I(14);
}
static void fx_add_i15 (void)
{
FX_ADD_I(15);
}
// 50-5f (ALT3) - adc #n - add with carry, register + immediate
#define FX_ADC_I(imm) \
int32 s = SUSEX16(SREG) + imm + SUSEX16(GSU.vCarry); \
GSU.vCarry = s >= 0x10000; \
GSU.vOverflow = ~(SREG ^ imm) & (imm ^ s) & 0x8000; \
GSU.vSign = s; \
GSU.vZero = s; \
R15++; \
DREG = s; \
TESTR14; \
CLRFLAGS
static void fx_adc_i0 (void)
{
FX_ADC_I(0);
}
static void fx_adc_i1 (void)
{
FX_ADC_I(1);
}
static void fx_adc_i2 (void)
{
FX_ADC_I(2);
}
static void fx_adc_i3 (void)
{
FX_ADC_I(3);
}
static void fx_adc_i4 (void)
{
FX_ADC_I(4);
}
static void fx_adc_i5 (void)
{
FX_ADC_I(5);
}
static void fx_adc_i6 (void)
{
FX_ADC_I(6);
}
static void fx_adc_i7 (void)
{
FX_ADC_I(7);
}
static void fx_adc_i8 (void)
{
FX_ADC_I(8);
}
static void fx_adc_i9 (void)
{
FX_ADC_I(9);
}
static void fx_adc_i10 (void)
{
FX_ADC_I(10);
}
static void fx_adc_i11 (void)
{
FX_ADC_I(11);
}
static void fx_adc_i12 (void)
{
FX_ADC_I(12);
}
static void fx_adc_i13 (void)
{
FX_ADC_I(13);
}
static void fx_adc_i14 (void)
{
FX_ADC_I(14);
}
static void fx_adc_i15 (void)
{
FX_ADC_I(15);
}
// 60-6f - sub rn - subtract, register - register
#define FX_SUB(reg) \
int32 s = SUSEX16(SREG) - SUSEX16(GSU.avReg[reg]); \
GSU.vCarry = s >= 0; \
GSU.vOverflow = (SREG ^ GSU.avReg[reg]) & (SREG ^ s) & 0x8000; \
GSU.vSign = s; \
GSU.vZero = s; \
R15++; \
DREG = s; \
TESTR14; \
CLRFLAGS
static void fx_sub_r0 (void)
{
FX_SUB(0);
}
static void fx_sub_r1 (void)
{
FX_SUB(1);
}
static void fx_sub_r2 (void)
{
FX_SUB(2);
}
static void fx_sub_r3 (void)
{
FX_SUB(3);
}
static void fx_sub_r4 (void)
{
FX_SUB(4);
}
static void fx_sub_r5 (void)
{
FX_SUB(5);
}
static void fx_sub_r6 (void)
{
FX_SUB(6);
}
static void fx_sub_r7 (void)
{
FX_SUB(7);
}
static void fx_sub_r8 (void)
{
FX_SUB(8);
}
static void fx_sub_r9 (void)
{
FX_SUB(9);
}
static void fx_sub_r10 (void)
{
FX_SUB(10);
}
static void fx_sub_r11 (void)
{
FX_SUB(11);
}
static void fx_sub_r12 (void)
{
FX_SUB(12);
}
static void fx_sub_r13 (void)
{
FX_SUB(13);
}
static void fx_sub_r14 (void)
{
FX_SUB(14);
}
static void fx_sub_r15 (void)
{
FX_SUB(15);
}
// 60-6f (ALT1) - sbc rn - subtract with carry, register - register
#define FX_SBC(reg) \
int32 s = SUSEX16(SREG) - SUSEX16(GSU.avReg[reg]) - (SUSEX16(GSU.vCarry ^ 1)); \
GSU.vCarry = s >= 0; \
GSU.vOverflow = (SREG ^ GSU.avReg[reg]) & (SREG ^ s) & 0x8000; \
GSU.vSign = s; \
GSU.vZero = s; \
R15++; \
DREG = s; \
TESTR14; \
CLRFLAGS
static void fx_sbc_r0 (void)
{
FX_SBC(0);
}
static void fx_sbc_r1 (void)
{
FX_SBC(1);
}
static void fx_sbc_r2 (void)
{
FX_SBC(2);
}
static void fx_sbc_r3 (void)
{
FX_SBC(3);
}
static void fx_sbc_r4 (void)
{
FX_SBC(4);
}
static void fx_sbc_r5 (void)
{
FX_SBC(5);
}
static void fx_sbc_r6 (void)
{
FX_SBC(6);
}
static void fx_sbc_r7 (void)
{
FX_SBC(7);
}
static void fx_sbc_r8 (void)
{
FX_SBC(8);
}
static void fx_sbc_r9 (void)
{
FX_SBC(9);
}
static void fx_sbc_r10 (void)
{
FX_SBC(10);
}
static void fx_sbc_r11 (void)
{
FX_SBC(11);
}
static void fx_sbc_r12 (void)
{
FX_SBC(12);
}
static void fx_sbc_r13 (void)
{
FX_SBC(13);
}
static void fx_sbc_r14 (void)
{
FX_SBC(14);
}
static void fx_sbc_r15 (void)
{
FX_SBC(15);
}
// 60-6f (ALT2) - sub #n - subtract, register - immediate
#define FX_SUB_I(imm) \
int32 s = SUSEX16(SREG) - imm; \
GSU.vCarry = s >= 0; \
GSU.vOverflow = (SREG ^ imm) & (SREG ^ s) & 0x8000; \
GSU.vSign = s; \
GSU.vZero = s; \
R15++; \
DREG = s; \
TESTR14; \
CLRFLAGS
static void fx_sub_i0 (void)
{
FX_SUB_I(0);
}
static void fx_sub_i1 (void)
{
FX_SUB_I(1);
}
static void fx_sub_i2 (void)
{
FX_SUB_I(2);
}
static void fx_sub_i3 (void)
{
FX_SUB_I(3);
}
static void fx_sub_i4 (void)
{
FX_SUB_I(4);
}
static void fx_sub_i5 (void)
{
FX_SUB_I(5);
}
static void fx_sub_i6 (void)
{
FX_SUB_I(6);
}
static void fx_sub_i7 (void)
{
FX_SUB_I(7);
}
static void fx_sub_i8 (void)
{
FX_SUB_I(8);
}
static void fx_sub_i9 (void)
{
FX_SUB_I(9);
}
static void fx_sub_i10 (void)
{
FX_SUB_I(10);
}
static void fx_sub_i11 (void)
{
FX_SUB_I(11);
}
static void fx_sub_i12 (void)
{
FX_SUB_I(12);
}
static void fx_sub_i13 (void)
{
FX_SUB_I(13);
}
static void fx_sub_i14 (void)
{
FX_SUB_I(14);
}
static void fx_sub_i15 (void)
{
FX_SUB_I(15);
}
// 60-6f (ALT3) - cmp rn - compare, register, register
#define FX_CMP(reg) \
int32 s = SUSEX16(SREG) - SUSEX16(GSU.avReg[reg]); \
GSU.vCarry = s >= 0; \
GSU.vOverflow = (SREG ^ GSU.avReg[reg]) & (SREG ^ s) & 0x8000; \
GSU.vSign = s; \
GSU.vZero = s; \
R15++; \
CLRFLAGS
static void fx_cmp_r0 (void)
{
FX_CMP(0);
}
static void fx_cmp_r1 (void)
{
FX_CMP(1);
}
static void fx_cmp_r2 (void)
{
FX_CMP(2);
}
static void fx_cmp_r3 (void)
{
FX_CMP(3);
}
static void fx_cmp_r4 (void)
{
FX_CMP(4);
}
static void fx_cmp_r5 (void)
{
FX_CMP(5);
}
static void fx_cmp_r6 (void)
{
FX_CMP(6);
}
static void fx_cmp_r7 (void)
{
FX_CMP(7);
}
static void fx_cmp_r8 (void)
{
FX_CMP(8);
}
static void fx_cmp_r9 (void)
{
FX_CMP(9);
}
static void fx_cmp_r10 (void)
{
FX_CMP(10);
}
static void fx_cmp_r11 (void)
{
FX_CMP(11);
}
static void fx_cmp_r12 (void)
{
FX_CMP(12);
}
static void fx_cmp_r13 (void)
{
FX_CMP(13);
}
static void fx_cmp_r14 (void)
{
FX_CMP(14);
}
static void fx_cmp_r15 (void)
{
FX_CMP(15);
}
// 70 - merge - R7 as upper byte, R8 as lower byte (used for texture-mapping)
static void fx_merge (void)
{
uint32 v = (R7 & 0xff00) | ((R8 & 0xff00) >> 8);
R15++;
DREG = v;
GSU.vOverflow = (v & 0xc0c0) << 16;
GSU.vZero = !(v & 0xf0f0);
GSU.vSign = ((v | (v << 8)) & 0x8000);
GSU.vCarry = (v & 0xe0e0) != 0;
TESTR14;
CLRFLAGS;
}
// 71-7f - and rn - reister & register
#define FX_AND(reg) \
uint32 v = SREG & GSU.avReg[reg]; \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_and_r1 (void)
{
FX_AND(1);
}
static void fx_and_r2 (void)
{
FX_AND(2);
}
static void fx_and_r3 (void)
{
FX_AND(3);
}
static void fx_and_r4 (void)
{
FX_AND(4);
}
static void fx_and_r5 (void)
{
FX_AND(5);
}
static void fx_and_r6 (void)
{
FX_AND(6);
}
static void fx_and_r7 (void)
{
FX_AND(7);
}
static void fx_and_r8 (void)
{
FX_AND(8);
}
static void fx_and_r9 (void)
{
FX_AND(9);
}
static void fx_and_r10 (void)
{
FX_AND(10);
}
static void fx_and_r11 (void)
{
FX_AND(11);
}
static void fx_and_r12 (void)
{
FX_AND(12);
}
static void fx_and_r13 (void)
{
FX_AND(13);
}
static void fx_and_r14 (void)
{
FX_AND(14);
}
static void fx_and_r15 (void)
{
FX_AND(15);
}
// 71-7f (ALT1) - bic rn - reister & ~register
#define FX_BIC(reg) \
uint32 v = SREG & ~GSU.avReg[reg]; \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_bic_r1 (void)
{
FX_BIC(1);
}
static void fx_bic_r2 (void)
{
FX_BIC(2);
}
static void fx_bic_r3 (void)
{
FX_BIC(3);
}
static void fx_bic_r4 (void)
{
FX_BIC(4);
}
static void fx_bic_r5 (void)
{
FX_BIC(5);
}
static void fx_bic_r6 (void)
{
FX_BIC(6);
}
static void fx_bic_r7 (void)
{
FX_BIC(7);
}
static void fx_bic_r8 (void)
{
FX_BIC(8);
}
static void fx_bic_r9 (void)
{
FX_BIC(9);
}
static void fx_bic_r10 (void)
{
FX_BIC(10);
}
static void fx_bic_r11 (void)
{
FX_BIC(11);
}
static void fx_bic_r12 (void)
{
FX_BIC(12);
}
static void fx_bic_r13 (void)
{
FX_BIC(13);
}
static void fx_bic_r14 (void)
{
FX_BIC(14);
}
static void fx_bic_r15 (void)
{
FX_BIC(15);
}
// 71-7f (ALT2) - and #n - reister & immediate
#define FX_AND_I(imm) \
uint32 v = SREG & imm; \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_and_i1 (void)
{
FX_AND_I(1);
}
static void fx_and_i2 (void)
{
FX_AND_I(2);
}
static void fx_and_i3 (void)
{
FX_AND_I(3);
}
static void fx_and_i4 (void)
{
FX_AND_I(4);
}
static void fx_and_i5 (void)
{
FX_AND_I(5);
}
static void fx_and_i6 (void)
{
FX_AND_I(6);
}
static void fx_and_i7 (void)
{
FX_AND_I(7);
}
static void fx_and_i8 (void)
{
FX_AND_I(8);
}
static void fx_and_i9 (void)
{
FX_AND_I(9);
}
static void fx_and_i10 (void)
{
FX_AND_I(10);
}
static void fx_and_i11 (void)
{
FX_AND_I(11);
}
static void fx_and_i12 (void)
{
FX_AND_I(12);
}
static void fx_and_i13 (void)
{
FX_AND_I(13);
}
static void fx_and_i14 (void)
{
FX_AND_I(14);
}
static void fx_and_i15 (void)
{
FX_AND_I(15);
}
// 71-7f (ALT3) - bic #n - reister & ~immediate
#define FX_BIC_I(imm) \
uint32 v = SREG & ~imm; \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_bic_i1 (void)
{
FX_BIC_I(1);
}
static void fx_bic_i2 (void)
{
FX_BIC_I(2);
}
static void fx_bic_i3 (void)
{
FX_BIC_I(3);
}
static void fx_bic_i4 (void)
{
FX_BIC_I(4);
}
static void fx_bic_i5 (void)
{
FX_BIC_I(5);
}
static void fx_bic_i6 (void)
{
FX_BIC_I(6);
}
static void fx_bic_i7 (void)
{
FX_BIC_I(7);
}
static void fx_bic_i8 (void)
{
FX_BIC_I(8);
}
static void fx_bic_i9 (void)
{
FX_BIC_I(9);
}
static void fx_bic_i10 (void)
{
FX_BIC_I(10);
}
static void fx_bic_i11 (void)
{
FX_BIC_I(11);
}
static void fx_bic_i12 (void)
{
FX_BIC_I(12);
}
static void fx_bic_i13 (void)
{
FX_BIC_I(13);
}
static void fx_bic_i14 (void)
{
FX_BIC_I(14);
}
static void fx_bic_i15 (void)
{
FX_BIC_I(15);
}
// 80-8f - mult rn - 8 bit to 16 bit signed multiply, register * register
#define FX_MULT(reg) \
uint32 v = (uint32) (SEX8(SREG) * SEX8(GSU.avReg[reg])); \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_mult_r0 (void)
{
FX_MULT(0);
}
static void fx_mult_r1 (void)
{
FX_MULT(1);
}
static void fx_mult_r2 (void)
{
FX_MULT(2);
}
static void fx_mult_r3 (void)
{
FX_MULT(3);
}
static void fx_mult_r4 (void)
{
FX_MULT(4);
}
static void fx_mult_r5 (void)
{
FX_MULT(5);
}
static void fx_mult_r6 (void)
{
FX_MULT(6);
}
static void fx_mult_r7 (void)
{
FX_MULT(7);
}
static void fx_mult_r8 (void)
{
FX_MULT(8);
}
static void fx_mult_r9 (void)
{
FX_MULT(9);
}
static void fx_mult_r10 (void)
{
FX_MULT(10);
}
static void fx_mult_r11 (void)
{
FX_MULT(11);
}
static void fx_mult_r12 (void)
{
FX_MULT(12);
}
static void fx_mult_r13 (void)
{
FX_MULT(13);
}
static void fx_mult_r14 (void)
{
FX_MULT(14);
}
static void fx_mult_r15 (void)
{
FX_MULT(15);
}
// 80-8f (ALT1) - umult rn - 8 bit to 16 bit unsigned multiply, register * register
#define FX_UMULT(reg) \
uint32 v = USEX8(SREG) * USEX8(GSU.avReg[reg]); \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_umult_r0 (void)
{
FX_UMULT(0);
}
static void fx_umult_r1 (void)
{
FX_UMULT(1);
}
static void fx_umult_r2 (void)
{
FX_UMULT(2);
}
static void fx_umult_r3 (void)
{
FX_UMULT(3);
}
static void fx_umult_r4 (void)
{
FX_UMULT(4);
}
static void fx_umult_r5 (void)
{
FX_UMULT(5);
}
static void fx_umult_r6 (void)
{
FX_UMULT(6);
}
static void fx_umult_r7 (void)
{
FX_UMULT(7);
}
static void fx_umult_r8 (void)
{
FX_UMULT(8);
}
static void fx_umult_r9 (void)
{
FX_UMULT(9);
}
static void fx_umult_r10 (void)
{
FX_UMULT(10);
}
static void fx_umult_r11 (void)
{
FX_UMULT(11);
}
static void fx_umult_r12 (void)
{
FX_UMULT(12);
}
static void fx_umult_r13 (void)
{
FX_UMULT(13);
}
static void fx_umult_r14 (void)
{
FX_UMULT(14);
}
static void fx_umult_r15 (void)
{
FX_UMULT(15);
}
// 80-8f (ALT2) - mult #n - 8 bit to 16 bit signed multiply, register * immediate
#define FX_MULT_I(imm) \
uint32 v = (uint32) (SEX8(SREG) * ((int32) imm)); \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_mult_i0 (void)
{
FX_MULT_I(0);
}
static void fx_mult_i1 (void)
{
FX_MULT_I(1);
}
static void fx_mult_i2 (void)
{
FX_MULT_I(2);
}
static void fx_mult_i3 (void)
{
FX_MULT_I(3);
}
static void fx_mult_i4 (void)
{
FX_MULT_I(4);
}
static void fx_mult_i5 (void)
{
FX_MULT_I(5);
}
static void fx_mult_i6 (void)
{
FX_MULT_I(6);
}
static void fx_mult_i7 (void)
{
FX_MULT_I(7);
}
static void fx_mult_i8 (void)
{
FX_MULT_I(8);
}
static void fx_mult_i9 (void)
{
FX_MULT_I(9);
}
static void fx_mult_i10 (void)
{
FX_MULT_I(10);
}
static void fx_mult_i11 (void)
{
FX_MULT_I(11);
}
static void fx_mult_i12 (void)
{
FX_MULT_I(12);
}
static void fx_mult_i13 (void)
{
FX_MULT_I(13);
}
static void fx_mult_i14 (void)
{
FX_MULT_I(14);
}
static void fx_mult_i15 (void)
{
FX_MULT_I(15);
}
// 80-8f (ALT3) - umult #n - 8 bit to 16 bit unsigned multiply, register * immediate
#define FX_UMULT_I(imm) \
uint32 v = USEX8(SREG) * ((uint32) imm); \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_umult_i0 (void)
{
FX_UMULT_I(0);
}
static void fx_umult_i1 (void)
{
FX_UMULT_I(1);
}
static void fx_umult_i2 (void)
{
FX_UMULT_I(2);
}
static void fx_umult_i3 (void)
{
FX_UMULT_I(3);
}
static void fx_umult_i4 (void)
{
FX_UMULT_I(4);
}
static void fx_umult_i5 (void)
{
FX_UMULT_I(5);
}
static void fx_umult_i6 (void)
{
FX_UMULT_I(6);
}
static void fx_umult_i7 (void)
{
FX_UMULT_I(7);
}
static void fx_umult_i8 (void)
{
FX_UMULT_I(8);
}
static void fx_umult_i9 (void)
{
FX_UMULT_I(9);
}
static void fx_umult_i10 (void)
{
FX_UMULT_I(10);
}
static void fx_umult_i11 (void)
{
FX_UMULT_I(11);
}
static void fx_umult_i12 (void)
{
FX_UMULT_I(12);
}
static void fx_umult_i13 (void)
{
FX_UMULT_I(13);
}
static void fx_umult_i14 (void)
{
FX_UMULT_I(14);
}
static void fx_umult_i15 (void)
{
FX_UMULT_I(15);
}
// 90 - sbk - store word to last accessed RAM address
static void fx_sbk (void)
{
RAM(GSU.vLastRamAdr) = (uint8) SREG;
RAM(GSU.vLastRamAdr ^ 1) = (uint8) (SREG >> 8);
CLRFLAGS;
R15++;
}
// 91-94 - link #n - R11 = R15 + immediate
#define FX_LINK_I(lkn) \
R11 = R15 + lkn; \
CLRFLAGS; \
R15++
static void fx_link_i1 (void)
{
FX_LINK_I(1);
}
static void fx_link_i2 (void)
{
FX_LINK_I(2);
}
static void fx_link_i3 (void)
{
FX_LINK_I(3);
}
static void fx_link_i4 (void)
{
FX_LINK_I(4);
}
// 95 - sex - sign extend 8 bit to 16 bit
static void fx_sex (void)
{
uint32 v = (uint32) SEX8(SREG);
R15++;
DREG = v;
GSU.vSign = v;
GSU.vZero = v;
TESTR14;
CLRFLAGS;
}
// 96 - asr - aritmetric shift right by one
static void fx_asr (void)
{
uint32 v;
GSU.vCarry = SREG & 1;
v = (uint32) (SEX16(SREG) >> 1);
R15++;
DREG = v;
GSU.vSign = v;
GSU.vZero = v;
TESTR14;
CLRFLAGS;
}
// 96 (ALT1) - div2 - aritmetric shift right by one
static void fx_div2 (void)
{
uint32 v;
int32 s = SEX16(SREG);
GSU.vCarry = s & 1;
if (s == -1)
v = 0;
else
v = (uint32) (s >> 1);
R15++;
DREG = v;
GSU.vSign = v;
GSU.vZero = v;
TESTR14;
CLRFLAGS;
}
// 97 - ror - rotate right by one
static void fx_ror (void)
{
uint32 v = (USEX16(SREG) >> 1) | (GSU.vCarry << 15);
GSU.vCarry = SREG & 1;
R15++;
DREG = v;
GSU.vSign = v;
GSU.vZero = v;
TESTR14;
CLRFLAGS;
}
// 98-9d - jmp rn - jump to address of register
#define FX_JMP(reg) \
R15 = GSU.avReg[reg]; \
CLRFLAGS
static void fx_jmp_r8 (void)
{
FX_JMP(8);
}
static void fx_jmp_r9 (void)
{
FX_JMP(9);
}
static void fx_jmp_r10 (void)
{
FX_JMP(10);
}
static void fx_jmp_r11 (void)
{
FX_JMP(11);
}
static void fx_jmp_r12 (void)
{
FX_JMP(12);
}
static void fx_jmp_r13 (void)
{
FX_JMP(13);
}
// 98-9d (ALT1) - ljmp rn - set program bank to source register and jump to address of register
#define FX_LJMP(reg) \
GSU.vPrgBankReg = GSU.avReg[reg] & 0x7f; \
GSU.pvPrgBank = GSU.apvRomBank[GSU.vPrgBankReg]; \
R15 = SREG; \
GSU.bCacheActive = FALSE; \
fx_cache(); \
R15--
static void fx_ljmp_r8 (void)
{
FX_LJMP(8);
}
static void fx_ljmp_r9 (void)
{
FX_LJMP(9);
}
static void fx_ljmp_r10 (void)
{
FX_LJMP(10);
}
static void fx_ljmp_r11 (void)
{
FX_LJMP(11);
}
static void fx_ljmp_r12 (void)
{
FX_LJMP(12);
}
static void fx_ljmp_r13 (void)
{
FX_LJMP(13);
}
// 9e - lob - set upper byte to zero (keep low byte)
static void fx_lob (void)
{
uint32 v = USEX8(SREG);
R15++;
DREG = v;
GSU.vSign = v << 8;
GSU.vZero = v << 8;
TESTR14;
CLRFLAGS;
}
// 9f - fmult - 16 bit to 32 bit signed multiplication, upper 16 bits only
static void fx_fmult (void)
{
uint32 v;
uint32 c = (uint32) (SEX16(SREG) * SEX16(R6));
v = c >> 16;
R15++;
DREG = v;
GSU.vSign = v;
GSU.vZero = v;
GSU.vCarry = (c >> 15) & 1;
TESTR14;
CLRFLAGS;
}
// 9f (ALT1) - lmult - 16 bit to 32 bit signed multiplication
static void fx_lmult (void)
{
uint32 v;
uint32 c = (uint32) (SEX16(SREG) * SEX16(R6));
R4 = c;
v = c >> 16;
R15++;
DREG = v;
GSU.vSign = v;
GSU.vZero = v;
// XXX: R6 or R4?
GSU.vCarry = (R4 >> 15) & 1; // should it be bit 15 of R4 instead ?
TESTR14;
CLRFLAGS;
}
// a0-af - ibt rn, #pp - immediate byte transfer
#define FX_IBT(reg) \
uint8 v = PIPE; \
R15++; \
FETCHPIPE; \
R15++; \
GSU.avReg[reg] = SEX8(v); \
CLRFLAGS
static void fx_ibt_r0 (void)
{
FX_IBT(0);
}
static void fx_ibt_r1 (void)
{
FX_IBT(1);
}
static void fx_ibt_r2 (void)
{
FX_IBT(2);
}
static void fx_ibt_r3 (void)
{
FX_IBT(3);
}
static void fx_ibt_r4 (void)
{
FX_IBT(4);
}
static void fx_ibt_r5 (void)
{
FX_IBT(5);
}
static void fx_ibt_r6 (void)
{
FX_IBT(6);
}
static void fx_ibt_r7 (void)
{
FX_IBT(7);
}
static void fx_ibt_r8 (void)
{
FX_IBT(8);
}
static void fx_ibt_r9 (void)
{
FX_IBT(9);
}
static void fx_ibt_r10 (void)
{
FX_IBT(10);
}
static void fx_ibt_r11 (void)
{
FX_IBT(11);
}
static void fx_ibt_r12 (void)
{
FX_IBT(12);
}
static void fx_ibt_r13 (void)
{
FX_IBT(13);
}
static void fx_ibt_r14 (void)
{
FX_IBT(14);
READR14;
}
static void fx_ibt_r15 (void)
{
FX_IBT(15);
}
// a0-af (ALT1) - lms rn, (yy) - load word from RAM (short address)
#define FX_LMS(reg) \
GSU.vLastRamAdr = ((uint32) PIPE) << 1; \
R15++; \
FETCHPIPE; \
R15++; \
GSU.avReg[reg] = (uint32) RAM(GSU.vLastRamAdr); \
GSU.avReg[reg] |= ((uint32) RAM(GSU.vLastRamAdr + 1)) << 8; \
CLRFLAGS
static void fx_lms_r0 (void)
{
FX_LMS(0);
}
static void fx_lms_r1 (void)
{
FX_LMS(1);
}
static void fx_lms_r2 (void)
{
FX_LMS(2);
}
static void fx_lms_r3 (void)
{
FX_LMS(3);
}
static void fx_lms_r4 (void)
{
FX_LMS(4);
}
static void fx_lms_r5 (void)
{
FX_LMS(5);
}
static void fx_lms_r6 (void)
{
FX_LMS(6);
}
static void fx_lms_r7 (void)
{
FX_LMS(7);
}
static void fx_lms_r8 (void)
{
FX_LMS(8);
}
static void fx_lms_r9 (void)
{
FX_LMS(9);
}
static void fx_lms_r10 (void)
{
FX_LMS(10);
}
static void fx_lms_r11 (void)
{
FX_LMS(11);
}
static void fx_lms_r12 (void)
{
FX_LMS(12);
}
static void fx_lms_r13 (void)
{
FX_LMS(13);
}
static void fx_lms_r14 (void)
{
FX_LMS(14);
READR14;
}
static void fx_lms_r15 (void)
{
FX_LMS(15);
}
// a0-af (ALT2) - sms (yy), rn - store word in RAM (short address)
// XXX: If rn == r15, is the value of r15 before or after the extra byte is read ?
#define FX_SMS(reg) \
uint32 v = GSU.avReg[reg]; \
GSU.vLastRamAdr = ((uint32) PIPE) << 1; \
R15++; \
FETCHPIPE; \
RAM(GSU.vLastRamAdr) = (uint8) v; \
RAM(GSU.vLastRamAdr + 1) = (uint8) (v >> 8); \
CLRFLAGS; \
R15++
static void fx_sms_r0 (void)
{
FX_SMS(0);
}
static void fx_sms_r1 (void)
{
FX_SMS(1);
}
static void fx_sms_r2 (void)
{
FX_SMS(2);
}
static void fx_sms_r3 (void)
{
FX_SMS(3);
}
static void fx_sms_r4 (void)
{
FX_SMS(4);
}
static void fx_sms_r5 (void)
{
FX_SMS(5);
}
static void fx_sms_r6 (void)
{
FX_SMS(6);
}
static void fx_sms_r7 (void)
{
FX_SMS(7);
}
static void fx_sms_r8 (void)
{
FX_SMS(8);
}
static void fx_sms_r9 (void)
{
FX_SMS(9);
}
static void fx_sms_r10 (void)
{
FX_SMS(10);
}
static void fx_sms_r11 (void)
{
FX_SMS(11);
}
static void fx_sms_r12 (void)
{
FX_SMS(12);
}
static void fx_sms_r13 (void)
{
FX_SMS(13);
}
static void fx_sms_r14 (void)
{
FX_SMS(14);
}
static void fx_sms_r15 (void)
{
FX_SMS(15);
}
// b0-bf - from rn - set source register
// b0-bf (B) - moves rn - move register to register, and set flags, (if B flag is set)
#define FX_FROM(reg) \
if (TF(B)) \
{ \
uint32 v = GSU.avReg[reg]; \
R15++; \
DREG = v; \
GSU.vOverflow = (v & 0x80) << 16; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS; \
} \
else \
{ \
GSU.pvSreg = &GSU.avReg[reg]; \
R15++; \
}
static void fx_from_r0 (void)
{
FX_FROM(0);
}
static void fx_from_r1 (void)
{
FX_FROM(1);
}
static void fx_from_r2 (void)
{
FX_FROM(2);
}
static void fx_from_r3 (void)
{
FX_FROM(3);
}
static void fx_from_r4 (void)
{
FX_FROM(4);
}
static void fx_from_r5 (void)
{
FX_FROM(5);
}
static void fx_from_r6 (void)
{
FX_FROM(6);
}
static void fx_from_r7 (void)
{
FX_FROM(7);
}
static void fx_from_r8 (void)
{
FX_FROM(8);
}
static void fx_from_r9 (void)
{
FX_FROM(9);
}
static void fx_from_r10 (void)
{
FX_FROM(10);
}
static void fx_from_r11 (void)
{
FX_FROM(11);
}
static void fx_from_r12 (void)
{
FX_FROM(12);
}
static void fx_from_r13 (void)
{
FX_FROM(13);
}
static void fx_from_r14 (void)
{
FX_FROM(14);
}
static void fx_from_r15 (void)
{
FX_FROM(15);
}
// c0 - hib - move high-byte to low-byte
static void fx_hib (void)
{
uint32 v = USEX8(SREG >> 8);
R15++;
DREG = v;
GSU.vSign = v << 8;
GSU.vZero = v << 8;
TESTR14;
CLRFLAGS;
}
// c1-cf - or rn
#define FX_OR(reg) \
uint32 v = SREG | GSU.avReg[reg]; \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_or_r1 (void)
{
FX_OR(1);
}
static void fx_or_r2 (void)
{
FX_OR(2);
}
static void fx_or_r3 (void)
{
FX_OR(3);
}
static void fx_or_r4 (void)
{
FX_OR(4);
}
static void fx_or_r5 (void)
{
FX_OR(5);
}
static void fx_or_r6 (void)
{
FX_OR(6);
}
static void fx_or_r7 (void)
{
FX_OR(7);
}
static void fx_or_r8 (void)
{
FX_OR(8);
}
static void fx_or_r9 (void)
{
FX_OR(9);
}
static void fx_or_r10 (void)
{
FX_OR(10);
}
static void fx_or_r11 (void)
{
FX_OR(11);
}
static void fx_or_r12 (void)
{
FX_OR(12);
}
static void fx_or_r13 (void)
{
FX_OR(13);
}
static void fx_or_r14 (void)
{
FX_OR(14);
}
static void fx_or_r15 (void)
{
FX_OR(15);
}
// c1-cf (ALT1) - xor rn
#define FX_XOR(reg) \
uint32 v = SREG ^ GSU.avReg[reg]; \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_xor_r1 (void)
{
FX_XOR(1);
}
static void fx_xor_r2 (void)
{
FX_XOR(2);
}
static void fx_xor_r3 (void)
{
FX_XOR(3);
}
static void fx_xor_r4 (void)
{
FX_XOR(4);
}
static void fx_xor_r5 (void)
{
FX_XOR(5);
}
static void fx_xor_r6 (void)
{
FX_XOR(6);
}
static void fx_xor_r7 (void)
{
FX_XOR(7);
}
static void fx_xor_r8 (void)
{
FX_XOR(8);
}
static void fx_xor_r9 (void)
{
FX_XOR(9);
}
static void fx_xor_r10 (void)
{
FX_XOR(10);
}
static void fx_xor_r11 (void)
{
FX_XOR(11);
}
static void fx_xor_r12 (void)
{
FX_XOR(12);
}
static void fx_xor_r13 (void)
{
FX_XOR(13);
}
static void fx_xor_r14 (void)
{
FX_XOR(14);
}
static void fx_xor_r15 (void)
{
FX_XOR(15);
}
// c1-cf (ALT2) - or #n
#define FX_OR_I(imm) \
uint32 v = SREG | imm; \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_or_i1 (void)
{
FX_OR_I(1);
}
static void fx_or_i2 (void)
{
FX_OR_I(2);
}
static void fx_or_i3 (void)
{
FX_OR_I(3);
}
static void fx_or_i4 (void)
{
FX_OR_I(4);
}
static void fx_or_i5 (void)
{
FX_OR_I(5);
}
static void fx_or_i6 (void)
{
FX_OR_I(6);
}
static void fx_or_i7 (void)
{
FX_OR_I(7);
}
static void fx_or_i8 (void)
{
FX_OR_I(8);
}
static void fx_or_i9 (void)
{
FX_OR_I(9);
}
static void fx_or_i10 (void)
{
FX_OR_I(10);
}
static void fx_or_i11 (void)
{
FX_OR_I(11);
}
static void fx_or_i12 (void)
{
FX_OR_I(12);
}
static void fx_or_i13 (void)
{
FX_OR_I(13);
}
static void fx_or_i14 (void)
{
FX_OR_I(14);
}
static void fx_or_i15 (void)
{
FX_OR_I(15);
}
// c1-cf (ALT3) - xor #n
#define FX_XOR_I(imm) \
uint32 v = SREG ^ imm; \
R15++; \
DREG = v; \
GSU.vSign = v; \
GSU.vZero = v; \
TESTR14; \
CLRFLAGS
static void fx_xor_i1 (void)
{
FX_XOR_I(1);
}
static void fx_xor_i2 (void)
{
FX_XOR_I(2);
}
static void fx_xor_i3 (void)
{
FX_XOR_I(3);
}
static void fx_xor_i4 (void)
{
FX_XOR_I(4);
}
static void fx_xor_i5 (void)
{
FX_XOR_I(5);
}
static void fx_xor_i6 (void)
{
FX_XOR_I(6);
}
static void fx_xor_i7 (void)
{
FX_XOR_I(7);
}
static void fx_xor_i8 (void)
{
FX_XOR_I(8);
}
static void fx_xor_i9 (void)
{
FX_XOR_I(9);
}
static void fx_xor_i10 (void)
{
FX_XOR_I(10);
}
static void fx_xor_i11 (void)
{
FX_XOR_I(11);
}
static void fx_xor_i12 (void)
{
FX_XOR_I(12);
}
static void fx_xor_i13 (void)
{
FX_XOR_I(13);
}
static void fx_xor_i14 (void)
{
FX_XOR_I(14);
}
static void fx_xor_i15 (void)
{
FX_XOR_I(15);
}
// d0-de - inc rn - increase by one
#define FX_INC(reg) \
GSU.avReg[reg] += 1; \
GSU.vSign = GSU.avReg[reg]; \
GSU.vZero = GSU.avReg[reg]; \
CLRFLAGS; \
R15++
static void fx_inc_r0 (void)
{
FX_INC(0);
}
static void fx_inc_r1 (void)
{
FX_INC(1);
}
static void fx_inc_r2 (void)
{
FX_INC(2);
}
static void fx_inc_r3 (void)
{
FX_INC(3);
}
static void fx_inc_r4 (void)
{
FX_INC(4);
}
static void fx_inc_r5 (void)
{
FX_INC(5);
}
static void fx_inc_r6 (void)
{
FX_INC(6);
}
static void fx_inc_r7 (void)
{
FX_INC(7);
}
static void fx_inc_r8 (void)
{
FX_INC(8);
}
static void fx_inc_r9 (void)
{
FX_INC(9);
}
static void fx_inc_r10 (void)
{
FX_INC(10);
}
static void fx_inc_r11 (void)
{
FX_INC(11);
}
static void fx_inc_r12 (void)
{
FX_INC(12);
}
static void fx_inc_r13 (void)
{
FX_INC(13);
}
static void fx_inc_r14 (void)
{
FX_INC(14);
READR14;
}
// df - getc - transfer ROM buffer to color register
static void fx_getc (void)
{
#ifndef FX_DO_ROMBUFFER
uint8 c = ROM(R14);
#else
uint8 c = GSU.vRomBuffer;
#endif
if (GSU.vPlotOptionReg & 0x04)
c = (c & 0xf0) | (c >> 4);
if (GSU.vPlotOptionReg & 0x08)
{
GSU.vColorReg &= 0xf0;
GSU.vColorReg |= c & 0x0f;
}
else
GSU.vColorReg = USEX8(c);
CLRFLAGS;
R15++;
}
// df (ALT2) - ramb - set current RAM bank
static void fx_ramb (void)
{
GSU.vRamBankReg = SREG & (FX_RAM_BANKS - 1);
GSU.pvRamBank = GSU.apvRamBank[GSU.vRamBankReg & 0x3];
CLRFLAGS;
R15++;
}
// df (ALT3) - romb - set current ROM bank
static void fx_romb (void)
{
GSU.vRomBankReg = USEX8(SREG) & 0x7f;
GSU.pvRomBank = GSU.apvRomBank[GSU.vRomBankReg];
CLRFLAGS;
R15++;
}
// e0-ee - dec rn - decrement by one
#define FX_DEC(reg) \
GSU.avReg[reg] -= 1; \
GSU.vSign = GSU.avReg[reg]; \
GSU.vZero = GSU.avReg[reg]; \
CLRFLAGS; \
R15++
static void fx_dec_r0 (void)
{
FX_DEC(0);
}
static void fx_dec_r1 (void)
{
FX_DEC(1);
}
static void fx_dec_r2 (void)
{
FX_DEC(2);
}
static void fx_dec_r3 (void)
{
FX_DEC(3);
}
static void fx_dec_r4 (void)
{
FX_DEC(4);
}
static void fx_dec_r5 (void)
{
FX_DEC(5);
}
static void fx_dec_r6 (void)
{
FX_DEC(6);
}
static void fx_dec_r7 (void)
{
FX_DEC(7);
}
static void fx_dec_r8 (void)
{
FX_DEC(8);
}
static void fx_dec_r9 (void)
{
FX_DEC(9);
}
static void fx_dec_r10 (void)
{
FX_DEC(10);
}
static void fx_dec_r11 (void)
{
FX_DEC(11);
}
static void fx_dec_r12 (void)
{
FX_DEC(12);
}
static void fx_dec_r13 (void)
{
FX_DEC(13);
}
static void fx_dec_r14 (void)
{
FX_DEC(14);
READR14;
}
// ef - getb - get byte from ROM at address R14
static void fx_getb (void)
{
uint32 v;
#ifndef FX_DO_ROMBUFFER
v = (uint32) ROM(R14);
#else
v = (uint32) GSU.vRomBuffer;
#endif
R15++;
DREG = v;
TESTR14;
CLRFLAGS;
}
// ef (ALT1) - getbh - get high-byte from ROM at address R14
static void fx_getbh (void)
{
uint32 v;
#ifndef FX_DO_ROMBUFFER
uint32 c = (uint32) ROM(R14);
#else
uint32 c = USEX8(GSU.vRomBuffer);
#endif
v = USEX8(SREG) | (c << 8);
R15++;
DREG = v;
TESTR14;
CLRFLAGS;
}
// ef (ALT2) - getbl - get low-byte from ROM at address R14
static void fx_getbl (void)
{
uint32 v;
#ifndef FX_DO_ROMBUFFER
uint32 c = (uint32) ROM(R14);
#else
uint32 c = USEX8(GSU.vRomBuffer);
#endif
v = (SREG & 0xff00) | c;
R15++;
DREG = v;
TESTR14;
CLRFLAGS;
}
// ef (ALT3) - getbs - get sign extended byte from ROM at address R14
static void fx_getbs (void)
{
uint32 v;
#ifndef FX_DO_ROMBUFFER
int8 c;
c = ROM(R14);
v = SEX8(c);
#else
v = SEX8(GSU.vRomBuffer);
#endif
R15++;
DREG = v;
TESTR14;
CLRFLAGS;
}
// f0-ff - iwt rn, #xx - immediate word transfer to register
#define FX_IWT(reg) \
uint32 v = PIPE; \
R15++; \
FETCHPIPE; \
R15++; \
v |= USEX8(PIPE) << 8; \
FETCHPIPE; \
R15++; \
GSU.avReg[reg] = v; \
CLRFLAGS
static void fx_iwt_r0 (void)
{
FX_IWT(0);
}
static void fx_iwt_r1 (void)
{
FX_IWT(1);
}
static void fx_iwt_r2 (void)
{
FX_IWT(2);
}
static void fx_iwt_r3 (void)
{
FX_IWT(3);
}
static void fx_iwt_r4 (void)
{
FX_IWT(4);
}
static void fx_iwt_r5 (void)
{
FX_IWT(5);
}
static void fx_iwt_r6 (void)
{
FX_IWT(6);
}
static void fx_iwt_r7 (void)
{
FX_IWT(7);
}
static void fx_iwt_r8 (void)
{
FX_IWT(8);
}
static void fx_iwt_r9 (void)
{
FX_IWT(9);
}
static void fx_iwt_r10 (void)
{
FX_IWT(10);
}
static void fx_iwt_r11 (void)
{
FX_IWT(11);
}
static void fx_iwt_r12 (void)
{
FX_IWT(12);
}
static void fx_iwt_r13 (void)
{
FX_IWT(13);
}
static void fx_iwt_r14 (void)
{
FX_IWT(14);
READR14;
}
static void fx_iwt_r15 (void)
{
FX_IWT(15);
}
// f0-ff (ALT1) - lm rn, (xx) - load word from RAM
#define FX_LM(reg) \
GSU.vLastRamAdr = PIPE; \
R15++; \
FETCHPIPE; \
R15++; \
GSU.vLastRamAdr |= USEX8(PIPE) << 8; \
FETCHPIPE; \
R15++; \
GSU.avReg[reg] = RAM(GSU.vLastRamAdr); \
GSU.avReg[reg] |= USEX8(RAM(GSU.vLastRamAdr ^ 1)) << 8; \
CLRFLAGS
static void fx_lm_r0 (void)
{
FX_LM(0);
}
static void fx_lm_r1 (void)
{
FX_LM(1);
}
static void fx_lm_r2 (void)
{
FX_LM(2);
}
static void fx_lm_r3 (void)
{
FX_LM(3);
}
static void fx_lm_r4 (void)
{
FX_LM(4);
}
static void fx_lm_r5 (void)
{
FX_LM(5);
}
static void fx_lm_r6 (void)
{
FX_LM(6);
}
static void fx_lm_r7 (void)
{
FX_LM(7);
}
static void fx_lm_r8 (void)
{
FX_LM(8);
}
static void fx_lm_r9 (void)
{
FX_LM(9);
}
static void fx_lm_r10 (void)
{
FX_LM(10);
}
static void fx_lm_r11 (void)
{
FX_LM(11);
}
static void fx_lm_r12 (void)
{
FX_LM(12);
}
static void fx_lm_r13 (void)
{
FX_LM(13);
}
static void fx_lm_r14 (void)
{
FX_LM(14);
READR14;
}
static void fx_lm_r15 (void)
{
FX_LM(15);
}
// f0-ff (ALT2) - sm (xx), rn - store word in RAM
// XXX: If rn == r15, is the value of r15 before or after the extra bytes are read ?
#define FX_SM(reg) \
uint32 v = GSU.avReg[reg]; \
GSU.vLastRamAdr = PIPE; \
R15++; \
FETCHPIPE; \
R15++; \
GSU.vLastRamAdr |= USEX8(PIPE) << 8; \
FETCHPIPE; \
RAM(GSU.vLastRamAdr) = (uint8) v; \
RAM(GSU.vLastRamAdr ^ 1) = (uint8) (v >> 8); \
CLRFLAGS; \
R15++
static void fx_sm_r0 (void)
{
FX_SM(0);
}
static void fx_sm_r1 (void)
{
FX_SM(1);
}
static void fx_sm_r2 (void)
{
FX_SM(2);
}
static void fx_sm_r3 (void)
{
FX_SM(3);
}
static void fx_sm_r4 (void)
{
FX_SM(4);
}
static void fx_sm_r5 (void)
{
FX_SM(5);
}
static void fx_sm_r6 (void)
{
FX_SM(6);
}
static void fx_sm_r7 (void)
{
FX_SM(7);
}
static void fx_sm_r8 (void)
{
FX_SM(8);
}
static void fx_sm_r9 (void)
{
FX_SM(9);
}
static void fx_sm_r10 (void)
{
FX_SM(10);
}
static void fx_sm_r11 (void)
{
FX_SM(11);
}
static void fx_sm_r12 (void)
{
FX_SM(12);
}
static void fx_sm_r13 (void)
{
FX_SM(13);
}
static void fx_sm_r14 (void)
{
FX_SM(14);
}
static void fx_sm_r15 (void)
{
FX_SM(15);
}
// GSU executions functions
uint32 fx_run (uint32 nInstructions)
{
GSU.vCounter = nInstructions;
READR14;
while (TF(G) && (GSU.vCounter-- > 0))
FX_STEP;
#if 0
#ifndef FX_ADDRESS_CHECK
GSU.vPipeAdr = USEX16(R15 - 1) | (USEX8(GSU.vPrgBankReg) << 16);
#endif
#endif
return (nInstructions - GSU.vInstCount);
}
/*
uint32 fx_run_to_breakpoint (uint32 nInstructions)
{
uint32 vCounter = 0;
while (TF(G) && vCounter < nInstructions)
{
vCounter++;
FX_STEP;
if (USEX16(R15) == GSU.vBreakPoint)
{
GSU.vErrorCode = FX_BREAKPOINT;
break;
}
}
#if 0
#ifndef FX_ADDRESS_CHECK
GSU.vPipeAdr = USEX16(R15 - 1) | (USEX8(GSU.vPrgBankReg) << 16);
#endif
#endif
return (vCounter);
}
*/
/*
uint32 fx_step_over (uint32 nInstructions)
{
uint32 vCounter = 0;
while (TF(G) && vCounter < nInstructions)
{
vCounter++;
FX_STEP;
if (USEX16(R15) == GSU.vBreakPoint)
{
GSU.vErrorCode = FX_BREAKPOINT;
break;
}
if (USEX16(R15) == GSU.vStepPoint)
break;
}
#if 0
#ifndef FX_ADDRESS_CHECK
GSU.vPipeAdr = USEX16(R15 - 1) | (USEX8(GSU.vPrgBankReg) << 16);
#endif
#endif
return (vCounter);
}
*/
// Special table for the different plot configurations
void (*fx_PlotTable[]) (void) =
{
&fx_plot_2bit, &fx_plot_4bit, &fx_plot_4bit, &fx_plot_8bit, &fx_plot_obj,
&fx_rpix_2bit, &fx_rpix_4bit, &fx_rpix_4bit, &fx_rpix_8bit, &fx_rpix_obj
};
// Opcode table
void (*fx_OpcodeTable[]) (void) =
{
// ALT0 Table
// 00 - 0f
&fx_stop, &fx_nop, &fx_cache, &fx_lsr, &fx_rol, &fx_bra, &fx_bge, &fx_blt,
&fx_bne, &fx_beq, &fx_bpl, &fx_bmi, &fx_bcc, &fx_bcs, &fx_bvc, &fx_bvs,
// 10 - 1f
&fx_to_r0, &fx_to_r1, &fx_to_r2, &fx_to_r3, &fx_to_r4, &fx_to_r5, &fx_to_r6, &fx_to_r7,
&fx_to_r8, &fx_to_r9, &fx_to_r10, &fx_to_r11, &fx_to_r12, &fx_to_r13, &fx_to_r14, &fx_to_r15,
// 20 - 2f
&fx_with_r0, &fx_with_r1, &fx_with_r2, &fx_with_r3, &fx_with_r4, &fx_with_r5, &fx_with_r6, &fx_with_r7,
&fx_with_r8, &fx_with_r9, &fx_with_r10, &fx_with_r11, &fx_with_r12, &fx_with_r13, &fx_with_r14, &fx_with_r15,
// 30 - 3f
&fx_stw_r0, &fx_stw_r1, &fx_stw_r2, &fx_stw_r3, &fx_stw_r4, &fx_stw_r5, &fx_stw_r6, &fx_stw_r7,
&fx_stw_r8, &fx_stw_r9, &fx_stw_r10, &fx_stw_r11, &fx_loop, &fx_alt1, &fx_alt2, &fx_alt3,
// 40 - 4f
&fx_ldw_r0, &fx_ldw_r1, &fx_ldw_r2, &fx_ldw_r3, &fx_ldw_r4, &fx_ldw_r5, &fx_ldw_r6, &fx_ldw_r7,
&fx_ldw_r8, &fx_ldw_r9, &fx_ldw_r10, &fx_ldw_r11, &fx_plot_2bit, &fx_swap, &fx_color, &fx_not,
// 50 - 5f
&fx_add_r0, &fx_add_r1, &fx_add_r2, &fx_add_r3, &fx_add_r4, &fx_add_r5, &fx_add_r6, &fx_add_r7,
&fx_add_r8, &fx_add_r9, &fx_add_r10, &fx_add_r11, &fx_add_r12, &fx_add_r13, &fx_add_r14, &fx_add_r15,
// 60 - 6f
&fx_sub_r0, &fx_sub_r1, &fx_sub_r2, &fx_sub_r3, &fx_sub_r4, &fx_sub_r5, &fx_sub_r6, &fx_sub_r7,
&fx_sub_r8, &fx_sub_r9, &fx_sub_r10, &fx_sub_r11, &fx_sub_r12, &fx_sub_r13, &fx_sub_r14, &fx_sub_r15,
// 70 - 7f
&fx_merge, &fx_and_r1, &fx_and_r2, &fx_and_r3, &fx_and_r4, &fx_and_r5, &fx_and_r6, &fx_and_r7,
&fx_and_r8, &fx_and_r9, &fx_and_r10, &fx_and_r11, &fx_and_r12, &fx_and_r13, &fx_and_r14, &fx_and_r15,
// 80 - 8f
&fx_mult_r0, &fx_mult_r1, &fx_mult_r2, &fx_mult_r3, &fx_mult_r4, &fx_mult_r5, &fx_mult_r6, &fx_mult_r7,
&fx_mult_r8, &fx_mult_r9, &fx_mult_r10, &fx_mult_r11, &fx_mult_r12, &fx_mult_r13, &fx_mult_r14, &fx_mult_r15,
// 90 - 9f
&fx_sbk, &fx_link_i1, &fx_link_i2, &fx_link_i3, &fx_link_i4, &fx_sex, &fx_asr, &fx_ror,
&fx_jmp_r8, &fx_jmp_r9, &fx_jmp_r10, &fx_jmp_r11, &fx_jmp_r12, &fx_jmp_r13, &fx_lob, &fx_fmult,
// a0 - af
&fx_ibt_r0, &fx_ibt_r1, &fx_ibt_r2, &fx_ibt_r3, &fx_ibt_r4, &fx_ibt_r5, &fx_ibt_r6, &fx_ibt_r7,
&fx_ibt_r8, &fx_ibt_r9, &fx_ibt_r10, &fx_ibt_r11, &fx_ibt_r12, &fx_ibt_r13, &fx_ibt_r14, &fx_ibt_r15,
// b0 - bf
&fx_from_r0, &fx_from_r1, &fx_from_r2, &fx_from_r3, &fx_from_r4, &fx_from_r5, &fx_from_r6, &fx_from_r7,
&fx_from_r8, &fx_from_r9, &fx_from_r10, &fx_from_r11, &fx_from_r12, &fx_from_r13, &fx_from_r14, &fx_from_r15,
// c0 - cf
&fx_hib, &fx_or_r1, &fx_or_r2, &fx_or_r3, &fx_or_r4, &fx_or_r5, &fx_or_r6, &fx_or_r7,
&fx_or_r8, &fx_or_r9, &fx_or_r10, &fx_or_r11, &fx_or_r12, &fx_or_r13, &fx_or_r14, &fx_or_r15,
// d0 - df
&fx_inc_r0, &fx_inc_r1, &fx_inc_r2, &fx_inc_r3, &fx_inc_r4, &fx_inc_r5, &fx_inc_r6, &fx_inc_r7,
&fx_inc_r8, &fx_inc_r9, &fx_inc_r10, &fx_inc_r11, &fx_inc_r12, &fx_inc_r13, &fx_inc_r14, &fx_getc,
// e0 - ef
&fx_dec_r0, &fx_dec_r1, &fx_dec_r2, &fx_dec_r3, &fx_dec_r4, &fx_dec_r5, &fx_dec_r6, &fx_dec_r7,
&fx_dec_r8, &fx_dec_r9, &fx_dec_r10, &fx_dec_r11, &fx_dec_r12, &fx_dec_r13, &fx_dec_r14, &fx_getb,
// f0 - ff
&fx_iwt_r0, &fx_iwt_r1, &fx_iwt_r2, &fx_iwt_r3, &fx_iwt_r4, &fx_iwt_r5, &fx_iwt_r6, &fx_iwt_r7,
&fx_iwt_r8, &fx_iwt_r9, &fx_iwt_r10, &fx_iwt_r11, &fx_iwt_r12, &fx_iwt_r13, &fx_iwt_r14, &fx_iwt_r15,
// ALT1 Table
// 00 - 0f
&fx_stop, &fx_nop, &fx_cache, &fx_lsr, &fx_rol, &fx_bra, &fx_bge, &fx_blt,
&fx_bne, &fx_beq, &fx_bpl, &fx_bmi, &fx_bcc, &fx_bcs, &fx_bvc, &fx_bvs,
// 10 - 1f
&fx_to_r0, &fx_to_r1, &fx_to_r2, &fx_to_r3, &fx_to_r4, &fx_to_r5, &fx_to_r6, &fx_to_r7,
&fx_to_r8, &fx_to_r9, &fx_to_r10, &fx_to_r11, &fx_to_r12, &fx_to_r13, &fx_to_r14, &fx_to_r15,
// 20 - 2f
&fx_with_r0, &fx_with_r1, &fx_with_r2, &fx_with_r3, &fx_with_r4, &fx_with_r5, &fx_with_r6, &fx_with_r7,
&fx_with_r8, &fx_with_r9, &fx_with_r10, &fx_with_r11, &fx_with_r12, &fx_with_r13, &fx_with_r14, &fx_with_r15,
// 30 - 3f
&fx_stb_r0, &fx_stb_r1, &fx_stb_r2, &fx_stb_r3, &fx_stb_r4, &fx_stb_r5, &fx_stb_r6, &fx_stb_r7,
&fx_stb_r8, &fx_stb_r9, &fx_stb_r10, &fx_stb_r11, &fx_loop, &fx_alt1, &fx_alt2, &fx_alt3,
// 40 - 4f
&fx_ldb_r0, &fx_ldb_r1, &fx_ldb_r2, &fx_ldb_r3, &fx_ldb_r4, &fx_ldb_r5, &fx_ldb_r6, &fx_ldb_r7,
&fx_ldb_r8, &fx_ldb_r9, &fx_ldb_r10, &fx_ldb_r11, &fx_rpix_2bit, &fx_swap, &fx_cmode, &fx_not,
// 50 - 5f
&fx_adc_r0, &fx_adc_r1, &fx_adc_r2, &fx_adc_r3, &fx_adc_r4, &fx_adc_r5, &fx_adc_r6, &fx_adc_r7,
&fx_adc_r8, &fx_adc_r9, &fx_adc_r10, &fx_adc_r11, &fx_adc_r12, &fx_adc_r13, &fx_adc_r14, &fx_adc_r15,
// 60 - 6f
&fx_sbc_r0, &fx_sbc_r1, &fx_sbc_r2, &fx_sbc_r3, &fx_sbc_r4, &fx_sbc_r5, &fx_sbc_r6, &fx_sbc_r7,
&fx_sbc_r8, &fx_sbc_r9, &fx_sbc_r10, &fx_sbc_r11, &fx_sbc_r12, &fx_sbc_r13, &fx_sbc_r14, &fx_sbc_r15,
// 70 - 7f
&fx_merge, &fx_bic_r1, &fx_bic_r2, &fx_bic_r3, &fx_bic_r4, &fx_bic_r5, &fx_bic_r6, &fx_bic_r7,
&fx_bic_r8, &fx_bic_r9, &fx_bic_r10, &fx_bic_r11, &fx_bic_r12, &fx_bic_r13, &fx_bic_r14, &fx_bic_r15,
// 80 - 8f
&fx_umult_r0, &fx_umult_r1, &fx_umult_r2, &fx_umult_r3, &fx_umult_r4, &fx_umult_r5, &fx_umult_r6, &fx_umult_r7,
&fx_umult_r8, &fx_umult_r9, &fx_umult_r10, &fx_umult_r11, &fx_umult_r12, &fx_umult_r13, &fx_umult_r14, &fx_umult_r15,
// 90 - 9f
&fx_sbk, &fx_link_i1, &fx_link_i2, &fx_link_i3, &fx_link_i4, &fx_sex, &fx_div2, &fx_ror,
&fx_ljmp_r8, &fx_ljmp_r9, &fx_ljmp_r10, &fx_ljmp_r11, &fx_ljmp_r12, &fx_ljmp_r13, &fx_lob, &fx_lmult,
// a0 - af
&fx_lms_r0, &fx_lms_r1, &fx_lms_r2, &fx_lms_r3, &fx_lms_r4, &fx_lms_r5, &fx_lms_r6, &fx_lms_r7,
&fx_lms_r8, &fx_lms_r9, &fx_lms_r10, &fx_lms_r11, &fx_lms_r12, &fx_lms_r13, &fx_lms_r14, &fx_lms_r15,
// b0 - bf
&fx_from_r0, &fx_from_r1, &fx_from_r2, &fx_from_r3, &fx_from_r4, &fx_from_r5, &fx_from_r6, &fx_from_r7,
&fx_from_r8, &fx_from_r9, &fx_from_r10, &fx_from_r11, &fx_from_r12, &fx_from_r13, &fx_from_r14, &fx_from_r15,
// c0 - cf
&fx_hib, &fx_xor_r1, &fx_xor_r2, &fx_xor_r3, &fx_xor_r4, &fx_xor_r5, &fx_xor_r6, &fx_xor_r7,
&fx_xor_r8, &fx_xor_r9, &fx_xor_r10, &fx_xor_r11, &fx_xor_r12, &fx_xor_r13, &fx_xor_r14, &fx_xor_r15,
// d0 - df
&fx_inc_r0, &fx_inc_r1, &fx_inc_r2, &fx_inc_r3, &fx_inc_r4, &fx_inc_r5, &fx_inc_r6, &fx_inc_r7,
&fx_inc_r8, &fx_inc_r9, &fx_inc_r10, &fx_inc_r11, &fx_inc_r12, &fx_inc_r13, &fx_inc_r14, &fx_getc,
// e0 - ef
&fx_dec_r0, &fx_dec_r1, &fx_dec_r2, &fx_dec_r3, &fx_dec_r4, &fx_dec_r5, &fx_dec_r6, &fx_dec_r7,
&fx_dec_r8, &fx_dec_r9, &fx_dec_r10, &fx_dec_r11, &fx_dec_r12, &fx_dec_r13, &fx_dec_r14, &fx_getbh,
// f0 - ff
&fx_lm_r0, &fx_lm_r1, &fx_lm_r2, &fx_lm_r3, &fx_lm_r4, &fx_lm_r5, &fx_lm_r6, &fx_lm_r7,
&fx_lm_r8, &fx_lm_r9, &fx_lm_r10, &fx_lm_r11, &fx_lm_r12, &fx_lm_r13, &fx_lm_r14, &fx_lm_r15,
// ALT2 Table
// 00 - 0f
&fx_stop, &fx_nop, &fx_cache, &fx_lsr, &fx_rol, &fx_bra, &fx_bge, &fx_blt,
&fx_bne, &fx_beq, &fx_bpl, &fx_bmi, &fx_bcc, &fx_bcs, &fx_bvc, &fx_bvs,
// 10 - 1f
&fx_to_r0, &fx_to_r1, &fx_to_r2, &fx_to_r3, &fx_to_r4, &fx_to_r5, &fx_to_r6, &fx_to_r7,
&fx_to_r8, &fx_to_r9, &fx_to_r10, &fx_to_r11, &fx_to_r12, &fx_to_r13, &fx_to_r14, &fx_to_r15,
// 20 - 2f
&fx_with_r0, &fx_with_r1, &fx_with_r2, &fx_with_r3, &fx_with_r4, &fx_with_r5, &fx_with_r6, &fx_with_r7,
&fx_with_r8, &fx_with_r9, &fx_with_r10, &fx_with_r11, &fx_with_r12, &fx_with_r13, &fx_with_r14, &fx_with_r15,
// 30 - 3f
&fx_stw_r0, &fx_stw_r1, &fx_stw_r2, &fx_stw_r3, &fx_stw_r4, &fx_stw_r5, &fx_stw_r6, &fx_stw_r7,
&fx_stw_r8, &fx_stw_r9, &fx_stw_r10, &fx_stw_r11, &fx_loop, &fx_alt1, &fx_alt2, &fx_alt3,
// 40 - 4f
&fx_ldw_r0, &fx_ldw_r1, &fx_ldw_r2, &fx_ldw_r3, &fx_ldw_r4, &fx_ldw_r5, &fx_ldw_r6, &fx_ldw_r7,
&fx_ldw_r8, &fx_ldw_r9, &fx_ldw_r10, &fx_ldw_r11, &fx_plot_2bit, &fx_swap, &fx_color, &fx_not,
// 50 - 5f
&fx_add_i0, &fx_add_i1, &fx_add_i2, &fx_add_i3, &fx_add_i4, &fx_add_i5, &fx_add_i6, &fx_add_i7,
&fx_add_i8, &fx_add_i9, &fx_add_i10, &fx_add_i11, &fx_add_i12, &fx_add_i13, &fx_add_i14, &fx_add_i15,
// 60 - 6f
&fx_sub_i0, &fx_sub_i1, &fx_sub_i2, &fx_sub_i3, &fx_sub_i4, &fx_sub_i5, &fx_sub_i6, &fx_sub_i7,
&fx_sub_i8, &fx_sub_i9, &fx_sub_i10, &fx_sub_i11, &fx_sub_i12, &fx_sub_i13, &fx_sub_i14, &fx_sub_i15,
// 70 - 7f
&fx_merge, &fx_and_i1, &fx_and_i2, &fx_and_i3, &fx_and_i4, &fx_and_i5, &fx_and_i6, &fx_and_i7,
&fx_and_i8, &fx_and_i9, &fx_and_i10, &fx_and_i11, &fx_and_i12, &fx_and_i13, &fx_and_i14, &fx_and_i15,
// 80 - 8f
&fx_mult_i0, &fx_mult_i1, &fx_mult_i2, &fx_mult_i3, &fx_mult_i4, &fx_mult_i5, &fx_mult_i6, &fx_mult_i7,
&fx_mult_i8, &fx_mult_i9, &fx_mult_i10, &fx_mult_i11, &fx_mult_i12, &fx_mult_i13, &fx_mult_i14, &fx_mult_i15,
// 90 - 9f
&fx_sbk, &fx_link_i1, &fx_link_i2, &fx_link_i3, &fx_link_i4, &fx_sex, &fx_asr, &fx_ror,
&fx_jmp_r8, &fx_jmp_r9, &fx_jmp_r10, &fx_jmp_r11, &fx_jmp_r12, &fx_jmp_r13, &fx_lob, &fx_fmult,
// a0 - af
&fx_sms_r0, &fx_sms_r1, &fx_sms_r2, &fx_sms_r3, &fx_sms_r4, &fx_sms_r5, &fx_sms_r6, &fx_sms_r7,
&fx_sms_r8, &fx_sms_r9, &fx_sms_r10, &fx_sms_r11, &fx_sms_r12, &fx_sms_r13, &fx_sms_r14, &fx_sms_r15,
// b0 - bf
&fx_from_r0, &fx_from_r1, &fx_from_r2, &fx_from_r3, &fx_from_r4, &fx_from_r5, &fx_from_r6, &fx_from_r7,
&fx_from_r8, &fx_from_r9, &fx_from_r10, &fx_from_r11, &fx_from_r12, &fx_from_r13, &fx_from_r14, &fx_from_r15,
// c0 - cf
&fx_hib, &fx_or_i1, &fx_or_i2, &fx_or_i3, &fx_or_i4, &fx_or_i5, &fx_or_i6, &fx_or_i7,
&fx_or_i8, &fx_or_i9, &fx_or_i10, &fx_or_i11, &fx_or_i12, &fx_or_i13, &fx_or_i14, &fx_or_i15,
// d0 - df
&fx_inc_r0, &fx_inc_r1, &fx_inc_r2, &fx_inc_r3, &fx_inc_r4, &fx_inc_r5, &fx_inc_r6, &fx_inc_r7,
&fx_inc_r8, &fx_inc_r9, &fx_inc_r10, &fx_inc_r11, &fx_inc_r12, &fx_inc_r13, &fx_inc_r14, &fx_ramb,
// e0 - ef
&fx_dec_r0, &fx_dec_r1, &fx_dec_r2, &fx_dec_r3, &fx_dec_r4, &fx_dec_r5, &fx_dec_r6, &fx_dec_r7,
&fx_dec_r8, &fx_dec_r9, &fx_dec_r10, &fx_dec_r11, &fx_dec_r12, &fx_dec_r13, &fx_dec_r14, &fx_getbl,
// f0 - ff
&fx_sm_r0, &fx_sm_r1, &fx_sm_r2, &fx_sm_r3, &fx_sm_r4, &fx_sm_r5, &fx_sm_r6, &fx_sm_r7,
&fx_sm_r8, &fx_sm_r9, &fx_sm_r10, &fx_sm_r11, &fx_sm_r12, &fx_sm_r13, &fx_sm_r14, &fx_sm_r15,
// ALT3 Table
// 00 - 0f
&fx_stop, &fx_nop, &fx_cache, &fx_lsr, &fx_rol, &fx_bra, &fx_bge, &fx_blt,
&fx_bne, &fx_beq, &fx_bpl, &fx_bmi, &fx_bcc, &fx_bcs, &fx_bvc, &fx_bvs,
// 10 - 1f
&fx_to_r0, &fx_to_r1, &fx_to_r2, &fx_to_r3, &fx_to_r4, &fx_to_r5, &fx_to_r6, &fx_to_r7,
&fx_to_r8, &fx_to_r9, &fx_to_r10, &fx_to_r11, &fx_to_r12, &fx_to_r13, &fx_to_r14, &fx_to_r15,
// 20 - 2f
&fx_with_r0, &fx_with_r1, &fx_with_r2, &fx_with_r3, &fx_with_r4, &fx_with_r5, &fx_with_r6, &fx_with_r7,
&fx_with_r8, &fx_with_r9, &fx_with_r10, &fx_with_r11, &fx_with_r12, &fx_with_r13, &fx_with_r14, &fx_with_r15,
// 30 - 3f
&fx_stb_r0, &fx_stb_r1, &fx_stb_r2, &fx_stb_r3, &fx_stb_r4, &fx_stb_r5, &fx_stb_r6, &fx_stb_r7,
&fx_stb_r8, &fx_stb_r9, &fx_stb_r10, &fx_stb_r11, &fx_loop, &fx_alt1, &fx_alt2, &fx_alt3,
// 40 - 4f
&fx_ldb_r0, &fx_ldb_r1, &fx_ldb_r2, &fx_ldb_r3, &fx_ldb_r4, &fx_ldb_r5, &fx_ldb_r6, &fx_ldb_r7,
&fx_ldb_r8, &fx_ldb_r9, &fx_ldb_r10, &fx_ldb_r11, &fx_rpix_2bit, &fx_swap, &fx_cmode, &fx_not,
// 50 - 5f
&fx_adc_i0, &fx_adc_i1, &fx_adc_i2, &fx_adc_i3, &fx_adc_i4, &fx_adc_i5, &fx_adc_i6, &fx_adc_i7,
&fx_adc_i8, &fx_adc_i9, &fx_adc_i10, &fx_adc_i11, &fx_adc_i12, &fx_adc_i13, &fx_adc_i14, &fx_adc_i15,
// 60 - 6f
&fx_cmp_r0, &fx_cmp_r1, &fx_cmp_r2, &fx_cmp_r3, &fx_cmp_r4, &fx_cmp_r5, &fx_cmp_r6, &fx_cmp_r7,
&fx_cmp_r8, &fx_cmp_r9, &fx_cmp_r10, &fx_cmp_r11, &fx_cmp_r12, &fx_cmp_r13, &fx_cmp_r14, &fx_cmp_r15,
// 70 - 7f
&fx_merge, &fx_bic_i1, &fx_bic_i2, &fx_bic_i3, &fx_bic_i4, &fx_bic_i5, &fx_bic_i6, &fx_bic_i7,
&fx_bic_i8, &fx_bic_i9, &fx_bic_i10, &fx_bic_i11, &fx_bic_i12, &fx_bic_i13, &fx_bic_i14, &fx_bic_i15,
// 80 - 8f
&fx_umult_i0, &fx_umult_i1, &fx_umult_i2, &fx_umult_i3, &fx_umult_i4, &fx_umult_i5, &fx_umult_i6, &fx_umult_i7,
&fx_umult_i8, &fx_umult_i9, &fx_umult_i10, &fx_umult_i11, &fx_umult_i12, &fx_umult_i13, &fx_umult_i14, &fx_umult_i15,
// 90 - 9f
&fx_sbk, &fx_link_i1, &fx_link_i2, &fx_link_i3, &fx_link_i4, &fx_sex, &fx_div2, &fx_ror,
&fx_ljmp_r8, &fx_ljmp_r9, &fx_ljmp_r10, &fx_ljmp_r11, &fx_ljmp_r12, &fx_ljmp_r13, &fx_lob, &fx_lmult,
// a0 - af
&fx_lms_r0, &fx_lms_r1, &fx_lms_r2, &fx_lms_r3, &fx_lms_r4, &fx_lms_r5, &fx_lms_r6, &fx_lms_r7,
&fx_lms_r8, &fx_lms_r9, &fx_lms_r10, &fx_lms_r11, &fx_lms_r12, &fx_lms_r13, &fx_lms_r14, &fx_lms_r15,
// b0 - bf
&fx_from_r0, &fx_from_r1, &fx_from_r2, &fx_from_r3, &fx_from_r4, &fx_from_r5, &fx_from_r6, &fx_from_r7,
&fx_from_r8, &fx_from_r9, &fx_from_r10, &fx_from_r11, &fx_from_r12, &fx_from_r13, &fx_from_r14, &fx_from_r15,
// c0 - cf
&fx_hib, &fx_xor_i1, &fx_xor_i2, &fx_xor_i3, &fx_xor_i4, &fx_xor_i5, &fx_xor_i6, &fx_xor_i7,
&fx_xor_i8, &fx_xor_i9, &fx_xor_i10, &fx_xor_i11, &fx_xor_i12, &fx_xor_i13, &fx_xor_i14, &fx_xor_i15,
// d0 - df
&fx_inc_r0, &fx_inc_r1, &fx_inc_r2, &fx_inc_r3, &fx_inc_r4, &fx_inc_r5, &fx_inc_r6, &fx_inc_r7,
&fx_inc_r8, &fx_inc_r9, &fx_inc_r10, &fx_inc_r11, &fx_inc_r12, &fx_inc_r13, &fx_inc_r14, &fx_romb,
// e0 - ef
&fx_dec_r0, &fx_dec_r1, &fx_dec_r2, &fx_dec_r3, &fx_dec_r4, &fx_dec_r5, &fx_dec_r6, &fx_dec_r7,
&fx_dec_r8, &fx_dec_r9, &fx_dec_r10, &fx_dec_r11, &fx_dec_r12, &fx_dec_r13, &fx_dec_r14, &fx_getbs,
// f0 - ff
&fx_lm_r0, &fx_lm_r1, &fx_lm_r2, &fx_lm_r3, &fx_lm_r4, &fx_lm_r5, &fx_lm_r6, &fx_lm_r7,
&fx_lm_r8, &fx_lm_r9, &fx_lm_r10, &fx_lm_r11, &fx_lm_r12, &fx_lm_r13, &fx_lm_r14, &fx_lm_r15
};