blob: 466611effb48c9163b1cdd6ee0efd99c048ee9b7 [file] [log] [blame] [raw]
/* m68k_cpu.c: 68k-CPU simulator
Copyright (c) 2009-2010 Holger Veit
Permission is hereby granted, free of charge, to any person obtaining a
copy of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the
Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
Holger Veit BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Except as contained in this notice, the name of Holger Veit et al shall not be
used in advertising or otherwise to promote the sale, use or other dealings
in this Software without prior written authorization from Holger Veit et al.
04-Oct-09 HV Initial version
25-Apr-10 HV Fixed LSR.W and ROXR.B instructions
26-Jun-10 HV Incomplete decoding of BCHG d,d instruction
15-Jul-10 HV IRQ logic loses lower prio interrupts
17-Jul-10 HV Implement Call/Exit Tracing with symbol table lookup
17-Jul-10 HV Mustn't grant interrupt at level == IPL
18-Jul-10 HV Broken address calculation for AIDX and EA_W_RMW, wonder why this didn't pop up earlier.
20-Jul-10 HV Corrected ADDQ.W/SUBQ.W for EA_ADIR, EOR.[WL]
23-Jul-10 HV Broken C code sequence in lsl.l
23-Jul-10 HV RTE didn't set/reset S bit
*/
#include "m68k_cpu.h"
#include <ctype.h>
#if defined(_WIN32)
#include <windows.h>
#else
#include <unistd.h>
#endif
/* status reg flags */
#define FLAG_C 0x0001
#define FLAG_V 0x0002
#define FLAG_Z 0x0004
#define FLAG_N 0x0008
#define FLAG_X 0x0010
#define FLAG_I0 0x0100
#define FLAG_I1 0x0200
#define FLAG_I2 0x0400
#define FLAG_IPL_MASK (FLAG_I0|FLAG_I1|FLAG_I2)
#define FLAG_S 0x2000
#define FLAG_T 0x8000
#define FLAG_T1 FLAG_T
#define FLAG_T0 0x4000
#define BIT7 0x80
#define BIT8 0x100
#define BIT15 0x8000
#define BIT16 0x10000
#define BIT31 0x80000000
#define BIT32 0x100000000LL
#define MASK_0(x) ((x) & 1)
#define MASK_8U(x) ((x) & 0xffffff00)
#define MASK_8L(x) ((x) & 0x000000ff)
#define MASK_8SGN(x) ((x) & BIT7)
#define MASK_9(x) ((x) & BIT8)
#define MASK_16U(x) ((x) & 0xffff0000)
#define MASK_16L(x) ((x) & 0x0000ffff)
#define MASK_16SGN(x) ((x) & BIT15)
#define MASK_17(x) ((x) & BIT16)
#define MASK_32U(x) (0)
#define MASK_32L(x) ((x) & 0xffffffff)
#define MASK_32SGN(x) ((x) & BIT31)
#define MASK_33(x64) ((x64) & BIT32)
#define COMBINE8(tgt,src) (MASK_8U(tgt) | MASK_8L(src))
#define COMBINE16(tgt,src) (MASK_16U(tgt) | MASK_16L(src))
#define COMBINE32(tgt,src) MASK_32L(src)
extern t_addr addrmask;
static t_addr addrmasks[] = {
0x00ffffff, /*68000*/
0x000fffff, /*68008*/
0x00ffffff, /*68010*/
0xffffffff, /*68020*/
0xffffffff /*68030*/
};
int16 cputype = CPU_TYPE_68000 >> UNIT_CPU_V_TYPE;
/* CPU data structures
* m68kcpu_dev CPU device descriptor
* m68kcpu_unit CPU unit descriptor
* m68kcpu_reg CPU register list
* m68kcpu_mod CPU modifiers list
*/
UNIT *m68kcpu_unit; /* must be set elsewhere */
DEVICE *m68kcpu_dev; /* must be set elsewhere */
void (*m68kcpu_trapcallback)(DEVICE* dptr,int trapnum) = 0;
/* register set */
int32 DR[8];
#define D0 DR[0]
#define D1 DR[1]
#define D2 DR[2]
#define D3 DR[3]
#define D4 DR[4]
#define D5 DR[5]
#define D6 DR[6]
#define D7 DR[7]
t_addr AR[8];
#define A0 AR[0]
#define A1 AR[1]
#define A2 AR[2]
#define A3 AR[3]
#define A4 AR[4]
#define A5 AR[5]
#define A6 AR[6]
#define A7 AR[7]
t_addr USP;
t_addr *cur_sp;
uint16 SR;
#define CCR_C (SR & FLAG_C)
#define CCR_V (SR & FLAG_V)
#define CCR_Z (SR & FLAG_Z)
#define CCR_N (SR & FLAG_N)
#define CCR_X (SR & FLAG_X)
#define SR_IPL ((SR & FLAG_IPL_MASK)>>8)
#define SR_S (SR & FLAG_S)
#define SR_T (SR & FLAG_T)
#define SR_T0 (SR & FLAG_T0)
#define SR_T1 (SR & FLAG_T1)
#define ONEF(flag) SR |= (flag)
#define CLRF(flag) SR &= ~(flag)
#define SETF(cond,flag) if (cond) SR |= (flag); else SR &= ~(flag)
#define SETZ8(cond) if (MASK_8L(cond)) SR &= ~FLAG_Z; else SR |= FLAG_Z
#define SETZ16(cond) if (MASK_16L(cond)) SR &= ~FLAG_Z; else SR |= FLAG_Z
#define SETZ32(cond) if (MASK_32L(cond)) SR &= ~FLAG_Z; else SR |= FLAG_Z
#define SETNZ8(cond) SETZ8(cond); if (MASK_8SGN(cond)) SR |= FLAG_N; else SR &= ~FLAG_N
#define SETNZ16(cond) SETZ16(cond); if (MASK_16SGN(cond)) SR |= FLAG_N; else SR &= ~FLAG_N
#define SETNZ32(cond) SETZ32(cond); if (MASK_32SGN(cond)) SR |= FLAG_N; else SR &= ~FLAG_N
#define SETV_ADD8(a1,a2,r) SETF(MASK_8SGN(((a1)^(r))&((a2)^(r))),FLAG_V);
#define SETV_ADD16(a1,a2,r) SETF(MASK_16SGN(((a1)^(r))&((a2)^(r))),FLAG_V);
#define SETV_ADD32(a1,a2,r) SETF(MASK_32SGN(((a1)^(r))&((a2)^(r))),FLAG_V);
#define SETV_SUB8(s,d,r) SETF(MASK_8SGN(((s)^(d))&((r)^(d))),FLAG_V)
#define SETV_SUB16(s,d,r) SETF(MASK_16SGN(((s)^(d))&((r)^(d))),FLAG_V)
#define SETV_SUB32(s,d,r) SETF(MASK_32SGN(((s)^(d))&((r)^(d))),FLAG_V)
#define ASSERT_PRIV() if (!SR_S) { rc = STOP_PRVIO; break; }
#define ASSERT_OK(func) if ((rc=(func)) != SCPE_OK) break
#define ASSERT_OKRET(func) if ((rc=(func)) != SCPE_OK) return rc
#define AREG(r) (r==7 ? cur_sp : &AR[r])
uint16 SFC;
uint16 DFC;
uint32 VBR;
t_addr saved_PC;
static t_bool intpending;
static int m68k_sublevel;
REG m68kcpu_reg[] = {
{ HRDATA (D0, DR[0], 32) },
{ HRDATA (D1, DR[1], 32) },
{ HRDATA (D2, DR[2], 32) },
{ HRDATA (D3, DR[3], 32) },
{ HRDATA (D4, DR[4], 32) },
{ HRDATA (D5, DR[5], 32) },
{ HRDATA (D6, DR[6], 32) },
{ HRDATA (D7, DR[7], 32) },
{ HRDATA (A0, AR[0], 32) },
{ HRDATA (A1, AR[1], 32) },
{ HRDATA (A2, AR[2], 32) },
{ HRDATA (A3, AR[3], 32) },
{ HRDATA (A4, AR[4], 32) },
{ HRDATA (A5, AR[5], 32) },
{ HRDATA (A6, AR[6], 32) },
{ HRDATA (A7, AR[7], 32) },
{ HRDATA (SSP, AR[7], 32) },
{ HRDATA (USP, USP, 32) },
{ HRDATA (PC, saved_PC, 32) },
{ HRDATA (SR, SR, 16) },
{ HRDATA (CCR, SR, 8) },
{ FLDATA (C, SR, 0) },
{ FLDATA (V, SR, 1) },
{ FLDATA (Z, SR, 2) },
{ FLDATA (N, SR, 3) },
{ FLDATA (X, SR, 4) },
{ GRDATA (IPL, SR, 8, 3, 8) },
{ FLDATA (S, SR, 13) },
{ FLDATA (T, SR, 15) },
{ HRDATA (SFC, SFC, 3), REG_HIDDEN },
{ HRDATA (DFC, DFC, 3), REG_HIDDEN },
{ HRDATA (VBR, VBR, 32), REG_RO },
{ FLDATA (IRQPEN, intpending, 0), REG_HIDDEN },
{ NULL }
};
DEBTAB m68kcpu_dt[] = {
{ "EXC", DBG_CPU_EXC },
{ "PC", DBG_CPU_PC },
{ "INT", DBG_CPU_INT },
{ "CTRACE", DBG_CPU_CTRACE },
{ "BTRACE", DBG_CPU_BTRACE },
{ NULL, 0 }
};
static const char *condnames[] = {
"RA", "SR", "HI", "LS", "CC", "CS", "NE", "EQ", "VC", "VS", "PL", "MI", "GE", "LT", "GT", "LE"
};
#if 0
/* sample code */
static MTAB m68kcpu_mod[] = {
M68KCPU_STDMOD,
{ 0 }
};
DEVICE m68kcpu_dev = {
"CPU", &m68kcpu_unit, m68kcpu_reg, m68kcpu_mod,
1, 16, 32, 2, 16, 16,
&m68kcpu_ex, &m68kcpu_dep, &m68kcpu_reset,
&m68kcpu_boot, NULL, NULL,
NULL, DEV_DEBUG, 0,
m68kcpu_dt, NULL, NULL
};
#endif
static DEVICE* cpudev_self = 0;
t_stat m68kcpu_peripheral_reset()
{
t_stat rc;
DEVICE** devs = sim_devices;
DEVICE* dptr;
while ((dptr = *devs) != NULL) {
if (dptr != cpudev_self) {
ASSERT_OKRET(dptr->reset(dptr));
}
devs++;
}
return SCPE_OK;
}
/* simple prefetch I cache */
#define CACHE_SIZE 16
#define CACHE_MASK 0x0f
static t_addr cache_pc;
static uint8 cache_line[CACHE_SIZE];
static t_stat ReadICache(t_addr tpc)
{
int i;
t_stat rc;
uint8* mem;
ASSERT_OKRET(Mem((tpc+CACHE_SIZE-1)&addrmask,&mem));
/* 68000/08/10 do not like unaligned access */
if (cputype < 3 && (tpc & 1)) return STOP_ERRADR;
for (i=CACHE_SIZE-1; i>=0; i--) {
cache_line[i] = *mem--;
}
// for (i=0; i<16; i++) printf("icache[%d]=0x%08x\n",i,cache_line[i]);
return SCPE_OK;
}
static t_stat ReadInstr(t_addr pc,uint32* inst)
{
t_stat rc;
t_addr tpc;
IOHANDLER* ioh;
if ((rc=TranslateAddr(pc & ~CACHE_MASK,&tpc,&ioh,MEM_READ,FALSE,FALSE)) != SCPE_OK)
return rc==SIM_ISIO ? STOP_PCIO : rc;
if (tpc != cache_pc) {
ASSERT_OKRET(ReadICache(tpc));
}
pc &= CACHE_MASK;
*inst = (cache_line[pc]<<8) | cache_line[pc+1];
return SCPE_OK;
}
static t_stat ReadInstrInc(t_addr* pc,uint32* inst)
{
t_stat rc;
ASSERT_OKRET(ReadInstr(*pc,inst));
*pc += 2;
return SCPE_OK;
}
static t_stat ReadInstrLongInc(t_addr* pc,uint32* inst)
{
t_stat rc;
uint32 val1,val2;
ASSERT_OKRET(ReadInstr(*pc,&val1));
*pc += 2;
ASSERT_OKRET(ReadInstr(*pc,&val2));
*pc += 2;
*inst = COMBINE16(val1<<16,val2);
return SCPE_OK;
}
void m68k_set_s(t_bool tf)
{
if (tf) {
SR |= FLAG_S;
cur_sp = &A7;
} else {
SR &= ~FLAG_S;
cur_sp = &USP;
}
}
void m68k_setipl(int ipl)
{
// printf("set ipl to %d\n",ipl);
SR &= ~FLAG_IPL_MASK;
SR |= (ipl & 7) << 8;
}
/* interrupt logic */
static int intvectors[8];
static t_stat m68k_irqinit()
{
int i;
for (i=0; i<8; i++) intvectors[i] = 0;
intpending = 0;
return SCPE_OK;
}
t_stat m68k_raise_vectorint(int level,int vector)
{
int mask = 1<<level;
IFDEBUG(DBG_CPU_INT,fprintf(sim_deb,"CPU : [0x%08x] Interrupt: request level=%d, IPL=%d, vec=%d, pending=%x\n",
saved_PC,level,SR_IPL,vector,intpending));
if ((intpending & mask) == 0) {
intvectors[level] = vector;
intpending |= mask;
}
return SCPE_OK;
}
t_stat m68k_raise_autoint(int level)
{
return m68k_raise_vectorint(level,level+24);
}
static void m68k_nocallback(DEVICE* dev,int trapnum)
{
/* do nothing */
}
/* reset and boot */
t_stat m68kcpu_reset(DEVICE* dptr)
{
t_stat rc;
uint32 dummy;
cpudev_self = dptr;
sim_brk_types = SWMASK('E')|SWMASK('R')|SWMASK('W');
sim_brk_dflt = SWMASK('E');
addrmask = addrmasks[cputype];
ASSERT_OKRET(m68k_alloc_mem());
ASSERT_OKRET(m68k_ioinit());
m68kcpu_trapcallback = &m68k_nocallback;
m68k_sublevel = 0;
/* TODO: 68010 VBR */
ReadPL(0,&A7);
ReadPL(4,&saved_PC);
ReadInstr(saved_PC,&dummy); /* fill prefetch cache */
m68k_irqinit(); /* reset interrupt flags */
m68k_set_s(TRUE); /* reset to supervisor mode */
return SCPE_OK;
}
t_stat m68kcpu_boot(int32 unitno,DEVICE* dptr)
{
return dptr->reset(dptr);
}
/* for instruction decoder */
#define IR_1512 (IR&0170000)
#define IR_1109 (IR&0007000)
#define IR_1108 (IR&0007400)
#define IR_1106 (IR&0007700)
#define IR_1103 (IR&0007770)
#define IR_08 (IR&0000400)
#define IR_0806 (IR&0000700)
#define IR_0803 (IR&0000770)
#define IR_0706 (IR&0000300)
#define IR_0703 (IR&0000370)
#define IR_0503 (IR&0000070)
#define IR_080403 (IR&0000430)
#define IR_08060403 (IR&0000730)
#define IR_0200 (IR&0000007)
#define IR_EAMOD (IR&0000070)
#define IR_0503 (IR&0000070)
#define IR_COND (IR&0007400)
#define IR_EA (IR&0000077)
#define IR_EAM12 (IR&0000060)
#define IR_EAREG (IR&0000007)
#define IR_DISP (IR&0000377)
#define IR_EATGT ((IR&0000700)>>3)
#define IR_REGX ((IR&0007000)>>9)
#define IR_REGY (IR&0000007)
#define IR_TRAP (IR&0000017)
#define IR_SIZE ((IR&0000300)>>6)
#define IR_DATA (IR&0000377)
#define IRE_DA (IRE&0100000)
#define IRE_REG ((IRE&0070000)>>12)
#define IRE_WL (IRE&0004000)
#define IRE_DISP (IRE&0000377)
/* EA modes */
#define EA_DDIR 0000
#define EA_ADIR 0010
#define EA_AIND 0020
#define EA_API 0030
#define EA_APD 0040
#define EA_AIDX 0050
#define EA_AXIDX 0060
#define EA_EXT 0070
#define EA_IMM 0074
#define EAX_AW 000
#define EAX_AL 001
#define EAX_PCIDX 002
#define EAX_PCXIDX 003
#define EAX_IMM 004
#define EXTB(x) ((int32)((int8)((x)&0xff)))
#define EXTW(x) ((int32)((int16)((x)&0xffff)))
#define DRX DR[IR_REGX]
#define DRY DR[IR_REGY]
static uint32 quickarg[] = { 8,1,2,3,4,5,6,7 };
static int32 shmask8[] = { 0x00,0x80,0xc0,0xe0,0xf0,0xf8,0xfc,0xfe,0xff };
static int32 shmask16[] = { 0x0000,
0x8000,0xc000,0xe000,0xf000,0xf800,0xfc00,0xfe00,0xff00,
0xff80,0xffc0,0xffe0,0xfff0,0xff80,0xffc0,0xffe0,0xffff,
0xffff };
static int32 shmask32[] = { 0x00000000,
0x80000000,0xc0000000,0xe0000000,0xf0000000,
0xf8000000,0xfc000000,0xfe000000,0xff000000,
0xff800000,0xffc00000,0xffe00000,0xfff00000,
0xfff80000,0xfffc0000,0xfffe0000,0xffff0000,
0xffff8000,0xffffc000,0xffffe000,0xfffff000,
0xfffff800,0xfffffc00,0xfffffe00,0xffffff00,
0xffffff80,0xffffffc0,0xffffffe0,0xfffffff0,
0xfffffff8,0xfffffffc,0xfffffffe,0xffffffff,
0xffffffff };
static int32 bitmask[] = { 0x00000000,
0x00000001,0x00000002,0x00000004,0x00000008,
0x00000010,0x00000020,0x00000040,0x00000080,
0x00000100,0x00000200,0x00000400,0x00000800,
0x00001000,0x00002000,0x00004000,0x00000800,
0x00010000,0x00020000,0x00040000,0x00008000,
0x00100000,0x00200000,0x00400000,0x00080000,
0x01000000,0x02000000,0x04000000,0x00800000,
0x10000000,0x20000000,0x40000000,0x80000000,
0x00000000 };
static t_addr saved_ea;
static t_stat ea_src_b(uint32 eamod,uint32 eareg,uint32* val,t_addr* pc)
{
t_stat rc = SCPE_OK;
uint32 reg, regno, IRE;
t_addr *areg;
// printf("src eamod=%x eareg=%x\n",eamod,eareg);
switch (eamod) {
case EA_DDIR:
*val = MASK_8L(DR[eareg]);
return SCPE_OK;
case EA_ADIR:
*val = MASK_8L(*AREG(eareg));
return SCPE_OK;
case EA_AIND:
return ReadVB(saved_ea = *AREG(eareg),val);
case EA_API:
areg = AREG(eareg);
rc = ReadVB(saved_ea = *areg,val);
*areg += (eareg==7 ? 2 : 1);
return rc;
case EA_APD:
areg = AREG(eareg);
*areg -= (eareg==7 ? 2 : 1);
return ReadVB(saved_ea = *areg,val);
case EA_AIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return ReadVB(saved_ea = *AREG(eareg)+EXTW(IRE),val);
case EA_AXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = IRE_DA ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
return ReadVB(saved_ea = *AREG(eareg) + EXTW(IRE_DISP) + reg, val);
case EA_EXT:
switch (eareg) {
case EAX_AW:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
saved_ea = EXTW(IRE);
rc = ReadVB(saved_ea, val);
return rc;
case EAX_AL:
ASSERT_OKRET(ReadPL(*pc,&IRE));
*pc += 4;
return ReadVB(saved_ea = IRE, val);
case EAX_PCIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return ReadVB(saved_ea = *pc-2 + EXTW(IRE), val);
case EAX_PCXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = (IRE_DA) ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
return ReadVB(saved_ea = *pc-2 + EXTW(IRE_DISP) + reg, val);
case EAX_IMM:
ASSERT_OKRET(ReadInstrInc(pc,val));
*val = MASK_8L(*val);
return SCPE_OK;
default:
return STOP_ERROP;
}
default:
return STOP_ERROP;
}
}
static t_stat ea_src_bs(uint32 eamod,uint32 eareg,uint32* val,t_addr* pc)
{
if (eamod==EA_EXT && eareg==EAX_IMM) {
*val = MASK_8L(SR);
return SCPE_OK;
}
return ea_src_b(eamod,eareg,val,pc);
}
static t_stat ea_src_w(uint32 eamod,uint32 eareg,uint32* val,t_addr* pc)
{
t_stat rc = SCPE_OK;
uint32 reg, regno, IRE;
t_addr *areg;
switch (eamod) {
case EA_DDIR:
*val = MASK_16L(DR[eareg]);
return SCPE_OK;
case EA_ADIR:
*val = MASK_16L(*AREG(eareg));
return SCPE_OK;
case EA_AIND:
return ReadVW(saved_ea = *AREG(eareg), val);
case EA_API:
areg = AREG(eareg);
rc = ReadVW(saved_ea = *areg, val);
*areg += 2;
return rc;
case EA_APD:
areg = AREG(eareg);
*areg -= 2;
return ReadVW(saved_ea = *areg, val);
case EA_AIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return ReadVW(saved_ea = *AREG(eareg) + EXTW(IRE), val);
case EA_AXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = IRE_DA ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
return ReadVW(saved_ea = *AREG(eareg) + EXTW(IRE_DISP) + reg, val);
case EA_EXT:
switch (eareg) {
case EAX_AW:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return ReadVW(saved_ea = EXTW(IRE), val);
case EAX_AL:
ASSERT_OKRET(ReadPL(*pc,&IRE));
*pc += 4;
return ReadVW(saved_ea = IRE, val);
case EAX_PCIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return ReadVW(saved_ea = *pc-2 + EXTW(IRE), val);
case EAX_PCXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = (IRE_DA) ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
return ReadVW(saved_ea = *pc-2 + EXTW(IRE_DISP) + reg, val);
case EAX_IMM:
return ReadInstrInc(pc,val);
default:
return STOP_ERROP;
}
default:
return STOP_ERROP;
}
}
static t_stat ea_src_ws(uint32 eamod,uint32 eareg,uint32* val,t_addr* pc)
{
if (eamod==EA_EXT && eareg==EAX_IMM) {
*val = SR;
return SCPE_OK;
}
return ea_src_w(eamod,eareg,val,pc);
}
/* non dereferencing version of ea_src_l, only accepts ea category control */
static t_stat ea_src_l_nd(uint32 eamod,uint32 eareg,uint32* val,t_addr* pc)
{
t_stat rc = SCPE_OK;
uint32 reg, regno, IRE;
switch (eamod) {
case EA_AIND:
*val = *AREG(eareg);
return SCPE_OK;
case EA_AIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
*val = *AREG(eareg) + EXTW(IRE);
return SCPE_OK;
case EA_AXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = IRE_DA ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
*val = *AREG(eareg) + EXTW(IRE_DISP) + reg;
return SCPE_OK;
case EA_EXT:
switch (eareg) {
case EAX_AW:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
*val = EXTW(IRE);
return SCPE_OK;
case EAX_AL:
ASSERT_OKRET(ReadPL(*pc,val));
*pc += 4;
return SCPE_OK;
case EAX_PCIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
*val = *pc-2 + EXTW(IRE);
return SCPE_OK;
case EAX_PCXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = (IRE_DA) ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
*val = *pc-2 + EXTW(IRE_DISP) + reg;
return SCPE_OK;
default:
return STOP_ERROP;
}
default:
return STOP_ERROP;
}
}
static t_stat ea_src_l(uint32 eamod,uint32 eareg,uint32* val,t_addr* pc)
{
t_stat rc = SCPE_OK;
uint32 reg, regno, IRE;
t_addr *areg;
switch (eamod) {
case EA_DDIR:
*val = DR[eareg];
return SCPE_OK;
case EA_ADIR:
*val = *AREG(eareg);
return SCPE_OK;
case EA_AIND:
return ReadVL(saved_ea = *AREG(eareg), val);
case EA_API:
areg = AREG(eareg);
rc = ReadVL(saved_ea = *areg, val);
*areg += 4;
return rc;
case EA_APD:
areg = AREG(eareg);
*areg -= 4;
return ReadVL(saved_ea = *areg, val);
case EA_AIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return ReadVL(saved_ea = *AREG(eareg) + EXTW(IRE), val);
case EA_AXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = IRE_DA ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
return ReadVL(saved_ea = *AREG(eareg) + EXTW(IRE_DISP) + reg, val);
case EA_EXT:
switch (eareg) {
case EAX_AW:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return ReadVL(saved_ea = EXTW(IRE), val);
case EAX_AL:
ASSERT_OKRET(ReadPL(*pc,&IRE));
*pc += 4;
return ReadVL(saved_ea = IRE, val);
case EAX_PCIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return ReadVL(saved_ea = *pc-2 + EXTW(IRE), val);
case EAX_PCXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = (IRE_DA) ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
return ReadVL(saved_ea = *pc-2 + EXTW(IRE_DISP) + reg, val);
case EAX_IMM:
ASSERT_OKRET(ReadVL(*pc,val));
*pc += 4;
return SCPE_OK;
default:
return STOP_ERROP;
}
default:
return STOP_ERROP;
}
}
static t_stat ea_src_l64(uint32 eamod,uint32 eareg,t_uint64* val64,t_addr* pc)
{
uint32 val32;
t_stat rc = ea_src_l(eamod,eareg,&val32,pc);
*val64 = (t_uint64)val32;
return rc;
}
t_stat ea_src(uint32 eamod,uint32 eareg,uint32* val,int sz,t_addr* pc)
{
switch (sz) {
case SZ_BYTE:
return ea_src_b(eamod,eareg,val,pc);
case SZ_WORD:
return ea_src_w(eamod,eareg,val,pc);
case SZ_LONG:
return ea_src_l(eamod,eareg,val,pc);
default:
return STOP_ERROP;
}
}
static t_stat ea_dst_b(uint32 eamod,uint32 eareg,uint32 val,t_addr* pc)
{
t_stat rc;
uint32 IRE,reg,regno;
t_addr *areg;
// printf("dst: eamod=%x eareg=%x\n",eamod,eareg);
// printf("val=%x\n",val);
switch (eamod) {
case EA_DDIR:
DR[eareg] = COMBINE8(DR[eareg],val);
return SCPE_OK;
case EA_AIND:
return WriteVB(*AREG(eareg), val);
case EA_API:
areg = AREG(eareg);
rc = WriteVB(*areg, val);
*areg += (eareg==7 ? 2 : 1);
return rc;
case EA_APD:
areg = AREG(eareg);
*areg -= (eareg==7 ? 2 : 1);
return WriteVB(*areg, val);
case EA_AIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return WriteVB(*AREG(eareg) + EXTW(IRE), val);
case EA_AXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = IRE_DA ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
return WriteVB(*AREG(eareg) + EXTW(IRE_DISP) + reg, val);
case EA_EXT:
switch (eareg) {
case EAX_AW:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return WriteVB(EXTW(IRE), val);
case EAX_AL:
ASSERT_OKRET(ReadPL(*pc,&IRE));
*pc += 4;
return WriteVB(IRE, val);
default:
return STOP_ERROP;
}
case EA_ADIR:
default:
return STOP_ERROP;
}
}
t_stat ea_dst_b_rmw(uint32 eamod,uint32 eareg,uint32 val)
{
switch (eamod) {
case EA_DDIR:
DR[eareg] = COMBINE8(DR[eareg],val);
return SCPE_OK;
case EA_AIND:
case EA_API:
case EA_APD:
case EA_AIDX:
case EA_AXIDX:
return WriteVB(saved_ea, val);
case EA_EXT:
switch (eareg) {
case EAX_AW:
case EAX_AL:
return WriteVB(saved_ea, val);
case EAX_IMM:
SR = COMBINE8(SR,val);
return SCPE_OK;
default:
return STOP_ERROP;
}
default:
return STOP_ERROP;
}
}
static t_stat ea_dst_w(uint32 eamod,uint32 eareg,uint32 val,t_addr* pc)
{
t_stat rc;
uint32 IRE,reg,regno;
t_addr *areg;
switch (eamod) {
case EA_DDIR:
DR[eareg] = COMBINE16(DR[eareg],val);
return SCPE_OK;
case EA_ADIR:
*AREG(eareg) = COMBINE16(*AREG(eareg),val);
// *AREG(eareg) = EXTW(val);
return SCPE_OK;
case EA_AIND:
return WriteVW(*AREG(eareg), val);
case EA_API:
areg = AREG(eareg);
rc = WriteVW(*areg, val);
*areg += 2;
return rc;
case EA_APD:
areg = AREG(eareg);
*areg -= 2;
return WriteVW(*areg, val);
case EA_AIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return WriteVW(*AREG(eareg) + EXTW(IRE), val);
case EA_AXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = IRE_DA ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
return WriteVW(*AREG(eareg) + EXTW(IRE_DISP) + reg, val);
case EA_EXT:
switch (eareg) {
case EAX_AW:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return WriteVW(EXTW(IRE), val);
case EAX_AL:
ASSERT_OKRET(ReadPL(*pc,&IRE));
*pc += 4;
return WriteVW(IRE, val);
default:
return STOP_ERROP;
}
default:
return STOP_ERROP;
}
}
static t_stat ea_dst_w_rmw(uint32 eamod,uint32 eareg,uint32 val)
{
switch (eamod) {
case EA_DDIR:
DR[eareg] = COMBINE16(DR[eareg],val);
return SCPE_OK;
case EA_ADIR:
printf("ea_dst_w_rmw EA_ADIR: pc=%x\n",saved_PC);
*AREG(eareg) = val; /* use full 32 bits even for word operand */
return SCPE_OK;
case EA_AIND:
case EA_API:
case EA_APD:
case EA_AIDX:
case EA_AXIDX:
return WriteVW(saved_ea, val);
case EA_EXT:
switch (eareg) {
case EAX_AW:
case EAX_AL:
return WriteVW(saved_ea, val);
case EAX_IMM:
SR = val;
return SCPE_OK;
default:
return STOP_ERROP;
}
default:
return STOP_ERROP;
}
}
static t_stat ea_dst_l(uint32 eamod,uint32 eareg,uint32 val,t_addr* pc)
{
t_stat rc;
uint32 IRE,reg,regno;
t_addr *areg;
switch (eamod) {
case EA_DDIR:
DR[eareg] = val;
return SCPE_OK;
case EA_ADIR:
*AREG(eareg) = val;
return SCPE_OK;
case EA_AIND:
return WriteVL(*AREG(eareg), val);
case EA_API:
areg = AREG(eareg);
rc = WriteVL(*areg, val);
*areg += 4;
return rc;
case EA_APD:
areg = AREG(eareg);
*areg -= 4;
return WriteVL(*areg, val);
case EA_AIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return WriteVL(*AREG(eareg) + EXTW(IRE), val);
case EA_AXIDX:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
regno = IRE_REG;
reg = IRE_DA ? *AREG(regno) : DR[regno];
if (!IRE_WL) reg = EXTW(reg);
return WriteVL(*AREG(eareg) + EXTW(IRE_DISP) + reg, val);
case EA_EXT:
switch (eareg) {
case EAX_AW:
ASSERT_OKRET(ReadInstrInc(pc,&IRE));
return WriteVL(EXTW(IRE), val);
case EAX_AL:
ASSERT_OKRET(ReadPL(*pc,&IRE));
*pc += 4;
return WriteVL(IRE, val);
default:
return STOP_ERROP;
}
default:
return STOP_ERROP;
}
}
t_stat ea_dst_l_rmw(uint32 eamod,uint32 eareg,uint32 val)
{
switch (eamod) {
case EA_DDIR:
DR[eareg] = val;
return SCPE_OK;
case EA_ADIR:
*AREG(eareg) = val;
return SCPE_OK;
case EA_AIND:
case EA_API:
case EA_APD:
case EA_AIDX:
case EA_AXIDX:
return WriteVL(saved_ea, val);
case EA_EXT:
switch (eareg) {
case EAX_AW:
case EAX_AL:
return WriteVL(saved_ea, val);
default:
return STOP_ERROP;
}
default:
return STOP_ERROP;
}
}
t_stat ea_dst(uint32 eamod,uint32 eareg,uint32 val,int sz,t_addr* pc)
{
switch (sz) {
case SZ_BYTE:
return ea_dst_b(eamod,eareg,val,pc);
case SZ_WORD:
return ea_dst_w(eamod,eareg,val,pc);
case SZ_LONG:
return ea_dst_l(eamod,eareg,val,pc);
default:
return STOP_ERROP;
}
}
static t_bool testcond(uint32 c)
{
int n,v;
switch (c) {
case 0x0000: /*T*/
return TRUE;
case 0x0100: /*F*/
return FALSE;
case 0x0200: /*HI*/
return !(CCR_C || CCR_Z);
case 0x0300: /*LS*/
return CCR_C || CCR_Z;
case 0x0400: /*CC*/
return !CCR_C;
case 0x0500: /*CS*/
return CCR_C;
case 0x0600: /*NE*/
return !CCR_Z;
case 0x0700: /*EQ*/
return CCR_Z;
case 0x0800: /*VC*/
return !CCR_V;
case 0x0900: /*VS*/
return CCR_V;
case 0x0a00: /*PL*/
return !CCR_N;
case 0x0b00: /*MI*/
return CCR_N;
case 0x0c00: /*GE*/
n = CCR_N; v = CCR_V;
return (n && v) || !(n || v);
case 0x0d00: /*LT*/
n = CCR_N; v = CCR_V;
return (n && !v) || (!n && v);
case 0x0e00: /*GT*/
n = CCR_N; v = CCR_V;
return !CCR_Z && (n || !v) && (!n || v);
case 0x0f00: /*LE*/
n = CCR_N; v = CCR_V;
return CCR_Z || (!n && v) || (n && !v);
default: /*notreached*/
return FALSE;
}
}
/* push/pop on supervisor sp */
static t_stat m68k_push16(uint32 data)
{
A7 -= 2;
return WriteVW(A7,data);
}
static t_stat m68k_push32(uint32 data)
{
A7 -= 4;
return WriteVL(A7,data);
}
static t_stat m68k_pop16(uint32* data)
{
A7 += 2;
return ReadVW(A7-2,data);
}
static t_stat m68k_pop32(uint32* data)
{
A7 += 4;
return ReadVL(A7-4,data);
}
/* push/pop on current sp */
t_stat m68k_cpush16(uint32 data)
{
*cur_sp -= 2;
return WriteVW(*cur_sp,data);
}
static t_stat m68k_cpush32(uint32 data)
{
*cur_sp -= 4;
return WriteVL(*cur_sp,data);
}
static t_stat m68k_cpop16(uint32* data)
{
*cur_sp += 2;
return ReadVW(*cur_sp-2,data);
}
static t_stat m68k_cpop32(uint32* data)
{
*cur_sp += 4;
return ReadVL(*cur_sp-4,data);
}
t_stat m68k_gen_exception(int vecno,t_addr* pc)
{
t_stat rc;
uint32 dummy;
t_addr oldpc = *pc;
char out[20];
/* @TODO VBR! */
if (cputype<2) {
ASSERT_OKRET(m68k_push32(*pc));
ASSERT_OKRET(m68k_push16(SR));
m68k_set_s(TRUE);
CLRF(FLAG_T0|FLAG_T1);
} else {
/* no support for 68010 and above yet */
return STOP_IMPL;
}
/* set the new PC */
ASSERT_OKRET(ReadPL(vecno<<2,pc));
IFDEBUG(DBG_CPU_EXC,fprintf(sim_deb,"CPU : [0x%08x] Exception: vec=%d to %s\n",oldpc,vecno,m68k_getsym(*pc,XFMT,out)));
return ReadInstr(*pc,&dummy); /* fill prefetch cache */
}
static uint32 m68k_add8(uint32 src1,uint32 src2,uint32 x)
{
uint32 res = MASK_8L(src1) + MASK_8L(src2) + x;
SETNZ8(res);
SETF(MASK_9(res),FLAG_C|FLAG_X);
SETV_ADD8(src1,src2,res);
return res;
}
static uint32 m68k_add16(uint32 src1,uint32 src2,uint32 x,t_bool chgflags)
{
uint32 res = MASK_16L(src1) + MASK_16L(src2) + x;
if (chgflags) {
SETNZ16(res);
SETF(MASK_17(res),FLAG_C|FLAG_X);
SETV_ADD16(src1,src2,res);
}
return res;
}
static uint32 m68k_add32(t_uint64 src1,t_uint64 src2,t_uint64 x,t_bool chgflags)
{
t_uint64 resx = MASK_32L(src1) + MASK_32L(src2) + x;
if (chgflags) {
SETNZ32(resx);
SETF(MASK_33(resx),FLAG_C|FLAG_X);
SETV_ADD32(src1,src2,resx);
}
return (uint32)resx;
}
static uint32 m68k_sub8(uint32 dst,uint32 src,uint32 x)
{
uint32 res = MASK_8L(dst) - MASK_8L(src) - x;
SETNZ8(res);
SETF(MASK_9(res),FLAG_C|FLAG_X);
SETV_SUB8(src,dst,res);
return res;
}
static uint32 m68k_sub16(uint32 dst,uint32 src,uint32 x,t_bool chgflags)
{
uint32 res = MASK_16L(dst) - MASK_16L(src) - x;
if (chgflags) {
SETNZ16(res);
SETF(MASK_17(res),FLAG_C|FLAG_X);
SETV_SUB16(src,dst,res);
}
return res;
}
static uint32 m68k_sub32(t_uint64 dst,t_uint64 src, t_uint64 x,t_bool chgflags)
{
t_uint64 resx = MASK_32L(dst) - MASK_32L(src) - x;
if (chgflags) {
SETNZ32(resx);
SETF(MASK_33(resx),FLAG_C|FLAG_X);
SETV_SUB32(src,dst,resx);
}
return (uint32)resx;
}
static uint32* movem_regs[] = {
(uint32*)&D0, (uint32*)&D1, (uint32*)&D2, (uint32*)&D3, (uint32*)&D4, (uint32*)&D5, (uint32*)&D6, (uint32*)&D7,
(uint32*)&A0, (uint32*)&A1, (uint32*)&A2, (uint32*)&A3, (uint32*)&A4, (uint32*)&A5, (uint32*)&A6, 0
};
static t_stat m68k_movem_r_pd(t_addr* areg,uint32 regs,t_bool sz)
{
int i;
t_stat rc;
t_addr ea = *areg;
movem_regs[15] = cur_sp;
for (i=0; i<16; i++) {
if (regs & (1<<i)) {
if (sz) {
ea -= 4;
ASSERT_OK(WriteVL(ea, *movem_regs[15-i]));
} else {
ea -= 2;
ASSERT_OK(WriteVW(ea, *movem_regs[15-i]));
}
}
}
*areg = ea;
return SCPE_OK;
}
static t_stat m68k_movem_r_ea(t_addr ea,uint32 regs,t_bool sz)
{
int i;
t_stat rc;
movem_regs[15] = cur_sp;
for (i=0; i<16; i++) {
if (regs & (1<<i)) {
if (sz) {
ASSERT_OK(WriteVL(ea, *movem_regs[i]));
ea += 4;
} else {
ASSERT_OK(WriteVW(ea, *movem_regs[i]));
ea += 2;
}
}
}
return SCPE_OK;
}
static t_stat m68k_movem_pi_r(t_addr* areg,uint32 regs,t_bool sz)
{
int i;
t_addr ea = *areg;
uint32 src;
t_stat rc;
movem_regs[15] = cur_sp;
for (i=0; i<16; i++) {
if (regs & (1<<i)) {
if (sz) {
ASSERT_OK(ReadVL(ea, movem_regs[i]));
ea += 4;
} else {
ASSERT_OK(ReadVW(ea, &src));
*movem_regs[i] = EXTW(src);
ea += 2;
}
}
}
*areg = ea;
return SCPE_OK;
}
static t_stat m68k_movem_ea_r(t_addr ea,uint32 regs,t_bool sz)
{
int i;
uint32 src;
t_stat rc;
movem_regs[15] = cur_sp;
for (i=0; i<16; i++) {
if (regs & (1<<i)) {
if (sz) {
ASSERT_OK(ReadVL(ea, movem_regs[i]));
ea += 4;
} else {
ASSERT_OK(ReadVW(ea, &src));
*movem_regs[i] = EXTW(src);
ea += 2;
}
}
}
return SCPE_OK;
}
static t_stat m68k_divu_w(uint32 divdr,int32* reg, t_addr* pc)
{
uint32 quo,rem,*dst;
dst = (uint32*)reg;
divdr = MASK_16L(divdr);
if (divdr==0) return m68k_gen_exception(5,pc);
quo = *dst / divdr;
rem = (*dst % divdr)<<16;
if (MASK_16U(quo)) ONEF(FLAG_V);
else {
SETNZ16(quo);
CLRF(FLAG_V|FLAG_C);
*dst = COMBINE16(rem,quo);
}
return SCPE_OK;
}
static t_stat m68k_divs_w(uint32 divdr,int32* reg, t_addr* pc)
{
int32 quo,rem,div;
div = EXTW(divdr);
if (div==0) return m68k_gen_exception(5,pc);
if (*reg==0x80000000 && div==0xffffffff) {
CLRF(FLAG_Z|FLAG_N|FLAG_V|FLAG_C);
*reg = 0;
return SCPE_OK;
}
quo = *reg / divdr;
rem = (*reg % divdr)<<16;
if (EXTW(quo) == quo) {
SETNZ16(quo);
CLRF(FLAG_V|FLAG_C);
*reg = COMBINE16(rem,quo);
} else ONEF(FLAG_V);
return SCPE_OK;
}
static t_bool m68k_checkints(t_addr* pc)
{
int i;
if (intpending) {
for (i=7; i>=1; i--) {
if (intpending & (1<<i) && (i==7 || i > SR_IPL)) {
/* found a pending irq at level i, that must be serviced now */
IFDEBUG(DBG_CPU_INT,fprintf(sim_deb,"CPU : [0x%08x] Interrupt: granting level=%d, IPL=%d, pending=%x\n",
*pc,i,SR_IPL,intpending));
m68k_gen_exception(intvectors[i],pc); /* generate an exception */
intpending &= ~(1<<i);
intvectors[i] = 0; /* mark it as handled */
m68k_setipl(i); /* set new interrupt prio, to leave out lower prio interrupts */
return TRUE;
}
}
}
return intpending != 0;
}
/* handle stop instruction */
static t_stat m68k_stop(t_addr* pc)
{
t_stat rc = SCPE_OK;
t_addr oldpc = *pc;
IFDEBUG(DBG_CPU_INT,fprintf(sim_deb,"CPU : [0x%08x] STOP: SR=0x%04x\n",oldpc-4,SR));
for (;;) {
/* is there an interrupt above IPL (checked in raise_vectorint) ? */
if (m68k_checkints(pc)) break;
/* loop until something is happening */
if (sim_interval <= 0 && (rc=sim_process_event())) break;
sim_interval--;
}
IFDEBUG(DBG_CPU_INT,fprintf(sim_deb,"CPU : [0x%08x] STOP: will continue at 0x%08x intpending=%x rc=%d\n",*pc,oldpc,intpending,rc));
return rc;
}
t_stat sim_instr()
{
t_stat rc;
uint32 IR, IRE, src1, src2, res, ea;
int32 sres, *reg, cnt;
t_uint64 resx, srcx1, srcx2;
t_addr PC, srca, *areg, oldpc;
t_bool isbsr,iscond;
uint16 tracet0;
char out[20];
/* restore state */
PC = saved_PC;
rc = 0;
tracet0 = 0;
/* the big main loop */
while (rc == SCPE_OK) {
saved_PC = PC;
/* expired? */
if (sim_interval <= 0) {
if ((rc = sim_process_event())) break;
}
/* process breakpoints */
if (sim_brk_summ && sim_brk_test(PC, E_BKPT_SPC|SWMASK('E'))) {
rc = STOP_IBKPT;
break;
}
/* opcode fetch */
ASSERT_OK(ReadInstrInc(&PC,&IR));
IFDEBUG(DBG_CPU_PC,fprintf(sim_deb,"DEBUG(PC): PC=%x IR=%x\n",PC-2,IR));
sim_interval--;
/* now decode instruction */
switch (IR_1512) {
/* 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 0 0 0 | Opcode | 0 |Length | effective address | addi,andi,cmpi,eori,ori,subi
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 0 0 0 | Register | 1 |Opcode | effective address<>001| bchg,bclr,bset,btst
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 0 0 0 | Register | 1 |Opcode | 0 0 1 | Register | movep
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0x0000:
switch (IR_1103) {
case 0000400: case 0001400: case 0002400: case 0003400:
case 0004400: case 0005400: case 0006400: case 0007400: /* btst d,d */
cnt = DRX & 0x1f;
goto do_btstd;
case 0004000: /* btst #,d */
ASSERT_OK(ReadInstrInc(&PC,&IRE));
cnt = IRE & 0x1f;
do_btstd: SETZ32(DRY & bitmask[cnt+1]);
rc = SCPE_OK; break;
case 0000420: case 0000430: case 0000440: case 0000450:
case 0000460: case 0000470: case 0001420: case 0001430:
case 0001440: case 0001450: case 0001460: case 0001470:
case 0002420: case 0002430: case 0002440: case 0002450:
case 0002460: case 0002470: case 0003420: case 0003430:
case 0003440: case 0003450: case 0003460: case 0003470:
case 0004420: case 0004430: case 0004440: case 0004450:
case 0004460: case 0004470: case 0005420: case 0005430:
case 0005440: case 0005450: case 0005460: case 0005470:
case 0006420: case 0006430: case 0006440: case 0006450:
case 0006460: case 0006470: case 0007420: case 0007430:
case 0007440: case 0007450: case 0007460: case 0007470: /* btst d,ea */
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
cnt = DRX & 7;
goto do_btst8;
case 0004020: case 0004030: case 0004040: case 0004050:
case 0004060: case 0004070: /* btst #,ea */
ASSERT_OK(ReadInstrInc(&PC,&IRE));
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
cnt = IRE & 7;
do_btst8: SETZ8(src1 & bitmask[cnt+1]);
rc = SCPE_OK;
break;
case 0000700: case 0001700: case 0002700: case 0003700:
case 0004700: case 0005700: case 0006700: case 0007700: /* bset d,d */
cnt = DRX & 0x1f;
src1 = bitmask[cnt+1];
goto do_bsetd;
case 0004300: /* bset #,d */
ASSERT_OK(ReadInstrInc(&PC,&IRE));
src1 = bitmask[(IRE & 0x1f)+1];
do_bsetd: reg = &DRY;
SETZ32(*reg & src1);
*reg |= src1;
rc = SCPE_OK; break;
case 0000720: case 0000730: case 0000740: case 0000750:
case 0000760: case 0000770: case 0001720: case 0001730:
case 0001740: case 0001750: case 0001760: case 0001770:
case 0002720: case 0002730: case 0002740: case 0002750:
case 0002760: case 0002770: case 0003720: case 0003730:
case 0003740: case 0003750: case 0003760: case 0003770:
case 0004720: case 0004730: case 0004740: case 0004750:
case 0004760: case 0004770: case 0005720: case 0005730:
case 0005740: case 0005750: case 0005760: case 0005770:
case 0006720: case 0006730: case 0006740: case 0006750:
case 0006760: case 0006770: case 0007720: case 0007730:
case 0007740: case 0007750: case 0007760: case 0007770: /* bset d,ea */
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&res,&PC));
cnt = DRY & 7;
src1 = bitmask[cnt+1];
goto do_bset8;
case 0004320: case 0004330: case 0004340: case 0004350:
case 0004360: case 0004370: /* bset # */
ASSERT_OK(ReadInstrInc(&PC,&IRE));
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&res,&PC));
src1 = bitmask[(IRE&7)+1];
do_bset8: SETZ8(res & src1);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res | src1); break;
case 0000500: case 0001500: case 0002500: case 0003500:
case 0004500: case 0005500: case 0006500: case 0007500: /* bchg d,d */
cnt = DRX & 0x1f;
src1 = bitmask[cnt+1];
goto do_bchgd;
case 0004100: /* bchg #,d */
ASSERT_OK(ReadInstrInc(&PC,&IRE));
src1 = bitmask[(IRE & 0x1f)+1];
do_bchgd: reg = &DRY;
SETZ32(*reg & src1);
*reg ^= src1;
rc = SCPE_OK; break;
case 0000520: case 0000530: case 0000540: case 0000550:
case 0000560: case 0000570: case 0001520: case 0001530:
case 0001540: case 0001550: case 0001560: case 0001570:
case 0002520: case 0002530: case 0002540: case 0002550:
case 0002560: case 0002570: case 0003520: case 0003530:
case 0003540: case 0003550: case 0003560: case 0003570:
case 0004520: case 0004530: case 0004540: case 0004550:
case 0004560: case 0004570: case 0005520: case 0005530:
case 0005540: case 0005550: case 0005560: case 0005570:
case 0006520: case 0006530: case 0006540: case 0006550:
case 0006560: case 0006570: case 0007520: case 0007530:
case 0007540: case 0007550: case 0007560: case 0007570: /* bchg d,ea */
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&res,&PC));
cnt = DRX & 7;
src1 = bitmask[cnt+1];
goto do_bchg8;
case 0004120: case 0004130: case 0004140: case 0004150:
case 0004160: case 0004170: /* bchg #,ea */
ASSERT_OK(ReadInstrInc(&PC,&IRE));
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&res,&PC));
src1 = bitmask[(IRE&7)+1];
do_bchg8: SETZ8(res & src1);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res ^ src1); break;
case 0000600: case 0001600: case 0002600: case 0003600:
case 0004600: case 0005600: case 0006600: case 0007600: /* bclr d,d */
cnt = DRX & 0x1f;
src1 = bitmask[cnt+1];
goto do_bclrd;
case 0004200: /* bclr #,d */
ASSERT_OK(ReadInstrInc(&PC,&IRE));
src1 = bitmask[(IRE & 0x1f)+1];
do_bclrd: reg = &DRY;
SETZ32(*reg & src1);
*reg &= ~src1;
rc = SCPE_OK; break;
case 0000620: case 0000630: case 0000640: case 0000650:
case 0000660: case 0000670: case 0001620: case 0001630:
case 0001640: case 0001650: case 0001660: case 0001670:
case 0002620: case 0002630: case 0002640: case 0002650:
case 0002660: case 0002670: case 0003620: case 0003630:
case 0003640: case 0003650: case 0003660: case 0003670:
case 0004620: case 0004630: case 0004640: case 0004650:
case 0004660: case 0004670: case 0005620: case 0005630:
case 0005640: case 0005650: case 0005660: case 0005670:
case 0006620: case 0006630: case 0006640: case 0006650:
case 0006660: case 0006670: case 0007620: case 0007630:
case 0007640: case 0007650: case 0007660: case 0007670: /* bclr d,ea */
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&res,&PC));
cnt = DRX & 7;
src1 = bitmask[cnt+1];
goto do_bclr8;
case 0004220: case 0004230: case 0004240: case 0004250:
case 0004260: case 0004270: /* bclr #,ea */
ASSERT_OK(ReadInstrInc(&PC,&IRE));
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&res,&PC));
src1 = bitmask[(IRE&7)+1];
do_bclr8: SETZ8(res & src1);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res & ~src1); break;
case 0000410: case 0001410: case 0002410: case 0003410:
case 0004410: case 0005410: case 0006410: case 0007410: /*movep.w m,r*/
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
ASSERT_OK(ReadVB(srca,&src1));
reg = &DRX;
*reg = src1<<8;
rc = ReadVB(srca+2,&src1);
*reg = COMBINE8(*reg,src1);
break;
case 0000510: case 0001510: case 0002510: case 0003510:
case 0004510: case 0005510: case 0006510: case 0007510: /*movep.l m,r*/
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
ASSERT_OK(ReadVB(srca,&src1));
reg = &DRX;
*reg = src1<<8;
ASSERT_OK(ReadVB(srca+2,&src1));
*reg = (COMBINE8(*reg,src1))<<8;
ASSERT_OK(ReadVB(srca+4,&src1));
*reg = (COMBINE8(*reg,src1))<<8;
rc = ReadVB(srca+6,&src1);
*reg = (COMBINE8(*reg,src1))<<8;
break;
case 0000610: case 0001610: case 0002610: case 0003610:
case 0004610: case 0005610: case 0006610: case 0007610: /*movep.w r,m*/
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
src1 = DRX;
ASSERT_OK(WriteVB(srca,src1>>8));
rc = WriteVB(srca+2,src1); break;
case 0000710: case 0001710: case 0002710: case 0003710:
case 0004710: case 0005710: case 0006710: case 0007710: /*movep.l r,m*/
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
ASSERT_OK(WriteVB(srca,src1>>24));
ASSERT_OK(WriteVB(srca+2,src1>>16));
ASSERT_OK(WriteVB(srca+4,src1>>8));
rc = WriteVB(srca+6,src1); break;
case 0000000: case 0000020: case 0000030: case 0000040:
case 0000050: case 0000060: case 0000070: /*ori.b*/
ASSERT_OK(ReadInstrInc(&PC,&src2));
ASSERT_OK(ea_src_bs(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 | src2;
if (IR_EA != EA_IMM) {
SETNZ8(res);
CLRF(FLAG_C|FLAG_V);
}
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
tracet0 = SR_T0; break;
case 0000100: case 0000120: case 0000130: case 0000140:
case 0000150: case 0000160: case 0000170: /*ori.w*/
if (IR_EA == EA_IMM) ASSERT_PRIV();
ASSERT_OK(ReadInstrInc(&PC,&src2));
ASSERT_OK(ea_src_ws(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 | src2;
if (IR_EA != EA_IMM) {
SETNZ16(res);
CLRF(FLAG_C|FLAG_V);
}
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
tracet0 = SR_T0; break;
case 0000200: case 0000220: case 0000230: case 0000240:
case 0000250: case 0000260: case 0000270: /*ori.l*/
ASSERT_OK(ReadInstrLongInc(&PC,&src2));
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 | src2;
SETNZ32(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res); break;
case 0001000: case 0001020: case 0001030: case 0001040:
case 0001050: case 0001060: case 0001070: /*andi.b*/
ASSERT_OK(ReadInstrInc(&PC,&src2));
ASSERT_OK(ea_src_bs(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 & src2;
if (IR_EA != EA_IMM) {
SETNZ8(res);
CLRF(FLAG_C|FLAG_V);
}
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
tracet0 = SR_T0; break;
case 0001100: case 0001120: case 0001130: case 0001140:
case 0001150: case 0001160: case 0001170: /*andi.w*/
if (IR_EA==EA_IMM) ASSERT_PRIV();
ASSERT_OK(ReadInstrInc(&PC,&src2));
ASSERT_OK(ea_src_ws(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 & src2;
if (IR_EA != EA_IMM) {
SETNZ16(res);
CLRF(FLAG_C|FLAG_V);
}
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
tracet0 = SR_T0; break;
case 0001200: case 0001220: case 0001230: case 0001240:
case 0001250: case 0001260: case 0001270: /*andi.l*/
ASSERT_OK(ReadInstrLongInc(&PC,&src2));
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 & src2;
SETNZ32(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0006000: case 0006020: case 0006030: case 0006040:
case 0006050: case 0006060: case 0006070: /*cmpi.b*/
case 0002000: case 0002020: case 0002030: case 0002040:
case 0002050: case 0002060: case 0002070: /*subi.b*/
ASSERT_OK(ReadInstrInc(&PC,&src2));
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_sub8(src1,src2,0);
rc = IR_1103 < 0006000 ? ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res) : SCPE_OK;
break;
case 0006100: case 0006120: case 0006130: case 0006140:
case 0006150: case 0006160: case 0006170: /*cmpi.w*/
case 0002100: case 0002120: case 0002130: case 0002140:
case 0002150: case 0002160: case 0002170: /*subi.w*/
ASSERT_OK(ReadInstrInc(&PC,&src2));
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_sub16(src1,src2,0,TRUE);
rc = IR_1103 < 0006000 ? ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res) : SCPE_OK;
break;
case 0006200: case 0006220: case 0006230: case 0006240:
case 0006250: case 0006260: case 0006270: /*cmpi.l*/
case 0002200: case 0002220: case 0002230: case 0002240:
case 0002250: case 0002260: case 0002270: /*subi.l*/
ASSERT_OK(ReadInstrLongInc(&PC,&src2));
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
res = m68k_sub32(srcx1,(t_uint64)src2,0,TRUE);
rc = IR_1103 < 0006000 ? ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res) : SCPE_OK;
break;
case 0003000: case 0003020: case 0003030: case 0003040:
case 0003050: case 0003060: case 0003070: /*addi.b*/
ASSERT_OK(ReadInstrInc(&PC,&src2));
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_add8(src1,src2,0);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res); break;
case 0003100: case 0003120: case 0003130: case 0003140:
case 0003150: case 0003160: case 0003170: /*addi.w*/
ASSERT_OK(ReadInstrInc(&PC,&src2));
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_add16(src1,src2,0,TRUE);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res); break;
case 0003200: case 0003220: case 0003230: case 0003240:
case 0003250: case 0003260: case 0003270: /*addi.l*/
ASSERT_OK(ReadInstrLongInc(&PC,&src2));
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
res = m68k_add32(srcx1,(t_uint64)src2,0,TRUE);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0005000: case 0005020: case 0005030: case 0005040:
case 0005050: case 0005060: case 0005070: /*eori.b*/
ASSERT_OK(ReadInstrInc(&PC,&src2));
ASSERT_OK(ea_src_bs(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 ^ src2;
if (IR_EA != EA_IMM) {
SETNZ8(res);
CLRF(FLAG_C|FLAG_V);
}
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
tracet0 = SR_T0; break;
case 0005100: case 0005120: case 0005130: case 0005140:
case 0005150: case 0005160: case 0005170: /*eori.w*/
if (IR_EA==EA_IMM) ASSERT_PRIV();
ASSERT_OK(ReadInstrInc(&PC,&src2));
ASSERT_OK(ea_src_ws(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 ^ src2;
if (IR_EA != EA_IMM) {
SETNZ16(res);
CLRF(FLAG_C|FLAG_V);
}
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
tracet0 = SR_T0; break;
case 0005200: case 0005220: case 0005230: case 0005240:
case 0005250: case 0005260: case 0005270: /*eori.l*/
ASSERT_OK(ReadInstrLongInc(&PC,&src2));
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 ^ src2;
SETNZ32(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res); break;
default:
rc = STOP_ERROP;
}
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 0 |Length2| TargetReg | TargetMode| SourceMode| SourceReg | move
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 0 |Length2| TargetReg | 0 0 1 | SourceMode| SourceReg | movea
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0x1000:
ea = IR_EATGT;
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
if (ea == EA_ADIR)
rc = STOP_ERROP; /* movea.b */
else {
ASSERT_OK(ea_dst_b(ea,IR_REGX,src1,&PC));
SETNZ8(src1);
}
break;
case 0x2000:
ea = IR_EATGT;
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
if (ea==EA_ADIR) { /* movea.l */
*AREG(IR_REGX) = src1;
rc = SCPE_OK;
} else {
rc = ea_dst_l(ea,IR_REGX,src1,&PC);
SETNZ32(src1);
}
break;
case 0x3000:
ea = IR_EATGT;
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
if (ea==EA_ADIR) { /* movea.w */
*AREG(IR_REGX) = EXTW(src1);
rc = SCPE_OK;
} else {
rc = ea_dst_w(ea,IR_REGX,src1,&PC);
SETNZ16(src1);
}
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 0 | Opcode | 0 |Length | effective address | clr,neg,negx,not,tst
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 0 | Opcode | 0 | 1 1 | effective address | moveccr,movesr
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 0 | Opcode | 0 |Mode | 0 0 0 | Register | ext
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 0 | Opcode | 0 |Opcode | effective address | jmp,jsr,movem,nbcd,pea,tas
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 0 | Opcode | 0 |Opcode | Vector | trap
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 0 | Opcode | 0 |Opcode | Register | link,moveusp,swap,unlink
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 0 | Opcode | 0 |Opcode | illegal,nop,reset,rte,rtr,rts,stop,trapv
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 0 | Register | 1 |Opcode | effective address | chk,lea
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0x4000:
switch (IR_1106) {
case 000600: case 001600: case 002600: case 003600:
case 004600: case 005600: case 006600: case 007600: /*chk*/
src1 = DRX;
SETF((src1 & BIT31) != 0,FLAG_N);
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&res,&PC));
rc = CCR_N || src1 > res ? m68k_gen_exception(6,&PC) : SCPE_OK;
break;
case 000700: case 001700: case 002700: case 003700:
case 004700: case 005700: case 006700: case 007700: /*lea*/
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
*AREG(IR_REGX) = srca;
rc = SCPE_OK;
break;
case 000300: /*move from sr*/
rc = ea_dst_w(IR_EAMOD,IR_EAREG,SR,&PC);
break;
case 001000: /*clr.b*/
ONEF(FLAG_Z);
CLRF(FLAG_N|FLAG_C|FLAG_V);
rc = ea_dst_b(IR_EAMOD,IR_EAREG,0,&PC);
break;
case 001100: /*clr.w*/
ONEF(FLAG_Z);
CLRF(FLAG_N|FLAG_C|FLAG_V);
rc = ea_dst_w(IR_EAMOD,IR_EAREG,0,&PC);
break;
case 001200: /*clr.l*/
ONEF(FLAG_Z);
CLRF(FLAG_N|FLAG_C|FLAG_V);
rc = ea_dst_l(IR_EAMOD,IR_EAREG,0,&PC);
break;
case 000000: /*negx.b*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
src1 += (CCR_X ? 1 : 0);
goto do_neg8;
case 002000: /*neg.b*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
do_neg8: res = m68k_sub8(0,src1,0);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 000100: /*negx.w*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
src1 += (CCR_X ? 1 : 0);
goto do_neg16;
case 002100: /*neg.w*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
do_neg16: res = m68k_sub16(0,src1,0,TRUE);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 000200: /*negx.l*/
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
srcx1 = (t_uint64)src1 + (CCR_X ? 1 : 0);
goto do_neg32;
case 002200: /*neg.l*/
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
do_neg32: res = m68k_sub32(0,srcx1,0,TRUE);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 002300: /*move to ccr*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
SR = COMBINE8(SR,src1);
break;
case 003000: /*not.b*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = ~src1;
SETNZ8(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 003100: /*not.w*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = ~src1;
SETNZ16(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
break;
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
case 003200: /*not.l*/
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
res = ~src1;
SETNZ32(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 003300: /*move to sr*/
ASSERT_PRIV();
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
SR = src1;
tracet0 = SR_T0;
break;
case 004000: /*nbcd*/
rc = STOP_IMPL;
break;
case 004100: /*pea or swap*/
if (IR_0503==000) { /*swap*/
reg = &DRY;
src1 = *reg << 16;
res = *reg >> 16;
*reg = COMBINE16(src1,res);
SETNZ32(*reg);
CLRF(FLAG_C|FLAG_V);
rc = SCPE_OK;
} else { /*pea*/
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
ASSERT_OK(m68k_cpush32(srca));
}
break;
case 004200: /*movem.w or ext*/
if (IR_0503==000) { /*ext.w*/
reg = &DRY;
res = EXTB(*reg);
*reg = COMBINE16(*reg,res);
SETNZ16(res);
CLRF(FLAG_C|FLAG_V);
rc = SCPE_OK;
} else { /*movem.w regs,ea*/
ASSERT_OK(ReadInstrInc(&PC,&IRE));
if (IR_EAMOD==EA_APD)
rc = m68k_movem_r_pd(AREG(IR_REGY),IRE,FALSE);
else {
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
rc = m68k_movem_r_ea(srca,IRE,FALSE);
}
}
break;
case 004300: /*movem or ext*/
if (IR_0503==000) { /*ext.l*/
reg = &DRY;
*reg = EXTW(*reg);
SETNZ32(*reg);
CLRF(FLAG_C|FLAG_V);
rc = SCPE_OK;
} else { /*movem.l regs,ea */
ASSERT_OK(ReadInstrInc(&PC,&IRE));
if (IR_EAMOD==EA_APD)
rc = m68k_movem_r_pd(AREG(IR_REGY),IRE,TRUE);
else {
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
rc = m68k_movem_r_ea(srca,IRE,TRUE);
}
}
break;
case 005000: /*tst.b*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
SETNZ8(src1);
CLRF(FLAG_V|FLAG_C);
break;
case 005100: /*tst.w*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
SETNZ16(src1);
CLRF(FLAG_V|FLAG_C);
break;
case 005200: /*tst.l*/
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
SETNZ32(src1);
CLRF(FLAG_V|FLAG_C);
break;
case 005300: /*tas or illegal*/
if (IR==045374) { /*illegal*/
rc = STOP_ERROP;
} else { /*tas*/
rc = STOP_IMPL;
}
break;
case 006200: /*movem.w ea,regs*/
ASSERT_OK(ReadInstrInc(&PC,&IRE));
if (IR_EAMOD==EA_API)
rc = m68k_movem_pi_r(AREG(IR_REGY),IRE,FALSE);
else {
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
rc = m68k_movem_ea_r(srca,IRE,FALSE);
}
break;
case 006300: /*movem.l ea,regs*/
ASSERT_OK(ReadInstrInc(&PC,&IRE));
if (IR_EAMOD==EA_API)
rc = m68k_movem_pi_r(AREG(IR_REGY),IRE,TRUE);
else {
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
rc = m68k_movem_ea_r(srca,IRE,TRUE);
}
break;
case 007100:
switch(IR_0503) {
case 000000:
case 000010: /*trap*/
(*m68kcpu_trapcallback)(m68kcpu_dev,IR_TRAP);
rc = m68k_gen_exception(32+IR_TRAP,&PC);
break;
case 000020: /*link*/
ASSERT_OK(ReadInstrInc(&PC,&IRE));
if (IR_REGY==7) {
*cur_sp -= 4;
ASSERT_OK(WriteVL(*cur_sp,*cur_sp));
} else {
areg = AREG(IR_REGY);
ASSERT_OK(m68k_cpush32(*areg));
*areg = *cur_sp;
}
*cur_sp += EXTW(IRE);
break;
case 000030: /*unlk*/
if (IR_REGY==7) {
ASSERT_OK(ReadVL(*cur_sp,&srca));
*cur_sp = srca;
} else {
areg = AREG(IR_REGY);
*cur_sp = *areg;
ASSERT_OK(m68k_cpop32(areg));
}
break;
case 000040: /*move to usp*/
ASSERT_PRIV();
USP = AR[IR_REGY];
tracet0 = SR_T0;
rc = SCPE_OK;
break;
case 000050: /*move from usp*/
ASSERT_PRIV();
AR[IR_REGY] = USP;
rc = SCPE_OK;
break;
case 000060:
switch(IR_0200) {
case 000000: /*reset*/
ASSERT_PRIV();
rc = m68kcpu_peripheral_reset();
break;
case 000001: /*nop*/
rc = SCPE_OK;
tracet0 = SR_T0;
break;
case 000002: /*stop*/
ASSERT_PRIV();
ASSERT_OKRET(ReadInstrInc(&PC,&IRE));
SR = (uint16)IRE;
rc = STOP_HALT;
tracet0 = SR_T0;
break;
case 000003: /*rte*/
ASSERT_PRIV();
ASSERT_OK(m68k_pop16(&src1));
SR = src1;
m68k_set_s(SR_S != 0);
oldpc = PC;
rc = m68k_pop32(&PC);
tracet0 = SR_T0;
IFDEBUG(DBG_CPU_EXC,fprintf(sim_deb,"CPU : [0x%08x] RTE to 0x%08x, IPL=%d S=%d\n",
oldpc-2,PC,SR_IPL,SR_S?1:0));
break;
case 000005: /*rts*/
oldpc = PC;
rc = m68k_cpop32(&PC);
m68k_sublevel--;
IFDEBUG(DBG_CPU_CTRACE,fprintf(sim_deb,"CPU : [0x%08x] <<< RTS to 0x%08x (level=%d)\n",
oldpc-2,PC,m68k_sublevel));
tracet0 = SR_T0;
break;
case 000006: /*trapv*/
rc = CCR_V ? m68k_gen_exception(7,&PC) : SCPE_OK;
break;
case 000007: /*rtr*/
ASSERT_OK(m68k_cpop16(&src1));
SR = COMBINE8(SR,src1);
oldpc = PC;
rc = m68k_cpop32(&PC);
tracet0 = SR_T0;
IFDEBUG(DBG_CPU_EXC,fprintf(sim_deb,"CPU : [0x%08x] RTR to 0x%08x\n",oldpc-2,PC));
break;
default:
rc = STOP_ERROP;
}
break;
default:
rc = STOP_ERROP;
}
break;
case 007200: /*jsr*/
oldpc = PC;
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
ASSERT_OK(m68k_cpush32(PC));
IFDEBUG(DBG_CPU_CTRACE,fprintf(sim_deb,"CPU : [0x%08x] >>> JSR %s (level=%d)\n",
oldpc-2,m68k_getsym(srca,XFMT,out),m68k_sublevel));
PC = srca;
m68k_sublevel++;
tracet0 = SR_T0;
break;
case 007300: /*jmp*/
oldpc = PC;
ASSERT_OK(ea_src_l_nd(IR_EAMOD,IR_EAREG,&srca,&PC));
IFDEBUG(DBG_CPU_BTRACE,fprintf(sim_deb,"CPU : [0x%08x] ||| JMP %s\n",
oldpc-2,m68k_getsym(srca,XFMT,out)));
PC = srca;
tracet0 = SR_T0;
break;
default:
rc = STOP_ERROP;
}
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 1 | Quickdata |Opc|Length | effective address<>001| addq,subq
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 1 | Condition | 1 1 0 0 1 | Register | dbcc
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 0 1 | Condition | 1 1 | effective address<>001| scc
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0x5000:
switch (IR_0806) {
case 0000300:
case 0000700:
if (IR_0503==010) { /*dbcc*/
if (!IR_COND || !testcond(IR_COND)) { /* dbt is a NOP */
reg = &DRY;
src1 = MASK_16L((*reg-1));
*reg = MASK_16U(*reg) | src1;
if (src1 != 0xffff) {
ASSERT_OK(ReadInstr(PC,&IRE));
PC += (EXTW(IRE));
rc = SCPE_OK;
tracet0 = SR_T0;
break;
} /* else loop terminated */
}
/* loop cond not met or dbt */
PC += 2;
rc = SCPE_OK;
} else { /*scc*/
src1 = testcond(IR_COND) ? 0xff : 0x00;
rc = ea_dst_b(IR_EAMOD,IR_EAREG,src1,&PC);
}
break;
case 0000000: /*addq.b*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_add8(src1,quickarg[IR_REGX],0);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000100: /*addq.w*/
if (IR_EAMOD == EA_ADIR) {
*AREG(IR_REGY) += EXTW(quickarg[IR_REGX]);
rc = SCPE_OK;
} else {
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_add16(src1,quickarg[IR_REGX],0,TRUE);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
}
break;
case 0000200: /*addq.l*/
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
res = m68k_add32(srcx1,(t_uint64)quickarg[IR_REGX],0,IR_EAMOD!=EA_ADIR);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000400: /*subq.b*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_sub8(src1,quickarg[IR_REGX],0);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000500: /*subq.w*/
if (IR_EAMOD == EA_ADIR) {
*AREG(IR_REGY) -= EXTW(quickarg[IR_REGX]);
rc = SCPE_OK;
} else {
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_sub16(src1,quickarg[IR_REGX],0,TRUE);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
}
break;
case 0000600: /*subq.l*/
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
res = m68k_sub32(srcx1,(t_uint64)quickarg[IR_REGX],0,IR_EAMOD!=EA_ADIR);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
}
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 1 0 | Condition | Displacement | Bcc,bra,bsr
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0x6000:
isbsr = IR_COND==0x100; /* is bsr */
iscond = isbsr || testcond(IR_COND); /* condition matched */
if (IR_DISP) {
if (iscond) {
if (isbsr) {
IFDEBUG(DBG_CPU_CTRACE,fprintf(sim_deb,"CPU : [0x%08x] >>> BSR %s (level=%d\n",
PC-2,m68k_getsym(PC+EXTB(IR_DISP),XFMT,out),m68k_sublevel));
ASSERT_OK(m68k_cpush32(PC)); /* save PC for BSR */
m68k_sublevel++;
} else {
IFDEBUG(DBG_CPU_BTRACE,fprintf(sim_deb,"CPU : [0x%08x] ||| B%s %s\n",
PC-2,condnames[IR_COND>>8],m68k_getsym(PC+EXTB(IR_DISP),XFMT,out)));
}
PC += EXTB(IR_DISP); /* go to new location */
} /* else condition not matched */
} else { /* 16 bit ext word */
if (iscond) {
ASSERT_OK(ReadInstr(PC,&IRE)); /* get extension word */
if (isbsr) {
IFDEBUG(DBG_CPU_CTRACE,fprintf(sim_deb,"CPU : [0x%08x] >>> BSR %s (level=%d)\n",
PC-2,m68k_getsym(PC+EXTW(IRE),XFMT,out),m68k_sublevel));
ASSERT_OK(m68k_cpush32(PC+2)); /* save PC for BSR */
m68k_sublevel++;
} else {
IFDEBUG(DBG_CPU_BTRACE,fprintf(sim_deb,"CPU : [0x%08x] ||| B%s %s\n",
PC-2,condnames[IR_COND>>8],m68k_getsym(PC+EXTW(IRE),XFMT,out)));
}
PC += EXTW(IRE); /* go to new location */
} else {
PC += 2; /* condition not matched */
}
}
tracet0 = SR_T0;
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 0 1 1 1 | Register | 0 | Data | moveq
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0x7000:
src1 = DRX = EXTB(IR_DATA);
SETNZ32(src1);
CLRF(FLAG_C|FLAG_V);
rc = SCPE_OK; break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 0 0 0 | Register |Opc|Length | effective address<>00x| or
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 0 0 0 | Reg X | 1 0 0 | 0 0 |R/M| Reg Y | sbcd
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 0 0 0 | Register |Opc| 1 1 | effective address | divs,divu
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0x8000:
switch(IR_0803) {
case 0000300: case 0000320: case 0000330: case 0000340:
case 0000350: case 0000360: case 0000370: /*divu.w*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
rc = m68k_divu_w(src1,&DR[IR_REGX], &PC);
break;
case 0000700: case 0000720: case 0000730: case 0000740:
case 0000750: case 0000760: case 0000770: /*divs.w*/
rc = m68k_divs_w(src1,&DR[IR_REGX], &PC);
break;
case 0000400: /*sbcd d*/
rc = STOP_IMPL; break;
case 0000410: /*sbcd a*/
rc = STOP_IMPL; break;
case 0000000: case 0000020: case 0000030: case 0000040:
case 0000050: case 0000060: case 0000070: /*or.b ->d*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = MASK_8L(src1 | DRX);
SETNZ8(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_b(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000100: case 0000120: case 0000130: case 0000140:
case 0000150: case 0000160: case 0000170: /*or.w ->d*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = MASK_16L(src1 | DRX);
SETNZ16(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_b(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000200: case 0000220: case 0000230: case 0000240:
case 0000250: case 0000260: case 0000270: /*or.l ->d*/
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 & DRX;
SETNZ32(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_l(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000420: case 0000430: case 0000440: case 0000450:
case 0000460: case 0000470: /*or.b ->ea*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 | DRX;
SETNZ8(res);
CLRF(FLAG_V|FLAG_C);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000520: case 0000530: case 0000540: case 0000550:
case 0000560: case 0000570: /*or.w ->ea*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 | DRX;
SETNZ16(res);
CLRF(FLAG_V|FLAG_C);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000620: case 0000630: case 0000640: case 0000650:
case 0000660: case 0000670: /*or.l ->ea*/
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 | DRX;
SETNZ32(res);
CLRF(FLAG_V|FLAG_C);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
default:
rc = STOP_ERROP; break;
}
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 0 0 1 | Register |Opc|Length | effective address<>00x| sub
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 0 0 1 | Register |Opc| 1 1 | effective address | suba
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 0 0 1 | Reg X | 1 |Length | 0 0 |R/M| Reg Y | subx
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0x9000:
switch (IR_0803) {
case 0000300: case 0000310: case 0000320: case 0000330:
case 0000340: case 0000350: case 0000360: case 0000370: /* suba.w */
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&srca,&PC));
*AREG(IR_REGX) -= EXTW(srca); /* note: no flag changes! */
break;
case 0000700: case 0000710: case 0000720: case 0000730:
case 0000740: case 0000750: case 0000760: case 0000770: /* suba.l */
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&srca,&PC));
*AREG(IR_REGX) -= srca; /* note: no flag changes! */
break;
case 0000400: /*subx.b d*/
res = m68k_sub8(MASK_8L(DRY),DRX,CCR_X?1:0);
rc = ea_dst_b(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000410: /*subx.b -a*/
ASSERT_OK(ea_src_b(EA_APD,IR_REGY,&src1,&PC));
ASSERT_OK(ea_src_b(EA_APD,IR_REGX,&src2,&PC));
res = m68k_sub8(src1,src2,CCR_X?1:0);
rc = ea_dst_b_rmw(EA_APD,IR_REGX,res);
break;
case 0000500: /*subx.w d*/
res = m68k_sub16(MASK_16L(DRY),DRX,CCR_X?1:0,TRUE);
rc = ea_dst_w(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000510: /*subx.w -a*/
ASSERT_OK(ea_src_w(EA_APD,IR_REGY,&src1,&PC));
ASSERT_OK(ea_src_w(EA_APD,IR_REGX,&src2,&PC));
res = m68k_sub16(src1,src2,CCR_X?1:0,TRUE);
rc = ea_dst_w_rmw(EA_APD,IR_REGX,res);
break;
case 0000600: /*subx.l d*/
res = m68k_sub32((t_uint64)DRY,(t_uint64)DRX,CCR_X?1:0,TRUE);
rc = ea_dst_l(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000610: /*subx.l -a*/
ASSERT_OK(ea_src_l64(EA_APD,IR_REGY,&srcx1,&PC));
ASSERT_OK(ea_src_l64(EA_APD,IR_REGX,&srcx2,&PC));
res = m68k_sub32(srcx1,srcx2,CCR_X?1:0,TRUE);
rc = ea_dst_l_rmw(EA_APD,IR_REGX,res);
break;
case 0000000: case 0000020: case 0000030: case 0000040:
case 0000050: case 0000060: case 0000070: /* sub.b ->d */
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_sub8(DRX,src1,0);
rc = ea_dst_b(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000100: case 0000110: case 0000120: case 0000130:
case 0000140: case 0000150: case 0000160: case 0000170: /* sub.w ->d */
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_sub16(DRX,src1,0,TRUE);
rc = ea_dst_w(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000200: case 0000210: case 0000220: case 0000230:
case 0000240: case 0000250: case 0000260: case 0000270: /* sub.l ->d */
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
res = m68k_sub32((t_uint64)DRX,srcx1,0,TRUE);
rc = ea_dst_l(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000420: case 0000430: case 0000440: case 0000450:
case 0000460: case 0000470: /* sub.b ->ea */
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_sub8(src1,DRX,0);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000520: case 0000530: case 0000540: case 0000550:
case 0000560: case 0000570: /* sub.w ->ea */
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_sub16(src1,DRX,0,TRUE);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000620: case 0000630: case 0000640: case 0000650:
case 0000660: case 0000670: /* sub.l ->ea */
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
res = m68k_sub32(srcx1,(t_uint64)DRX,0,TRUE);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
default:
rc = STOP_ERROP;
}
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 0 1 0 | Opcode | trapa
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0xa000:
rc = m68k_gen_exception(10,&PC);
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 0 1 1 | Register | 0 |Length | effective address | cmp,cmpa
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 0 1 1 | Register | 1 |Length | effective address<>001| eor
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 0 1 1 | Reg X | 1 |Length | 0 0 1 | Reg Y | cmpm
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0xb000:
switch (IR_0803) {
case 0000410: /*cmpm.b*/
rc = STOP_IMPL; break;
case 0000510: /*cmpm.w*/
rc = STOP_IMPL; break;
case 0000610: /*cmpm.l*/
rc = STOP_IMPL; break;
case 0000400: case 0000420: case 0000430: case 0000440:
case 0000450: case 0000460: case 0000470: /*eor.b*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 ^ DRX;
SETNZ8(res);
CLRF(FLAG_V|FLAG_C);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000500: case 0000520: case 0000530: case 0000540:
case 0000550: case 0000560: case 0000570: /*eor.w*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 ^ DRX;
SETNZ16(res);
CLRF(FLAG_V|FLAG_C);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000600: case 0000620: case 0000630: case 0000640:
case 0000650: case 0000660: case 0000670: /*eor.l*/
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 ^ DRX;
SETNZ32(res);
CLRF(FLAG_V|FLAG_C);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000000: case 0000020: case 0000030: case 0000040:
case 0000050: case 0000060: case 0000070: /*cmp.b*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
(void)m68k_sub8(DRX,src1,0);
break;
case 0000100: case 0000110: case 0000120: case 0000130:
case 0000140: case 0000150: case 0000160: case 0000170: /*cmp.w*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
(void)m68k_sub16(DRX,src1,0,TRUE);
break;
case 0000200: case 0000210: case 0000220: case 0000230:
case 0000240: case 0000250: case 0000260: case 0000270: /*cmp.l*/
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
(void)m68k_sub32((t_uint64)DRX,srcx1,0,TRUE);
break;
case 0000300: case 0000310: case 0000320: case 0000330:
case 0000340: case 0000350: case 0000360: case 0000370: /*cmpa.w*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
areg = AREG(IR_REGX);
(void)m68k_sub32((t_uint64)EXTW(*areg),(t_uint64)src1,0,TRUE);
break;
case 0000700: case 0000710: case 0000720: case 0000730:
case 0000740: case 0000750: case 0000760: case 0000770: /*cmpa.l*/
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
(void)m68k_sub32((t_uint64)*AREG(IR_REGX),srcx1,0,TRUE);
break;
default:
rc = STOP_ERROP;
}
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 1 0 0 | Register |Opc|Length | effective address<>00x| and
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 1 0 0 | Reg X | 1 0 0 | 0 0 |R/M| Reg Y | abcd
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 1 0 0 | Reg X | 1 |Opcode | 0 0 |Opc| Reg Y | exg
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 1 0 0 | Register |Opc| 1 1 | effective address | muls,mulu
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0xc000:
switch(IR_0803) {
case 0000300: case 0000310: case 0000320: case 0000330:
case 0000340: case 0000350: case 0000360: case 0000370: /*mulu*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = (uint16)MASK_16L(src1) * (uint16)MASK_16L(DRX);
DRX = res;
SETNZ32(res);
CLRF(FLAG_C|FLAG_V);
break;
case 0000700: case 0000710: case 0000720: case 0000730:
case 0000740: case 0000750: case 0000760: case 0000770: /*muls*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
sres = (int16)MASK_16L(src1) * (int16)MASK_16L(DRX);
DRX = (uint32)sres;
SETNZ32(sres);
CLRF(FLAG_C|FLAG_V);
break;
case 0000500: /* exg d,d */
res = DRX; DRX = DRY; DRY = res;
rc = SCPE_OK; break;
case 0000510: /* exg a,a */
srca = *AREG(IR_REGX); *AREG(IR_REGX) = *AREG(IR_REGY); *AREG(IR_REGY) = srca;
rc = SCPE_OK; break;
case 0000610: /* exg a,d */
res = DRX; DRX = (uint32)*AREG(IR_REGY); *AREG(IR_REGY) = (t_addr)res;
rc = SCPE_OK; break;
case 0000400: /* abcd d */
rc = STOP_IMPL; break;
case 0000410: /* abcd a */
rc = STOP_IMPL; break;
case 0000000: case 00000020: case 0000030: case 0000040:
case 0000050: case 00000060: case 0000070: /* and.b -> d*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 & DRX;
SETNZ8(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_b(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000100: case 00000120: case 0000130: case 0000140:
case 0000150: case 00000160: case 0000170: /* and.w -> d*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 & DRX;
SETNZ16(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_w(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000200: case 00000220: case 0000230: case 0000240:
case 0000250: case 00000260: case 0000270: /* and.l -> d*/
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1 & DRX;
SETNZ32(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_l(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000420: case 00000430: case 0000440: case 0000450:
case 0000460: case 00000470: /* and.b -> ea*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = DRX & src1;
SETNZ8(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000520: case 00000530: case 0000540: case 0000550:
case 0000560: case 00000570: /* and.w -> ea*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = DRX & src1;
SETNZ16(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000620: case 00000630: case 0000640: case 0000650:
case 0000660: case 00000670: /* and.l -> ea*/
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&src1,&PC));
res = DRX & src1;
SETNZ32(res);
CLRF(FLAG_C|FLAG_V);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
default:
rc = STOP_ERROP;
}
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 1 0 1 | Register |Opc| 1 1 | effective address<>00x| add,adda
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 1 0 1 | Reg X | 1 |Length | 0 0 |R/M| Reg Y | addx
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0xd000:
switch (IR_0803) {
case 0000300: case 0000310: case 0000320: case 0000330:
case 0000340: case 0000350: case 0000360: case 0000370: /*adda.w*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&srca,&PC));
*AREG(IR_REGX) += EXTW(srca); /* note: no flag changes! */
break;
case 0000700: case 0000710: case 0000720: case 0000730:
case 0000740: case 0000750: case 0000760: case 0000770: /*adda.l*/
ASSERT_OK(ea_src_l(IR_EAMOD,IR_EAREG,&srca,&PC));
*AREG(IR_REGX) += srca; /* note: no flag changes! */
break;
case 0000400: /* addx.b d*/
res = m68k_add8(MASK_8L(DRY),DRX,CCR_X?1:0);
rc = ea_dst_b(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000410: /* addx.b -a*/
ASSERT_OK(ea_src_b(EA_APD,IR_REGY,&src1,&PC));
ASSERT_OK(ea_src_b(EA_APD,IR_REGX,&src2,&PC));
res = m68k_add8(src1,src2,CCR_X?1:0);
rc = ea_dst_b_rmw(EA_APD,IR_REGX,res);
break;
case 0000500: /* addx.w d*/
res = m68k_add16(MASK_16L(DRY),DRX,CCR_X?1:0,TRUE);
rc = ea_dst_w(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000510: /* addx.w -a*/
ASSERT_OK(ea_src_w(EA_APD,IR_REGY,&src1,&PC));
ASSERT_OK(ea_src_w(EA_APD,IR_REGX,&src2,&PC));
res = m68k_add16(src1,src2,CCR_X?1:0,TRUE);
rc = ea_dst_w_rmw(EA_APD,IR_REGX,res);
break;
case 0000600: /* addx.l d*/
res = m68k_add32((t_uint64)DRY,(t_uint64)DRX,CCR_X?1:0,TRUE);
rc = ea_dst_l(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000610: /* addx.l -a*/
ASSERT_OK(ea_src_l64(EA_APD,IR_REGY,&srcx1,&PC));
ASSERT_OK(ea_src_l64(EA_APD,IR_REGX,&srcx2,&PC));
res = m68k_add32(srcx1,srcx2,CCR_X?1:0,TRUE);
rc = ea_dst_l_rmw(EA_APD,IR_REGX,res);
break;
case 0000000: case 0000010: case 0000020: case 0000030:
case 0000040: case 0000050: case 0000060: case 0000070: /*add.b ->d*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_add8(src1,DRX,0);
rc = ea_dst_b(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000100: case 0000110: case 0000120: case 0000130:
case 0000140: case 0000150: case 0000160: case 0000170: /*add.w ->d*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_add16(src1,DRX,0,TRUE);
rc = ea_dst_w(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000200: case 0000210: case 0000220: case 0000230:
case 0000240: case 0000250: case 0000260: case 0000270: /*add.l ->d*/
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
res = m68k_add32(srcx1,(t_uint64)DRX,0,TRUE);
rc = ea_dst_l(EA_DDIR,IR_REGX,res,&PC);
break;
case 0000420: case 0000430: case 0000440: case 0000450:
case 0000460: case 0000470: /*add.b ->ea*/
ASSERT_OK(ea_src_b(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_add8(src1,DRX,0);
rc = ea_dst_b_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000520: case 0000530: case 0000540: case 0000550:
case 0000560: case 0000570: /*add.w ->ea*/
ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = m68k_add16(src1,DRX,0,TRUE);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
break;
case 0000620: case 0000630: case 0000640: case 0000650:
case 0000660: case 0000670: /*add.l ->ea*/
ASSERT_OK(ea_src_l64(IR_EAMOD,IR_EAREG,&srcx1,&PC));
res = m68k_add32(srcx1,(t_uint64)DRX,0,TRUE);
rc = ea_dst_l_rmw(IR_EAMOD,IR_EAREG,res);
break;
default:
rc = STOP_ERROP;
}
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 1 1 0 |Size/Reg X |dir|Length |i/r|Opcode2| Reg Y | asl,asr,lsl,lsr,rol,ror,roxl,roxr
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 1 1 0 | Opcode |dir| 1 1 | effective address | asl,asr,lsl,lsr,rol,ror,roxl,roxr
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0xe000:
switch (IR_1103) {
case 000040: case 001040: case 002040: case 003040:
case 004040: case 005040: case 006040: case 007040: /*asr.b r*/
cnt = DRX & 077;
goto do_asr8;
case 000000: case 001000: case 002000: case 003000:
case 004000: case 005000: case 006000: case 007000: /*asr.b #*/
cnt = quickarg[IR_REGX];
do_asr8: reg = DR+IR_REGY;
res = src1 = MASK_8L(*reg);
if (cnt) {
if (cnt<8) {
res >>= cnt;
if (MASK_8SGN(src1)) res |= shmask8[cnt];
SETF(src1&bitmask[cnt],FLAG_C|FLAG_X);
} else {
res = MASK_8SGN(src1) ? 0xff : 0x00;
SETF(res,FLAG_C|FLAG_X);
}
*reg = COMBINE8(*reg,res);
} else
CLRF(FLAG_C);
SETNZ8(res);
CLRF(FLAG_V);
rc =SCPE_OK; break;
case 000320: case 000330: case 000340: case 000350:
case 000360: case 000370: /*asr*/
cnt = 1;
goto do_asr16;
case 000140: case 001140: case 002140: case 003140:
case 004140: case 005140: case 006140: case 007140: /*asr.w r*/
cnt = DRX & 077;
IR = EA_DDIR | IR_REGY;
goto do_asr16;
case 000100: case 001100: case 002100: case 003100:
case 004100: case 005100: case 006100: case 007100: /*asr.w #*/
cnt = quickarg[IR_REGX];
IR = EA_DDIR | IR_REGY;
do_asr16: ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
if (cnt) {
if (cnt<16) {
res = src1 >> cnt;
if (MASK_16SGN(src1)) res |= shmask16[cnt];
SETF(src1&bitmask[cnt],FLAG_C|FLAG_X);
} else {
res = MASK_16SGN(src1) ? 0xffff : 0x0000;
SETF(res,FLAG_C|FLAG_X);
}
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
} else {
CLRF(FLAG_C);
res = src1;
rc = SCPE_OK;
}
SETNZ16(res);
CLRF(FLAG_V);
break;
case 000240: case 001240: case 002240: case 003240:
case 004240: case 005240: case 006240: case 007240: /*asr.l r*/
cnt = DRX & 077;
goto do_asr32;
case 000200: case 001200: case 002200: case 003200:
case 004200: case 005200: case 006200: case 007200: /*asr.l #*/
cnt = quickarg[IR_REGX];
do_asr32: reg = DR+IR_REGY;
res = src1 = *reg;
if (cnt) {
if (cnt < 32) {
res >>= cnt;
if (MASK_32SGN(src1)) res |= shmask32[cnt];
SETF(src1&bitmask[cnt],FLAG_C|FLAG_X);
} else {
res = MASK_32SGN(src1) ? 0xffffffff : 0x00000000;
SETF(res,FLAG_C|FLAG_X);
}
*reg = res;
} else CLRF(FLAG_C);
SETNZ32(res);
CLRF(FLAG_V);
rc = SCPE_OK; break;
case 000440: case 001440: case 002440: case 003440:
case 004440: case 005440: case 006440: case 007440: /*asl.b r*/
cnt = DRX & 077;
goto do_asl8;
case 000400: case 001400: case 002400: case 003400:
case 004400: case 005400: case 006400: case 007400: /*asl.b #*/
cnt = quickarg[IR_REGX];
do_asl8: reg = DR+IR_REGY;
res = src1 = MASK_8L(*reg);
if (cnt) {
if (cnt<8) {
res = src1 << cnt;
SETF(MASK_9(res),FLAG_C|FLAG_X);
src1 &= shmask8[cnt+1];
SETF(src1 && src1 != shmask8[cnt+1],FLAG_V);
} else {
res = 0;
SETF(cnt==8?(src1 & 1):0,FLAG_C|FLAG_X);
SETF(src1,FLAG_V);
}
*reg = COMBINE8(*reg,res);
} else CLRF(FLAG_C|FLAG_V);
SETNZ8(res);
rc = SCPE_OK; break;
case 000720: case 000730: case 000740: case 000750:
case 000760: case 000770: /*asl*/
cnt = 1;
goto do_asl16;
case 000540: case 001540: case 002540: case 003540:
case 004540: case 005540: case 006540: case 007540: /*asl.w r*/
cnt = DRX & 077;
IR = EA_DDIR | IR_REGY;
goto do_asl16;
case 000500: case 001500: case 002500: case 003500:
case 004500: case 005500: case 006500: case 007500: /*asl.w #*/
cnt = quickarg[IR_REGX];
IR = EA_DDIR | IR_REGY;
do_asl16: ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
if (cnt) {
if (cnt<16) {
res = src1 << cnt;
SETF(MASK_17(res),FLAG_C|FLAG_X);
src1 &= shmask16[cnt+1];
SETF(src1 && src1 != shmask16[cnt+1],FLAG_V);
} else {
res = 0;
SETF(cnt==16?(src1 & 1):0,FLAG_C|FLAG_X);
SETF(src1,FLAG_V);
}
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
} else {
CLRF(FLAG_C|FLAG_V);
rc = SCPE_OK;
}
SETNZ16(res);
break;
case 000640: case 001640: case 002640: case 003640:
case 004640: case 005640: case 006640: case 007640: /*asl.l r*/
cnt = DRX & 077;
goto do_asl32;
case 000600: case 001600: case 002600: case 003600:
case 004600: case 005600: case 006600: case 007600: /*asl.l #*/
cnt = quickarg[IR_REGX];
do_asl32: reg = DR+IR_REGY;
res = src1 = *reg;
if (cnt) {
if (cnt<32) {
res <<= cnt;
SETF(src1 & bitmask[32-cnt],FLAG_C|FLAG_X);
src1 &= shmask32[cnt+1];
SETF(src1 && src1 != shmask32[cnt+1],FLAG_V);
} else {
res = 0;
SETF(cnt==16?(src1 & 1):0,FLAG_C|FLAG_X);
SETF(src1,FLAG_V);
}
*reg = res;
} else CLRF(FLAG_C|FLAG_V);
SETNZ32(res);
rc = SCPE_OK; break;
case 000050: case 001050: case 002050: case 003050:
case 004050: case 005050: case 006050: case 007050: /*lsr.b r*/
cnt = DRX & 077;
goto do_lsr8;
case 000010: case 001010: case 002010: case 003010:
case 004010: case 005010: case 006010: case 007010: /*lsr.b #*/
cnt = quickarg[IR_REGX];
do_lsr8: reg = DR+IR_REGY;
res = src1 = MASK_8L(*reg);
if (cnt) {
if (cnt <= 8) {
res = src1 >> cnt;
SETF(src1&bitmask[cnt],FLAG_C|FLAG_X);
} else {
res = 0;
CLRF(FLAG_X|FLAG_C);
}
*reg = COMBINE8(*reg,res);
} else CLRF(FLAG_C);
CLRF(FLAG_V);
SETNZ8(res);
rc = SCPE_OK; break;
case 001320: case 001330: case 001340: case 001350:
case 001360: case 001370: /*lsr*/
cnt = 1;
goto do_lsr16;
case 000150: case 001150: case 002150: case 003150:
case 004150: case 005150: case 006150: case 007150: /*lsr.w r*/
cnt = DRX & 077;
IR = EA_DDIR | IR_REGY;
goto do_lsr16;
case 000110: case 001110: case 002110: case 003110:
case 004110: case 005110: case 006110: case 007110: /*lsr.w #*/
cnt = quickarg[IR_REGX];
IR = EA_DDIR | IR_REGY;
do_lsr16: ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
if (cnt) {
if (cnt <= 16) {
res = src1 >> cnt;
SETF(src1&bitmask[cnt],FLAG_C|FLAG_X);
} else {
res = 0;
CLRF(FLAG_X|FLAG_C);
}
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
} else {
CLRF(FLAG_C);
rc = SCPE_OK;
}
CLRF(FLAG_V);
SETNZ16(res);
break;
case 000250: case 001250: case 002250: case 003250:
case 004250: case 005250: case 006250: case 007250: /*lsr.l r*/
cnt = DRX & 077;
goto do_lsr32;
case 000210: case 001210: case 002210: case 003210:
case 004210: case 005210: case 006210: case 007210: /*lsr.l #*/
cnt = quickarg[IR_REGX];
do_lsr32: reg = DR+IR_REGY;
res = src1 = *reg;
if (cnt) {
if (cnt <= 32) {
res = src1 >> cnt;
SETF(src1&bitmask[cnt],FLAG_C|FLAG_X);
} else {
res = 0;
CLRF(FLAG_X|FLAG_C);
}
*reg = res;
} else CLRF(FLAG_C);
CLRF(FLAG_V);
SETNZ32(res);
rc = SCPE_OK;
break;
case 000450: case 001450: case 002450: case 003450:
case 004450: case 005450: case 006450: case 007450: /*lsl.b r*/
cnt = DRX & 077;
goto do_lsl8;
case 000410: case 001410: case 002410: case 003410:
case 004410: case 005410: case 006410: case 007410: /*lsl.b #*/
cnt = quickarg[IR_REGX];
do_lsl8: reg = DR+IR_REGY;
res = src1 = MASK_8L(*reg);
if (cnt) {
if (cnt <= 8) {
res = src1 << cnt;
SETF(src1&bitmask[9-cnt],FLAG_C|FLAG_X);
} else {
res = 0;
CLRF(FLAG_X|FLAG_C);
}
*reg = COMBINE8(*reg,res);
} else CLRF(FLAG_C);
SETNZ8(res);
CLRF(FLAG_V);
rc = SCPE_OK; break;
case 001720: case 001730: case 001740: case 001750:
case 001760: case 001770: /*lsl*/
cnt = 1;
goto do_lsl16;
case 000550: case 001550: case 002550: case 003550:
case 004550: case 005550: case 006550: case 007550: /*lsl.w r*/
cnt = DRX & 077;
IR = EA_DDIR | IR_REGY;
goto do_lsl16;
case 000510: case 001510: case 002510: case 003510:
case 004510: case 005510: case 006510: case 007510: /*lsl.w #*/
cnt = quickarg[IR_REGX];
IR = EA_DDIR | IR_REGY;
do_lsl16: ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&src1,&PC));
res = src1;
if (cnt) {
if (cnt <= 16) {
res = src1 << cnt;
SETF(src1&bitmask[17-cnt],FLAG_C|FLAG_X);
} else {
res = 0;
CLRF(FLAG_X|FLAG_C);
}
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
} else {
CLRF(FLAG_C);
rc = SCPE_OK;
}
SETNZ16(res);
CLRF(FLAG_V);
break;
case 000650: case 001650: case 002650: case 003650:
case 004650: case 005650: case 006650: case 007650: /*lsl.l r*/
cnt = DRX & 077;
goto do_lsl32;
case 000610: case 001610: case 002610: case 003610:
case 004610: case 005610: case 006610: case 007610: /*lsl.l #*/
cnt = quickarg[IR_REGX];
IR = EA_DDIR | IR_REGY;
do_lsl32: reg = DR+IR_REGY;
res = src1 = *reg;
if (cnt) {
if (cnt <= 32) {
res = src1 << cnt;
SETF(src1&bitmask[33-cnt],FLAG_C|FLAG_X);
} else {
res = 0;
CLRF(FLAG_X|FLAG_C);
}
*reg = res;
} else {
CLRF(FLAG_C);
rc = SCPE_OK;
}
SETNZ32(res);
CLRF(FLAG_V);
break;
case 000060: case 001060: case 002060: case 003060:
case 004060: case 005060: case 006060: case 007060: /*roxr.b r*/
cnt = DRX & 077;
goto do_roxr8;
case 000020: case 001020: case 002020: case 003020:
case 004020: case 005020: case 006020: case 007020: /*roxr.b #*/
cnt = quickarg[IR_REGX];
do_roxr8: reg = DR+IR_REGY;
res = MASK_8L(*reg);
if (cnt) {
cnt %= 9;
if (CCR_X) res |= BIT8;
res = (res>>cnt) | (res<<(9-cnt));
*reg = COMBINE8(*reg,res);
SETF(MASK_9(res),FLAG_X|FLAG_C);
} else SETF(CCR_X,FLAG_C);
SETNZ8(res);
CLRF(FLAG_V);
rc = SCPE_OK; break;
case 002320: case 002330: case 002340: case 002350:
case 002360: case 002370: /*roxr*/
cnt = 1;
goto do_roxr16;
case 000160: case 001160: case 002160: case 003160:
case 004160: case 005160: case 006160: case 007160: /*roxr.w r*/
cnt = DRX & 077;
IR = EA_DDIR | IR_REGY;
goto do_roxr16;
case 000120: case 001120: case 002120: case 003120:
case 004120: case 005120: case 006120: case 007120: /*roxr.w #*/
cnt = quickarg[IR_REGX];
IR = EA_DDIR | IR_REGY;
do_roxr16: ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&res,&PC));
if (cnt) {
cnt %= 17;
if (CCR_X) res |= BIT16;
res = (res>>cnt) | (res<<(17-cnt));
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
SETF(MASK_17(res),FLAG_X|FLAG_C);
} else {
SETF(CCR_X,FLAG_C);
rc = SCPE_OK;
}
SETNZ16(res);
CLRF(FLAG_V);
break;
case 000260: case 001260: case 002260: case 003260:
case 004260: case 005260: case 006260: case 007260: /*roxr.l r*/
cnt = DRX & 077;
goto do_roxr32;
case 000220: case 001220: case 002220: case 003220:
case 004220: case 005220: case 006220: case 007220: /*roxr.l #*/
cnt = quickarg[IR_REGX];
do_roxr32: reg = DR+IR_REGY;
resx = *reg;
if (cnt) {
cnt %= 33;
if (CCR_X) resx |= BIT32;
resx = (resx>>cnt) | (resx<<(33-cnt));
*reg = MASK_32L(resx);
SETF(MASK_33(resx),FLAG_X|FLAG_C);
} else SETF(CCR_X,FLAG_C);
SETNZ32(resx);
CLRF(FLAG_V);
rc = SCPE_OK; break;
case 000460: case 001460: case 002460: case 003460:
case 004460: case 005460: case 006460: case 007460: /*roxl.b r*/
cnt = DRX & 077;
goto do_roxl8;
case 000420: case 001420: case 002420: case 003420:
case 004420: case 005420: case 006420: case 007420: /*roxl.b #*/
cnt = quickarg[IR_REGX];
do_roxl8: reg = DR+IR_REGY;
res = MASK_8L(*reg);
if (cnt) {
cnt %= 9;
if (CCR_X) res |= BIT8;
res = (res<<cnt) | (res>>(9-cnt));
*reg = COMBINE8(*reg,res);
SETF(MASK_9(res),FLAG_X|FLAG_C);
} else SETF(CCR_X,FLAG_C);
SETNZ8(res);
CLRF(FLAG_V);
rc = SCPE_OK; break;
case 002720: case 002730: case 002740: case 002750:
case 002760: case 002770: /*roxl*/
cnt = 1;
goto do_roxl16;
case 000560: case 001560: case 002560: case 003560:
case 004560: case 005560: case 006560: case 007560: /*roxl.w r*/
cnt = DRX & 077;
IR = EA_DDIR | IR_REGY;
goto do_roxl16;
case 000520: case 001520: case 002520: case 003520:
case 004520: case 005520: case 006520: case 007520: /*roxl.w #*/
cnt = quickarg[IR_REGX];
IR = EA_DDIR | IR_REGY;
do_roxl16: ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&res,&PC));
if (cnt) {
cnt %= 17;
if (CCR_X) res |= BIT16;
res = (res<<cnt) | (res>>(17-cnt));
SETF(MASK_17(res),FLAG_X|FLAG_C);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
} else {
SETF(CCR_X,FLAG_C);
rc = SCPE_OK;
}
SETNZ16(res);
CLRF(FLAG_V);
break;
case 000660: case 001660: case 002660: case 003660:
case 004660: case 005660: case 006660: case 007660: /*roxl.l r*/
cnt = DRX & 077;
goto do_roxl32;
case 000620: case 001620: case 002620: case 003620:
case 004620: case 005620: case 006620: case 007620: /*roxl.l #*/
cnt = quickarg[IR_REGX];
do_roxl32: reg = DR+IR_REGY;
resx = *reg;
if (cnt) {
cnt %= 33;
if (CCR_X) resx |= BIT32;
resx = (resx<<cnt) | (resx>>(33-cnt));
SETF(MASK_33(resx),FLAG_X|FLAG_C);
*reg = MASK_32L(resx);
} else SETF(CCR_X,FLAG_C);
SETNZ32(resx);
CLRF(FLAG_V);
rc = SCPE_OK; break;
case 000070: case 001070: case 002070: case 003070:
case 004070: case 005070: case 006070: case 007070: /*ror.b r*/
cnt = DRX & 077;
goto do_ror8;
case 000030: case 001030: case 002030: case 003030:
case 004030: case 005030: case 006030: case 007030: /*ror.b #*/
cnt = quickarg[IR_REGX];
do_ror8: reg = DR+IR_REGY;
res = MASK_8L(*reg);
if (cnt) {
cnt &= 7;
res = (res>>cnt) | (res<<(8-cnt));
SETF(MASK_9(res),FLAG_C);
*reg = COMBINE8(*reg,res);
} else CLRF(FLAG_C);
SETNZ8(res);
CLRF(FLAG_V);
rc = SCPE_OK; break;
case 003320: case 003330: case 003340: case 003350:
case 003360: case 003370: /*ror*/
cnt = 1;
goto do_ror16;
case 000170: case 001170: case 002170: case 003170:
case 004170: case 005170: case 006170: case 007170: /*ror.w r*/
cnt = DRX & 077;
IR = EA_DDIR | IR_REGY;
goto do_ror16;
case 000130: case 001130: case 002130: case 003130:
case 004130: case 005130: case 006130: case 007130: /*ror.w #*/
cnt = quickarg[IR_REGX];
IR = EA_DDIR | IR_REGY;
do_ror16: ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&res,&PC));
if (cnt) {
cnt &= 15;
res = (res>>cnt) | (res<<(16-cnt));
SETF(MASK_17(res),FLAG_C);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
} else {
CLRF(FLAG_C);
rc = SCPE_OK;
}
SETNZ16(res);
CLRF(FLAG_V);
break;
case 000270: case 001270: case 002270: case 003270:
case 004270: case 005270: case 006270: case 007270: /*ror.l r*/
cnt = DRX & 077;
goto do_ror32;
case 000230: case 001230: case 002230: case 003230:
case 004230: case 005230: case 006230: case 007230: /*ror.l #*/
cnt = quickarg[IR_REGX];
do_ror32: reg = DR+IR_REGY;
resx = *reg;
if (cnt) {
cnt &= 31;
resx = (resx>>cnt) | (resx<<(32-cnt));
SETF(MASK_33(resx),FLAG_C);
*reg = (int32)resx;
} else {
CLRF(FLAG_C);
rc = SCPE_OK;
}
SETNZ32(resx);
CLRF(FLAG_V);
rc = SCPE_OK; break;
case 000470: case 001470: case 002470: case 003470:
case 004470: case 005470: case 006470: case 007470: /*rol.b r*/
cnt = DRX & 077;
goto do_rol8;
case 000430: case 001430: case 002430: case 003430:
case 004430: case 005430: case 006430: case 007430: /*rol.b #*/
cnt = quickarg[IR_REGX];
do_rol8: reg = DR+IR_REGY;
res = MASK_8L(*reg);
if (cnt) {
cnt &= 7;
res = (res<<cnt) | (res>>(8-cnt));
SETF(MASK_9(res),FLAG_C);
*reg = COMBINE8(*reg,res);
} else CLRF(FLAG_C);
SETNZ8(res);
CLRF(FLAG_V);
rc = SCPE_OK; break;
case 003720: case 003730: case 003740: case 003750:
case 003760: case 003770: /*rol*/
cnt = 1;
goto do_rol16;
case 000570: case 001570: case 002570: case 003570:
case 004570: case 005570: case 006570: case 007570: /*rol.w r*/
cnt = DRX & 077;
IR = EA_DDIR | IR_REGY;
goto do_rol16;
case 000530: case 001530: case 002530: case 003530:
case 004530: case 005530: case 006530: case 007530: /*rol.w #*/
cnt = quickarg[IR_REGX];
IR = EA_DDIR | IR_REGY;
do_rol16: ASSERT_OK(ea_src_w(IR_EAMOD,IR_EAREG,&res,&PC));
if (cnt) {
cnt &= 15;
res = (res<<cnt) | (res>>(16-cnt));
SETF(MASK_17(res),FLAG_C);
rc = ea_dst_w_rmw(IR_EAMOD,IR_EAREG,res);
} else {
CLRF(FLAG_C);
rc = SCPE_OK;
}
SETNZ16(res);
CLRF(FLAG_V);
break;
case 000670: case 001670: case 002670: case 003670:
case 004670: case 005670: case 006670: case 007670: /*rol.l r*/
cnt = DRX & 077;
goto do_rol32;
case 000630: case 001630: case 002630: case 003630:
case 004630: case 005630: case 006630: case 007630: /*rol.l #*/
cnt = quickarg[IR_REGX];
do_rol32: reg = DR+IR_REGY;
resx = (uint32)*reg;
if (cnt) {
cnt &= 31;
resx = (resx<<cnt) | (resx>>(32-cnt));
SETF(MASK_32L(resx),FLAG_C);
*reg = MASK_32L(resx);
} else CLRF(FLAG_C);
SETNZ32(resx);
CLRF(FLAG_V);
rc = SCPE_OK; break;
default:
rc = STOP_ERROP;
}
break;
/* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
* | 1 1 1 1 | Opcode | trapf
* +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---*/
case 0xf000:
rc = m68k_gen_exception(11,&PC); break;
/* unreachable */
default:
rc = STOP_ERROP; break;
}
/* handle tracing */
if (tracet0 || SR_T1) {
if (m68kcpu_unit->flags & UNIT_CPU_TRACE) {
/* leave loop */
sim_interval = -1;
rc = STOP_TRACE;
break;
}
IFDEBUG(DBG_CPU_EXC,fprintf(sim_deb,"CPU : [0x%08x] Exception: Tracebit set\n",PC));
ASSERT_OK(m68k_gen_exception(9,&PC));
/* remain in loop */
}
tracet0 = 0;
/* handle interrupts (sets/resets intpending) */
m68k_checkints(&PC);
/* handle STOP instr */
if (rc==STOP_HALT) {
if (m68kcpu_unit->flags & UNIT_CPU_STOP) {
PC -= 4; /* correct PC to point to STOP instr */
break;
}
if ((rc = m68k_stop(&PC)) != SCPE_OK)
break; /* does not return until interrupt occurs, will react to CTRL-E */
}
}
/* handle various exit codes */
switch (rc) {
case STOP_ERRADR: /* address error */
if ((m68kcpu_unit->flags & UNIT_CPU_EXC)==0) {
IFDEBUG(DBG_CPU_EXC,fprintf(sim_deb,"CPU : [0x%08x] Exception: Address error\n",PC));
if ((rc = m68k_gen_exception(3,&PC)) != SCPE_OK) {
/* double bus fault */
rc = STOP_DBF; /* cannot be masked, will stop simulator */
}
}
return rc;
case STOP_PCIO: /* cannot be masked, will stop simulator */
return rc;
case STOP_ERRIO: /* bus error */
if ((m68kcpu_unit->flags & UNIT_CPU_EXC)==0) {
IFDEBUG(DBG_CPU_EXC,fprintf(sim_deb,"CPU : [0x%08x] Exception: Bus error\n",PC));
if ((rc = m68k_gen_exception(2,&PC)) != SCPE_OK) {
/* double bus fault */
rc = STOP_DBF; /* cannot be masked, will stop simulator */
}
}
return rc;
case STOP_ERROP: /* illegal opcode */
if (!(m68kcpu_unit->flags & UNIT_CPU_EXC)) {
IFDEBUG(DBG_CPU_EXC,fprintf(sim_deb,"CPU : [0x%08x] Exception: Illegal opcode\n",PC));
rc = m68k_gen_exception(4,&PC);
}
return rc;
case STOP_PRVIO: /* privilege violation */
if (!(m68kcpu_unit->flags & UNIT_CPU_PRVIO)) {
IFDEBUG(DBG_CPU_EXC,fprintf(sim_deb,"CPU : [0x%08x] Exception: Privilege violation\n",PC));
rc = m68k_gen_exception(8,&PC);
}
break;
case STOP_IMPL:
return rc; /* leave sim_instr */
default:
return rc; /* leave sim_instr */
}
/* save state */
saved_PC = PC;
return rc;
}