/* * HalfNES by Andrew Hoffman * Licensed under the GNU GPL Version 3. See LICENSE file */ package com.grapeshot.halfnes; import java.io.File; import java.io.FileOutputStream; import java.io.FileWriter; import java.io.IOException; import java.io.OutputStreamWriter; import java.nio.charset.StandardCharsets; public final class CPU { private final CPURAM ram; private int cycles; //increment to steal cycles from cpu public int clocks; //use for synchronizing with cpu private int A, X, Y, S; // registers public int PC; private boolean carryFlag = false, zeroFlag = false, interruptsDisabled = true, decimalModeFlag = false; private boolean overflowFlag = false, negativeFlag = false, previntflag = false, nmi = false, prevnmi = false, logging = false; private int pb = 0;// set to 1 if access crosses page boundary public int interrupt = 0; public boolean nmiNext = false, idle = false; private final static boolean decimalModeEnable = false, idleLoopSkip = true; //NES 6502 is missing decimal mode, but most other 6502s have it private boolean interruptDelay = false; private final static String[] opcodes = opcodes(); //Battletoads Hack until I get around to making a truly cycle accurate CPU core. //Delays the write of a STA, STX, or STY until the first cycle of the NEXT instruction //which is enough to move it a few PPU clocks after the scroll is changed //making sure that Battletoads gets its sprite 0 hit. final private static boolean battletoadsHackOn = true; private boolean dirtyBattletoadsHack = false; private int hackAddr = 0; private int hackData = 0; private static enum dummy { ONCARRY, ALWAYS; //type of dummy read } OutputStreamWriter w; //debug log writer public CPU(final CPURAM cpuram) { ram = cpuram; //ram is the ONLY thing the cpu tries to talk to. if (logging) { startLog(); } } public void startLog() { logging = true; try { w = new OutputStreamWriter(new FileOutputStream(new File("nesdebug.txt")), StandardCharsets.UTF_8); } catch (IOException e) { System.err.println("Cannot create debug log" + e.getLocalizedMessage()); } } public void startLog(String path) { logging = true; try { w = new OutputStreamWriter(new FileOutputStream(new File(path)), StandardCharsets.UTF_8); } catch (IOException e) { System.err.println("Cannot create debug log" + e.getLocalizedMessage()); } } public void stopLog() { logging = false; flushLog(); } public void init() { init(null); } public void init(Integer initialPC) {// different than reset // puts RAM in NES poweron state for (int i = 0; i < 0x800; ++i) { ram.write(i, 0xFF); } ram.write(0x0008, 0xF7); ram.write(0x0009, 0xEF); ram.write(0x000A, 0xDF); ram.write(0x000F, 0xBF); for (int i = 0x4000; i <= 0x400F; ++i) { ram.write(i, 0x00); } ram.write(0x4015, 0x00); ram.write(0x4017, 0x00); //clocks = 27393; //correct for position we start vblank in A = 0; X = 0; Y = 0; S = 0xFD; if (initialPC == null) { PC = ram.read(0xFFFD) * 256 + ram.read(0xFFFC); } else { PC = initialPC; } } public void reset() { PC = ram.read(0xFFFD) * 256 + ram.read(0xFFFC); ram.write(0x4015, 0); ram.write(0x4017, ram.read(0x4017)); //disable audio on reset S -= 3; S &= 0xff; interruptsDisabled = true; } public void modcycles() { //System.err.println(clocks); clocks = 0; } public void stealcycles(int cyclestosteal) { cycles += cyclestosteal; log("**STEAL " + cyclestosteal + "**"); } public final void runcycle(final int scanline, final int pixel) { ram.read(0x4000); //attempt to sync the APU every cycle and make dmc irqs work properly, which they still don't. Feh. ++clocks; //guard against overflows // if ((A & 0xff) != A) { // System.err.println("houston we have A problem"); // } // if ((X & 0xff) != X) { // System.err.println("houston we have X problem"); // } // if ((Y & 0xff) != Y) { // System.err.println("houston we have Y problem"); // } // if ((S & 0xff) != S) { // System.err.println("houston we have S problem"); // } // if ((PC & 0xffff) != PC) { // System.err.println("houston we have PC problem"); // } if (ram.apu.sprdma_count > 0) { ram.apu.sprdma_count--; if (ram.apu.sprdma_count == 0) { cycles += 513; } //this doesn't look right any more //who patched this in and when? (wasn't me, it was for some bug...) } if (dirtyBattletoadsHack && cycles == 1) { ram.write(hackAddr, hackData); dirtyBattletoadsHack = false; } if (cycles-- > 0) { //count down cycles until there is work to do again return; } //now we're at the start of a new instruction //handle nmi requests (NMI line is edge sensitive not level sensitive) if (nmiNext) { nmi(); nmiNext = false; } if (nmi && !prevnmi) {//only trigger on positive rising edge of NMI nmiNext = true; } prevnmi = nmi; if (interrupt > 0) { if (!interruptsDisabled && !interruptDelay) { interrupt(); cycles += 7; return; } else if (interruptDelay) { interruptDelay = false; if (!previntflag) { interrupt(); cycles += 7; return; } } } else { interruptDelay = false; } //Idle loop skipping if (idle && idleLoopSkip) { cycles += 3; //not accurate should depend on type of instr we skip decoding return; } pb = 0; final int instr = ram.read(PC++); //note: if (logging) { //that looks redundant, but this is a really expensive operation to create the log string //also, logging *might* trigger side effects if logging while executing //code from i/o registers (reading twice). So we don't want to do it always. //TODO: Optimize this! It gets called a LOT //and slows logging to 16 fps //even when not actually writing anything String op = String.format(opcodes[instr], ram.read(PC), ram.read(PC + 1), PC + (byte) (ram.read(PC)) + 1); log(utils.hex(PC - 1) + " " + utils.hex(instr) + String.format(" %-14s ", op) + status() + " CYC:" + pixel + " SL:" + scanline + "\n"); } if (cycles == 0) { flushLog(); } switch (instr) { // ADC case 0x69: adc(imm()); cycles += 2; break; case 0x65: adc(zpg()); cycles += 3; break; case 0x75: adc(zpg(X)); cycles += 4; break; case 0x6d: adc(abs()); cycles += 4; break; case 0x7d: adc(abs(X, dummy.ONCARRY)); cycles += 4 + pb; break; case 0x79: adc(abs(Y, dummy.ONCARRY)); cycles += 4 + pb; break; case 0x61: adc(indX()); cycles += 6; break; case 0x71: adc(indY(dummy.ONCARRY)); cycles += 5 + pb; break; // AHX (unofficial) case 0x93: ahx(indY(dummy.ALWAYS)); cycles += 6; break; case 0x9f: ahx(abs(Y, dummy.ALWAYS)); cycles += 5; break; // ALR (unofficial) case 0x4b: alr(imm()); cycles += 2; break; // ANC (unofficial) case 0x0b: anc(imm()); cycles += 2; break; case 0x2b: anc(imm()); cycles += 2; break; // AND case 0x29: and(imm()); cycles += 2; break; case 0x25: and(zpg()); cycles += 3; break; case 0x35: and(zpg(X)); cycles += 4; break; case 0x2D: and(abs()); cycles += 4; break; case 0x3D: and(abs(X, dummy.ONCARRY)); cycles += 4 + pb; break; case 0x39: and(abs(Y, dummy.ONCARRY)); cycles += 4 + pb; break; case 0x21: and(indX()); cycles += 6; break; case 0x31: and(indY(dummy.ONCARRY)); cycles += 5 + pb; break; // ARR (unofficial) case 0x6b: arr(imm()); cycles += 2; break; // ASL case 0x0A: aslA(); cycles += 2; break; case 0x06: asl(zpg()); cycles += 5; break; case 0x16: asl(zpg(X)); cycles += 6; break; case 0x0e: asl(abs()); cycles += 6; break; case 0x1e: asl(abs(X, dummy.ALWAYS)); cycles += 7; break; // AXS (unofficial) case 0xcb: axs(imm()); cycles += 2; break; // BIT case 0x24: bit(zpg()); cycles += 3; break; case 0x2c: bit(abs()); cycles += 4; break; // Branches: every branch uses rel. addressing case 0x10: branch(!negativeFlag); cycles += 2 + pb; break; case 0x30: branch(negativeFlag); cycles += 2 + pb; break; case 0x50: branch(!overflowFlag); cycles += 2 + pb; break; case 0x70: branch(overflowFlag); cycles += 2 + pb; break; case 0x90: branch(!carryFlag); cycles += 2 + pb; break; case 0xB0: branch(carryFlag); cycles += 2 + pb; break; case 0xD0: branch(!zeroFlag); cycles += 2 + pb; break; case 0xF0: branch(zeroFlag); cycles += 2 + pb; break; // BRK case 0x00: //System.err.println("Hey! A break!"); breakinterrupt(); cycles += 7; break; // CMP case 0xc9: cmp(A, imm()); cycles += 2; break; case 0xc5: cmp(A, zpg()); cycles += 3; break; case 0xd5: cmp(A, zpg(X)); cycles += 4; break; case 0xcd: cmp(A, abs()); cycles += 4; break; case 0xdd: cmp(A, abs(X, dummy.ONCARRY)); cycles += 4 + pb; break; case 0xd9: cmp(A, abs(Y, dummy.ONCARRY)); cycles += 4 + pb; break; case 0xc1: cmp(A, indX()); cycles += 6; break; case 0xd1: cmp(A, indY(dummy.ONCARRY)); cycles += 5 + pb; break; // CPX case 0xe0: cmp(X, imm()); cycles += 2; break; case 0xe4: cmp(X, zpg()); cycles += 3; break; case 0xec: cmp(X, abs()); cycles += 4; break; // CPY case 0xc0: cmp(Y, imm()); cycles += 2; break; case 0xc4: cmp(Y, zpg()); cycles += 3; break; case 0xcc: cmp(Y, abs()); cycles += 4; break; // DEC case 0xc6: dec(zpg()); cycles += 5; break; case 0xd6: dec(zpg(X)); cycles += 6; break; case 0xce: dec(abs()); cycles += 6; break; case 0xde: dec(abs(X, dummy.ALWAYS)); cycles += 7; break; // DCP (unofficial) case 0xc3: dcp(A, indX()); cycles += 8; break; case 0xd3: dcp(A, indY(dummy.ALWAYS)); cycles += 8; break; case 0xc7: dcp(A, zpg()); cycles += 5; break; case 0xd7: dcp(A, zpg(X)); cycles += 6; break; case 0xdb: dcp(A, abs(Y, dummy.ALWAYS)); cycles += 7; break; case 0xcf: dcp(A, abs()); cycles += 6; break; case 0xdf: dcp(A, abs(X, dummy.ALWAYS)); cycles += 7; break; // EOR case 0x49: eor(imm()); cycles += 2; break; case 0x45: eor(zpg()); cycles += 3; break; case 0x55: eor(zpg(X)); cycles += 4; break; case 0x4d: eor(abs()); cycles += 4; break; case 0x5d: eor(abs(X, dummy.ONCARRY)); cycles += 4 + pb; break; case 0x59: eor(abs(Y, dummy.ONCARRY)); cycles += 4 + pb; break; case 0x41: eor(indX()); cycles += 6; break; case 0x51: eor(indY(dummy.ONCARRY)); cycles += 5 + pb; break; // Flag set/clear case 0x18: carryFlag = false; cycles += 2; break; case 0x38: carryFlag = true; cycles += 2; break; case 0x58: //cli //interrupts shouldn't fire for 1 cycle after cli delayInterrupt(); interruptsDisabled = false; cycles += 2; break; case 0x78: //sei delayInterrupt(); interruptsDisabled = true; cycles += 2; break; case 0xb8: overflowFlag = false; cycles += 2; break; case 0xd8: decimalModeFlag = false; cycles += 2; break;// decimal mode doesnt case 0xf8: decimalModeFlag = true; cycles += 2; break;// do anything on NES // INC case 0xe6: inc(zpg()); cycles += 5; break; case 0xf6: inc(zpg(X)); cycles += 6; break; case 0xee: inc(abs()); cycles += 6; break; case 0xfe: inc(abs(X, dummy.ALWAYS)); cycles += 7; break; // ISC (unofficial) case 0xe3: isc(indX()); cycles += 8; break; case 0xf3: isc(indY(dummy.ALWAYS)); cycles += 8; break; case 0xe7: isc(zpg()); cycles += 5; break; case 0xf7: isc(zpg(X)); cycles += 6; break; case 0xfb: isc(abs(Y, dummy.ALWAYS)); cycles += 7; break; case 0xef: isc(abs()); cycles += 6; break; case 0xff: isc(abs(X, dummy.ALWAYS)); cycles += 7; break; // JMP case 0x4c: int tempe = PC; PC = abs(); if (PC == (tempe - 1)) { idle = true; } cycles += 3; break; case 0x6c: int tempf = PC; PC = ind(); if (PC == (tempf - 1)) { idle = true; } cycles += 5; break; // JSR case 0x20: jsr(abs()); cycles += 6; break; // KIL (unofficial) case 0x02: case 0x12: case 0x22: case 0x32: case 0x42: case 0x52: case 0x62: case 0x72: case 0x92: case 0xb2: case 0xd2: case 0xf2: System.err.println("KIL - CPU locked"); flushLog(); ram.apu.nes.runEmulation = false; break; // LAS (unofficial) case 0xbb: las(abs(Y, dummy.ONCARRY)); cycles += 4 + pb; break; // LAX (unofficial) case 0xa3: lax(indX()); cycles += 6; break; case 0xb3: lax(indY(dummy.ONCARRY)); cycles += 5 + pb; break; case 0xa7: lax(zpg()); cycles += 3; break; case 0xb7: lax(zpg(Y)); cycles += 4; break; case 0xab: lax(imm()); cycles += 2; break; case 0xaf: lax(abs()); cycles += 4; break; case 0xbf: lax(abs(Y, dummy.ONCARRY)); cycles += 4 + pb; break; // LDA case 0xa9: lda(imm()); cycles += 2; break; case 0xa5: lda(zpg()); cycles += 3; break; case 0xb5: lda(zpg(X)); cycles += 4; break; case 0xad: lda(abs()); cycles += 4; break; case 0xbd: lda(abs(X, dummy.ONCARRY)); cycles += 4 + pb; break; case 0xb9: lda(abs(Y, dummy.ONCARRY)); cycles += 4 + pb; break; case 0xa1: lda(indX()); cycles += 6; break; case 0xb1: lda(indY(dummy.ONCARRY)); cycles += 5 + pb; break; // LDX case 0xa2: ldx(imm()); cycles += 2; break; case 0xa6: ldx(zpg()); cycles += 3; break; case 0xb6: ldx(zpg(Y)); cycles += 4; break; case 0xae: ldx(abs()); cycles += 4; break; case 0xbe: ldx(abs(Y, dummy.ONCARRY)); cycles += 4 + pb; break; // LDY case 0xa0: ldy(imm()); cycles += 2; break; case 0xa4: ldy(zpg()); cycles += 3; break; case 0xb4: ldy(zpg(X)); cycles += 4; break; case 0xac: ldy(abs()); cycles += 4; break; case 0xbc: ldy(abs(X, dummy.ONCARRY)); cycles += 4 + pb; break; // LSR case 0x4a: lsrA(); cycles += 2; break; case 0x46: lsr(zpg()); cycles += 5; break; case 0x56: lsr(zpg(X)); cycles += 6; break; case 0x4e: lsr(abs()); cycles += 6; break; case 0x5e: lsr(abs(X, dummy.ALWAYS)); cycles += 7; break; // NOP case 0x1a: case 0x3a: case 0x5a: case 0x7a: case 0xda: case 0xEA: case 0xfa: cycles += 2; break; case 0x80: case 0x82: case 0xc2: case 0xe2: case 0x89: imm(); cycles += 2; break; case 0x04: case 0x44: case 0x64: zpg(); cycles += 3; break; case 0x14: case 0x34: case 0x54: case 0x74: case 0xd4: case 0xf4: zpg(X); cycles += 4; break; case 0x0C: abs(); cycles += 4; break; case 0x1c: case 0x3c: case 0x5c: case 0x7c: case 0xdc: case 0xfc: abs(X, dummy.ONCARRY); cycles += 4 + pb; break; // ORA case 0x09: ora(imm()); cycles += 2; break; case 0x05: ora(zpg()); cycles += 3; break; case 0x15: ora(zpg(X)); cycles += 4; break; case 0x0d: ora(abs()); cycles += 4; break; case 0x1d: ora(abs(X, dummy.ONCARRY)); cycles += 4 + pb; break; case 0x19: ora(abs(Y, dummy.ONCARRY)); cycles += 4 + pb; break; case 0x01: ora(indX()); cycles += 6; break; case 0x11: ora(indY(dummy.ONCARRY)); cycles += 5 + pb; break; // Register instrs. case 0xAA: X = A; cycles += 2; setflags(A); break; case 0x8a: A = X; cycles += 2; setflags(A); break; case 0xca: X--; X &= 0xFF; setflags(X); cycles += 2; break; case 0xe8: X++; X &= 0xFF; setflags(X); cycles += 2; break; case 0xa8: Y = A; cycles += 2; setflags(A); break; case 0x98: A = Y; cycles += 2; setflags(A); break; case 0x88: Y--; Y &= 0xFF; setflags(Y); cycles += 2; break; case 0xc8: Y++; Y &= 0xFF; setflags(Y); cycles += 2; break; // RLA (unofficial) case 0x23: rla(indX()); cycles += 8; break; case 0x33: rla(indY(dummy.ALWAYS)); cycles += 8; break; case 0x27: rla(zpg()); cycles += 5; break; case 0x37: rla(zpg(X)); cycles += 6; break; case 0x3b: rla(abs(Y, dummy.ALWAYS)); cycles += 7; break; case 0x2f: rla(abs()); cycles += 6; break; case 0x3f: rla(abs(X, dummy.ALWAYS)); cycles += 7; break; // ROL case 0x2a: rolA(); cycles += 2; break; case 0x26: rol(zpg()); cycles += 5; break; case 0x36: rol(zpg(X)); cycles += 6; break; case 0x2e: rol(abs()); cycles += 6; break; case 0x3e: rol(abs(X, dummy.ALWAYS)); cycles += 7; break; // ROR case 0x6a: rorA(); cycles += 2; break; case 0x66: ror(zpg()); cycles += 5; break; case 0x76: ror(zpg(X)); cycles += 6; break; case 0x6e: ror(abs()); cycles += 6; break; case 0x7e: ror(abs(X, dummy.ALWAYS)); cycles += 7; break; // RRA (unofficial) case 0x63: rra(indX()); cycles += 8; break; case 0x73: rra(indY(dummy.ALWAYS)); cycles += 8; break; case 0x67: rra(zpg()); cycles += 5; break; case 0x77: rra(zpg(X)); cycles += 6; break; case 0x7b: rra(abs(Y, dummy.ALWAYS)); cycles += 7; break; case 0x6f: rra(abs()); cycles += 6; break; case 0x7f: rra(abs(X, dummy.ALWAYS)); cycles += 7; break; // RTI case 0x40: rti(); cycles += 6; break; // RTS case 0x60: rts(); cycles += 6; break; // SAX (unofficial) case 0x83: sax(indX()); cycles += 6; break; case 0x87: sax(zpg()); cycles += 3; break; case 0x97: sax(zpg(Y)); cycles += 4; break; case 0x8f: sax(abs()); cycles += 4; break; // SBC case 0xE1: sbc(indX()); cycles += 6; break; case 0xF1: sbc(indY(dummy.ONCARRY)); cycles += 5 + pb; break; case 0xE5: sbc(zpg()); cycles += 3; break; case 0xF5: sbc(zpg(X)); cycles += 4; break; case 0xE9: sbc(imm()); cycles += 2; break; case 0xF9: sbc(abs(Y, dummy.ONCARRY)); cycles += 4 + pb; break; case 0xeb: sbc(imm()); cycles += 2; break; case 0xEd: sbc(abs()); cycles += 4; break; case 0xFd: sbc(abs(X, dummy.ONCARRY)); cycles += 4 + pb; break; // SHX (unofficial) case 0x9e: shx(abs(Y, dummy.ALWAYS)); cycles += 5; break; // SHY (unofficial) case 0x9c: shy(abs(X, dummy.ALWAYS)); cycles += 5; break; // SLO (unofficial) case 0x03: slo(indX()); cycles += 8; break; case 0x07: slo(zpg()); cycles += 5; break; case 0x0f: slo(abs()); cycles += 6; break; case 0x13: slo(indY(dummy.ALWAYS)); cycles += 8; break; case 0x17: slo(zpg(X)); cycles += 6; break; case 0x1b: slo(abs(Y, dummy.ALWAYS)); cycles += 7; break; case 0x1f: slo(abs(X, dummy.ALWAYS)); cycles += 7; break; // SRE (unofficial) case 0x43: sre(indX()); cycles += 8; break; case 0x53: sre(indY(dummy.ALWAYS)); cycles += 8; break; case 0x47: sre(zpg()); cycles += 5; break; case 0x57: sre(zpg(X)); cycles += 6; break; case 0x5b: sre(abs(Y, dummy.ALWAYS)); cycles += 7; break; case 0x4f: sre(abs()); cycles += 6; break; case 0x5f: sre(abs(X, dummy.ALWAYS)); cycles += 7; break; // STA case 0x85: sta(zpg()); cycles += 3; break; case 0x95: sta(zpg(X)); cycles += 4; break; case 0x8d: sta(abs()); cycles += 4; break; case 0x9d: sta(abs(X, dummy.ALWAYS)); cycles += 5; break; case 0x99: sta(abs(Y, dummy.ALWAYS)); cycles += 5; break; case 0x81: sta(indX()); cycles += 6; break; case 0x91: sta(indY(dummy.ALWAYS)); cycles += 6; break; // Stack instructions case 0x9A: S = X; cycles += 2; break; case 0xBA: X = S; cycles += 2; setflags(X); break; case 0x48: ram.read(PC + 1); //dummy fetch push(A); cycles += 3; break; case 0x68: ram.read(PC + 1); //dummy fetch A = pop(); setflags(A); cycles += 4; break; case 0x08: ram.read(PC + 1); //dummy fetch push(flagstobyte() | utils.BIT4); cycles += 3; break; case 0x28: //plp delayInterrupt(); ram.read(PC + 1); //dummy fetch bytetoflags(pop()); cycles += 4; break; // STX case 0x86: stx(zpg()); cycles += 3; break; case 0x96: stx(zpg(Y)); cycles += 4; break; case 0x8E: stx(abs()); cycles += 4; break; // STY case 0x84: sty(zpg()); cycles += 3; break; case 0x94: sty(zpg(X)); cycles += 4; break; case 0x8c: sty(abs()); cycles += 4; break; // TAS (unofficial) case 0x9b: tas(abs(Y, dummy.ALWAYS)); cycles += 5; break; // XAA (unofficial) case 0x8b: xaa(imm()); cycles += 2; break; default: cycles += 2; System.err.println("Illegal opcode:" + utils.hex(instr) + " @ " + utils.hex(PC - 1)); break; } pb = 0; PC &= 0xffff; } /* really every instruction should be reading from or writing something to memory every cycle. Even when all that's happening that cycle is the processor updating state internally Fetching the next opcode cn overlap with last cycle of prev instruction if that last cycle is purely internal. but since the second cycle of all instructions (even single byte ones) is reading the nest byte after the PC, the fastest we can do even a single byte NOP instruction is still 2 cycles. that's where the dummy reads+writes come from. how to represent this in the smallest space possible? probably the way they did it on the real chip: using a PLA that does certain things conditionally based on bits of the current opcode and the current cycle (up to 7 i suppose) Bisqwit did some really nifty template stuff with his C==10 emu that I can't match. */ private void delayInterrupt() { interruptDelay = true; previntflag = interruptsDisabled; } private void rol(final int addr) { int data = (ram.read(addr)); ram.write(addr, data); //dummy write data = (data << 1) | (carryFlag ? 1 : 0); carryFlag = ((data & (utils.BIT8)) != 0); data &= 0xFF; setflags(data); ram.write(addr, data); } private void rolA() { A = A << 1 | (carryFlag ? 1 : 0); carryFlag = ((A & (utils.BIT8)) != 0); A &= 0xFF; setflags(A); } private void ror(final int addr) { int data = ram.read(addr); ram.write(addr, data); //dummy write final boolean tmp = carryFlag; carryFlag = ((data & (utils.BIT0)) != 0); data >>= 1; data &= 0x7F; data |= (tmp ? 0x80 : 0); setflags(data); ram.write(addr, data); } private void rorA() { final boolean tmp = carryFlag; carryFlag = ((A & (utils.BIT0)) != 0); A >>= 1; A &= 0x7F; A |= (tmp ? 128 : 0); setflags(A); } public void setNMI(boolean val) { this.nmi = val; } private void nmi() { idle = false; log("**NMI**"); //System.err.println(" NMI"); push(PC >> 8); // high bit 1st push((PC) & 0xFF);// check that this pushes right address push(flagstobyte() & ~utils.BIT4); PC = ram.read(0xFFFA) + (ram.read(0xFFFB) << 8); cycles += 7; interruptsDisabled = true; } private void interrupt() { idle = false; log("**INTERRUPT**"); //System.err.println("IRQ " + interrupt); push(PC >> 8); // high bit 1st push(PC & 0xFF);// check that this pushes right address push(flagstobyte() & ~utils.BIT4); //jump to reset vector PC = ram.read(0xFFFE) + (ram.read(0xFFFF) << 8); interruptsDisabled = true; } private void breakinterrupt() { //same as interrupt but BRK flag is turned on log("**BREAK**"); ram.read(PC++); //dummy fetch push(PC >> 8); // high bit 1st push(PC & 0xFF);// check that this pushes right address push(flagstobyte() | utils.BIT4 | utils.BIT5);//push byte w/bits 4+5 set PC = ram.read(0xFFFE) + (ram.read(0xFFFF) << 8); interruptsDisabled = true; } private void lsr(final int addr) { int data = ram.read(addr); ram.write(addr, data); //dummy write carryFlag = ((data & (utils.BIT0)) != 0); data >>= 1; data &= 0x7F; ram.write(addr, data); setflags(data); } private void lsrA() { carryFlag = ((A & (utils.BIT0)) != 0); A >>= 1; A &= 0x7F; setflags(A); } private void eor(final int addr) { A ^= ram.read(addr); A &= 0xff; setflags(A); } private void ora(final int addr) { A |= ram.read(addr); A &= 0xff; setflags(A); } // Instructions private void bit(final int addr) { final int data = ram.read(addr); zeroFlag = ((data & A) == 0); negativeFlag = ((data & (utils.BIT7)) != 0); overflowFlag = ((data & (utils.BIT6)) != 0); } private void jsr(final int addr) { PC--; ram.read(PC); //dummy fetch push(PC >> 8); // high bit 1st push(PC & 0xFF);// check that this pushes right address PC = addr; } private void rts() { ram.read(PC++); //dummy fetch PC = (pop() & 0xff) | (pop() << 8);// page crossing bug again? PC++; } private void rti() { //System.err.println("RTI"); ram.read(PC++); //dummy fetch bytetoflags(pop()); PC = (pop() & 0xff) | (pop() << 8); // not plus one } private int pop() { ++S; S &= 0xff; return ram.read(0x100 + S); } public void push(final int byteToPush) { ram.write((0x100 + (S & 0xff)), byteToPush); --S; S &= 0xff; } private void branch(final boolean isTaken) { if (isTaken) { final int pcprev = PC + 1;// store prev. PC PC = rel(); // System.err.println(pcprev + " "+ PC); //page boundary penalty if ((pcprev & 0xff00) != (PC & 0xff00)) { pb = 2;//page crossing for branch takes 2 cycles } else { cycles++; } if ((pcprev - 2) == PC) { idle = true; } } else { rel(); // have to do the memory access even if we're not branching } } private void inc(final int addr) { int tmp = ram.read(addr); ram.write(addr, tmp); //dummy write ++tmp; tmp &= 0xff; ram.write(addr, tmp); //THEN real write setflags(tmp); } private void dec(final int addr) { int tmp = ram.read(addr); ram.write(addr, tmp); //dummy write --tmp; tmp &= 0xff; ram.write(addr, tmp); //THEN real write setflags(tmp); } private void adc(final int addr) { final int value = ram.read(addr); int result; if (decimalModeFlag && decimalModeEnable) { int AL = (A & 0xF) + (value & 0xF) + (carryFlag ? 1 : 0); if (AL >= 0x0A) { AL = ((AL + 0x6) & 0xF) + 0x10; } result = (A & 0xF0) + (value & 0xF0) + AL; if (result >= 0xA0) { result += 0x60; } } else { result = value + A + (carryFlag ? 1 : 0); } carryFlag = (result >> 8 != 0); // set overflow flag overflowFlag = (((A ^ value) & 0x80) == 0) && (((A ^ result) & 0x80) != 0); A = result & 0xff; setflags(A);// set other flags } private void sbc(final int addr) { final int value = ram.read(addr); int result; if (decimalModeFlag && decimalModeEnable) { int AL = (A & 0xF) - (value & 0xF) + (carryFlag ? 1 : 0) - 1; if (AL < 0) { AL = ((AL - 0x6) & 0xF) - 0x10; } result = (A & 0xF0) + (value & 0xF0) + AL; if (result < 0) { result -= 0x60; } } else { result = A - value - (carryFlag ? 0 : 1); } carryFlag = (result >> 8 == 0); // set overflow flag overflowFlag = (((A ^ value) & 0x80) != 0) && (((A ^ result) & 0x80) != 0); A = result & 0xff; setflags(A);// set other flags } private void and(final int addr) { A &= ram.read(addr); setflags(A); } private void asl(final int addr) { int data = ram.read(addr); ram.write(addr, data); //dummy write carryFlag = ((data & (utils.BIT7)) != 0); data = data << 1; data &= 0xff; setflags(data); ram.write(addr, data); } private void aslA() { carryFlag = ((A & (utils.BIT7)) != 0); A <<= 1; A &= 0xff; setflags(A); } private void cmp(final int regval, final int addr) { final int result = regval - ram.read(addr); if (result < 0) { negativeFlag = ((result & (utils.BIT7)) != 0); carryFlag = false; zeroFlag = false; } else if (result == 0) { negativeFlag = false; carryFlag = true; zeroFlag = true; } else { negativeFlag = ((result & (utils.BIT7)) != 0); carryFlag = true; zeroFlag = false; } } private void lda(final int addr) { A = ram.read(addr); setflags(A); } private void ldx(final int addr) { X = ram.read(addr); setflags(X); } private void ldy(final int addr) { Y = ram.read(addr); setflags(Y); } private void setflags(final int result) { zeroFlag = (result == 0); negativeFlag = ((result & (utils.BIT7)) != 0); } private void sta(final int addr) { if (!battletoadsHackOn) { ram.write(addr, A); } else { hackAddr = addr; hackData = A; dirtyBattletoadsHack = true; } } private void stx(final int addr) { if (!battletoadsHackOn) { ram.write(addr, X); } else { hackAddr = addr; hackData = X; dirtyBattletoadsHack = true; } } private void sty(final int addr) { if (!battletoadsHackOn) { ram.write(addr, Y); } else { hackAddr = addr; hackData = Y; dirtyBattletoadsHack = true; } } // Unofficial opcodes private void ahx(final int addr) { final int data = (A & X & ((addr >> 8) + 1)) & 0xFF; final int tmp = (addr - Y) & 0xFF; if ((Y + tmp) <= 0xFF) { ram.write(addr, data); } else { ram.write(addr, ram.read(addr)); } } private void alr(final int addr) { and(addr); lsrA(); } private void anc(final int addr) { and(addr); carryFlag = negativeFlag; } private void arr(final int addr) { A = (((ram.read(addr) & A) >> 1) | (carryFlag ? 0x80 : 0x00)); setflags(A); carryFlag = ((A & (utils.BIT6)) != 0); overflowFlag = carryFlag ^ ((A & (utils.BIT5)) != 0); } private void axs(final int addr) { X = ((A & X) - ram.read(addr)) & 0xff; setflags(X); carryFlag = (X >= 0); } private void dcp(final int regval, final int addr) { dec(addr); cmp(regval, addr); } private void las(final int addr) { S &= ram.read(addr); A = X = S; setflags(S); } private void lax(final int addr) { A = X = ram.read(addr); setflags(A); } private void isc(final int addr) { inc(addr); sbc(addr); } private void rla(final int addr) { rol(addr); and(addr); } private void rra(int addr) { ror(addr); adc(addr); } private void sax(int addr) { ram.write(addr, (A & X) & 0xFF); } private void shx(final int addr) { final int data = (X & ((addr >> 8) + 1)) & 0xFF; final int tmp = (addr - Y) & 0xFF; if ((Y + tmp) <= 0xFF) { ram.write(addr, data); } else { ram.write(addr, ram.read(addr)); } } private void shy(final int addr) { final int data = (Y & ((addr >> 8) + 1)) & 0xFF; final int tmp = (addr - X) & 0xFF; if ((X + tmp) <= 0xFF) { ram.write(addr, data); } else { ram.write(addr, ram.read(addr)); } } private void slo(int addr) { asl(addr); ora(addr); } private void sre(int addr) { lsr(addr); eor(addr); } private void tas(int addr) { S = A & X; final int data = (S & ((addr >> 8) + 1)) & 0xFF; final int tmp = (addr - Y) & 0xFF; if ((Y + tmp) <= 0xFF) { ram.write(addr, data); } else { ram.write(addr, ram.read(addr)); } } private void xaa(int addr) { A = X & ram.read(addr); setflags(A); } // Functions for memory address types; each returns the _memory_address_ for // the next fn private int imm() { return PC++; } private int zpg() { // zero page mode return ram.read(PC++); } private int zpg(final int reg) { // zero page added to register (modulus page boundary) return (ram.read(PC++) + reg) & 0xff; } private int rel() { // returns actual value of PC, not memory location to look at // because only branches use this return ((byte) ram.read(PC++)) + PC; } private int abs() { // absolute mode return ram.read(PC++) + (ram.read(PC++) << 8); } private int abs(final int reg, final dummy dummy) { // absolute plus value from reg final int addr = (ram.read(PC++) | (ram.read(PC++) << 8)); if (addr >> 8 != (addr + reg) >> 8) { pb = 1; } if ((addr & 0xFF00) != ((addr + reg) & 0xFF00) && dummy == dummy.ONCARRY) { ram.read((addr & 0xFF00) | ((addr + reg) & 0xFF)); } if (dummy == dummy.ALWAYS) { ram.read((addr & 0xFF00) | ((addr + reg) & 0xFF)); } return (addr + reg) & 0xffff; } private int ind() { // weird mode. only used by jmp final int readloc = abs(); return ram.read(readloc) + (ram.read(((readloc & 0xff) == 0xff) ? readloc - 0xff : readloc + 1) << 8); //if reading from the last byte in a page, high bit of address //is taken from first byte on the page, not first byte on NEXT page. } private int indX() { // indirect mode final int arg = ram.read(PC++); return ram.read((arg + X) & 0xff) + (ram.read((arg + 1 + X) & 0xff) << 8); // doesn't suffer from the same bug as jump indirect } private int indY(final dummy dummy) { final int arg = ram.read(PC++); final int addr = (ram.read((arg) & 0xff) | (ram.read((arg + 1) & 0xff) << 8)); if (addr >> 8 != (addr + Y) >> 8) { pb = 1; } if ((addr & 0xFF00) != ((addr + Y) & 0xFF00) && dummy == dummy.ONCARRY) { ram.read((addr & 0xFF00) | ((addr + Y) & 0xFF)); } if (dummy == dummy.ALWAYS) { ram.read((addr & 0xFF00) | ((addr + Y) & 0xFF)); } return (addr + Y) & 0xffff; } public final int flagstobyte() { return ((negativeFlag ? utils.BIT7 : 0) | (overflowFlag ? utils.BIT6 : 0) | utils.BIT5 | (decimalModeFlag ? utils.BIT3 : 0) | (interruptsDisabled ? utils.BIT2 : 0) | (zeroFlag ? utils.BIT1 : 0) | (carryFlag ? utils.BIT0 : 0)); } private void bytetoflags(final int statusbyte) { negativeFlag = ((statusbyte & utils.BIT7) != 0); overflowFlag = ((statusbyte & utils.BIT6) != 0); //breakFlag = ((b & 32) != 0); // unusedFlag = ((b & 16) != 0); // actually nestest wants the unused flag to always be zero, // and doesn't set the break flag with a plp decimalModeFlag = ((statusbyte & utils.BIT3) != 0); interruptsDisabled = ((statusbyte & utils.BIT2) != 0); zeroFlag = ((statusbyte & utils.BIT1) != 0); carryFlag = ((statusbyte & utils.BIT0) != 0); } public String status() { //TODO: convert to format string. lots of wasted strings return " PC:" + utils.hex(PC) + " A:" + utils.hex(A) + " X:" + utils.hex(X) + " Y:" + utils.hex(Y) + " P:" + utils.hex(flagstobyte()) + " SP:" + utils.hex(S); } public static String[] opcodes() { //%1 1st byte, %2 2nd byte, %3 relative offset from PC //odd combination of format string and eventual syntax in file here. String[] op = new String[0x100]; op[0x00] = "BRK"; op[0x01] = "ORA $(%2$02X%1$02X,x)"; op[0x02] = "KIL"; op[0x03] = "SLO $(%2$02X%1$02X,x)"; op[0x04] = "NOP $%1$02X"; op[0x05] = "ORA $%1$02X"; op[0x06] = "ASL $%1$02X"; op[0x07] = "SLO $%1$02X"; op[0x08] = "PHP"; op[0x09] = "ORA #$%1$02X"; op[0x0A] = "ASL A"; op[0x0B] = "ANC #$%1$02X"; op[0x0C] = "NOP $%2$02X%1$02X"; op[0x0D] = "ORA $%2$02X%1$02X"; op[0x0E] = "ASL $%2$02X%1$02X"; op[0x0F] = "SLO $%2$02X%1$02X"; op[0x10] = "BPL $%3$02X"; op[0x11] = "ORA ($%1$02X), y"; op[0x12] = "KIL"; op[0x13] = "SLO ($%1$02X), y"; op[0x14] = "NOP $%1$02X,x"; op[0x15] = "ORA $%1$02X,x"; op[0x16] = "ASL $%1$02X,x"; op[0x17] = "SLO $%1$02X,x"; op[0x18] = "CLC"; op[0x19] = "ORA $%2$02X%1$02X,y"; op[0x1A] = "NOP"; op[0x1B] = "SLO $%2$02X%1$02X,y"; op[0x1C] = "NOP $%2$02X%1$02X,x"; op[0x1D] = "ORA $%2$02X%1$02X,x"; op[0x1E] = "ASL $%2$02X%1$02X,x"; op[0x1F] = "SLO $%2$02X%1$02X,x"; op[0x20] = "JSR $%2$02X%1$02X"; op[0x21] = "AND $(%2$02X%1$02X,x)"; op[0x22] = "KIL"; op[0x23] = "RLA $(%2$02X%1$02X,x)"; op[0x24] = "BIT $%1$02X"; op[0x25] = "AND $%1$02X"; op[0x26] = "ROL $%1$02X"; op[0x27] = "RLA $%1$02X"; op[0x28] = "PLP"; op[0x29] = "AND #$%1$02X"; op[0x2A] = "ROL"; op[0x2B] = "ANC #$%1$02X"; op[0x2C] = "BIT $%2$02X%1$02X"; op[0x2D] = "AND $%2$02X%1$02X"; op[0x2E] = "ROL $%2$02X%1$02X"; op[0x2F] = "RLA $%2$02X%1$02X"; op[0x30] = "BMI $%3$02X"; op[0x31] = "AND ($%1$02X), y"; op[0x32] = "KIL"; op[0x33] = "RLA ($%1$02X), y"; op[0x34] = "NOP $%1$02X,x"; op[0x35] = "AND $%1$02X,x"; op[0x36] = "ROL $%1$02X,x"; op[0x37] = "RLA $%1$02X,x"; op[0x38] = "SEC"; op[0x39] = "AND $%2$02X%1$02X,y"; op[0x3A] = "NOP"; op[0x3B] = "RLA $%2$02X%1$02X,y"; op[0x3C] = "NOP $%2$02X%1$02X,x"; op[0x3D] = "AND $%2$02X%1$02X,x"; op[0x3E] = "ROL $%2$02X%1$02X,x"; op[0x3F] = "RLA $%2$02X%1$02X,x"; op[0x40] = "RTI"; op[0x41] = "EOR $(%2$02X%1$02X,x)"; op[0x42] = "KIL"; op[0x43] = "SRE $(%2$02X%1$02X,x)"; op[0x44] = "NOP $%1$02X"; op[0x45] = "EOR $%1$02X"; op[0x46] = "LSR $%1$02X"; op[0x47] = "SRE $%1$02X"; op[0x48] = "PHA"; op[0x49] = "EOR #$%1$02X"; op[0x4A] = "LSR"; op[0x4B] = "ALR #$%1$02X"; op[0x4C] = "JMP $%2$02X%1$02X"; op[0x4D] = "EOR $%2$02X%1$02X"; op[0x4E] = "LSR $%2$02X%1$02X"; op[0x4F] = "SRE $%2$02X%1$02X"; op[0x50] = "BVC $%3$02X"; op[0x51] = "EOR ($%1$02X), y"; op[0x52] = "KIL"; op[0x53] = "SRE ($%1$02X), y"; op[0x54] = "NOP $%1$02X,x"; op[0x55] = "EOR $%1$02X,x"; op[0x56] = "LSR $%1$02X,x"; op[0x57] = "SRE $%1$02X,x"; op[0x58] = "CLI"; op[0x59] = "EOR $%2$02X%1$02X,y"; op[0x5A] = "NOP"; op[0x5B] = "SRE $%2$02X%1$02X,y"; op[0x5C] = "NOP $%2$02X%1$02X,x"; op[0x5D] = "EOR $%2$02X%1$02X,x"; op[0x5E] = "LSR $%2$02X%1$02X,x"; op[0x5F] = "SRE $%2$02X%1$02X,x"; op[0x60] = "RTS"; op[0x61] = "ADC $(%2$02X%1$02X,x)"; op[0x62] = "KIL"; op[0x63] = "RRA $(%2$02X%1$02X,x)"; op[0x64] = "NOP $%1$02X"; op[0x65] = "ADC $%1$02X"; op[0x66] = "ROR $%1$02X"; op[0x67] = "RRA $%1$02X"; op[0x68] = "PLA"; op[0x69] = "ADC #$%1$02X"; op[0x6A] = "ROR"; op[0x6B] = "ARR #$%1$02X"; op[0x6C] = "JMP ($%2$02X%1$02X)"; op[0x6D] = "ADC $%2$02X%1$02X"; op[0x6E] = "ROR $%2$02X%1$02X"; op[0x6F] = "RRA $%2$02X%1$02X"; op[0x70] = "BVS $%3$02X"; op[0x71] = "ADC ($%1$02X), y"; op[0x72] = "KIL"; op[0x73] = "RRA ($%1$02X), y"; op[0x74] = "NOP $%1$02X,x"; op[0x75] = "ADC $%1$02X,x"; op[0x76] = "ROR $%1$02X,x"; op[0x77] = "RRA $%1$02X,x"; op[0x78] = "SEI"; op[0x79] = "ADC $%2$02X%1$02X,y"; op[0x7A] = "NOP"; op[0x7B] = "RRA $%2$02X%1$02X,y"; op[0x7C] = "NOP $%2$02X%1$02X,x"; op[0x7D] = "ADC $%2$02X%1$02X,x"; op[0x7E] = "ROR $%2$02X%1$02X,x"; op[0x7F] = "RRA $%2$02X%1$02X,x"; op[0x80] = "NOP #$%1$02X"; op[0x81] = "STA $(%2$02X%1$02X,x)"; op[0x82] = "NOP #$%1$02X"; op[0x83] = "SAX $(%2$02X%1$02X,x)"; op[0x84] = "STY $%1$02X"; op[0x85] = "STA $%1$02X"; op[0x86] = "STX $%1$02X"; op[0x87] = "SAX $%1$02X"; op[0x88] = "DEY"; op[0x89] = "NOP #$%1$02X"; op[0x8A] = "TXA"; op[0x8B] = "XAA #$%1$02X"; op[0x8C] = "STY $%2$02X%1$02X"; op[0x8D] = "STA $%2$02X%1$02X"; op[0x8E] = "STX $%2$02X%1$02X"; op[0x8F] = "SAX $%2$02X%1$02X"; op[0x90] = "BCC $%3$02X"; op[0x91] = "STA ($%1$02X), y"; op[0x92] = "KIL"; op[0x93] = "AHX ($%1$02X), y"; op[0x94] = "STY $%1$02X,x"; op[0x95] = "STA $%1$02X,x"; op[0x96] = "STX $%1$02X,y"; op[0x97] = "SAX $%1$02X,y"; op[0x98] = "TYA"; op[0x99] = "STA $%2$02X%1$02X,y"; op[0x9A] = "TXS"; op[0x9B] = "TAS $%2$02X%1$02X,y"; op[0x9C] = "SHY $%2$02X%1$02X,x"; op[0x9D] = "STA $%2$02X%1$02X,x"; op[0x9E] = "SHX $%2$02X%1$02X,y"; op[0x9F] = "AHX $%2$02X%1$02X,y"; op[0xA0] = "LDY #$%1$02X"; op[0xA1] = "LDA $(%2$02X%1$02X,x)"; op[0xA2] = "LDX #$%1$02X"; op[0xA3] = "LAX $(%2$02X%1$02X,x)"; op[0xA4] = "LDY $%1$02X"; op[0xA5] = "LDA $%1$02X"; op[0xA6] = "LDX $%1$02X"; op[0xA7] = "LAX $%1$02X"; op[0xA8] = "TAY"; op[0xA9] = "LDA #$%1$02X"; op[0xAA] = "TAX"; op[0xAB] = "LAX #$%1$02X"; op[0xAC] = "LDY $%2$02X%1$02X"; op[0xAD] = "LDA $%2$02X%1$02X"; op[0xAE] = "LDX $%2$02X%1$02X"; op[0xAF] = "LAX $%2$02X%1$02X"; op[0xB0] = "BCS $%3$02X"; op[0xB1] = "LDA ($%1$02X), y"; op[0xB2] = "KIL"; op[0xB3] = "LAX ($%1$02X), y"; op[0xB4] = "LDY $%1$02X,x"; op[0xB5] = "LDA $%1$02X,x"; op[0xB6] = "LDX $%1$02X,y"; op[0xB7] = "LAX $%1$02X,y"; op[0xB8] = "CLV"; op[0xB9] = "LDA $%2$02X%1$02X,y"; op[0xBA] = "TSX"; op[0xBB] = "LAS $%2$02X%1$02X,y"; op[0xBC] = "LDY $%2$02X%1$02X,x"; op[0xBD] = "LDA $%2$02X%1$02X,x"; op[0xBE] = "LDX $%2$02X%1$02X,y"; op[0xBF] = "LAX $%2$02X%1$02X,y"; op[0xC0] = "CPY #$%1$02X"; op[0xC1] = "CMP $(%2$02X%1$02X,x)"; op[0xC2] = "NOP #$%1$02X"; op[0xC3] = "DCP $(%2$02X%1$02X,x)"; op[0xC4] = "CPY $%1$02X"; op[0xC5] = "CMP $%1$02X"; op[0xC6] = "DEC $%1$02X"; op[0xC7] = "DCP $%1$02X"; op[0xC8] = "INY"; op[0xC9] = "CMP #$%1$02X"; op[0xCA] = "DEX"; op[0xCB] = "AXS #$%1$02X"; op[0xCC] = "CPY $%2$02X%1$02X"; op[0xCD] = "CMP $%2$02X%1$02X"; op[0xCE] = "DEC $%2$02X%1$02X"; op[0xCF] = "DCP $%2$02X%1$02X"; op[0xD0] = "BNE $%3$02X"; op[0xD1] = "CMP ($%1$02X), y"; op[0xD2] = "KIL"; op[0xD3] = "DCP ($%1$02X), y"; op[0xD4] = "NOP $%1$02X,x"; op[0xD5] = "CMP $%1$02X,x"; op[0xD6] = "DEC $%1$02X,x"; op[0xD7] = "DCP $%1$02X,x"; op[0xD8] = "CLD"; op[0xD9] = "CMP $%2$02X%1$02X,y"; op[0xDA] = "NOP"; op[0xDB] = "DCP $%2$02X%1$02X,y"; //did i delete this line somehow? op[0xDC] = "NOP $%2$02X%1$02X,x"; op[0xDD] = "CMP $%2$02X%1$02X,x"; op[0xDE] = "DEC $%2$02X%1$02X,x"; op[0xDF] = "DCP $%2$02X%1$02X,x"; op[0xE0] = "CPX #$%1$02X"; op[0xE1] = "SBC $(%2$02X%1$02X,x)"; op[0xE2] = "NOP #$%1$02X"; op[0xE3] = "ISC $(%2$02X%1$02X,x)"; op[0xE4] = "CPX $%1$02X"; op[0xE5] = "SBC $%1$02X"; op[0xE6] = "INC $%1$02X"; op[0xE7] = "ISC $%1$02X"; op[0xE8] = "INX"; op[0xE9] = "SBC #$%1$02X"; op[0xEA] = "NOP"; op[0xEB] = "SBC #$%1$02X"; op[0xEC] = "CPX $%2$02X%1$02X"; op[0xED] = "SBC $%2$02X%1$02X"; op[0xEE] = "INC $%2$02X%1$02X"; op[0xEF] = "ISC $%2$02X%1$02X"; op[0xF0] = "BEQ $%3$02X"; op[0xF1] = "SBC ($%1$02X), y"; op[0xF2] = "KIL"; op[0xF3] = "ISC ($%1$02X), y"; op[0xF4] = "NOP $%1$02X,x"; op[0xF5] = "SBC $%1$02X,x"; op[0xF6] = "INC $%1$02X,x"; op[0xF7] = "ISC $%1$02X,x"; op[0xF8] = "SED"; op[0xF9] = "SBC $%2$02X%1$02X,y"; op[0xFA] = "NOP"; op[0xFB] = "ISC $%2$02X%1$02X,y"; op[0xFC] = "NOP $%2$02X%1$02X,x"; op[0xFD] = "SBC $%2$02X%1$02X,x"; op[0xFE] = "INC $%2$02X%1$02X,x"; op[0xFF] = "ISC $%2$02X%1$02X,x"; return op; } //these methods are needed for NSF playing use public void setRegA(int value) { A = value & 0xff; } public void setRegX(int value) { X = value & 0xff; } public void setPC(int value) { PC = value & 0xffff; idle = false; log("**PC SET**"); } public final void log(String tolog) { if (logging) { try { w.write(tolog); } catch (IOException e) { System.err.println("Cannot write to debug log" + e.getLocalizedMessage()); } } } private void flushLog() { if (logging) { try { w.flush(); } catch (IOException e) { System.err.println("Cannot write to debug log" + e.getLocalizedMessage()); } } } }