Mercurial > hg > Members > kono > os9 > sbc09
view src/engine.c @ 85:4652761a60f9
mmeu fix
author | Shinji KONO <kono@ie.u-ryukyu.ac.jp> |
---|---|
date | Sun, 12 Aug 2018 12:34:56 +0900 |
parents | 9b661787d5ed |
children | ef64e3f4e229 |
line wrap: on
line source
/* 6809 Simulator V09. created 1994,1995 by L.C. Benschop. copyleft (c) 1994-2014 by the sbc09 team, see AUTHORS for more details. license: GNU General Public License version 2, see LICENSE for more details. This program simulates a 6809 processor. System dependencies: short must be 16 bits. char must be 8 bits. long must be more than 16 bits. arrays up to 65536 bytes must be supported. machine must be twos complement. Most Unix machines will work. For MSODS you need long pointers and you may have to malloc() the mem array of 65536 bytes. Special instructions: SWI2 writes char to stdout from register B. SWI3 reads char from stdout to register B, sets carry at EOF. (or when no key available when using term control). SWI retains its normal function. CWAI and SYNC stop simulator. Note: special instructions are gone for now. ACIA emulation at port $E000 Note: BIG_ENDIAN option is no longer needed. */ #include <stdio.h> #include <unistd.h> #define engine #include "v09.h" #define USLEEP 1000 Byte aca,acb; Byte *breg=&aca,*areg=&acb; static int tracetrick=0; extern long romstart; #ifndef USE_MMU static Byte mem1(Word adr) { if ((adr&0xfe00)==(IOPAGE&0xfe00)) return do_input(adr&0x1ff); return mem[adr]; } static void SETBYTE1(Word a,Byte n) { if ((a&0xfe00)==(IOPAGE&0xfe00)) do_output(a&0x1ff,n); if(!(a>=romstart))mem[a]=n; } #define mem(a) mem1(a) #define SETBYTE(a,n) SETBYTE1(a,n); #else int paddr(Word adr, Byte *immu) { if ((adr&0xfe00)==(IOPAGE&0xfe00)) return 0x38*0x2000+adr; return (immu[ (adr) >> 13 ] <<13 ) + ((adr) & 0x1fff ); } Byte * mem0(Byte *iphymem, Word adr, Byte *immu) { return & iphymem[ paddr(adr,immu) ]; } Byte mem1(Byte *iphymem, Word adr, Byte *immu) { if ((adr&0xfe00)==(IOPAGE&0xfe00)) return do_input(adr&0x1ff); Byte *p = mem0(iphymem, adr, immu); if(!(p-phymem>=rommemsize)) { return *p; } else { return 0xff; } } #define mem(a) mem1(iphymem,a,immu) Byte * SETBYTE1(Word a,Byte n, Byte *iphymem, Byte *immu) { Word adr = a; if ((adr&0xfe00)==(IOPAGE&0xfe00)) { do_output(adr&0x1ff,n); return mmu; } else { Byte *p = mem0(iphymem, adr,immu); if(!(p-phymem>=romstart)) { *p=n; } } return immu; } #define SETBYTE(a,n) immu=SETBYTE1(a,n,iphymem,immu); #endif #define GETWORD(a) (mem(a)<<8|mem((a)+1)) #define SETWORD(a,n) {Word a1=a;SETBYTE(a1,n>>8);SETBYTE(a1+1,n);} /* Macros for load and store of accumulators. Can be modified to check for port addresses */ // #define LOADAC(reg) if((eaddr&0xff00)!=(IOPAGE&0xff00))reg=mem(eaddr);else\ // reg=do_input(eaddr&0xff); // #define STOREAC(reg) if((eaddr&0xff00)!=(IOPAGE&0xff00))SETBYTE(eaddr,reg)else\ // do_output(eaddr&0xff,reg); /* Two bytes of a word are fetched separately because of the possible wrap-around at address $ffff and alignment */ #define IMMBYTE(b) b=mem(ipcreg++); #define IMMWORD(w) {w=GETWORD(ipcreg);ipcreg+=2;} #define PUSHBYTE(b) {--isreg;SETBYTE(isreg,b)} #define PUSHWORD(w) {isreg-=2;SETWORD(isreg,w)} #define PULLBYTE(b) b=mem(isreg++); #define PULLWORD(w) {w=GETWORD(isreg);isreg+=2;} #define PSHUBYTE(b) {--iureg;SETBYTE(iureg,b)} #define PSHUWORD(w) {iureg-=2;SETWORD(iureg,w)} #define PULUBYTE(b) b=mem(iureg++); #define PULUWORD(w) {w=GETWORD(iureg);iureg+=2;} #define SIGNED(b) ((Word)(b&0x80?b|0xff00:b)) #define GETDREG ((iareg<<8)|ibreg) #define SETDREG(n) {iareg=(n)>>8;ibreg=(n);} /* Macros for addressing modes (postbytes have their own code) */ #define DIRECT {IMMBYTE(eaddr) eaddr|=(idpreg<<8);} #define IMM8 {eaddr=ipcreg++;} #define IMM16 {eaddr=ipcreg;ipcreg+=2;} #define EXTENDED {IMMWORD(eaddr)} /* macros to set status flags */ #define SEC iccreg|=0x01; #define CLC iccreg&=0xfe; #define SEZ iccreg|=0x04; #define CLZ iccreg&=0xfb; #define SEN iccreg|=0x08; #define CLN iccreg&=0xf7; #define SEV iccreg|=0x02; #define CLV iccreg&=0xfd; #define SEH iccreg|=0x20; #define CLH iccreg&=0xdf; /* set N and Z flags depending on 8 or 16 bit result */ #define SETNZ8(b) {if(b)CLZ else SEZ if(b&0x80)SEN else CLN} #define SETNZ16(b) {if(b)CLZ else SEZ if(b&0x8000)SEN else CLN} #define SETSTATUS(a,b,res) if((a^b^res)&0x10) SEH else CLH \ if((a^b^res^(res>>1))&0x80)SEV else CLV \ if(res&0x100)SEC else CLC SETNZ8((Byte)res) #define SETSTATUSD(a,b,res) {if(res&0x10000) SEC else CLC \ if(((res>>1)^a^b^res)&0x8000) SEV else CLV \ SETNZ16((Word)res)} /* Macros for branch instructions */ #define BRANCH(f) if(!iflag){IMMBYTE(tb) if(f)ipcreg+=SIGNED(tb);}\ else{IMMWORD(tw) if(f)ipcreg+=tw;} #define NXORV ((iccreg&0x08)^((iccreg&0x02)<<2)) /* MAcros for setting/getting registers in TFR/EXG instructions */ #define GETREG(val,reg) switch(reg) {\ case 0: val=GETDREG;break;\ case 1: val=ixreg;break;\ case 2: val=iyreg;break;\ case 3: val=iureg;break;\ case 4: val=isreg;break;\ case 5: val=ipcreg;break;\ case 8: val=iareg;break;\ case 9: val=ibreg;break;\ case 10: val=iccreg;break;\ case 11: val=idpreg;break;} #define SETREG(val,reg) switch(reg) {\ case 0: SETDREG(val) break;\ case 1: ixreg=val;break;\ case 2: iyreg=val;break;\ case 3: iureg=val;break;\ case 4: isreg=val;break;\ case 5: ipcreg=val;break;\ case 8: iareg=val;break;\ case 9: ibreg=val;break;\ case 10: iccreg=val;break;\ case 11: idpreg=val;break;} #define LOADAC(reg) reg=mem(eaddr); #define STOREAC(reg) SETBYTE(eaddr,reg); #define LOADREGS ixreg=xreg;iyreg=yreg;\ iureg=ureg;isreg=sreg;\ ipcreg=pcreg;\ iareg=*areg;ibreg=*breg;\ idpreg=dpreg;iccreg=ccreg;immu=mmu; #define SAVEREGS xreg=ixreg;yreg=iyreg;\ ureg=iureg;sreg=isreg;\ pcreg=ipcreg;\ *areg=iareg;*breg=ibreg;\ dpreg=idpreg;ccreg=iccreg;mmu=immu; unsigned char haspostbyte[] = { /*0*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*1*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*2*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*3*/ 1,1,1,1,0,0,0,0,0,0,0,0,0,0,0,0, /*4*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*5*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*6*/ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /*7*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*8*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*9*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*A*/ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /*B*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*C*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*D*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /*E*/ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /*F*/ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, }; extern char *prog ; void interpr(void) { Word ixreg,iyreg,iureg,isreg,ipcreg; Byte idpreg,iccreg,iareg,ibreg; /* Make local variables for the registers. On a real processor (non-Intel) these could be implemented as fast registers. */ Word eaddr; /* effective address */ Byte ireg; /* instruction register */ Byte iflag; /* flag to indicate $10 or $11 prebyte */ Byte tb;Word tw; Byte *immu = 0; #ifdef USE_MMU Byte *iphymem = (Byte *)phymem; #endif LOADREGS for(;;){ if(attention) { if(tracing && ipcreg>=tracelo && ipcreg<=tracehi) { SAVEREGS #ifdef USE_MMU Byte *phyadr = mem0(phymem,ipcreg,immu); prog = (char *)(phyadr - ipcreg); #endif do_trace(tracefile); } if(irq) { if(irq==1&&!(iccreg&0x10)) { /* standard IRQ */ PUSHWORD(ipcreg) PUSHWORD(iureg) PUSHWORD(iyreg) PUSHWORD(ixreg) PUSHBYTE(idpreg) PUSHBYTE(ibreg) PUSHBYTE(iareg) iccreg|=0x80; PUSHBYTE(iccreg) iccreg|=0x90; ipcreg=GETWORD(0xfff8); } if(irq==2&&!(iccreg&0x40)) { /* Fast IRQ */ PUSHWORD(ipcreg) iccreg&=0x7f; PUSHBYTE(iccreg) iccreg|=0x50; ipcreg=GETWORD(0xfff6); } if(!tracing)attention=0; irq=0; } } if(escape){ SAVEREGS do_escape(); LOADREGS } iflag=0; flaginstr: /* $10 and $11 instructions return here */ ireg=mem(ipcreg++); if(haspostbyte[ireg]) { Byte postbyte=mem(ipcreg++); switch(postbyte) { case 0x00: eaddr=ixreg;break; case 0x01: eaddr=ixreg+1;break; case 0x02: eaddr=ixreg+2;break; case 0x03: eaddr=ixreg+3;break; case 0x04: eaddr=ixreg+4;break; case 0x05: eaddr=ixreg+5;break; case 0x06: eaddr=ixreg+6;break; case 0x07: eaddr=ixreg+7;break; case 0x08: eaddr=ixreg+8;break; case 0x09: eaddr=ixreg+9;break; case 0x0A: eaddr=ixreg+10;break; case 0x0B: eaddr=ixreg+11;break; case 0x0C: eaddr=ixreg+12;break; case 0x0D: eaddr=ixreg+13;break; case 0x0E: eaddr=ixreg+14;break; case 0x0F: eaddr=ixreg+15;break; case 0x10: eaddr=ixreg-16;break; case 0x11: eaddr=ixreg-15;break; case 0x12: eaddr=ixreg-14;break; case 0x13: eaddr=ixreg-13;break; case 0x14: eaddr=ixreg-12;break; case 0x15: eaddr=ixreg-11;break; case 0x16: eaddr=ixreg-10;break; case 0x17: eaddr=ixreg-9;break; case 0x18: eaddr=ixreg-8;break; case 0x19: eaddr=ixreg-7;break; case 0x1A: eaddr=ixreg-6;break; case 0x1B: eaddr=ixreg-5;break; case 0x1C: eaddr=ixreg-4;break; case 0x1D: eaddr=ixreg-3;break; case 0x1E: eaddr=ixreg-2;break; case 0x1F: eaddr=ixreg-1;break; case 0x20: eaddr=iyreg;break; case 0x21: eaddr=iyreg+1;break; case 0x22: eaddr=iyreg+2;break; case 0x23: eaddr=iyreg+3;break; case 0x24: eaddr=iyreg+4;break; case 0x25: eaddr=iyreg+5;break; case 0x26: eaddr=iyreg+6;break; case 0x27: eaddr=iyreg+7;break; case 0x28: eaddr=iyreg+8;break; case 0x29: eaddr=iyreg+9;break; case 0x2A: eaddr=iyreg+10;break; case 0x2B: eaddr=iyreg+11;break; case 0x2C: eaddr=iyreg+12;break; case 0x2D: eaddr=iyreg+13;break; case 0x2E: eaddr=iyreg+14;break; case 0x2F: eaddr=iyreg+15;break; case 0x30: eaddr=iyreg-16;break; case 0x31: eaddr=iyreg-15;break; case 0x32: eaddr=iyreg-14;break; case 0x33: eaddr=iyreg-13;break; case 0x34: eaddr=iyreg-12;break; case 0x35: eaddr=iyreg-11;break; case 0x36: eaddr=iyreg-10;break; case 0x37: eaddr=iyreg-9;break; case 0x38: eaddr=iyreg-8;break; case 0x39: eaddr=iyreg-7;break; case 0x3A: eaddr=iyreg-6;break; case 0x3B: eaddr=iyreg-5;break; case 0x3C: eaddr=iyreg-4;break; case 0x3D: eaddr=iyreg-3;break; case 0x3E: eaddr=iyreg-2;break; case 0x3F: eaddr=iyreg-1;break; case 0x40: eaddr=iureg;break; case 0x41: eaddr=iureg+1;break; case 0x42: eaddr=iureg+2;break; case 0x43: eaddr=iureg+3;break; case 0x44: eaddr=iureg+4;break; case 0x45: eaddr=iureg+5;break; case 0x46: eaddr=iureg+6;break; case 0x47: eaddr=iureg+7;break; case 0x48: eaddr=iureg+8;break; case 0x49: eaddr=iureg+9;break; case 0x4A: eaddr=iureg+10;break; case 0x4B: eaddr=iureg+11;break; case 0x4C: eaddr=iureg+12;break; case 0x4D: eaddr=iureg+13;break; case 0x4E: eaddr=iureg+14;break; case 0x4F: eaddr=iureg+15;break; case 0x50: eaddr=iureg-16;break; case 0x51: eaddr=iureg-15;break; case 0x52: eaddr=iureg-14;break; case 0x53: eaddr=iureg-13;break; case 0x54: eaddr=iureg-12;break; case 0x55: eaddr=iureg-11;break; case 0x56: eaddr=iureg-10;break; case 0x57: eaddr=iureg-9;break; case 0x58: eaddr=iureg-8;break; case 0x59: eaddr=iureg-7;break; case 0x5A: eaddr=iureg-6;break; case 0x5B: eaddr=iureg-5;break; case 0x5C: eaddr=iureg-4;break; case 0x5D: eaddr=iureg-3;break; case 0x5E: eaddr=iureg-2;break; case 0x5F: eaddr=iureg-1;break; case 0x60: eaddr=isreg;break; case 0x61: eaddr=isreg+1;break; case 0x62: eaddr=isreg+2;break; case 0x63: eaddr=isreg+3;break; case 0x64: eaddr=isreg+4;break; case 0x65: eaddr=isreg+5;break; case 0x66: eaddr=isreg+6;break; case 0x67: eaddr=isreg+7;break; case 0x68: eaddr=isreg+8;break; case 0x69: eaddr=isreg+9;break; case 0x6A: eaddr=isreg+10;break; case 0x6B: eaddr=isreg+11;break; case 0x6C: eaddr=isreg+12;break; case 0x6D: eaddr=isreg+13;break; case 0x6E: eaddr=isreg+14;break; case 0x6F: eaddr=isreg+15;break; case 0x70: eaddr=isreg-16;break; case 0x71: eaddr=isreg-15;break; case 0x72: eaddr=isreg-14;break; case 0x73: eaddr=isreg-13;break; case 0x74: eaddr=isreg-12;break; case 0x75: eaddr=isreg-11;break; case 0x76: eaddr=isreg-10;break; case 0x77: eaddr=isreg-9;break; case 0x78: eaddr=isreg-8;break; case 0x79: eaddr=isreg-7;break; case 0x7A: eaddr=isreg-6;break; case 0x7B: eaddr=isreg-5;break; case 0x7C: eaddr=isreg-4;break; case 0x7D: eaddr=isreg-3;break; case 0x7E: eaddr=isreg-2;break; case 0x7F: eaddr=isreg-1;break; case 0x80: eaddr=ixreg;ixreg++;break; case 0x81: eaddr=ixreg;ixreg+=2;break; case 0x82: ixreg--;eaddr=ixreg;break; case 0x83: ixreg-=2;eaddr=ixreg;break; case 0x84: eaddr=ixreg;break; case 0x85: eaddr=ixreg+SIGNED(ibreg);break; case 0x86: eaddr=ixreg+SIGNED(iareg);break; case 0x87: eaddr=0;break; /*ILELGAL*/ case 0x88: IMMBYTE(eaddr);eaddr=ixreg+SIGNED(eaddr);break; case 0x89: IMMWORD(eaddr);eaddr+=ixreg;break; case 0x8A: eaddr=0;break; /*ILLEGAL*/ case 0x8B: eaddr=ixreg+GETDREG;break; case 0x8C: IMMBYTE(eaddr);eaddr=ipcreg+SIGNED(eaddr);break; case 0x8D: IMMWORD(eaddr);eaddr+=ipcreg;break; case 0x8E: eaddr=0;break; /*ILLEGAL*/ case 0x8F: IMMWORD(eaddr);break; case 0x90: eaddr=ixreg;ixreg++;eaddr=GETWORD(eaddr);break; case 0x91: eaddr=ixreg;ixreg+=2;eaddr=GETWORD(eaddr);break; case 0x92: ixreg--;eaddr=ixreg;eaddr=GETWORD(eaddr);break; case 0x93: ixreg-=2;eaddr=ixreg;eaddr=GETWORD(eaddr);break; case 0x94: eaddr=ixreg;eaddr=GETWORD(eaddr);break; case 0x95: eaddr=ixreg+SIGNED(ibreg);eaddr=GETWORD(eaddr);break; case 0x96: eaddr=ixreg+SIGNED(iareg);eaddr=GETWORD(eaddr);break; case 0x97: eaddr=0;break; /*ILELGAL*/ case 0x98: IMMBYTE(eaddr);eaddr=ixreg+SIGNED(eaddr); eaddr=GETWORD(eaddr);break; case 0x99: IMMWORD(eaddr);eaddr+=ixreg;eaddr=GETWORD(eaddr);break; case 0x9A: eaddr=0;break; /*ILLEGAL*/ case 0x9B: eaddr=ixreg+GETDREG;eaddr=GETWORD(eaddr);break; case 0x9C: IMMBYTE(eaddr);eaddr=ipcreg+SIGNED(eaddr); eaddr=GETWORD(eaddr);break; case 0x9D: IMMWORD(eaddr);eaddr+=ipcreg;eaddr=GETWORD(eaddr);break; case 0x9E: eaddr=0;break; /*ILLEGAL*/ case 0x9F: IMMWORD(eaddr);eaddr=GETWORD(eaddr);break; case 0xA0: eaddr=iyreg;iyreg++;break; case 0xA1: eaddr=iyreg;iyreg+=2;break; case 0xA2: iyreg--;eaddr=iyreg;break; case 0xA3: iyreg-=2;eaddr=iyreg;break; case 0xA4: eaddr=iyreg;break; case 0xA5: eaddr=iyreg+SIGNED(ibreg);break; case 0xA6: eaddr=iyreg+SIGNED(iareg);break; case 0xA7: eaddr=0;break; /*ILELGAL*/ case 0xA8: IMMBYTE(eaddr);eaddr=iyreg+SIGNED(eaddr);break; case 0xA9: IMMWORD(eaddr);eaddr+=iyreg;break; case 0xAA: eaddr=0;break; /*ILLEGAL*/ case 0xAB: eaddr=iyreg+GETDREG;break; case 0xAC: IMMBYTE(eaddr);eaddr=ipcreg+SIGNED(eaddr);break; case 0xAD: IMMWORD(eaddr);eaddr+=ipcreg;break; case 0xAE: eaddr=0;break; /*ILLEGAL*/ case 0xAF: IMMWORD(eaddr);break; case 0xB0: eaddr=iyreg;iyreg++;eaddr=GETWORD(eaddr);break; case 0xB1: eaddr=iyreg;iyreg+=2;eaddr=GETWORD(eaddr);break; case 0xB2: iyreg--;eaddr=iyreg;eaddr=GETWORD(eaddr);break; case 0xB3: iyreg-=2;eaddr=iyreg;eaddr=GETWORD(eaddr);break; case 0xB4: eaddr=iyreg;eaddr=GETWORD(eaddr);break; case 0xB5: eaddr=iyreg+SIGNED(ibreg);eaddr=GETWORD(eaddr);break; case 0xB6: eaddr=iyreg+SIGNED(iareg);eaddr=GETWORD(eaddr);break; case 0xB7: eaddr=0;break; /*ILELGAL*/ case 0xB8: IMMBYTE(eaddr);eaddr=iyreg+SIGNED(eaddr); eaddr=GETWORD(eaddr);break; case 0xB9: IMMWORD(eaddr);eaddr+=iyreg;eaddr=GETWORD(eaddr);break; case 0xBA: eaddr=0;break; /*ILLEGAL*/ case 0xBB: eaddr=iyreg+GETDREG;eaddr=GETWORD(eaddr);break; case 0xBC: IMMBYTE(eaddr);eaddr=ipcreg+SIGNED(eaddr); eaddr=GETWORD(eaddr);break; case 0xBD: IMMWORD(eaddr);eaddr+=ipcreg;eaddr=GETWORD(eaddr);break; case 0xBE: eaddr=0;break; /*ILLEGAL*/ case 0xBF: IMMWORD(eaddr);eaddr=GETWORD(eaddr);break; case 0xC0: eaddr=iureg;iureg++;break; case 0xC1: eaddr=iureg;iureg+=2;break; case 0xC2: iureg--;eaddr=iureg;break; case 0xC3: iureg-=2;eaddr=iureg;break; case 0xC4: eaddr=iureg;break; case 0xC5: eaddr=iureg+SIGNED(ibreg);break; case 0xC6: eaddr=iureg+SIGNED(iareg);break; case 0xC7: eaddr=0;break; /*ILELGAL*/ case 0xC8: IMMBYTE(eaddr);eaddr=iureg+SIGNED(eaddr);break; case 0xC9: IMMWORD(eaddr);eaddr+=iureg;break; case 0xCA: eaddr=0;break; /*ILLEGAL*/ case 0xCB: eaddr=iureg+GETDREG;break; case 0xCC: IMMBYTE(eaddr);eaddr=ipcreg+SIGNED(eaddr);break; case 0xCD: IMMWORD(eaddr);eaddr+=ipcreg;break; case 0xCE: eaddr=0;break; /*ILLEGAL*/ case 0xCF: IMMWORD(eaddr);break; case 0xD0: eaddr=iureg;iureg++;eaddr=GETWORD(eaddr);break; case 0xD1: eaddr=iureg;iureg+=2;eaddr=GETWORD(eaddr);break; case 0xD2: iureg--;eaddr=iureg;eaddr=GETWORD(eaddr);break; case 0xD3: iureg-=2;eaddr=iureg;eaddr=GETWORD(eaddr);break; case 0xD4: eaddr=iureg;eaddr=GETWORD(eaddr);break; case 0xD5: eaddr=iureg+SIGNED(ibreg);eaddr=GETWORD(eaddr);break; case 0xD6: eaddr=iureg+SIGNED(iareg);eaddr=GETWORD(eaddr);break; case 0xD7: eaddr=0;break; /*ILELGAL*/ case 0xD8: IMMBYTE(eaddr);eaddr=iureg+SIGNED(eaddr); eaddr=GETWORD(eaddr);break; case 0xD9: IMMWORD(eaddr);eaddr+=iureg;eaddr=GETWORD(eaddr);break; case 0xDA: eaddr=0;break; /*ILLEGAL*/ case 0xDB: eaddr=iureg+GETDREG;eaddr=GETWORD(eaddr);break; case 0xDC: IMMBYTE(eaddr);eaddr=ipcreg+SIGNED(eaddr); eaddr=GETWORD(eaddr);break; case 0xDD: IMMWORD(eaddr);eaddr+=ipcreg;eaddr=GETWORD(eaddr);break; case 0xDE: eaddr=0;break; /*ILLEGAL*/ case 0xDF: IMMWORD(eaddr);eaddr=GETWORD(eaddr);break; case 0xE0: eaddr=isreg;isreg++;break; case 0xE1: eaddr=isreg;isreg+=2;break; case 0xE2: isreg--;eaddr=isreg;break; case 0xE3: isreg-=2;eaddr=isreg;break; case 0xE4: eaddr=isreg;break; case 0xE5: eaddr=isreg+SIGNED(ibreg);break; case 0xE6: eaddr=isreg+SIGNED(iareg);break; case 0xE7: eaddr=0;break; /*ILELGAL*/ case 0xE8: IMMBYTE(eaddr);eaddr=isreg+SIGNED(eaddr);break; case 0xE9: IMMWORD(eaddr);eaddr+=isreg;break; case 0xEA: eaddr=0;break; /*ILLEGAL*/ case 0xEB: eaddr=isreg+GETDREG;break; case 0xEC: IMMBYTE(eaddr);eaddr=ipcreg+SIGNED(eaddr);break; case 0xED: IMMWORD(eaddr);eaddr+=ipcreg;break; case 0xEE: eaddr=0;break; /*ILLEGAL*/ case 0xEF: IMMWORD(eaddr);break; case 0xF0: eaddr=isreg;isreg++;eaddr=GETWORD(eaddr);break; case 0xF1: eaddr=isreg;isreg+=2;eaddr=GETWORD(eaddr);break; case 0xF2: isreg--;eaddr=isreg;eaddr=GETWORD(eaddr);break; case 0xF3: isreg-=2;eaddr=isreg;eaddr=GETWORD(eaddr);break; case 0xF4: eaddr=isreg;eaddr=GETWORD(eaddr);break; case 0xF5: eaddr=isreg+SIGNED(ibreg);eaddr=GETWORD(eaddr);break; case 0xF6: eaddr=isreg+SIGNED(iareg);eaddr=GETWORD(eaddr);break; case 0xF7: eaddr=0;break; /*ILELGAL*/ case 0xF8: IMMBYTE(eaddr);eaddr=isreg+SIGNED(eaddr); eaddr=GETWORD(eaddr);break; case 0xF9: IMMWORD(eaddr);eaddr+=isreg;eaddr=GETWORD(eaddr);break; case 0xFA: eaddr=0;break; /*ILLEGAL*/ case 0xFB: eaddr=isreg+GETDREG;eaddr=GETWORD(eaddr);break; case 0xFC: IMMBYTE(eaddr);eaddr=ipcreg+SIGNED(eaddr); eaddr=GETWORD(eaddr);break; case 0xFD: IMMWORD(eaddr);eaddr+=ipcreg;eaddr=GETWORD(eaddr);break; case 0xFE: eaddr=0;break; /*ILLEGAL*/ case 0xFF: IMMWORD(eaddr);eaddr=GETWORD(eaddr);break; } } switch(ireg) { case 0x00: /*NEG direct*/ DIRECT tw=-mem(eaddr);SETSTATUS(0,mem(eaddr),tw) SETBYTE(eaddr,tw)break; case 0x01: break;/*ILLEGAL*/ case 0x02: break;/*ILLEGAL*/ case 0x03: /*COM direct*/ DIRECT tb=~mem(eaddr);SETNZ8(tb);SEC CLV SETBYTE(eaddr,tb)break; case 0x04: /*LSR direct*/ DIRECT tb=mem(eaddr);if(tb&0x01)SEC else CLC if(tb&0x10)SEH else CLH tb>>=1;SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x05: break;/* ILLEGAL*/ case 0x06: /*ROR direct*/ DIRECT tb=(iccreg&0x01)<<7; if(mem(eaddr)&0x01)SEC else CLC tw=(mem(eaddr)>>1)+tb;SETNZ8(tw) SETBYTE(eaddr,tw) break; case 0x07: /*ASR direct*/ DIRECT tb=mem(eaddr);if(tb&0x01)SEC else CLC if(tb&0x10)SEH else CLH tb>>=1; if(tb&0x40)tb|=0x80;SETBYTE(eaddr,tb)SETNZ8(tb) break; case 0x08: /*ASL direct*/ DIRECT tw=mem(eaddr)<<1; SETSTATUS(mem(eaddr),mem(eaddr),tw) SETBYTE(eaddr,tw)break; case 0x09: /*ROL direct*/ DIRECT tb=mem(eaddr);tw=iccreg&0x01; if(tb&0x80)SEC else CLC if((tb&0x80)^((tb<<1)&0x80))SEV else CLV tb=(tb<<1)+tw;SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x0A: /*DEC direct*/ DIRECT tb=mem(eaddr)-1;if(tb==0x7F)SEV else CLV SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x0B: break; /*ILLEGAL*/ case 0x0C: /*INC direct*/ DIRECT tb=mem(eaddr)+1;if(tb==0x80)SEV else CLV SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x0D: /*TST direct*/ DIRECT tb=mem(eaddr);SETNZ8(tb) break; case 0x0E: /*JMP direct*/ DIRECT ipcreg=eaddr;break; case 0x0F: /*CLR direct*/ DIRECT SETBYTE(eaddr,0);CLN CLV SEZ CLC break; case 0x10: /* flag10 */ iflag=1;goto flaginstr; case 0x11: /* flag11 */ iflag=2;goto flaginstr; case 0x12: /* NOP */ break; case 0x13: /* SYNC */ do usleep(USLEEP); /* Wait for IRQ */ while(!irq && !attention); if(iccreg&0x40)tracetrick=1; break; case 0x14: break; /*ILLEGAL*/ case 0x15: break; /*ILLEGAL*/ case 0x16: /*LBRA*/ IMMWORD(eaddr) ipcreg+=eaddr;break; case 0x17: /*LBSR*/ IMMWORD(eaddr) PUSHWORD(ipcreg) ipcreg+=eaddr;break; case 0x18: break; /*ILLEGAL*/ case 0x19: /* DAA*/ tw=iareg; if(iccreg&0x20)tw+=6; if((tw&0x0f)>9)tw+=6; if(iccreg&0x01)tw+=0x60; if((tw&0xf0)>0x90)tw+=0x60; if(tw&0x100)SEC iareg=tw;break; case 0x1A: /* ORCC*/ IMMBYTE(tb) iccreg|=tb;break; case 0x1B: break; /*ILLEGAL*/ case 0x1C: /* ANDCC*/ IMMBYTE(tb) iccreg&=tb;break; case 0x1D: /* SEX */ tw=SIGNED(ibreg); SETNZ16(tw) SETDREG(tw) break; case 0x1E: /* EXG */ IMMBYTE(tb) {Word t2;GETREG(tw,tb>>4) GETREG(t2,tb&15) SETREG(t2,tb>>4) SETREG(tw,tb&15) } break; case 0x1F: /* TFR */ IMMBYTE(tb) GETREG(tw,tb>>4) SETREG(tw,tb&15) break; case 0x20: /* (L)BRA*/ BRANCH(1) break; case 0x21: /* (L)BRN*/ BRANCH(0) break; case 0x22: /* (L)BHI*/ BRANCH(!(iccreg&0x05)) break; case 0x23: /* (L)BLS*/ BRANCH(iccreg&0x05) break; case 0x24: /* (L)BCC*/ BRANCH(!(iccreg&0x01)) break; case 0x25: /* (L)BCS*/ BRANCH(iccreg&0x01) break; case 0x26: /* (L)BNE*/ BRANCH(!(iccreg&0x04)) break; case 0x27: /* (L)BEQ*/ BRANCH(iccreg&0x04) break; case 0x28: /* (L)BVC*/ BRANCH(!(iccreg&0x02)) break; case 0x29: /* (L)BVS*/ BRANCH(iccreg&0x02) break; case 0x2A: /* (L)BPL*/ BRANCH(!(iccreg&0x08)) break; case 0x2B: /* (L)BMI*/ BRANCH(iccreg&0x08) break; case 0x2C: /* (L)BGE*/ BRANCH(!NXORV) break; case 0x2D: /* (L)BLT*/ BRANCH(NXORV) break; case 0x2E: /* (L)BGT*/ BRANCH(!(NXORV||iccreg&0x04)) break; case 0x2F: /* (L)BLE*/ BRANCH(NXORV||iccreg&0x04) break; case 0x30: /* LEAX*/ ixreg=eaddr; if(ixreg) CLZ else SEZ break; case 0x31: /* LEAY*/ iyreg=eaddr; if(iyreg) CLZ else SEZ break; case 0x32: /* LEAS*/ isreg=eaddr;break; case 0x33: /* LEAU*/ iureg=eaddr;break; case 0x34: /* PSHS*/ IMMBYTE(tb) if(tb&0x80)PUSHWORD(ipcreg) if(tb&0x40)PUSHWORD(iureg) if(tb&0x20)PUSHWORD(iyreg) if(tb&0x10)PUSHWORD(ixreg) if(tb&0x08)PUSHBYTE(idpreg) if(tb&0x04)PUSHBYTE(ibreg) if(tb&0x02)PUSHBYTE(iareg) if(tb&0x01)PUSHBYTE(iccreg) break; case 0x35: /* PULS*/ IMMBYTE(tb) if(tb&0x01)PULLBYTE(iccreg) if(tb&0x02)PULLBYTE(iareg) if(tb&0x04)PULLBYTE(ibreg) if(tb&0x08)PULLBYTE(idpreg) if(tb&0x10)PULLWORD(ixreg) if(tb&0x20)PULLWORD(iyreg) if(tb&0x40)PULLWORD(iureg) if(tb&0x80)PULLWORD(ipcreg) if(tracetrick&&tb==0xff) { /* Arrange fake FIRQ after next insn for hardware tracing */ tracetrick=0; irq=2; attention=1; goto flaginstr; } break; case 0x36: /* PSHU*/ IMMBYTE(tb) if(tb&0x80)PSHUWORD(ipcreg) if(tb&0x40)PSHUWORD(isreg) if(tb&0x20)PSHUWORD(iyreg) if(tb&0x10)PSHUWORD(ixreg) if(tb&0x08)PSHUBYTE(idpreg) if(tb&0x04)PSHUBYTE(ibreg) if(tb&0x02)PSHUBYTE(iareg) if(tb&0x01)PSHUBYTE(iccreg) break; case 0x37: /* PULU*/ IMMBYTE(tb) if(tb&0x01)PULUBYTE(iccreg) if(tb&0x02)PULUBYTE(iareg) if(tb&0x04)PULUBYTE(ibreg) if(tb&0x08)PULUBYTE(idpreg) if(tb&0x10)PULUWORD(ixreg) if(tb&0x20)PULUWORD(iyreg) if(tb&0x40)PULUWORD(isreg) if(tb&0x80)PULUWORD(ipcreg) break; case 0x39: /* RTS*/ PULLWORD(ipcreg) break; case 0x3A: /* ABX*/ ixreg+=ibreg; break; case 0x3B: /* RTI*/ PULLBYTE(iccreg) tb=iccreg&0x80; if(tb) { PULLBYTE(iareg) PULLBYTE(ibreg) PULLBYTE(idpreg) PULLWORD(ixreg) PULLWORD(iyreg) PULLWORD(iureg) } PULLWORD(ipcreg) break; case 0x3C: /* CWAI*/ IMMBYTE(tb) PUSHWORD(ipcreg) PUSHWORD(iureg) PUSHWORD(iyreg) PUSHWORD(ixreg) PUSHBYTE(idpreg) PUSHBYTE(ibreg) PUSHBYTE(iareg) PUSHBYTE(iccreg) iccreg&=tb; iccreg|=0x80; do usleep(USLEEP); /* Wait for IRQ */ while(!attention && !((irq==1&&!(iccreg&0x10))||(irq==2&&!(iccreg&0x040)))); if(irq==1)ipcreg=GETWORD(0xfff8); else ipcreg=GETWORD(0xfff6); irq=0; if(!tracing)attention=0; break; case 0x3D: /* MUL*/ tw=iareg*ibreg; if(tw)CLZ else SEZ if(tw&0x80) SEC else CLC SETDREG(tw) break; case 0x3E: break; /*ILLEGAL*/ case 0x3F: /* SWI (SWI2 SWI3)*/ { PUSHWORD(ipcreg) PUSHWORD(iureg) PUSHWORD(iyreg) PUSHWORD(ixreg) PUSHBYTE(idpreg) PUSHBYTE(ibreg) PUSHBYTE(iareg) iccreg|=0x80; PUSHBYTE(iccreg) if(!iflag)iccreg|=0x50; switch(iflag) { case 0:ipcreg=GETWORD(0xfffa);break; case 1:ipcreg=GETWORD(0xfff4);break; case 2:ipcreg=GETWORD(0xfff2);break; } }break; case 0x40: /*NEGA*/ tw=-iareg;SETSTATUS(0,iareg,tw) iareg=tw;break; case 0x41: break;/*ILLEGAL*/ case 0x42: break;/*ILLEGAL*/ case 0x43: /*COMA*/ tb=~iareg;SETNZ8(tb);SEC CLV iareg=tb;break; case 0x44: /*LSRA*/ tb=iareg;if(tb&0x01)SEC else CLC if(tb&0x10)SEH else CLH tb>>=1;SETNZ8(tb) iareg=tb;break; case 0x45: break;/* ILLEGAL*/ case 0x46: /*RORA*/ tb=(iccreg&0x01)<<7; if(iareg&0x01)SEC else CLC iareg=(iareg>>1)+tb;SETNZ8(iareg) break; case 0x47: /*ASRA*/ tb=iareg;if(tb&0x01)SEC else CLC if(tb&0x10)SEH else CLH tb>>=1; if(tb&0x40)tb|=0x80;iareg=tb;SETNZ8(tb) break; case 0x48: /*ASLA*/ tw=iareg<<1; SETSTATUS(iareg,iareg,tw) iareg=tw;break; case 0x49: /*ROLA*/ tb=iareg;tw=iccreg&0x01; if(tb&0x80)SEC else CLC if((tb&0x80)^((tb<<1)&0x80))SEV else CLV tb=(tb<<1)+tw;SETNZ8(tb) iareg=tb;break; case 0x4A: /*DECA*/ tb=iareg-1;if(tb==0x7F)SEV else CLV SETNZ8(tb) iareg=tb;break; case 0x4B: break; /*ILLEGAL*/ case 0x4C: /*INCA*/ tb=iareg+1;if(tb==0x80)SEV else CLV SETNZ8(tb) iareg=tb;break; case 0x4D: /*TSTA*/ SETNZ8(iareg) break; case 0x4E: break; /*ILLEGAL*/ case 0x4F: /*CLRA*/ iareg=0;CLN CLV SEZ CLC break; case 0x50: /*NEGB*/ tw=-ibreg;SETSTATUS(0,ibreg,tw) ibreg=tw;break; case 0x51: break;/*ILLEGAL*/ case 0x52: break;/*ILLEGAL*/ case 0x53: /*COMB*/ tb=~ibreg;SETNZ8(tb);SEC CLV ibreg=tb;break; case 0x54: /*LSRB*/ tb=ibreg;if(tb&0x01)SEC else CLC if(tb&0x10)SEH else CLH tb>>=1;SETNZ8(tb) ibreg=tb;break; case 0x55: break;/* ILLEGAL*/ case 0x56: /*RORB*/ tb=(iccreg&0x01)<<7; if(ibreg&0x01)SEC else CLC ibreg=(ibreg>>1)+tb;SETNZ8(ibreg) break; case 0x57: /*ASRB*/ tb=ibreg;if(tb&0x01)SEC else CLC if(tb&0x10)SEH else CLH tb>>=1; if(tb&0x40)tb|=0x80;ibreg=tb;SETNZ8(tb) break; case 0x58: /*ASLB*/ tw=ibreg<<1; SETSTATUS(ibreg,ibreg,tw) ibreg=tw;break; case 0x59: /*ROLB*/ tb=ibreg;tw=iccreg&0x01; if(tb&0x80)SEC else CLC if((tb&0x80)^((tb<<1)&0x80))SEV else CLV tb=(tb<<1)+tw;SETNZ8(tb) ibreg=tb;break; case 0x5A: /*DECB*/ tb=ibreg-1;if(tb==0x7F)SEV else CLV SETNZ8(tb) ibreg=tb;break; case 0x5B: break; /*ILLEGAL*/ case 0x5C: /*INCB*/ tb=ibreg+1;if(tb==0x80)SEV else CLV SETNZ8(tb) ibreg=tb;break; case 0x5D: /*TSTB*/ SETNZ8(ibreg) break; case 0x5E: break; /*ILLEGAL*/ case 0x5F: /*CLRB*/ ibreg=0;CLN CLV SEZ CLC break; case 0x60: /*NEG indexed*/ tw=-mem(eaddr);SETSTATUS(0,mem(eaddr),tw) SETBYTE(eaddr,tw)break; case 0x61: break;/*ILLEGAL*/ case 0x62: break;/*ILLEGAL*/ case 0x63: /*COM indexed*/ tb=~mem(eaddr);SETNZ8(tb);SEC CLV SETBYTE(eaddr,tb)break; case 0x64: /*LSR indexed*/ tb=mem(eaddr);if(tb&0x01)SEC else CLC if(tb&0x10)SEH else CLH tb>>=1;SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x65: break;/* ILLEGAL*/ case 0x66: /*ROR indexed*/ tb=(iccreg&0x01)<<7; if(mem(eaddr)&0x01)SEC else CLC tw=(mem(eaddr)>>1)+tb;SETNZ8(tw) SETBYTE(eaddr,tw) break; case 0x67: /*ASR indexed*/ tb=mem(eaddr);if(tb&0x01)SEC else CLC if(tb&0x10)SEH else CLH tb>>=1; if(tb&0x40)tb|=0x80;SETBYTE(eaddr,tb)SETNZ8(tb) break; case 0x68: /*ASL indexed*/ tw=mem(eaddr)<<1; SETSTATUS(mem(eaddr),mem(eaddr),tw) SETBYTE(eaddr,tw)break; case 0x69: /*ROL indexed*/ tb=mem(eaddr);tw=iccreg&0x01; if(tb&0x80)SEC else CLC if((tb&0x80)^((tb<<1)&0x80))SEV else CLV tb=(tb<<1)+tw;SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x6A: /*DEC indexed*/ tb=mem(eaddr)-1;if(tb==0x7F)SEV else CLV SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x6B: break; /*ILLEGAL*/ case 0x6C: /*INC indexed*/ tb=mem(eaddr)+1;if(tb==0x80)SEV else CLV SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x6D: /*TST indexed*/ tb=mem(eaddr);SETNZ8(tb) break; case 0x6E: /*JMP indexed*/ ipcreg=eaddr;break; case 0x6F: /*CLR indexed*/ SETBYTE(eaddr,0)CLN CLV SEZ CLC break; case 0x70: /*NEG ext*/ EXTENDED tw=-mem(eaddr);SETSTATUS(0,mem(eaddr),tw) SETBYTE(eaddr,tw)break; case 0x71: break;/*ILLEGAL*/ case 0x72: break;/*ILLEGAL*/ case 0x73: /*COM ext*/ EXTENDED tb=~mem(eaddr);SETNZ8(tb);SEC CLV SETBYTE(eaddr,tb)break; case 0x74: /*LSR ext*/ EXTENDED tb=mem(eaddr);if(tb&0x01)SEC else CLC if(tb&0x10)SEH else CLH tb>>=1;SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x75: break;/* ILLEGAL*/ case 0x76: /*ROR ext*/ EXTENDED tb=(iccreg&0x01)<<7; if(mem(eaddr)&0x01)SEC else CLC tw=(mem(eaddr)>>1)+tb;SETNZ8(tw) SETBYTE(eaddr,tw) break; case 0x77: /*ASR ext*/ EXTENDED tb=mem(eaddr);if(tb&0x01)SEC else CLC if(tb&0x10)SEH else CLH tb>>=1; if(tb&0x40)tb|=0x80;SETBYTE(eaddr,tb)SETNZ8(tb) break; case 0x78: /*ASL ext*/ EXTENDED tw=mem(eaddr)<<1; SETSTATUS(mem(eaddr),mem(eaddr),tw) SETBYTE(eaddr,tw)break; case 0x79: /*ROL ext*/ EXTENDED tb=mem(eaddr);tw=iccreg&0x01; if(tb&0x80)SEC else CLC if((tb&0x80)^((tb<<1)&0x80))SEV else CLV tb=(tb<<1)+tw;SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x7A: /*DEC ext*/ EXTENDED tb=mem(eaddr)-1;if(tb==0x7F)SEV else CLV SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x7B: break; /*ILLEGAL*/ case 0x7C: /*INC ext*/ EXTENDED tb=mem(eaddr)+1;if(tb==0x80)SEV else CLV SETNZ8(tb) SETBYTE(eaddr,tb)break; case 0x7D: /*TST ext*/ EXTENDED tb=mem(eaddr);SETNZ8(tb) break; case 0x7E: /*JMP ext*/ EXTENDED ipcreg=eaddr;break; case 0x7F: /*CLR ext*/ EXTENDED SETBYTE(eaddr,0)CLN CLV SEZ CLC break; case 0x80: /*SUBA immediate*/ IMM8 tw=iareg-mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0x81: /*CMPA immediate*/ IMM8 tw=iareg-mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) break; case 0x82: /*SBCA immediate*/ IMM8 tw=iareg-mem(eaddr)-(iccreg&0x01); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0x83: /*SUBD (CMPD CMPU) immediate*/ IMM16 {unsigned long res,dreg,breg; if(iflag==2)dreg=iureg;else dreg=GETDREG; breg=GETWORD(eaddr); res=dreg-breg; SETSTATUSD(dreg,breg,res) if(iflag==0) SETDREG(res) }break; case 0x84: /*ANDA immediate*/ IMM8 iareg=iareg&mem(eaddr);SETNZ8(iareg) CLV break; case 0x85: /*BITA immediate*/ IMM8 tb=iareg&mem(eaddr);SETNZ8(tb) CLV break; case 0x86: /*LDA immediate*/ IMM8 LOADAC(iareg) CLV SETNZ8(iareg) break; case 0x87: /*STA immediate (for the sake of orthogonality) */ IMM8 SETNZ8(iareg) CLV STOREAC(iareg) break; case 0x88: /*EORA immediate*/ IMM8 iareg=iareg^mem(eaddr);SETNZ8(iareg) CLV break; case 0x89: /*ADCA immediate*/ IMM8 tw=iareg+mem(eaddr)+(iccreg&0x01); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0x8A: /*ORA immediate*/ IMM8 iareg=iareg|mem(eaddr);SETNZ8(iareg) CLV break; case 0x8B: /*ADDA immediate*/ IMM8 tw=iareg+mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0x8C: /*CMPX (CMPY CMPS) immediate */ IMM16 {unsigned long dreg,breg,res; if(iflag==0)dreg=ixreg;else if(iflag==1) dreg=iyreg;else dreg=isreg;breg=GETWORD(eaddr); res=dreg-breg; SETSTATUSD(dreg,breg,res) }break; case 0x8D: /*BSR */ IMMBYTE(tb) PUSHWORD(ipcreg) ipcreg+=SIGNED(tb); break; case 0x8E: /* LDX (LDY) immediate */ IMM16 tw=GETWORD(eaddr); CLV SETNZ16(tw) if(!iflag)ixreg=tw; else iyreg=tw;break; case 0x8F: /* STX (STY) immediate (orthogonality) */ IMM16 if(!iflag) tw=ixreg; else tw=iyreg; CLV SETNZ16(tw) SETWORD(eaddr,tw) break; case 0x90: /*SUBA direct*/ DIRECT tw=iareg-mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0x91: /*CMPA direct*/ DIRECT tw=iareg-mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) break; case 0x92: /*SBCA direct*/ DIRECT tw=iareg-mem(eaddr)-(iccreg&0x01); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0x93: /*SUBD (CMPD CMPU) direct*/ DIRECT {unsigned long res,dreg,breg; if(iflag==2)dreg=iureg;else dreg=GETDREG; breg=GETWORD(eaddr); res=dreg-breg; SETSTATUSD(dreg,breg,res) if(iflag==0) SETDREG(res) }break; case 0x94: /*ANDA direct*/ DIRECT iareg=iareg&mem(eaddr);SETNZ8(iareg) CLV break; case 0x95: /*BITA direct*/ DIRECT tb=iareg&mem(eaddr);SETNZ8(tb) CLV break; case 0x96: /*LDA direct*/ DIRECT LOADAC(iareg) CLV SETNZ8(iareg) break; case 0x97: /*STA direct */ DIRECT SETNZ8(iareg) CLV STOREAC(iareg) break; case 0x98: /*EORA direct*/ DIRECT iareg=iareg^mem(eaddr);SETNZ8(iareg) CLV break; case 0x99: /*ADCA direct*/ DIRECT tw=iareg+mem(eaddr)+(iccreg&0x01); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0x9A: /*ORA direct*/ DIRECT iareg=iareg|mem(eaddr);SETNZ8(iareg) CLV break; case 0x9B: /*ADDA direct*/ DIRECT tw=iareg+mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0x9C: /*CMPX (CMPY CMPS) direct */ DIRECT {unsigned long dreg,breg,res; if(iflag==0)dreg=ixreg;else if(iflag==1) dreg=iyreg;else dreg=isreg;breg=GETWORD(eaddr); res=dreg-breg; SETSTATUSD(dreg,breg,res) }break; case 0x9D: /*JSR direct */ DIRECT PUSHWORD(ipcreg) ipcreg=eaddr; break; case 0x9E: /* LDX (LDY) direct */ DIRECT tw=GETWORD(eaddr); CLV SETNZ16(tw) if(!iflag)ixreg=tw; else iyreg=tw;break; case 0x9F: /* STX (STY) direct */ DIRECT if(!iflag) tw=ixreg; else tw=iyreg; CLV SETNZ16(tw) SETWORD(eaddr,tw) break; case 0xA0: /*SUBA indexed*/ tw=iareg-mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0xA1: /*CMPA indexed*/ tw=iareg-mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) break; case 0xA2: /*SBCA indexed*/ tw=iareg-mem(eaddr)-(iccreg&0x01); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0xA3: /*SUBD (CMPD CMPU) indexed*/ {unsigned long res,dreg,breg; if(iflag==2)dreg=iureg;else dreg=GETDREG; breg=GETWORD(eaddr); res=dreg-breg; SETSTATUSD(dreg,breg,res) if(iflag==0) SETDREG(res) }break; case 0xA4: /*ANDA indexed*/ iareg=iareg&mem(eaddr);SETNZ8(iareg) CLV break; case 0xA5: /*BITA indexed*/ tb=iareg&mem(eaddr);SETNZ8(tb) CLV break; case 0xA6: /*LDA indexed*/ LOADAC(iareg) CLV SETNZ8(iareg) break; case 0xA7: /*STA indexed */ SETNZ8(iareg) CLV STOREAC(iareg) break; case 0xA8: /*EORA indexed*/ iareg=iareg^mem(eaddr);SETNZ8(iareg) CLV break; case 0xA9: /*ADCA indexed*/ tw=iareg+mem(eaddr)+(iccreg&0x01); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0xAA: /*ORA indexed*/ iareg=iareg|mem(eaddr);SETNZ8(iareg) CLV break; case 0xAB: /*ADDA indexed*/ tw=iareg+mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0xAC: /*CMPX (CMPY CMPS) indexed */ {unsigned long dreg,breg,res; if(iflag==0)dreg=ixreg;else if(iflag==1) dreg=iyreg;else dreg=isreg;breg=GETWORD(eaddr); res=dreg-breg; SETSTATUSD(dreg,breg,res) }break; case 0xAD: /*JSR indexed */ PUSHWORD(ipcreg) ipcreg=eaddr; break; case 0xAE: /* LDX (LDY) indexed */ tw=GETWORD(eaddr); CLV SETNZ16(tw) if(!iflag)ixreg=tw; else iyreg=tw;break; case 0xAF: /* STX (STY) indexed */ if(!iflag) tw=ixreg; else tw=iyreg; CLV SETNZ16(tw) SETWORD(eaddr,tw) break; case 0xB0: /*SUBA ext*/ EXTENDED tw=iareg-mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0xB1: /*CMPA ext*/ EXTENDED tw=iareg-mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) break; case 0xB2: /*SBCA ext*/ EXTENDED tw=iareg-mem(eaddr)-(iccreg&0x01); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0xB3: /*SUBD (CMPD CMPU) ext*/ EXTENDED {unsigned long res,dreg,breg; if(iflag==2)dreg=iureg;else dreg=GETDREG; breg=GETWORD(eaddr); res=dreg-breg; SETSTATUSD(dreg,breg,res) if(iflag==0) SETDREG(res) }break; case 0xB4: /*ANDA ext*/ EXTENDED iareg=iareg&mem(eaddr);SETNZ8(iareg) CLV break; case 0xB5: /*BITA ext*/ EXTENDED tb=iareg&mem(eaddr);SETNZ8(tb) CLV break; case 0xB6: /*LDA ext*/ EXTENDED LOADAC(iareg) CLV SETNZ8(iareg) break; case 0xB7: /*STA ext */ EXTENDED SETNZ8(iareg) CLV STOREAC(iareg) break; case 0xB8: /*EORA ext*/ EXTENDED iareg=iareg^mem(eaddr);SETNZ8(iareg) CLV break; case 0xB9: /*ADCA ext*/ EXTENDED tw=iareg+mem(eaddr)+(iccreg&0x01); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0xBA: /*ORA ext*/ EXTENDED iareg=iareg|mem(eaddr);SETNZ8(iareg) CLV break; case 0xBB: /*ADDA ext*/ EXTENDED tw=iareg+mem(eaddr); SETSTATUS(iareg,mem(eaddr),tw) iareg=tw;break; case 0xBC: /*CMPX (CMPY CMPS) ext */ EXTENDED {unsigned long dreg,breg,res; if(iflag==0)dreg=ixreg;else if(iflag==1) dreg=iyreg;else dreg=isreg;breg=GETWORD(eaddr); res=dreg-breg; SETSTATUSD(dreg,breg,res) }break; case 0xBD: /*JSR ext */ EXTENDED PUSHWORD(ipcreg) ipcreg=eaddr; break; case 0xBE: /* LDX (LDY) ext */ EXTENDED tw=GETWORD(eaddr); CLV SETNZ16(tw) if(!iflag)ixreg=tw; else iyreg=tw;break; case 0xBF: /* STX (STY) ext */ EXTENDED if(!iflag) tw=ixreg; else tw=iyreg; CLV SETNZ16(tw) SETWORD(eaddr,tw) break; case 0xC0: /*SUBB immediate*/ IMM8 tw=ibreg-mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xC1: /*CMPB immediate*/ IMM8 tw=ibreg-mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) break; case 0xC2: /*SBCB immediate*/ IMM8 tw=ibreg-mem(eaddr)-(iccreg&0x01); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xC3: /*ADDD immediate*/ IMM16 {unsigned long res,dreg,breg; dreg=GETDREG; breg=GETWORD(eaddr); res=dreg+breg; SETSTATUSD(dreg,breg,res) SETDREG(res) }break; case 0xC4: /*ANDB immediate*/ IMM8 ibreg=ibreg&mem(eaddr);SETNZ8(ibreg) CLV break; case 0xC5: /*BITB immediate*/ IMM8 tb=ibreg&mem(eaddr);SETNZ8(tb) CLV break; case 0xC6: /*LDB immediate*/ IMM8 LOADAC(ibreg) CLV SETNZ8(ibreg) break; case 0xC7: /*STB immediate (for the sake of orthogonality) */ IMM8 SETNZ8(ibreg) CLV STOREAC(ibreg) break; case 0xC8: /*EORB immediate*/ IMM8 ibreg=ibreg^mem(eaddr);SETNZ8(ibreg) CLV break; case 0xC9: /*ADCB immediate*/ IMM8 tw=ibreg+mem(eaddr)+(iccreg&0x01); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xCA: /*ORB immediate*/ IMM8 ibreg=ibreg|mem(eaddr);SETNZ8(ibreg) CLV break; case 0xCB: /*ADDB immediate*/ IMM8 tw=ibreg+mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xCC: /*LDD immediate */ IMM16 tw=GETWORD(eaddr);SETNZ16(tw) CLV SETDREG(tw) break; case 0xCD: /*STD immediate (orthogonality) */ IMM16 tw=GETDREG; SETNZ16(tw) CLV SETWORD(eaddr,tw) break; case 0xCE: /* LDU (LDS) immediate */ IMM16 tw=GETWORD(eaddr); CLV SETNZ16(tw) if(!iflag)iureg=tw; else isreg=tw;break; case 0xCF: /* STU (STS) immediate (orthogonality) */ IMM16 if(!iflag) tw=iureg; else tw=isreg; CLV SETNZ16(tw) SETWORD(eaddr,tw) break; case 0xD0: /*SUBB direct*/ DIRECT tw=ibreg-mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xD1: /*CMPB direct*/ DIRECT tw=ibreg-mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) break; case 0xD2: /*SBCB direct*/ DIRECT tw=ibreg-mem(eaddr)-(iccreg&0x01); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xD3: /*ADDD direct*/ DIRECT {unsigned long res,dreg,breg; dreg=GETDREG; breg=GETWORD(eaddr); res=dreg+breg; SETSTATUSD(dreg,breg,res) SETDREG(res) }break; case 0xD4: /*ANDB direct*/ DIRECT ibreg=ibreg&mem(eaddr);SETNZ8(ibreg) CLV break; case 0xD5: /*BITB direct*/ DIRECT tb=ibreg&mem(eaddr);SETNZ8(tb) CLV break; case 0xD6: /*LDB direct*/ DIRECT LOADAC(ibreg) CLV SETNZ8(ibreg) break; case 0xD7: /*STB direct */ DIRECT SETNZ8(ibreg) CLV STOREAC(ibreg) break; case 0xD8: /*EORB direct*/ DIRECT ibreg=ibreg^mem(eaddr);SETNZ8(ibreg) CLV break; case 0xD9: /*ADCB direct*/ DIRECT tw=ibreg+mem(eaddr)+(iccreg&0x01); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xDA: /*ORB direct*/ DIRECT ibreg=ibreg|mem(eaddr);SETNZ8(ibreg) CLV break; case 0xDB: /*ADDB direct*/ DIRECT tw=ibreg+mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xDC: /*LDD direct */ DIRECT tw=GETWORD(eaddr);SETNZ16(tw) CLV SETDREG(tw) break; case 0xDD: /*STD direct */ DIRECT tw=GETDREG; SETNZ16(tw) CLV #ifdef USE_MMU STOREAC((tw>>8)&0x0ff); eaddr++; STOREAC(tw&0x0ff); break; #else SETWORD(eaddr,tw) break; #endif case 0xDE: /* LDU (LDS) direct */ DIRECT tw=GETWORD(eaddr); CLV SETNZ16(tw) if(!iflag)iureg=tw; else isreg=tw;break; case 0xDF: /* STU (STS) direct */ DIRECT if(!iflag) tw=iureg; else tw=isreg; CLV SETNZ16(tw) SETWORD(eaddr,tw) break; case 0xE0: /*SUBB indexed*/ tw=ibreg-mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xE1: /*CMPB indexed*/ tw=ibreg-mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) break; case 0xE2: /*SBCB indexed*/ tw=ibreg-mem(eaddr)-(iccreg&0x01); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xE3: /*ADDD indexed*/ {unsigned long res,dreg,breg; dreg=GETDREG; breg=GETWORD(eaddr); res=dreg+breg; SETSTATUSD(dreg,breg,res) SETDREG(res) }break; case 0xE4: /*ANDB indexed*/ ibreg=ibreg&mem(eaddr);SETNZ8(ibreg) CLV break; case 0xE5: /*BITB indexed*/ tb=ibreg&mem(eaddr);SETNZ8(tb) CLV break; case 0xE6: /*LDB indexed*/ LOADAC(ibreg) CLV SETNZ8(ibreg) break; case 0xE7: /*STB indexed */ SETNZ8(ibreg) CLV STOREAC(ibreg) break; case 0xE8: /*EORB indexed*/ ibreg=ibreg^mem(eaddr);SETNZ8(ibreg) CLV break; case 0xE9: /*ADCB indexed*/ tw=ibreg+mem(eaddr)+(iccreg&0x01); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xEA: /*ORB indexed*/ ibreg=ibreg|mem(eaddr);SETNZ8(ibreg) CLV break; case 0xEB: /*ADDB indexed*/ tw=ibreg+mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xEC: /*LDD indexed */ tw=GETWORD(eaddr);SETNZ16(tw) CLV SETDREG(tw) break; case 0xED: /*STD indexed */ tw=GETDREG; SETNZ16(tw) CLV #ifdef USE_MMU STOREAC((tw>>8)&0x0ff); eaddr++; STOREAC(tw&0x0ff); break; #else SETWORD(eaddr,tw) break; #endif case 0xEE: /* LDU (LDS) indexed */ tw=GETWORD(eaddr); CLV SETNZ16(tw) if(!iflag)iureg=tw; else isreg=tw;break; case 0xEF: /* STU (STS) indexed */ if(!iflag) tw=iureg; else tw=isreg; CLV SETNZ16(tw) SETWORD(eaddr,tw) break; case 0xF0: /*SUBB ext*/ EXTENDED tw=ibreg-mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xF1: /*CMPB ext*/ EXTENDED tw=ibreg-mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) break; case 0xF2: /*SBCB ext*/ EXTENDED tw=ibreg-mem(eaddr)-(iccreg&0x01); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xF3: /*ADDD ext*/ EXTENDED {unsigned long res,dreg,breg; dreg=GETDREG; breg=GETWORD(eaddr); res=dreg+breg; SETSTATUSD(dreg,breg,res) SETDREG(res) }break; case 0xF4: /*ANDB ext*/ EXTENDED ibreg=ibreg&mem(eaddr);SETNZ8(ibreg) CLV break; case 0xF5: /*BITB ext*/ EXTENDED tb=ibreg&mem(eaddr);SETNZ8(tb) CLV break; case 0xF6: /*LDB ext*/ EXTENDED LOADAC(ibreg) CLV SETNZ8(ibreg) break; case 0xF7: /*STB ext */ EXTENDED SETNZ8(ibreg) CLV STOREAC(ibreg) break; case 0xF8: /*EORB ext*/ EXTENDED ibreg=ibreg^mem(eaddr);SETNZ8(ibreg) CLV break; case 0xF9: /*ADCB ext*/ EXTENDED tw=ibreg+mem(eaddr)+(iccreg&0x01); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xFA: /*ORB ext*/ EXTENDED ibreg=ibreg|mem(eaddr);SETNZ8(ibreg) CLV break; case 0xFB: /*ADDB ext*/ EXTENDED tw=ibreg+mem(eaddr); SETSTATUS(ibreg,mem(eaddr),tw) ibreg=tw;break; case 0xFC: /*LDD ext */ EXTENDED tw=GETWORD(eaddr);SETNZ16(tw) CLV SETDREG(tw) break; case 0xFD: /*STD ext */ EXTENDED tw=GETDREG; SETNZ16(tw) CLV #ifdef USE_MMU STOREAC((tw>>8)&0x0ff); eaddr++; STOREAC(tw&0x0ff); break; #else SETWORD(eaddr,tw) break; #endif case 0xFE: /* LDU (LDS) ext */ EXTENDED tw=GETWORD(eaddr); CLV SETNZ16(tw) if(!iflag)iureg=tw; else isreg=tw;break; case 0xFF: /* STU (STS) ext */ EXTENDED if(!iflag) tw=iureg; else tw=isreg; CLV SETNZ16(tw) SETWORD(eaddr,tw) break; } } }