// // // File Name : MCL65+.ino // Used on : MCL65+ // Author : Ted Fried, MicroCore Labs // Creation : 1/1/2021 // // Description: // ============ // // MOS 6502 emulator with bus interface. // // Runs on a Teensy 4.1 where it can emulate the 6502's bus signals. // // When used with the MCL65+ PCB it can be used as a drop-in replacement // for the MOS 6502 processor. // //------------------------------------------------------------------------ // // Modification History: // ===================== // // Revision 1 1/1/2021 // Initial revision // // //------------------------------------------------------------------------ // // Copyright (c) 2021 Ted Fried // // 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 THE // AUTHORS OR COPYRIGHT HOLDERS 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. // //------------------------------------------------------------------------ #include // Teensy 4.1 pin assignments // #define PIN_CLK0 24 #define PIN_RESET 40 #define PIN_READY_n 26 #define PIN_IRQ 25 #define PIN_NMI 41 #define PIN_RDWR_n 12 #define PIN_SYNC 39 #define PIN_ADDR0 27 #define PIN_ADDR1 38 #define PIN_ADDR2 28 #define PIN_ADDR3 37 #define PIN_ADDR4 29 #define PIN_ADDR5 36 #define PIN_ADDR6 30 #define PIN_ADDR7 35 #define PIN_ADDR8 31 #define PIN_ADDR9 34 #define PIN_ADDR10 32 #define PIN_ADDR11 33 #define PIN_ADDR12 1 #define PIN_ADDR13 0 #define PIN_ADDR14 2 #define PIN_ADDR15 23 #define PIN_DATAIN0 14 #define PIN_DATAIN1 15 #define PIN_DATAIN2 16 #define PIN_DATAIN3 17 #define PIN_DATAIN4 18 #define PIN_DATAIN5 19 #define PIN_DATAIN6 20 #define PIN_DATAIN7 21 #define PIN_DATAOUT0 11 #define PIN_DATAOUT1 10 #define PIN_DATAOUT2 9 #define PIN_DATAOUT3 8 #define PIN_DATAOUT4 7 #define PIN_DATAOUT5 6 #define PIN_DATAOUT6 5 #define PIN_DATAOUT7 4 #define PIN_DATAOUT_OE_n 3 // 6502 Flags // #define flag_n (register_flags & 0x80) >> 7 // register_flags[7] #define flag_v (register_flags & 0x40) >> 6 // register_flags[6] #define flag_b (register_flags & 0x10) >> 4 // register_flags[4] #define flag_d (register_flags & 0x08) >> 3 // register_flags[3] #define flag_i (register_flags & 0x04) >> 2 // register_flags[2] #define flag_z (register_flags & 0x02) >> 1 // register_flags[1] #define flag_c (register_flags & 0x01) >> 0 // register_flags[0] // 6502 stack always in Page 1 // #define register_sp_fixed (0x0100 | register_sp) // CPU register for direct reads of the GPIOs // uint8_t register_flags=0x34; uint8_t next_instruction; uint8_t internal_memory_range=0; uint8_t nmi_n_old=1; uint8_t register_a=0; uint8_t register_x=0; uint8_t register_y=0; uint8_t register_sp=0xFF; uint8_t direct_datain=0; uint8_t direct_reset=0; uint8_t direct_ready_n=0; uint8_t direct_irq=0; uint8_t direct_nmi=0; uint8_t assert_sync=0; uint8_t global_temp=0; uint8_t last_access_internal_RAM=0; uint8_t internal_RAM[65536]; uint16_t register_pc=0; uint16_t current_address=0; uint16_t effective_address=0; // VIC20 Cartridge ROMs // uint8_t cartridge_RAM_A[0x2FFF]; uint8_t cartridge_RAM_6[0x2FFF]; uint8_t cartridge_RAM_2[0x2FFF]; // Apple II+ ROMS // uint8_t AppleIIP_ROM_D0[0x0800]; uint8_t AppleIIP_ROM_D8[0x0800]; uint8_t AppleIIP_ROM_E0[0x0800]; uint8_t AppleIIP_ROM_E8[0x0800]; uint8_t AppleIIP_ROM_F0[0x0800]; uint8_t AppleIIP_ROM_F8[0x0800]; // ------------------------------------------------------------------------------ // ------------------------------------------------------------------------------ // Setup Teensy 4.1 IO's // void setup() { pinMode(PIN_CLK0, INPUT); pinMode(PIN_RESET, INPUT); pinMode(PIN_READY_n, INPUT); pinMode(PIN_IRQ, INPUT); pinMode(PIN_NMI, INPUT); pinMode(PIN_RDWR_n, OUTPUT); pinMode(PIN_SYNC, OUTPUT); pinMode(PIN_ADDR0, OUTPUT); pinMode(PIN_ADDR1, OUTPUT); pinMode(PIN_ADDR2, OUTPUT); pinMode(PIN_ADDR3, OUTPUT); pinMode(PIN_ADDR4, OUTPUT); pinMode(PIN_ADDR5, OUTPUT); pinMode(PIN_ADDR6, OUTPUT); pinMode(PIN_ADDR7, OUTPUT); pinMode(PIN_ADDR8, OUTPUT); pinMode(PIN_ADDR9, OUTPUT); pinMode(PIN_ADDR10, OUTPUT); pinMode(PIN_ADDR11, OUTPUT); pinMode(PIN_ADDR12, OUTPUT); pinMode(PIN_ADDR13, OUTPUT); pinMode(PIN_ADDR14, OUTPUT); pinMode(PIN_ADDR15, OUTPUT); pinMode(PIN_DATAIN0, INPUT); pinMode(PIN_DATAIN1, INPUT); pinMode(PIN_DATAIN2, INPUT); pinMode(PIN_DATAIN3, INPUT); pinMode(PIN_DATAIN4, INPUT); pinMode(PIN_DATAIN5, INPUT); pinMode(PIN_DATAIN6, INPUT); pinMode(PIN_DATAIN7, INPUT); pinMode(PIN_DATAOUT0, OUTPUT); pinMode(PIN_DATAOUT1, OUTPUT); pinMode(PIN_DATAOUT2, OUTPUT); pinMode(PIN_DATAOUT3, OUTPUT); pinMode(PIN_DATAOUT4, OUTPUT); pinMode(PIN_DATAOUT5, OUTPUT); pinMode(PIN_DATAOUT6, OUTPUT); pinMode(PIN_DATAOUT7, OUTPUT); pinMode(PIN_DATAOUT_OE_n, OUTPUT); // Copy VIC20 cartridge ROM arrays into the emlulator's main RAM array. // //for (int u=0; u<=0x1FFF ; u++) { internal_RAM[0x2000+u] = cartridge_RAM_2[u+2]; } //for (int u=0; u<=0x1FFF ; u++) { internal_RAM[0x6000+u] = cartridge_RAM_6[u+2]; } //for (int u=0; u<=0x1FFF ; u++) { internal_RAM[0xa000+u] = cartridge_RAM_A[u+2]; } //for (uint32_t u=0; u<=0x07FF ; u++) { internal_RAM[0xD000+u] = AppleIIP_ROM_D0[u]; } //for (uint32_t u=0; u<=0x07FF ; u++) { internal_RAM[0xD800+u] = AppleIIP_ROM_D8[u]; } //for (uint32_t u=0; u<=0x07FF ; u++) { internal_RAM[0xE000+u] = AppleIIP_ROM_E0[u]; } //for (uint32_t u=0; u<=0x07FF ; u++) { internal_RAM[0xE800+u] = AppleIIP_ROM_E8[u]; } //for (uint32_t u=0; u<=0x07FF ; u++) { internal_RAM[0xF000+u] = AppleIIP_ROM_F0[u]; } //for (uint32_t u=0; u<=0x07FF ; u++) { internal_RAM[0xF800+u] = AppleIIP_ROM_F8[u]; } } // -------------------------------------------------------------------------------------------------- // -------------------------------------------------------------------------------------------------- // // Begin 6502 Bus Interface Unit // // -------------------------------------------------------------------------------------------------- // -------------------------------------------------------------------------------------------------- // ---------------------------------------------------------- // Address range check // Return: 0x0 for exernal memory // 0x1 for internal memory running cycle accurate // 0x2 for internal memory accelerated // ---------------------------------------------------------- inline uint8_t internal_address_check(uint16_t local_address) { //if ( (local_address>=0x2000) && (local_address<0x8000) ) return 0x2; // VIC20 RAM Expansion //if ( (local_address>=0xA000) && (local_address<0xC000) ) return 0x2; // VIC20 Cartridge ROM if ( (local_address>=0x0000) && (local_address <0x0400) ) return 0x0; // 6502 ZeroPage and Stack if ( (local_address>=0x0400) && (local_address <0x07FF) ) return 0x0; // Apple II Text Page 1 if ( (local_address>=0x0800) && (local_address <0xC000) ) return 0x0; // Apple II RAM // Skip Apple II I/O Range 0xC000 if ( (local_address>=0xD000) && (local_address <0xE000) ) return 0x0; // Apple II ROM D0/B8 if ( (local_address>=0xE000) && (local_address <0xF000) ) return 0x0; // Apple II ROM E0/E8 if ( (local_address>=0xF000) && (local_address<=0xFFFF) ) return 0x0; // Apple II ROM F0/F8 return 0x0; } // ------------------------------------------------- // Wait for the CLK1 rising edge and sample signals // ------------------------------------------------- inline void wait_for_CLK_rising_edge() { register uint32_t GPIO6_data=0; uint32_t d10, d2, d3, d4, d5, d76; while (((GPIO6_DR >> 12) & 0x1)!=0) {} // Teensy 4.1 Pin-24 GPIO6_DR[12] CLK //while (((GPIO6_DR >> 12) & 0x1)==0) {GPIO6_data=GPIO6_DR;} // This method is ok for VIC-20 and Apple-II+ non-DRAM ranges do { GPIO6_data_d1=GPIO6_DR; } while (((GPIO6_data_d1 >> 12) & 0x1)==0); // This method needed to support Apple-II+ DRAM read data setup time GPIO6_data=GPIO6_data_d1; d10 = (GPIO6_data&0x000C0000) >> 18; // Teensy 4.1 Pin-14 GPIO6_DR[19:18] D1:D0 d2 = (GPIO6_data&0x00800000) >> 21; // Teensy 4.1 Pin-16 GPIO6_DR[23] D2 d3 = (GPIO6_data&0x00400000) >> 19; // Teensy 4.1 Pin-17 GPIO6_DR[22] D3 d4 = (GPIO6_data&0x00020000) >> 13; // Teensy 4.1 Pin-18 GPIO6_DR[17] D4 d5 = (GPIO6_data&0x00010000) >> 11; // Teensy 4.1 Pin-19 GPIO6_DR[16] D5 d76 = (GPIO6_data&0x0C000000) >> 20; // Teensy 4.1 Pin-20 GPIO6_DR[27:26] D7:D6 direct_irq = (GPIO6_data&0x00002000) >> 13; // Teensy 4.1 Pin-25 GPIO6_DR[13] IRQ direct_ready_n = (GPIO6_data&0x40000000) >> 30; // Teensy 4.1 Pin-26 GPIO6_DR[30] READY direct_reset = (GPIO6_data&0x00100000) >> 20; // Teensy 4.1 Pin-40 GPIO6_DR[20] RESET direct_nmi = (GPIO6_data&0x00200000) >> 21; // Teensy 4.1 Pin-41 GPIO6_DR[21] NMI direct_datain = d76 | d5 | d4 | d3 | d2 | d10; return; } // ------------------------------------------------- // Wait for the CLK1 falling edge // ------------------------------------------------- inline void wait_for_CLK_falling_edge() { while (((GPIO6_DR >> 12) & 0x1)==0) {} // Teensy 4.1 Pin-24 GPIO6_DR[12] CLK while (((GPIO6_DR >> 12) & 0x1)!=0) {} return; } // ------------------------------------------------- // Drive the 6502 Address pins // ------------------------------------------------- inline void send_address(uint32_t local_address) { register uint32_t writeback_data=0; writeback_data = (0x6DFFFFF3 & GPIO6_DR); // Read in current GPIOx register value and clear the bits we intend to update writeback_data = writeback_data | (local_address & 0x8000)<<10 ; // 6502_Address[15] TEENSY_PIN23 GPIO6_DR[25] writeback_data = writeback_data | (local_address & 0x2000)>>10 ; // 6502_Address[13] TEENSY_PIN0 GPIO6_DR[3] writeback_data = writeback_data | (local_address & 0x1000)>>10 ; // 6502_Address[12] TEENSY_PIN1 GPIO6_DR[2] writeback_data = writeback_data | (local_address & 0x0002)<<27 ; // 6502_Address[1] TEENSY_PIN38 GPIO6_DR[28] GPIO6_DR = writeback_data | (local_address & 0x0001)<<31 ; // 6502_Address[0] TEENSY_PIN27 GPIO6_DR[31] writeback_data = (0xCFF3EFFF & GPIO7_DR); // Read in current GPIOx register value and clear the bits we intend to update writeback_data = writeback_data | (local_address & 0x0400)<<2 ; // 6502_Address[10] TEENSY_PIN32 GPIO7_DR[12] writeback_data = writeback_data | (local_address & 0x0200)<<20 ; // 6502_Address[9] TEENSY_PIN34 GPIO7_DR[29] writeback_data = writeback_data | (local_address & 0x0080)<<21 ; // 6502_Address[7] TEENSY_PIN35 GPIO7_DR[28] writeback_data = writeback_data | (local_address & 0x0020)<<13 ; // 6502_Address[5] TEENSY_PIN36 GPIO7_DR[18] GPIO7_DR = writeback_data | (local_address & 0x0008)<<16 ; // 6502_Address[3] TEENSY_PIN37 GPIO7_DR[19] writeback_data = (0xFF3BFFFF & GPIO8_DR); // Read in current GPIOx register value and clear the bits we intend to update writeback_data = writeback_data | (local_address & 0x0100)<<14 ; // 6502_Address[8] TEENSY_PIN31 GPIO8_DR[22] writeback_data = writeback_data | (local_address & 0x0040)<<17 ; // 6502_Address[6] TEENSY_PIN30 GPIO8_DR[23] GPIO8_DR = writeback_data | (local_address & 0x0004)<<16 ; // 6502_Address[2] TEENSY_PIN28 GPIO8_DR[18] writeback_data = (0x7FFFFF6F & GPIO9_DR); // Read in current GPIOx register value and clear the bits we intend to update writeback_data = writeback_data | (local_address & 0x4000)>>10 ; // 6502_Address[14] TEENSY_PIN2 GPIO9_DR[4] writeback_data = writeback_data | (local_address & 0x0800)>>4 ; // 6502_Address[11] TEENSY_PIN33 GPIO9_DR[7] GPIO9_DR = writeback_data | (local_address & 0x0010)<<27 ; // 6502_Address[4] TEENSY_PIN29 GPIO9_DR[31] return; } // ------------------------------------------------- // Send the address for a read cyle // ------------------------------------------------- inline void start_read(uint32_t local_address) { current_address = local_address; if (internal_address_check(current_address)>0x1) { //last_access_internal_RAM=1; } else { if (last_access_internal_RAM==1) wait_for_CLK_rising_edge(); last_access_internal_RAM=0; digitalWriteFast(PIN_RDWR_n, 0x1); //digitalWriteFast(PIN_SYNC, assert_sync); send_address(local_address); } return; } // ------------------------------------------------- // On the rising CLK edge, read in the data // ------------------------------------------------- inline uint8_t finish_read_byte() { if (internal_address_check(current_address)>0x1) { last_access_internal_RAM=1; return internal_RAM[current_address]; } else { if (last_access_internal_RAM==1) wait_for_CLK_rising_edge(); last_access_internal_RAM=0; do { wait_for_CLK_rising_edge(); } while (direct_ready_n == 0x1); // Delay a clock cycle until ready is active if (internal_address_check(current_address)>0x0) { return internal_RAM[current_address]; } else { return direct_datain; } } } // ------------------------------------------------- // Full read cycle with address and data read in // ------------------------------------------------- inline uint8_t read_byte(uint16_t local_address) { if (internal_address_check(local_address)>0x1) { last_access_internal_RAM=1; return internal_RAM[local_address]; } else { if (last_access_internal_RAM==1) wait_for_CLK_rising_edge(); last_access_internal_RAM=0; start_read(local_address); do { wait_for_CLK_rising_edge(); } while (direct_ready_n == 0x1); // Delay a clock cycle until ready is active if (internal_address_check(current_address)>0x0) { return internal_RAM[current_address]; } else { return direct_datain; } } } // ------------------------------------------------- // Full write cycle with address and data written // ------------------------------------------------- inline void write_byte(uint16_t local_address , uint8_t local_write_data) { // Internal RAM // if (internal_address_check(local_address)>0x1) { last_access_internal_RAM=1; internal_RAM[local_address] = local_write_data; } else { if (last_access_internal_RAM==1) wait_for_CLK_rising_edge(); last_access_internal_RAM=0; internal_RAM[local_address] = local_write_data; digitalWriteFast(PIN_RDWR_n, 0x0); //digitalWriteFast(PIN_SYNC, 0x0); send_address(local_address); // Drive the data bus pins from the Teensy to the bus driver which is inactive // digitalWriteFast(PIN_DATAOUT0, (local_write_data & 0x01) ); digitalWriteFast(PIN_DATAOUT1, (local_write_data & 0x02)>>1 ); digitalWriteFast(PIN_DATAOUT2, (local_write_data & 0x04)>>2 ); digitalWriteFast(PIN_DATAOUT3, (local_write_data & 0x08)>>3 ); digitalWriteFast(PIN_DATAOUT4, (local_write_data & 0x10)>>4 ); digitalWriteFast(PIN_DATAOUT5, (local_write_data & 0x20)>>5 ); digitalWriteFast(PIN_DATAOUT6, (local_write_data & 0x40)>>6 ); digitalWriteFast(PIN_DATAOUT7, (local_write_data & 0x80)>>7 ); // During the second CLK phase, enable the data bus output drivers // wait_for_CLK_falling_edge(); digitalWriteFast(PIN_DATAOUT_OE_n, 0x0 ); wait_for_CLK_rising_edge(); digitalWriteFast(PIN_DATAOUT_OE_n, 0x1 ); } return; } // -------------------------------------------------------------------------------------------------- // -------------------------------------------------------------------------------------------------- // // End 6502 Bus Interface Unit // // -------------------------------------------------------------------------------------------------- // -------------------------------------------------------------------------------------------------- void push(uint8_t push_data) { write_byte(register_sp_fixed, push_data); register_sp = register_sp - 1; return; } uint8_t pop() { uint8_t temp=0; register_sp = register_sp + 1; temp = read_byte(register_sp_fixed); return temp; } void Calc_Flags_NEGATIVE_ZERO(uint8_t local_data) { if (0x80&local_data) register_flags = register_flags | 0x80; // Set the N flag else register_flags = register_flags & 0x7F; // Clear the N flag if (local_data==0) register_flags = register_flags | 0x02; // Set the Z flag else register_flags = register_flags & 0xFD; // Clear the Z flag return; } uint16_t Sign_Extend16(uint16_t reg_data) { if ((reg_data&0x0080)== 0x0080) { return (reg_data | 0xFF00); } else { return (reg_data & 0x00FF); } } void Begin_Fetch_Next_Opcode() { register_pc++; assert_sync=1; start_read(register_pc); return; } // ------------------------------------------------- // Addressing Modes // ------------------------------------------------- uint8_t Fetch_Immediate() { register_pc++; return read_byte(register_pc); } uint8_t Fetch_ZeroPage() { effective_address = Fetch_Immediate(); return read_byte(effective_address); } uint8_t Fetch_ZeroPage_X() { uint16_t bal; bal = Fetch_Immediate(); read_byte(register_pc+1); effective_address = (0x00FF & (bal + register_x)); return read_byte(effective_address); } uint8_t Fetch_ZeroPage_Y() { uint16_t bal; bal = Fetch_Immediate(); read_byte(register_pc+1); effective_address = (0x00FF & (bal + register_y)); return read_byte(effective_address); } uint16_t Calculate_Absolute() { uint16_t adl, adh; adl = Fetch_Immediate(); adh = Fetch_Immediate()<<8; effective_address = adl + adh; return effective_address; } uint8_t Fetch_Absolute() { uint16_t adl, adh; adl = Fetch_Immediate(); adh = Fetch_Immediate()<<8; effective_address = adl + adh; return read_byte(effective_address); } uint8_t Fetch_Absolute_X(uint8_t page_cross_check) { uint16_t bal, bah; uint8_t local_data; bal = Fetch_Immediate(); bah = Fetch_Immediate()<<8; effective_address = bah + bal + register_x; local_data = read_byte(effective_address ); if ( page_cross_check==1 && ( (0xFF00&effective_address) != (0xFF00&bah) ) ) { local_data = read_byte(effective_address ); } return local_data; } uint8_t Fetch_Absolute_Y(uint8_t page_cross_check) { uint16_t bal, bah; uint8_t local_data; bal = Fetch_Immediate(); bah = Fetch_Immediate()<<8; effective_address = bah + bal + register_y; local_data = read_byte(effective_address ); if ( page_cross_check==1 && ( (0xFF00&effective_address) != (0xFF00&bah) ) ) { local_data = read_byte(effective_address ); } return local_data; } uint8_t Fetch_Indexed_Indirect_X() { uint16_t bal; uint16_t adl, adh; uint8_t local_data; bal = Fetch_Immediate() + register_x; read_byte(bal); adl = read_byte(0xFF&bal); adh = read_byte(0xFF&(bal+1)) << 8; effective_address = adh + adl ; local_data = read_byte(effective_address); return local_data; } uint8_t Fetch_Indexed_Indirect_Y(uint8_t page_cross_check) { uint16_t ial, bah, bal; uint8_t local_data; ial = Fetch_Immediate(); bal = read_byte(0xFF&ial); bah = read_byte(0xFF&(ial+1)) << 8; effective_address = bah + bal + register_y; local_data = read_byte(effective_address); if ( page_cross_check==1 && ((0xFF00&effective_address) != (0xFF00&bah)) ) { local_data = read_byte(effective_address); } return local_data; } void Write_ZeroPage(uint8_t local_data) { effective_address = Fetch_Immediate(); write_byte(effective_address , local_data); return; } void Write_Absolute(uint8_t local_data) { effective_address = Fetch_Immediate(); effective_address = (Fetch_Immediate() << 8) + effective_address; write_byte(effective_address , local_data ); return; } void Write_ZeroPage_X(uint8_t local_data) { effective_address = Fetch_Immediate(); read_byte(effective_address); write_byte( (0x00FF&(effective_address + register_x)) , local_data ); return; } void Write_ZeroPage_Y(uint8_t local_data) { effective_address = Fetch_Immediate(); read_byte(effective_address); write_byte( (0x00FF&(effective_address + register_y)) , local_data ); return; } void Write_Absolute_X(uint8_t local_data) { uint16_t bal,bah; bal = Fetch_Immediate(); bah = Fetch_Immediate()<<8; effective_address = bal + bah + register_x; read_byte(effective_address); write_byte(effective_address , local_data ); return; } void Write_Absolute_Y(uint8_t local_data) { uint16_t bal,bah; bal = Fetch_Immediate(); bah = Fetch_Immediate()<<8; effective_address = bal + bah + register_y; read_byte(effective_address); if ( (0xFF00&effective_address) != (0xFF00&bah) ) { read_byte(effective_address); } write_byte(effective_address , local_data ); return; } void Write_Indexed_Indirect_X(uint8_t local_data) { uint16_t bal; uint16_t adl, adh; bal = Fetch_Immediate(); read_byte(bal); adl = read_byte(0xFF&(bal+register_x)); adh = read_byte(0xFF&(bal+register_x+1)) << 8; effective_address = adh + adl; write_byte(effective_address , local_data ); return; } void Write_Indexed_Indirect_Y(uint8_t local_data) { uint16_t ial; uint16_t bal, bah; ial = Fetch_Immediate(); bal = read_byte(ial); bah = read_byte(ial+1)<<8; effective_address = bah + bal + register_y; read_byte(effective_address); write_byte(effective_address , local_data ); return; } void Double_WriteBack(uint8_t local_data) { write_byte(effective_address , local_data); write_byte(effective_address , local_data); return; } // ------------------------------------------------- // Reset sequence for the 6502 // ------------------------------------------------- void reset_sequence() { uint16_t temp1, temp2; while (digitalReadFast(PIN_RESET)!=0) {} // Stay here until RESET deasserts digitalWriteFast(PIN_RDWR_n, 0x1); digitalWriteFast(PIN_DATAOUT_OE_n, 0x1 ); temp1 = read_byte(register_pc); // Address ?? temp1 = read_byte(register_pc+1); // Address ?? + 1 temp1 = read_byte(register_sp_fixed); // Address SP temp1 = read_byte(register_sp_fixed-1); // Address SP - 1 temp1 = read_byte(register_sp_fixed-2); // Address SP - 2 temp1 = read_byte(0xFFFC); // Fetch Vector PCL temp2 = read_byte(0xFFFD); // Fetch Vector PCH register_flags = 0x34; // Set the I and B flags register_pc = (temp2<<8) | temp1; assert_sync=1; start_read(register_pc); // Fetch first opcode at vector PCH,PCL return; } // ------------------------------------------------- // NMI Interrupt Processing // ------------------------------------------------- void nmi_handler() { uint16_t temp1, temp2; wait_for_CLK_rising_edge(); // Begin processing on next CLK edge register_flags = register_flags | 0x20; // Set the flag[5] register_flags = register_flags & 0xEF; // Clear the B flag read_byte(register_pc+1); // Fetch PC+1 (Discard) push(register_pc>>8); // Push PCH push(register_pc); // Push PCL push(register_flags); // Push P temp1 = read_byte(0xFFFA); // Fetch Vector PCL temp2 = read_byte(0xFFFB); // Fetch Vector PCH register_flags = register_flags | 0x34; // Set the I flag and restore the B flag register_pc = (temp2<<8) | temp1; assert_sync=1; start_read(register_pc); // Fetch first opcode at vector PCH,PCL return; } // ------------------------------------------------- // BRK & IRQ Interrupt Processing // ------------------------------------------------- void irq_handler(uint8_t opcode_is_brk) { uint16_t temp1, temp2; wait_for_CLK_rising_edge(); // Begin processing on next CLK edge register_flags = register_flags | 0x20; // Set the flag[5] if (opcode_is_brk==1) register_flags = register_flags | 0x10; // Set the B flag else register_flags = register_flags & 0xEF; // Clear the B flag read_byte(register_pc+1); // Fetch PC+1 (Discard) push(register_pc>>8); // Push PCH push(register_pc); // Push PCL push(register_flags); // Push P temp1 = read_byte(0xFFFE); // Fetch Vector PCL temp2 = read_byte(0xFFFF); // Fetch Vector PCH register_flags = register_flags | 0x34; // Set the I flag and restore the B flag register_pc = (temp2<<8) | temp1; assert_sync=1; start_read(register_pc); // Fetch first opcode at vector PCH,PCL return; } // ------------------------------------------------- // // 6502 Opcodes // // ------------------------------------------------- // ------------------------------------------------- // 0x0A - ASL A - Arithmetic Shift Left - Accumulator // ------------------------------------------------- void opcode_0x0A() { read_byte(register_pc); Begin_Fetch_Next_Opcode(); if (0x80®ister_a) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag register_a = register_a << 1; Calc_Flags_NEGATIVE_ZERO(register_a); return; } // ------------------------------------------------- // 0x4A - LSR A - Logical Shift Right - Accumulator // ------------------------------------------------- void opcode_0x4A() { read_byte(register_pc); Begin_Fetch_Next_Opcode(); if (0x01®ister_a) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag register_a = register_a >> 1; Calc_Flags_NEGATIVE_ZERO(register_a); return; } // ------------------------------------------------- // 0x6A - ROR A - Rotate Right - Accumulator // ------------------------------------------------- void opcode_0x6A() { uint8_t old_carry_flag=0; read_byte(register_pc); Begin_Fetch_Next_Opcode(); old_carry_flag = register_flags << 7; // Shift the old carry flag to bit[8] to be rotated in if (0x01®ister_a) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag register_a = ( old_carry_flag | (register_a>>1) ); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // ------------------------------------------------- // 0x2A - ROL A - Rotate Left - Accumulator // ------------------------------------------------- void opcode_0x2A() { uint8_t old_carry_flag=0; read_byte(register_pc); Begin_Fetch_Next_Opcode(); old_carry_flag = 0x1 & register_flags; // Store the old carry flag to be rotated in if (0x80®ister_a) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag register_a = (register_a<<1) | old_carry_flag; Calc_Flags_NEGATIVE_ZERO(register_a); return; } // ------------------------------------------------- // ADC // ------------------------------------------------- void Calculate_ADC(uint16_t local_data) { uint16_t total=0; uint16_t bcd_low=0; uint16_t bcd_high=0; uint16_t bcd_total=0; uint8_t operand0=0; uint8_t operand1=0; uint8_t result=0; uint8_t low_carry=0; uint8_t high_carry=0; Begin_Fetch_Next_Opcode(); if ((flag_d)==1) { bcd_low = (0x0F®ister_a) + (0x0F&local_data) + (flag_c) ; if (bcd_low>0x9) { low_carry=0x10; bcd_low = bcd_low - 0xA ; } bcd_high = (0xF0®ister_a) + (0xF0&local_data) + low_carry; if (bcd_high>0x90) { high_carry=1; bcd_high = bcd_high - 0xA0 ; } register_flags = register_flags & 0xFE; // Clear the C flag if ((0x00FF&bcd_total) > 0x09) { bcd_total=bcd_total+0x010; bcd_total=bcd_total-0x0A; } if (high_carry==1) { bcd_total=bcd_total-0xA0; register_flags = register_flags | 0x01; } // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag total = (0xFF & (bcd_low + bcd_high)); } else { total = register_a + local_data + (flag_c); if (total>255) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag } operand0 = (register_a & 0x80); operand1 = (local_data & 0x80); result = (total & 0x80); if (operand0==0 && operand1==0 && result!=0) register_flags = register_flags | 0x40; // Set the V flag else if (operand0!=0 && operand1!=0 && result==0) register_flags = register_flags | 0x40; else register_flags = register_flags & 0xBF; // Clear the V flag register_a = (0xFF & total); Calc_Flags_NEGATIVE_ZERO(register_a); return; } void opcode_0x69() { Calculate_ADC(Fetch_Immediate()); return; } // 0x69 - ADC - Immediate - Binary void opcode_0x65() { Calculate_ADC(Fetch_ZeroPage()); return; } // 0x65 - ADC - ZeroPage void opcode_0x75() { Calculate_ADC(Fetch_ZeroPage_X()); return; } // 0x75 - ADC - ZeroPage , X void opcode_0x6D() { Calculate_ADC(Fetch_Absolute()); return; } // 0x6D - ADC - Absolute void opcode_0x7D() { Calculate_ADC(Fetch_Absolute_X(1)); return; } // 0x7D - ADC - Absolute , X void opcode_0x79() { Calculate_ADC(Fetch_Absolute_Y(1)); return; } // 0x79 - ADC - Absolute , Y void opcode_0x61() { Calculate_ADC(Fetch_Indexed_Indirect_X()); return; } // 0x61 - ADC - Indexed Indirect X void opcode_0x71() { Calculate_ADC(Fetch_Indexed_Indirect_Y(1)); return; } // 0x71 - ADC - Indirect Indexed Y // ------------------------------------------------- // SBC // ------------------------------------------------- void Calculate_SBC(uint16_t local_data) { uint16_t total=0; uint16_t bcd_low=0; uint16_t bcd_high=0; uint16_t bcd_total=0; int16_t signed_total=0; uint8_t operand0=0; uint8_t operand1=0; uint8_t result=0; uint8_t flag_c_invert=0; uint8_t low_carry=0; uint8_t high_carry=0; Begin_Fetch_Next_Opcode(); if (flag_c!=0) flag_c_invert=0; else flag_c_invert=1; if ((flag_d)==1) { bcd_low = (0x0F®ister_a) - (0x0F&local_data) - flag_c_invert ; if (bcd_low>0x9) { low_carry=0x10; bcd_low = bcd_low + 0xA ; } bcd_high = (0xF0®ister_a) - (0xF0&local_data) - low_carry; if (bcd_high>0x90) { high_carry=1; bcd_high = bcd_high + 0xA0 ; } register_flags = register_flags & 0xFE; // Clear the C flag if ((0x00FF&bcd_total) > 0x09) { bcd_total=bcd_total+0x010; bcd_total=bcd_total-0x0A; } if (high_carry==0) { bcd_total=bcd_total-0xA0; register_flags = register_flags | 0x01; } // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag total = (0xFF & (bcd_low + bcd_high)); } else { total = register_a - local_data - flag_c_invert; signed_total = (int16_t)register_a - (int16_t)(local_data ) - flag_c_invert; if (signed_total>=0) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag } operand0 = (register_a & 0x80); operand1 = (local_data & 0x80); result = (total & 0x80); if (operand0==0 && operand1!=0 && result!=0) register_flags = register_flags | 0x40; // Set the V flag else if (operand0!=0 && operand1==0 && result==0) register_flags = register_flags | 0x40; else register_flags = register_flags & 0xBF; // Clear the V flag register_a = (0xFF & total); Calc_Flags_NEGATIVE_ZERO(register_a); return; } void opcode_0xE9() { Calculate_SBC(Fetch_Immediate()); return; } // 0xE9 - SBC - Immediate void opcode_0xE5() { Calculate_SBC(Fetch_ZeroPage()); return; } // 0xE5 - SBC - ZeroPage void opcode_0xF5() { Calculate_SBC(Fetch_ZeroPage_X()); return; } // 0xF5 - SBC - ZeroPage , X void opcode_0xED() { Calculate_SBC(Fetch_Absolute()); return; } // 0xED - SBC - Absolute void opcode_0xFD() { Calculate_SBC(Fetch_Absolute_X(1)); return; } // 0xFD - SBC - Absolute , X void opcode_0xF9() { Calculate_SBC(Fetch_Absolute_Y(1)); return; } // 0xF9 - SBC - Absolute , Y void opcode_0xE1() { Calculate_SBC(Fetch_Indexed_Indirect_X()); return; } // 0xE1 - SBC - Indexed Indirect X void opcode_0xF1() { Calculate_SBC(Fetch_Indexed_Indirect_Y(1)); return; } // 0xF1 - SBC - Indirect Indexed Y // ------------------------------------------------- // Flag set/resets and NOP // ------------------------------------------------- void opcode_0xEA() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); return; } // 0xEA - NOP void opcode_0x18() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_flags=register_flags&0xFE; return; } // 0x18 - CLC - Clear Carry Flag void opcode_0xD8() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_flags=register_flags&0xF7; return; } // 0xD8 - CLD - Clear Decimal Mode void opcode_0x58() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_flags=register_flags&0xFB; return; } // 0x58 - CLI - Clear Interrupt Flag void opcode_0xB8() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_flags=register_flags&0xBF; return; } // 0xB8 - CLV - Clear Overflow Flag void opcode_0x38() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_flags=register_flags|0x01; return; } // 0x38 - SEC - Set Carry Flag void opcode_0x78() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_flags=register_flags|0x04; return; } // 0x78 - SEI - Set Interrupt Flag void opcode_0xF8() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_flags=register_flags|0x08; return; } // 0xF8 - SED - Set Decimal Mode // ------------------------------------------------- // Increment/Decrements // ------------------------------------------------- void opcode_0xCA() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_x=register_x-1; Calc_Flags_NEGATIVE_ZERO(register_x); return; } // 0xCA - DEX - Decrement X void opcode_0x88() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_y=register_y-1; Calc_Flags_NEGATIVE_ZERO(register_y); return; } // 0x88 - DEY - Decrement Y void opcode_0xE8() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_x=register_x+1; Calc_Flags_NEGATIVE_ZERO(register_x); return; } // 0xE8 - INX - Increment X void opcode_0xC8() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_y=register_y+1; Calc_Flags_NEGATIVE_ZERO(register_y); return; } // 0xC8 - INY - Increment Y // ------------------------------------------------- // Transfers // ------------------------------------------------- void opcode_0xAA() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_x=register_a; Calc_Flags_NEGATIVE_ZERO(register_x); return; } // 0xAA - TAX - Transfer Accumulator to X void opcode_0xA8() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_y=register_a; Calc_Flags_NEGATIVE_ZERO(register_y); return; } // 0xA8 - TAY - Transfer Accumulator to Y void opcode_0xBA() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_x=register_sp; Calc_Flags_NEGATIVE_ZERO(register_x); return; } // 0xBA - TSX - Transfer Stack Pointer to X void opcode_0x8A() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_a=register_x; Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x8A - TXA - Transfer X to Accumulator void opcode_0x9A() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_sp=register_x; return; } // 0x9A - TXS - Transfer X to Stack Pointer void opcode_0x98() { read_byte(register_pc+1); Begin_Fetch_Next_Opcode(); register_a=register_y; Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x98 - TYA - Transfer Y to Accumulator // ------------------------------------------------- // PUSH/POP Flags and Accumulator // ------------------------------------------------- void opcode_0x08() { read_byte(register_pc+1); push(register_flags|0x30); Begin_Fetch_Next_Opcode(); return; } // 0x08 - PHP - Push Flags to Stack void opcode_0x48() { read_byte(register_pc+1); push(register_a); Begin_Fetch_Next_Opcode(); return; } // 0x48 - PHA - Push Accumulator to the stack void opcode_0x28() { read_byte(register_pc+1); read_byte(register_sp_fixed); register_flags=(pop()|0x30); Begin_Fetch_Next_Opcode(); return; } // 0x28 - PLP - Pop Flags from Stack void opcode_0x68() { read_byte(register_pc+1); read_byte(register_sp_fixed); register_a=pop(); Calc_Flags_NEGATIVE_ZERO(register_a); Begin_Fetch_Next_Opcode(); return; } // 0x68 - PLA - Pop Accumulator from Stack // ------------------------------------------------- // AND // ------------------------------------------------- void opcode_0x29() { register_a=register_a&(Fetch_Immediate()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x29 - AND - Immediate void opcode_0x25() { register_a=register_a&(Fetch_ZeroPage()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x25 - AND - ZeroPage void opcode_0x35() { register_a=register_a&(Fetch_ZeroPage_X()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x35 - AND - ZeroPage , X void opcode_0x2D() { register_a=register_a&(Fetch_Absolute()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x2D - AND - Absolute void opcode_0x3D() { register_a=register_a&(Fetch_Absolute_X(1)); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x3D - AND - Absolute , X void opcode_0x39() { register_a=register_a&(Fetch_Absolute_Y(1)); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x19 - OR - Absolute , Y void opcode_0x21() { register_a=register_a&(Fetch_Indexed_Indirect_X()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x21 - AND - Indexed Indirect X void opcode_0x31() { register_a=register_a&(Fetch_Indexed_Indirect_Y(1)); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x31 - AND - Indirect Indexed Y // ------------------------------------------------- // ORA // ------------------------------------------------- void opcode_0x09() { register_a=register_a|(Fetch_Immediate()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x09 - OR - Immediate void opcode_0x05() { register_a=register_a|(Fetch_ZeroPage()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x05 - OR - ZeroPage void opcode_0x15() { register_a=register_a|(Fetch_ZeroPage_X()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x15 - OR - ZeroPage , X void opcode_0x0D() { register_a=register_a|(Fetch_Absolute()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x0D - OR - Absolute void opcode_0x1D() { register_a=register_a|(Fetch_Absolute_X(1)); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x1D - OR - Absolute , X void opcode_0x19() { register_a=register_a|(Fetch_Absolute_Y(1)); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x19 - OR - Absolute , Y void opcode_0x01() { register_a=register_a|(Fetch_Indexed_Indirect_X()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x01 - OR - Indexed Indirect X void opcode_0x11() { register_a=register_a|(Fetch_Indexed_Indirect_Y(1)); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x11 - OR - Indirect Indexed Y // ------------------------------------------------- // EOR // ------------------------------------------------- void opcode_0x49() { register_a=register_a^(Fetch_Immediate()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x49 - EOR - Immediate void opcode_0x45() { register_a=register_a^(Fetch_ZeroPage()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x45 - EOR - ZeroPage void opcode_0x55() { register_a=register_a^(Fetch_ZeroPage_X()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x55 - EOR - ZeroPage , X void opcode_0x4D() { register_a=register_a^(Fetch_Absolute()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x4D - EOR - Absolute void opcode_0x5D() { register_a=register_a^(Fetch_Absolute_X(1)); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x5D - EOR - Absolute , X void opcode_0x59() { register_a=register_a^(Fetch_Absolute_Y(1)); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x59 - EOR - Absolute , Y void opcode_0x41() { register_a=register_a^(Fetch_Indexed_Indirect_X()); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x41 - EOR - Indexed Indirect X void opcode_0x51() { register_a=register_a^(Fetch_Indexed_Indirect_Y(1)); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0x51 - EOR - Indirect Indexed Y // ------------------------------------------------- // LDA // ------------------------------------------------- void opcode_0xA9() { register_a=Fetch_Immediate(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xA9 - LDA - Immediate void opcode_0xA5() { register_a=Fetch_ZeroPage(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xA5 - LDA - ZeroPage void opcode_0xB5() { register_a=Fetch_ZeroPage_X(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xB5 - LDA - ZeroPage , X void opcode_0xAD() { register_a=Fetch_Absolute(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xAD - LDA - Absolute void opcode_0xBD() { register_a=Fetch_Absolute_X(1); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xBD - LDA - Absolute , X void opcode_0xB9() { register_a=Fetch_Absolute_Y(1); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xB9 - LDA - Absolute , Y void opcode_0xA1() { register_a=Fetch_Indexed_Indirect_X(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xA1 - LDA - Indexed Indirect X void opcode_0xB1() { register_a=Fetch_Indexed_Indirect_Y(1); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xB1 - LDA - Indirect Indexed Y // ------------------------------------------------- // LDX // ------------------------------------------------- void opcode_0xA2() { register_x=Fetch_Immediate(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_x); return; } // 0xA2 - LDX - Immediate void opcode_0xA6() { register_x=Fetch_ZeroPage(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_x); return; } // 0xA6 - LDX - ZeroPage void opcode_0xB6() { register_x=Fetch_ZeroPage_Y(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_x); return; } // 0xB6 - LDX - ZeroPage , Y void opcode_0xAE() { register_x=Fetch_Absolute(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_x); return; } // 0xAE - LDX - Absolute void opcode_0xBE() { register_x=Fetch_Absolute_Y(1); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_x); return; } // 0xBE - LDX - Absolute , Y // ------------------------------------------------- // LDY // ------------------------------------------------- void opcode_0xA0() { register_y=Fetch_Immediate(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_y);return; } // 0xA0 - LDY - Immediate void opcode_0xA4() { register_y=Fetch_ZeroPage(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_y);return; } // 0xA4 - LDY - ZeroPage void opcode_0xB4() { register_y=Fetch_ZeroPage_X(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_y);return; } // 0xB4 - LDY - ZeroPage , X void opcode_0xAC() { register_y=Fetch_Absolute(); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_y);return; } // 0xAC - LDY - Absolute void opcode_0xBC() { register_y=Fetch_Absolute_X(1); Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_y);return; } // 0xBC - LDY - Absolute , X // ------------------------------------------------- // BIT // ------------------------------------------------- void Calculate_BIT(uint8_t local_data) { uint8_t temp=0; Begin_Fetch_Next_Opcode(); register_flags = (register_flags & 0x3F) | (local_data & 0xC0); // Copy fetched memory[7:6] to C,V flags temp = local_data & register_a; if (temp==0) register_flags = register_flags | 0x02; // Set the Z flag else register_flags = register_flags & 0xFD; // Clear the Z flag return; } void opcode_0x24() { Calculate_BIT(Fetch_ZeroPage()); return; } // 0x24 - BIT - ZeroPage void opcode_0x2C() { Calculate_BIT(Fetch_Absolute()); return; } // 0x2C - BIT - Absolute // ------------------------------------------------- // CMP // ------------------------------------------------- void Calculate_CMP(uint8_t local_data) { int16_t temp=0; Begin_Fetch_Next_Opcode(); temp = register_a - local_data; if (register_a >= local_data) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag Calc_Flags_NEGATIVE_ZERO(temp); return; } void opcode_0xC9() { Calculate_CMP(Fetch_Immediate()); return; } // 0xC9 - CMP - Immediate void opcode_0xC5() { Calculate_CMP(Fetch_ZeroPage()); return; } // 0xC5 - CMP - ZeroPage void opcode_0xD5() { Calculate_CMP(Fetch_ZeroPage_X()); return; } // 0xD5 - CMP - ZeroPage , X void opcode_0xCD() { Calculate_CMP(Fetch_Absolute()); return; } // 0xCD - CMP - Absolute void opcode_0xDD() { Calculate_CMP(Fetch_Absolute_X(1)); return; } // 0xDD - CMP - Absolute , X void opcode_0xD9() { Calculate_CMP(Fetch_Absolute_Y(1)); return; } // 0xD9 - CMP - Absolute , Y void opcode_0xC1() { Calculate_CMP(Fetch_Indexed_Indirect_X()); return; } // 0xC1 - CMP - Indexed Indirect X void opcode_0xD1() { Calculate_CMP(Fetch_Indexed_Indirect_Y(1)); return; } // 0xD1 - CMP - Indirect Indexed Y // ------------------------------------------------- // CPX // ------------------------------------------------- void Calculate_CPX(uint8_t local_data) { uint16_t temp=0; Begin_Fetch_Next_Opcode(); temp = register_x - local_data; if (register_x >= local_data) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag Calc_Flags_NEGATIVE_ZERO(temp); return; } void opcode_0xE0() { Calculate_CPX(Fetch_Immediate()); return; } // 0xE0 - CPX - Immediate void opcode_0xE4() { Calculate_CPX(Fetch_ZeroPage()); return; } // 0xE4 - CPX - ZeroPage void opcode_0xEC() { Calculate_CPX(Fetch_Absolute()); return; } // 0xEC - CPX - Absolute // ------------------------------------------------- // CPY // ------------------------------------------------- void Calculate_CPY(uint8_t local_data) { uint16_t temp=0; Begin_Fetch_Next_Opcode(); temp = register_y - local_data; if (register_y >= local_data) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag Calc_Flags_NEGATIVE_ZERO(temp); return; } void opcode_0xC0() { Calculate_CPY(Fetch_Immediate()); return; } // 0xC0 - CPY - Immediate void opcode_0xC4() { Calculate_CPY(Fetch_ZeroPage()); return; } // 0xC4 - CPY - ZeroPage void opcode_0xCC() { Calculate_CPY(Fetch_Absolute()); return; } // 0xCC - CPY - Absolute // ------------------------------------------------- // Store Operations // ------------------------------------------------- void opcode_0x85() { Write_ZeroPage(register_a); Begin_Fetch_Next_Opcode(); return; } // 0x85 - STA - ZeroPage void opcode_0x8D() { Write_Absolute(register_a); Begin_Fetch_Next_Opcode(); return; } // 0x8D - STA - Absolute void opcode_0x95() { Write_ZeroPage_X(register_a); Begin_Fetch_Next_Opcode(); return; } // 0x95 - STA - ZeroPage , X void opcode_0x9D() { Write_Absolute_X(register_a); Begin_Fetch_Next_Opcode(); return; } // 0x9D - STA - Absolute , X void opcode_0x99() { Write_Absolute_Y(register_a); Begin_Fetch_Next_Opcode(); return; } // 0x99 - STA - Absolute , Y void opcode_0x81() { Write_Indexed_Indirect_X(register_a); Begin_Fetch_Next_Opcode(); return; } // 0x81 - STA - Indexed Indirect X void opcode_0x91() { Write_Indexed_Indirect_Y(register_a); Begin_Fetch_Next_Opcode(); return; } // 0x91 - STA - Indirect Indexed Y void opcode_0x86() { Write_ZeroPage(register_x); Begin_Fetch_Next_Opcode(); return; } // 0x86 - STX - ZeroPage void opcode_0x96() { Write_ZeroPage_Y(register_x); Begin_Fetch_Next_Opcode(); return; } // 0x96 - STX - ZeroPage , Y void opcode_0x8E() { Write_Absolute(register_x); Begin_Fetch_Next_Opcode(); return; } // 0x8E - STX - Absolute void opcode_0x84() { Write_ZeroPage(register_y); Begin_Fetch_Next_Opcode(); return; } // 0x84 - STY - ZeroPage void opcode_0x94() { Write_ZeroPage_X(register_y); Begin_Fetch_Next_Opcode(); return; } // 0x94 - STY - ZeroPage , X void opcode_0x8C() { Write_Absolute(register_y); Begin_Fetch_Next_Opcode(); return; } // 0x8C - STY - Absolute // ------------------------------------------------- // ASL - Arithmetic Shift Left - Memory // ------------------------------------------------- uint8_t Calculate_ASL(uint8_t local_data) { if ((0x80&local_data)==0x80) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag local_data = ((local_data << 1) & 0xFE); Calc_Flags_NEGATIVE_ZERO(local_data); return local_data; } // ------------------------------------------------- // ASL - Read-modify-write Operations // ------------------------------------------------- void opcode_0x06() { Double_WriteBack(Calculate_ASL(Fetch_ZeroPage())); Begin_Fetch_Next_Opcode(); return; } // 0x06 - ASL - Arithmetic Shift Left - ZeroPage void opcode_0x16() { Double_WriteBack(Calculate_ASL(Fetch_ZeroPage_X())); Begin_Fetch_Next_Opcode(); return; } // 0x16 - ASL - Arithmetic Shift Left - ZeroPage , X void opcode_0x0E() { Double_WriteBack(Calculate_ASL(Fetch_Absolute())); Begin_Fetch_Next_Opcode(); return; } // 0x0E - ASL - Arithmetic Shift Left - Absolute void opcode_0x1E() { Double_WriteBack(Calculate_ASL(Fetch_Absolute_X(0))); Begin_Fetch_Next_Opcode(); return; } // 0x1E - ASL - Arithmetic Shift Left - Absolute , X // ------------------------------------------------- // INC - Memory // ------------------------------------------------- uint8_t Calculate_INC(uint8_t local_data) { local_data = local_data + 1; global_temp = local_data; Calc_Flags_NEGATIVE_ZERO(local_data); return local_data; } void opcode_0xE6() { Double_WriteBack(Calculate_INC(Fetch_ZeroPage())); Begin_Fetch_Next_Opcode(); return; } // 0xE6 - INC - ZeroPage void opcode_0xF6() { Double_WriteBack(Calculate_INC(Fetch_ZeroPage_X())); Begin_Fetch_Next_Opcode(); return; } // 0xF6 - INC - ZeroPage , X void opcode_0xEE() { Double_WriteBack(Calculate_INC(Fetch_Absolute())); Begin_Fetch_Next_Opcode(); return; } // 0xEE - INC - Absolute void opcode_0xFE() { Double_WriteBack(Calculate_INC(Fetch_Absolute_X(0))); Begin_Fetch_Next_Opcode(); return; } // 0xFE - INC - Absolute , X // ------------------------------------------------- // DEC - Memory // ------------------------------------------------- uint8_t Calculate_DEC(uint8_t local_data) { local_data = local_data - 1; global_temp = local_data; Calc_Flags_NEGATIVE_ZERO(local_data); return local_data; } void opcode_0xC6() { Double_WriteBack(Calculate_DEC(Fetch_ZeroPage())); Begin_Fetch_Next_Opcode(); return; } // 0xC6 - DEC - ZeroPage void opcode_0xD6() { Double_WriteBack(Calculate_DEC(Fetch_ZeroPage_X())); Begin_Fetch_Next_Opcode(); return; } // 0xD6 - DEC - ZeroPage , X void opcode_0xCE() { Double_WriteBack(Calculate_DEC(Fetch_Absolute())); Begin_Fetch_Next_Opcode(); return; } // 0xCE - DEC - Absolute void opcode_0xDE() { Double_WriteBack(Calculate_DEC(Fetch_Absolute_X(0))); Begin_Fetch_Next_Opcode(); return; } // 0xDE - DEC - Absolute , X // ------------------------------------------------- // LSR - Memory // ------------------------------------------------- uint8_t Calculate_LSR(uint8_t local_data) { if ((0x01&local_data)==0x01) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag local_data = (0x7F& (local_data >> 1)); Calc_Flags_NEGATIVE_ZERO(local_data); return local_data; } void opcode_0x46() { Double_WriteBack(Calculate_LSR(Fetch_ZeroPage())); Begin_Fetch_Next_Opcode(); return; } // 0x46 - LSR - Logical Shift Right - ZeroPage void opcode_0x56() { Double_WriteBack(Calculate_LSR(Fetch_ZeroPage_X())); Begin_Fetch_Next_Opcode(); return; } // 0x56 - LSR - Logical Shift Right - ZeroPage , X void opcode_0x4E() { Double_WriteBack(Calculate_LSR(Fetch_Absolute())); Begin_Fetch_Next_Opcode(); return; } // 0x4E - LSR - Logical Shift Right - Absolute void opcode_0x5E() { Double_WriteBack(Calculate_LSR(Fetch_Absolute_X(0))); Begin_Fetch_Next_Opcode(); return; } // 0x5E - LSR - Logical Shift Right - Absolute , X // ------------------------------------------------- // ROR - Memory // ------------------------------------------------- uint8_t Calculate_ROR(uint8_t local_data) { uint8_t old_carry_flag=0; old_carry_flag = register_flags << 7; // Shift the old carry flag to bit[8] to be rotated in if ((0x01&local_data)==0x01) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag local_data = ( old_carry_flag | (local_data>>1) ); Calc_Flags_NEGATIVE_ZERO(local_data); return local_data; } void opcode_0x66() { Double_WriteBack(Calculate_ROR(Fetch_ZeroPage())); Begin_Fetch_Next_Opcode(); return; } // 0x66 - ROR - Rotate Right - ZeroPage void opcode_0x76() { Double_WriteBack(Calculate_ROR(Fetch_ZeroPage_X())); Begin_Fetch_Next_Opcode(); return; } // 0x76 - ROR - Rotate Right - ZeroPage , X void opcode_0x6E() { Double_WriteBack(Calculate_ROR(Fetch_Absolute())); Begin_Fetch_Next_Opcode(); return; } // 0x6E - ROR - Rotate Right - Absolute void opcode_0x7E() { Double_WriteBack(Calculate_ROR(Fetch_Absolute_X(0))); Begin_Fetch_Next_Opcode(); return; } // 0x7E - ROR - Rotate Right - Absolute , X // ------------------------------------------------- // ROL - Memory // ------------------------------------------------- uint8_t Calculate_ROL(uint8_t local_data) { uint8_t old_carry_flag=0; old_carry_flag = 0x1 & register_flags; // Store the old carry flag to be rotated in if (0x80&local_data) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag local_data = (local_data<<1) | old_carry_flag; Calc_Flags_NEGATIVE_ZERO(local_data); return local_data; } void opcode_0x26() { Double_WriteBack(Calculate_ROL(Fetch_ZeroPage())); Begin_Fetch_Next_Opcode(); return; } // 0x26 - ROL - Rotate Left - ZeroPage void opcode_0x36() { Double_WriteBack(Calculate_ROL(Fetch_ZeroPage_X())); Begin_Fetch_Next_Opcode(); return; } // 0x36 - ROL - Rotate Left - ZeroPage , X void opcode_0x2E() { Double_WriteBack(Calculate_ROL(Fetch_Absolute())); Begin_Fetch_Next_Opcode(); return; } // 0x2E - ROL - Rotate Left - Absolute void opcode_0x3E() { Double_WriteBack(Calculate_ROL(Fetch_Absolute_X(0))); Begin_Fetch_Next_Opcode(); return; } // 0x3E - ROL - Rotate Left - Absolute , X // ------------------------------------------------- // Branches // ------------------------------------------------- void Branch_Taken() { effective_address = Sign_Extend16(Fetch_Immediate()); effective_address = (register_pc+1) + effective_address; if ( (0xFF00®ister_pc) == (0xFF00&effective_address) ) { Fetch_Immediate(); } // Page boundary not crossed else { Fetch_Immediate(); Fetch_Immediate(); } // Page boundary crossed register_pc = effective_address; assert_sync=1; start_read(register_pc); return; } void opcode_0xB0() { if ((flag_c)==1) Branch_Taken(); else { Fetch_Immediate(); Begin_Fetch_Next_Opcode();} return; } // 0xB0 - BCS - Branch on Carry Set void opcode_0x90() { if ((flag_c)==0) Branch_Taken(); else { Fetch_Immediate(); Begin_Fetch_Next_Opcode();} return; } // 0x90 - BCC - Branch on Carry Clear void opcode_0xF0() { if ((flag_z)==1) Branch_Taken(); else { Fetch_Immediate(); Begin_Fetch_Next_Opcode();} return; } // 0xF0 - BEQ - Branch on Zero Set void opcode_0xD0() { if ((flag_z)==0) Branch_Taken(); else { Fetch_Immediate(); Begin_Fetch_Next_Opcode();} return; } // 0xD0 - BNE - Branch on Zero Clear void opcode_0x70() { if ((flag_v)==1) Branch_Taken(); else { Fetch_Immediate(); Begin_Fetch_Next_Opcode();} return; } // 0x70 - BVS - Branch on Overflow Set void opcode_0x50() { if ((flag_v)==0) Branch_Taken(); else { Fetch_Immediate(); Begin_Fetch_Next_Opcode();} return; } // 0x50 - BVC - Branch on Overflow Clear void opcode_0x30() { if ((flag_n)==1) Branch_Taken(); else { Fetch_Immediate(); Begin_Fetch_Next_Opcode();} return; } // 0x30 - BMI - Branch on Minus (N Flag Set) void opcode_0x10() { if ((flag_n)==0) Branch_Taken(); else { Fetch_Immediate(); Begin_Fetch_Next_Opcode();} return; } // 0x10 - BPL - Branch on Plus (N Flag Clear) // ------------------------------------------------- // Jumps and Returns // ------------------------------------------------- void opcode_0x4C() { register_pc=Calculate_Absolute(); assert_sync=1; start_read(register_pc); return; } // 0x4C - JMP - Jump Absolute // ------------------------------------------------- // 0x6C - JMP - Jump Indirect // ------------------------------------------------- void opcode_0x6C() { uint16_t lal, lah; uint16_t adl, adh; lal = Fetch_Immediate(); lah = Fetch_Immediate()<<8; adl = read_byte(lah + lal); adh = read_byte(lah + lal + 1)<<8; effective_address = adh+adl; register_pc = (0xFF00&adh) + (0x00FF&effective_address) ; // 6502 page wrapping bug assert_sync=1; start_read(register_pc); return ; } // ------------------------------------------------- // 0x20 - JSR - Jump to Subroutine // ------------------------------------------------- void opcode_0x20() { uint16_t adl, adh; adl = Fetch_Immediate(); adh = Fetch_Immediate()<<8; read_byte(register_sp_fixed); push((0xFF00®ister_pc)>>8); push(0x00FF®ister_pc); register_pc = adh+adl; assert_sync=1; start_read(register_pc); return ; } // ------------------------------------------------- // 0x40 - RTI - Return from Interrupt // ------------------------------------------------- void opcode_0x40() { uint16_t pcl, pch; Fetch_Immediate(); read_byte(register_sp_fixed); register_flags = pop(); pcl = pop(); pch = pop()<<8; register_pc = pch+pcl; assert_sync=1; start_read(register_pc); return ; } // ------------------------------------------------- // 0x60 - RTS - Return from Subroutine // ------------------------------------------------- void opcode_0x60() { uint16_t pcl, pch; Fetch_Immediate(); read_byte(register_sp_fixed); pcl = pop(); pch = pop()<<8; register_pc = pch+pcl+1; read_byte(register_pc); assert_sync=1; start_read(register_pc); return ; } // ------------------------------------------------- // // *** Undocumented 6502 Opcodes *** // // ------------------------------------------------- // -------------------------------------------------------------------------------------------------- // SLO - Shift left one bit in memory, then OR accumulator with memory. // -------------------------------------------------------------------------------------------------- uint8_t Calculate_SLO(uint8_t local_data) { if ((0x80&local_data)==0x80) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag local_data = ((local_data << 1) & 0xFE); register_a = register_a | local_data; Calc_Flags_NEGATIVE_ZERO(register_a); return local_data; } void opcode_0x07() { Double_WriteBack(Calculate_SLO(Fetch_ZeroPage())); Begin_Fetch_Next_Opcode(); return; } // 0x07 - SLO - ZeroPage void opcode_0x17() { Double_WriteBack(Calculate_SLO(Fetch_ZeroPage_X())); Begin_Fetch_Next_Opcode(); return; } // 0x17 - SLO - ZeroPage , X void opcode_0x03() { Double_WriteBack(Calculate_SLO(Fetch_Indexed_Indirect_X())); Begin_Fetch_Next_Opcode(); return; } // 0x03 - SLO - Indexed Indirect X void opcode_0x13() { Double_WriteBack(Calculate_SLO(Fetch_Indexed_Indirect_Y(1))); Begin_Fetch_Next_Opcode(); return; } // 0x13 - SLO - Indirect Indexed Y void opcode_0x0F() { Double_WriteBack(Calculate_SLO(Fetch_Absolute())); Begin_Fetch_Next_Opcode(); return; } // 0x0F - SLO - Absolute void opcode_0x1F() { Double_WriteBack(Calculate_SLO(Fetch_Absolute_X(1))); Begin_Fetch_Next_Opcode(); return; } // 0x1F - SLO - Absolute , X void opcode_0x1B() { Double_WriteBack(Calculate_SLO(Fetch_Absolute_Y(1))); Begin_Fetch_Next_Opcode(); return; } // 0x1B - SLO - Absolute , Y // -------------------------------------------------------------------------------------------------- // RLA - Rotate one bit left in memory, then AND accumulator with memory. // -------------------------------------------------------------------------------------------------- uint8_t Calculate_RLA(uint8_t local_data) { uint8_t old_carry_flag=0; old_carry_flag = 0x1 & register_flags; // Store the old carry flag to be rotated in if (0x80&local_data) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag local_data = (local_data<<1) | old_carry_flag; register_a = register_a & local_data; Calc_Flags_NEGATIVE_ZERO(register_a); return local_data; } void opcode_0x27() { Double_WriteBack(Calculate_RLA(Fetch_ZeroPage())); Begin_Fetch_Next_Opcode(); return; } // 0x27 - RLA - ZeroPage void opcode_0x37() { Double_WriteBack(Calculate_RLA(Fetch_ZeroPage_X())); Begin_Fetch_Next_Opcode(); return; } // 0x37 - RLA - ZeroPage , X void opcode_0x23() { Double_WriteBack(Calculate_RLA(Fetch_Indexed_Indirect_X())); Begin_Fetch_Next_Opcode(); return; } // 0x23 - RLA - Indexed Indirect X void opcode_0x33() { Double_WriteBack(Calculate_RLA(Fetch_Indexed_Indirect_Y(1))); Begin_Fetch_Next_Opcode(); return; } // 0x33 - RLA - Indirect Indexed Y void opcode_0x2F() { Double_WriteBack(Calculate_RLA(Fetch_Absolute())); Begin_Fetch_Next_Opcode(); return; } // 0x2F - RLA - Absolute void opcode_0x3F() { Double_WriteBack(Calculate_RLA(Fetch_Absolute_X(1))); Begin_Fetch_Next_Opcode(); return; } // 0x3F - RLA - Absolute , X void opcode_0x3B() { Double_WriteBack(Calculate_RLA(Fetch_Absolute_Y(1))); Begin_Fetch_Next_Opcode(); return; } // 0x3B - RLA - Absolute , Y // -------------------------------------------------------------------------------------------------- // SRE - Shift right one bit in memory, then EOR accumulator with memory. // -------------------------------------------------------------------------------------------------- uint8_t Calculate_SRE(uint8_t local_data) { if ((0x01&local_data)==0x01) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag local_data = (0x7F& (local_data >> 1)); register_a = register_a ^ local_data; Calc_Flags_NEGATIVE_ZERO(register_a); return local_data; } void opcode_0x47() { Double_WriteBack(Calculate_SRE(Fetch_ZeroPage())); Begin_Fetch_Next_Opcode(); return; } // 0x47 - SRE - ZeroPage void opcode_0x57() { Double_WriteBack(Calculate_SRE(Fetch_ZeroPage_X())); Begin_Fetch_Next_Opcode(); return; } // 0x57 - SRE - ZeroPage , X void opcode_0x43() { Double_WriteBack(Calculate_SRE(Fetch_Indexed_Indirect_X())); Begin_Fetch_Next_Opcode(); return; } // 0x43 - SRE - Indexed Indirect X void opcode_0x53() { Double_WriteBack(Calculate_SRE(Fetch_Indexed_Indirect_Y(1))); Begin_Fetch_Next_Opcode(); return; } // 0x53 - SRE - Indirect Indexed Y void opcode_0x4F() { Double_WriteBack(Calculate_SRE(Fetch_Absolute())); Begin_Fetch_Next_Opcode(); return; } // 0x4F - SRE - Absolute void opcode_0x5F() { Double_WriteBack(Calculate_SRE(Fetch_Absolute_X(1))); Begin_Fetch_Next_Opcode(); return; } // 0x5F - SRE - Absolute , X void opcode_0x5B() { Double_WriteBack(Calculate_SRE(Fetch_Absolute_Y(1))); Begin_Fetch_Next_Opcode(); return; } // 0x5B - SRE - Absolute , Y // -------------------------------------------------------------------------------------------------- // RRA - Rotate one bit right in memory, then add memory to accumulator (with carry). // -------------------------------------------------------------------------------------------------- uint8_t Calculate_RRA(uint8_t local_data) { if ((0x01&local_data)==0x01) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag local_data = (0x7F& (local_data >> 1)); global_temp = local_data; return local_data; } void opcode_0x67() { Double_WriteBack(Calculate_RRA(Fetch_ZeroPage())); Calculate_ADC(global_temp); return; } // 0x67 - RRA - ZeroPage void opcode_0x77() { Double_WriteBack(Calculate_RRA(Fetch_ZeroPage_X())); Calculate_ADC(global_temp); return; } // 0x77 - RRA - ZeroPage , X void opcode_0x63() { Double_WriteBack(Calculate_RRA(Fetch_Indexed_Indirect_X())); Calculate_ADC(global_temp); return; } // 0x63 - RRA - Indexed Indirect X void opcode_0x73() { Double_WriteBack(Calculate_RRA(Fetch_Indexed_Indirect_Y(1))); Calculate_ADC(global_temp); return; } // 0x73 - RRA - Indirect Indexed Y void opcode_0x6F() { Double_WriteBack(Calculate_RRA(Fetch_Absolute())); Calculate_ADC(global_temp); return; } // 0x6F - RRA - Absolute void opcode_0x7F() { Double_WriteBack(Calculate_RRA(Fetch_Absolute_X(1))); Calculate_ADC(global_temp); return; } // 0x7F - RRA - Absolute , X void opcode_0x7B() { Double_WriteBack(Calculate_RRA(Fetch_Absolute_Y(1))); Calculate_ADC(global_temp); return; } // 0x7B - RRA - Absolute , Y // -------------------------------------------------------------------------------------------------- // AND the contents of the A and X registers (without changing the contents of either register) and // stores the result in memory. // -------------------------------------------------------------------------------------------------- void opcode_0x87() { Write_ZeroPage(register_a®ister_x); Begin_Fetch_Next_Opcode(); return; } // 0x87 - SAX - ZeroPage void opcode_0x97() { Write_ZeroPage_Y(register_a®ister_x); Begin_Fetch_Next_Opcode(); return; } // 0x97 - SAX - ZeroPage , Y void opcode_0x83() { Write_Indexed_Indirect_X(register_a®ister_x); Begin_Fetch_Next_Opcode(); return; } // 0x83 - SAX - Indexed Indirect X void opcode_0x8F() { Write_Absolute(register_a®ister_x); Begin_Fetch_Next_Opcode(); return; } // 0x8F - SAX - Absolute // -------------------------------------------------------------------------------------------------- // Load both the accumulator and the X register with the contents of a memory location. // -------------------------------------------------------------------------------------------------- void opcode_0xA7() { register_a=Fetch_ZeroPage(); register_x=register_a; Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xA7 - LAX - ZeroPage void opcode_0xB7() { register_a=Fetch_ZeroPage_Y(); register_x=register_a; Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xB7 - LAX - ZeroPage , Y void opcode_0xA3() { register_a=Fetch_Indexed_Indirect_X(); register_x=register_a; Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xA3 - LAX - Indexed Indirect X void opcode_0xB3() { register_a=Fetch_Indexed_Indirect_Y(1); register_x=register_a; Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xB3 - LAX - Indirect Indexed Y void opcode_0xAF() { register_a=Fetch_Absolute(); register_x=register_a; Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xAF - LAX - Absolute void opcode_0xBF() { register_a=Fetch_Absolute_Y(1); register_x=register_a; Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xBF - LAX - Absolute , Y // -------------------------------------------------------------------------------------------------- // Decrement the contents of a memory location and then compare the result with the A register. // -------------------------------------------------------------------------------------------------- void opcode_0xC7() { Double_WriteBack(Calculate_DEC(Fetch_ZeroPage())); Calculate_CMP(global_temp); return; } // 0xC7 - DCP - ZeroPage void opcode_0xD7() { Double_WriteBack(Calculate_DEC(Fetch_ZeroPage_X())); Calculate_CMP(global_temp); return; } // 0xD7 - DCP - ZeroPage , X void opcode_0xC3() { Double_WriteBack(Calculate_DEC(Fetch_Indexed_Indirect_X())); Calculate_CMP(global_temp); return; } // 0xC3 - DCP - Indexed Indirect X void opcode_0xD3() { Double_WriteBack(Calculate_DEC(Fetch_Indexed_Indirect_Y(0))); Calculate_CMP(global_temp); return; } // 0xD3 - DCP - Indirect Indexed Y void opcode_0xCF() { Double_WriteBack(Calculate_DEC(Fetch_Absolute())); Calculate_CMP(global_temp); return; } // 0xCF - DCP - Absolute void opcode_0xDF() { Double_WriteBack(Calculate_DEC(Fetch_Absolute_X(0))); Calculate_CMP(global_temp); return; } // 0xDF - DCP - Absolute , X void opcode_0xDB() { Double_WriteBack(Calculate_DEC(Fetch_Absolute_Y(0))); Calculate_CMP(global_temp); return; } // 0xDB - DCP - Absolute , Y // -------------------------------------------------------------------------------------------------- // ISC - Increase memory by one, then subtract memory from accumulator (with borrow). // -------------------------------------------------------------------------------------------------- void opcode_0xE7() { Double_WriteBack(Calculate_INC(Fetch_ZeroPage())); Calculate_SBC(global_temp); return; } // 0xE7 - ISC - ZeroPage void opcode_0xF7() { Double_WriteBack(Calculate_INC(Fetch_ZeroPage_X())); Calculate_SBC(global_temp); return; } // 0xF7 - ISC - ZeroPage , X void opcode_0xE3() { Double_WriteBack(Calculate_INC(Fetch_Indexed_Indirect_X())); Calculate_SBC(global_temp); return; } // 0xE3 - ISC - Indexed Indirect X void opcode_0xF3() { Double_WriteBack(Calculate_INC(Fetch_Indexed_Indirect_Y(0))); Calculate_SBC(global_temp); return; } // 0xF3 - ISC - Indirect Indexed Y void opcode_0xEF() { Double_WriteBack(Calculate_INC(Fetch_Absolute())); Calculate_SBC(global_temp); return; } // 0xEF - ISC - Absolute void opcode_0xFF() { Double_WriteBack(Calculate_INC(Fetch_Absolute_X(0))); Calculate_SBC(global_temp); return; } // 0xFF - ISC - Absolute , X void opcode_0xFB() { Double_WriteBack(Calculate_INC(Fetch_Absolute_Y(0))); Calculate_SBC(global_temp); return; } // 0xFB - ISC - Absolute , Y // -------------------------------------------------------------------------------------------------- // ANC - ANDs the contents of the A register with an immediate value and then moves bit 7 of A // into the Carry flag. // -------------------------------------------------------------------------------------------------- void Calculate_ANC(uint8_t local_data) { Begin_Fetch_Next_Opcode(); register_a = register_a & local_data; if ((0x80®ister_a)==0x80) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag Calc_Flags_NEGATIVE_ZERO(register_a); return; } void opcode_0x0B() { Calculate_ANC(Fetch_Immediate()); return; } // 0x0B - ANC - Immediate void opcode_0x2B() { Calculate_ANC(Fetch_Immediate()); return; } // 0x2B - ANC - Immediate // -------------------------------------------------------------------------------------------------- // ALR - AND the contents of the A register with an immediate value and then LSRs the result. // -------------------------------------------------------------------------------------------------- void Calculate_ALR(uint8_t local_data) { Begin_Fetch_Next_Opcode(); register_a = register_a & local_data; if ((0x01®ister_a)==0x01) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag register_a = (0x7F& (register_a >> 1)); Calc_Flags_NEGATIVE_ZERO(register_a); return; } void opcode_0x4B() { Calculate_ALR(Fetch_Immediate()); return; } // 0x4B - ALR - Immediate // -------------------------------------------------------------------------------------------------- // ARR - ANDs the accumulator with an immediate value and then rotates the content right. // -------------------------------------------------------------------------------------------------- void Calculate_ARR(uint8_t local_data) { Begin_Fetch_Next_Opcode(); register_a = register_a & local_data; register_a = (0x7F& (register_a >> 1)); register_flags = register_flags & 0xBE; // Pre-clear the C and V flags if ( (0xC0 & register_a) == 0x40) { register_flags = register_flags | 0x40; } // Set the V flag if ( (0xC0 & register_a) == 0x80) { register_flags = register_flags | 0x41; } // Set the C and V flags if ( (0xC0 & register_a) == 0xC0) { register_flags = register_flags | 0x01; } // Set the C flag Calc_Flags_NEGATIVE_ZERO(register_a); return; } void opcode_0x6B() { Calculate_ARR(Fetch_Immediate()); return; } // 0x6B - ARR - Immediate // -------------------------------------------------------------------------------------------------- // SBX - ANDs the contents of the A and X registers (leaving the contents of A intact), // subtracts an immediate value, and then stores the result in X. // -------------------------------------------------------------------------------------------------- void Calculate_SBX(uint16_t local_data) { int16_t signed_total=0; Begin_Fetch_Next_Opcode(); register_x = register_a & register_x; register_x = register_x - local_data; signed_total = (int16_t)register_x - (int16_t)(local_data ); if (signed_total>=0) register_flags = register_flags | 0x01; // Set the C flag else register_flags = register_flags & 0xFE; // Clear the C flag register_x = (0xFF & register_x); Calc_Flags_NEGATIVE_ZERO(register_x); return; } void opcode_0xCB() { Calculate_SBX(Fetch_Immediate()); return; } // 0xCB - SBX - Immediate // -------------------------------------------------------------------------------------------------- // LAS - AND memory with stack pointer, transfer result to accumulator, X register and stack pointer. // -------------------------------------------------------------------------------------------------- void opcode_0xBB() { register_sp=(register_sp&Fetch_Absolute_Y(1)); register_a=register_sp; register_x=register_sp; Begin_Fetch_Next_Opcode(); Calc_Flags_NEGATIVE_ZERO(register_a); return; } // 0xBB - LAS - Absolute , Y // -------------------------------------------------------------------------------------------------- // NOP - Fetch Immediate // -------------------------------------------------------------------------------------------------- void opcode_0x80() { Fetch_Immediate(); Begin_Fetch_Next_Opcode(); return; } // 0x80 - NOP - Immediate void opcode_0x82() { Fetch_Immediate(); Begin_Fetch_Next_Opcode(); return; } // 0x82 - NOP - Immediate void opcode_0xC2() { Fetch_Immediate(); Begin_Fetch_Next_Opcode(); return; } // 0xC2 - NOP - Immediate void opcode_0xE2() { Fetch_Immediate(); Begin_Fetch_Next_Opcode(); return; } // 0xE2 - NOP - Immediate void opcode_0x89() { Fetch_Immediate(); Begin_Fetch_Next_Opcode(); return; } // 0x89 - NOP - Immediate // -------------------------------------------------------------------------------------------------- // NOP - Fetch ZeroPage // -------------------------------------------------------------------------------------------------- void opcode_0x04() { Fetch_ZeroPage(); Begin_Fetch_Next_Opcode(); return; } // 0x04 - NOP - ZeroPage void opcode_0x44() { Fetch_ZeroPage(); Begin_Fetch_Next_Opcode(); return; } // 0x44 - NOP - ZeroPage void opcode_0x64() { Fetch_ZeroPage(); Begin_Fetch_Next_Opcode(); return; } // 0x64 - NOP - ZeroPage // -------------------------------------------------------------------------------------------------- // NOP - Fetch ZeroPage , X // -------------------------------------------------------------------------------------------------- void opcode_0x14() { Fetch_ZeroPage_X(); Begin_Fetch_Next_Opcode(); return; } // 0x14 - NOP - ZeroPage , X void opcode_0x34() { Fetch_ZeroPage_X(); Begin_Fetch_Next_Opcode(); return; } // 0x34 - NOP - ZeroPage , X void opcode_0x54() { Fetch_ZeroPage_X(); Begin_Fetch_Next_Opcode(); return; } // 0x54 - NOP - ZeroPage , X void opcode_0x74() { Fetch_ZeroPage_X(); Begin_Fetch_Next_Opcode(); return; } // 0x74 - NOP - ZeroPage , X void opcode_0xD4() { Fetch_ZeroPage_X(); Begin_Fetch_Next_Opcode(); return; } // 0xD4 - NOP - ZeroPage , X void opcode_0xF4() { Fetch_ZeroPage_X(); Begin_Fetch_Next_Opcode(); return; } // 0xF4 - NOP - ZeroPage , X // -------------------------------------------------------------------------------------------------- // NOP - Fetch Absolute // -------------------------------------------------------------------------------------------------- void opcode_0x0C() { Fetch_Absolute(); Begin_Fetch_Next_Opcode(); return; } // 0x0C - NOP - Absolute // -------------------------------------------------------------------------------------------------- // NOP - Fetch Absolute , X // -------------------------------------------------------------------------------------------------- void opcode_0x1C() { Fetch_Absolute_X(1); Begin_Fetch_Next_Opcode(); return; } // 0x1C - NOP - Absolute , X void opcode_0x3C() { Fetch_Absolute_X(1); Begin_Fetch_Next_Opcode(); return; } // 0x3C - NOP - Absolute , X void opcode_0x5C() { Fetch_Absolute_X(1); Begin_Fetch_Next_Opcode(); return; } // 0x5C - NOP - Absolute , X void opcode_0x7C() { Fetch_Absolute_X(1); Begin_Fetch_Next_Opcode(); return; } // 0x7C - NOP - Absolute , X void opcode_0xDC() { Fetch_Absolute_X(1); Begin_Fetch_Next_Opcode(); return; } // 0xDC - NOP - Absolute , X void opcode_0xFC() { Fetch_Absolute_X(1); Begin_Fetch_Next_Opcode(); return; } // 0xFC - NOP - Absolute , X // -------------------------------------------------------------------------------------------------- // JAM - Lock up the processor // -------------------------------------------------------------------------------------------------- void opcode_0x02() { Fetch_Immediate(); while (1) {} return; } // 0x02 - JAM void opcode_0x12() { Fetch_Immediate(); while (1) {} return; } // 0x12 - JAM void opcode_0x22() { Fetch_Immediate(); while (1) {} return; } // 0x22 - JAM void opcode_0x32() { Fetch_Immediate(); while (1) {} return; } // 0x32 - JAM void opcode_0x42() { Fetch_Immediate(); while (1) {} return; } // 0x42 - JAM void opcode_0x52() { Fetch_Immediate(); while (1) {} return; } // 0x52 - JAM void opcode_0x62() { Fetch_Immediate(); while (1) {} return; } // 0x62 - JAM void opcode_0x72() { Fetch_Immediate(); while (1) {} return; } // 0x72 - JAM void opcode_0x92() { Fetch_Immediate(); while (1) {} return; } // 0x92 - JAM void opcode_0xB2() { Fetch_Immediate(); while (1) {} return; } // 0xB2 - JAM void opcode_0xD2() { Fetch_Immediate(); while (1) {} return; } // 0xD2 - JAM void opcode_0xF2() { Fetch_Immediate(); while (1) {} return; } // 0xF2 - JAM // -------------------------------------------------------------------------------------------------- // Unstable 6502 opcodes // -------------------------------------------------------------------------------------------------- void opcode_0x93() { Fetch_ZeroPage_Y(); Begin_Fetch_Next_Opcode(); return; } // 0x93 - SHA - ZeroPage , Y - Implelented here as a size 2 NOP void opcode_0x9F() { Fetch_Absolute_Y(0); Begin_Fetch_Next_Opcode(); return; } // 0x9F - SHA - Absolute , Y - Implelented here as a size 3 NOP void opcode_0x9E() { Fetch_Absolute_Y(0); Begin_Fetch_Next_Opcode(); return; } // 0x9E - SHX - Absolute , Y - Implelented here as a size 3 NOP void opcode_0x9C() { Fetch_Absolute_X(0); Begin_Fetch_Next_Opcode(); return; } // 0x9C - SHY - Absolute , X - Implelented here as a size 3 NOP void opcode_0x9B() { Fetch_Absolute_Y(0); Begin_Fetch_Next_Opcode(); return; } // 0x9B - TAS - Absolute , Y - Implelented here as a size 3 NOP void opcode_0x8B() { Fetch_Immediate(); Begin_Fetch_Next_Opcode(); return; } // 0x8B - ANE - Immediate - Implelented here as a size 2 NOP void opcode_0xAB() { Fetch_Immediate(); Begin_Fetch_Next_Opcode(); return; } // 0xAB - LAX - Immediate - Implelented here as a size 2 NOP // -------------------------------------------------------------------------------------------------- // -------------------------------------------------------------------------------------------------- // ------------------------------------------------- // // Main loop // // ------------------------------------------------- void loop() { setup(); // Give Teensy 4.1 a moment delay (50); wait_for_CLK_rising_edge(); wait_for_CLK_rising_edge(); wait_for_CLK_rising_edge(); reset_sequence(); while (1) { if (direct_reset==1) reset_sequence(); // Poll for NMI and IRQ // nmi_n_old = direct_nmi; if (nmi_n_old==0 && direct_nmi==1) nmi_handler(); if (direct_irq==0x1 && (flag_i)==0x0) irq_handler(0x0); next_instruction = finish_read_byte(); assert_sync=0; switch (next_instruction){ case 0x00: irq_handler(0x1); break; // BRK - Break case 0x01: opcode_0x01(); break; // OR - Indexed Indirect X case 0x02: opcode_0x02(); break; // JAM case 0x03: opcode_0x03(); break; // SLO - Indexed Indirect X case 0x04: opcode_0x04(); break; // NOP - ZeroPage case 0x05: opcode_0x05(); break; // OR ZeroPage case 0x06: opcode_0x06(); break; // ASL A - Arithmetic Shift Left - ZeroPage case 0x07: opcode_0x07(); break; // SLO - ZeroPage case 0x08: opcode_0x08(); break; // PHP - Push processor status to the stack case 0x09: opcode_0x09(); break; // OR - Immediate case 0x0A: opcode_0x0A(); break; // ASL A case 0x0B: opcode_0x0B(); break; // ANC - Immediate case 0x0C: opcode_0x0C(); break; // NOP - Absolute case 0x0D: opcode_0x0D(); break; // OR - Absolute case 0x0E: opcode_0x0E(); break; // ASL A - Arithmetic Shift Left - Absolute case 0x0F: opcode_0x0F(); break; // SLO - Absolute case 0x10: opcode_0x10(); break; // BNE - Branch on Zero Clear case 0x11: opcode_0x11(); break; // OR Indirect Indexed Y case 0x12: opcode_0x12(); break; // JAM case 0x13: opcode_0x13(); break; // Indirect Indexed Y case 0x14: opcode_0x14(); break; // NOP - ZeroPage , X case 0x15: opcode_0x15(); break; // OR - ZeroPage,X case 0x16: opcode_0x16(); break; // ASL A - Arithmetic Shift Left - ZeroPage , X case 0x17: opcode_0x17(); break; // SLO - ZeroPage , X case 0x18: opcode_0x18(); break; // CLC case 0x19: opcode_0x19(); break; // OR - Absolute,Y case 0x1A: opcode_0xEA(); break; // NOP case 0x1B: opcode_0x1B(); break; // SLO - Absolute , Y case 0x1C: opcode_0x1C(); break; // NOP - Absolute , X case 0x1D: opcode_0x1D(); break; // OR - Absolute,X case 0x1E: opcode_0x1E(); break; // ASL A - Arithmetic Shift Left - Absolute , X case 0x1F: opcode_0x1F(); break; // SLO - Absolute , X case 0x20: opcode_0x20(); break; // JSR - Jump to Subroutine case 0x21: opcode_0x21(); break; // AND - Indexed Indirect case 0x22: opcode_0x22(); break; // JAM case 0x23: opcode_0x23(); break; // RLA - Indexed Indirect X case 0x24: opcode_0x24(); break; // BIT - ZeroPage case 0x25: opcode_0x25(); break; // AND - ZeroPage case 0x26: opcode_0x26(); break; // ROL - Rotate Left - ZeroPage case 0x27: opcode_0x27(); break; // RLA - ZeroPage case 0x28: opcode_0x28(); break; // PLP - Pop processor status from the stack case 0x29: opcode_0x29(); break; // AND - Immediate case 0x2A: opcode_0x2A(); break; // ROL A case 0x2B: opcode_0x2B(); break; // ANC - Immediate case 0x2C: opcode_0x2C(); break; // BIT - Absolute case 0x2D: opcode_0x2D(); break; // AND - Absolute case 0x2E: opcode_0x2E(); break; // ROL - Rotate Left - Absolute case 0x2F: opcode_0x2F(); break; // RLA - Absolute case 0x30: opcode_0x30(); break; // BMI - Branch on Minus (N Flag Set) case 0x31: opcode_0x31(); break; // AND - Indirect Indexed case 0x32: opcode_0x32(); break; // JAM case 0x33: opcode_0x33(); break; // RLA - Indirect Indexed Y case 0x34: opcode_0x34(); break; // NOP - ZeroPage , X case 0x35: opcode_0x35(); break; // AND - ZeroPage,X case 0x36: opcode_0x36(); break; // ROL - Rotate Left - ZeroPage , X case 0x37: opcode_0x37(); break; // RLA - ZeroPage , X case 0x38: opcode_0x38(); break; // SEC case 0x39: opcode_0x39(); break; // AND - Absolute,Y case 0x3A: opcode_0xEA(); break; // NOP case 0x3B: opcode_0x3B(); break; // RLA - Absolute , Y case 0x3C: opcode_0x3C(); break; // NOP - Absolute , X case 0x3D: opcode_0x3D(); break; // AND - Absolute,X case 0x3E: opcode_0x3E(); break; // ROL - Rotate Left - Absolute , X case 0x3F: opcode_0x3F(); break; // RLA - Absolute , X case 0x40: opcode_0x40(); break; // RTI - Return from Interrupt case 0x41: opcode_0x41(); break; // EOR - Indexed Indirect X case 0x42: opcode_0x42(); break; // JAM case 0x43: opcode_0x43(); break; // SRE - Indexed Indirect X case 0x44: opcode_0x44(); break; // NOP - ZeroPage case 0x45: opcode_0x45(); break; // EOR - ZeroPage case 0x46: opcode_0x46(); break; // LSR - Logical Shift Right - ZeroPage case 0x47: opcode_0x47(); break; // SRE - ZeroPage case 0x48: opcode_0x48(); break; // PHA - Push Accumulator to the stack case 0x49: opcode_0x49(); break; // EOR - Immediate case 0x4A: opcode_0x4A(); break; // LSR A case 0x4B: opcode_0x4B(); break; // ALR - Immediate case 0x4C: opcode_0x4C(); break; // JMP - Jump Absolute case 0x4D: opcode_0x4D(); break; // EOR - Absolute case 0x4E: opcode_0x4E(); break; // LSR - Logical Shift Right - Absolute case 0x4F: opcode_0x4F(); break; // SRE - Absolute case 0x50: opcode_0x50(); break; // BVC - Branch on Overflow Clear case 0x51: opcode_0x51(); break; // EOR - Indirect Indexed Y case 0x52: opcode_0x52(); break; // JAM case 0x53: opcode_0x53(); break; // SRE - Indirect Indexed Y case 0x54: opcode_0x54(); break; // NOP - ZeroPage , X case 0x55: opcode_0x55(); break; // EOR - ZeroPage,X case 0x56: opcode_0x56(); break; // LSR - Logical Shift Right - ZeroPage , X case 0x57: opcode_0x57(); break; // SRE - ZeroPage , X case 0x58: opcode_0x58(); break; // CLI case 0x59: opcode_0x59(); break; // EOR - Absolute,Y case 0x5A: opcode_0xEA(); break; // NOP case 0x5B: opcode_0x5B(); break; // RE - Absolute , Y case 0x5C: opcode_0x5C(); break; // NOP - Absolute , X case 0x5D: opcode_0x5D(); break; // EOR - Absolute,X case 0x5E: opcode_0x5E(); break; // LSR - Logical Shift Right - Absolute , X case 0x5F: opcode_0x5F(); break; // SRE - Absolute , X case 0x60: opcode_0x60(); break; // RTS - Return from Subroutine case 0x61: opcode_0x61(); break; // ADC - Indexed Indirect X case 0x62: opcode_0x62(); break; // JAM case 0x63: opcode_0x63(); break; // RRA - Indexed Indirect X case 0x64: opcode_0x64(); break; // NOP - ZeroPage case 0x65: opcode_0x65(); break; // ADC - ZeroPage case 0x66: opcode_0x66(); break; // ROR - Rotate Right - ZeroPage case 0x67: opcode_0x67(); break; // RRA - ZeroPage case 0x68: opcode_0x68(); break; // PLA - Pop Accumulator from the stack case 0x69: opcode_0x69(); break; // ADC - Immediate case 0x6A: opcode_0x6A(); break; // ROR A case 0x6B: opcode_0x6B(); break; // ARR - Immediate case 0x6C: opcode_0x6C(); break; // JMP - Jump Indirect case 0x6D: opcode_0x6D(); break; // ADC - Absolute case 0x6E: opcode_0x6E(); break; // ROR - Rotate Right - Absolute case 0x6F: opcode_0x6F(); break; // RRA - Absolute case 0x70: opcode_0x70(); break; // BVS - Branch on Overflow Set case 0x71: opcode_0x71(); break; // ADC - Indirect Indexed Y case 0x72: opcode_0x72(); break; // JAM case 0x73: opcode_0x73(); break; // RRA - Indirect Indexed Y case 0x74: opcode_0x74(); break; // NOP - ZeroPage , X case 0x75: opcode_0x75(); break; // ADC - ZeroPage , X case 0x76: opcode_0x76(); break; // ROR - Rotate Right - ZeroPage , X case 0x77: opcode_0x77(); break; // RRA - ZeroPage , X case 0x78: opcode_0x78(); break; // SEI case 0x79: opcode_0x79(); break; // ADC - Absolute , Y case 0x7A: opcode_0xEA(); break; // NOP case 0x7B: opcode_0x7B(); break; // RRA - Absolute , Y case 0x7C: opcode_0x7C(); break; // NOP - Absolute , X case 0x7D: opcode_0x7D(); break; // ADC - Absolute , X case 0x7E: opcode_0x7E(); break; // ROR - Rotate Right - Absolute , X case 0x7F: opcode_0x7F(); break; // RRA - Absolute , X case 0x80: opcode_0x80(); break; // NOP - Immediate case 0x81: opcode_0x81(); break; // STA - Indexed Indirect X case 0x82: opcode_0x82(); break; // NOP - Immediate case 0x83: opcode_0x83(); break; // SAX - Indexed Indirect X case 0x84: opcode_0x84(); break; // STY - ZeroPage case 0x85: opcode_0x85(); break; // STA - ZeroPage case 0x86: opcode_0x86(); break; // STX - ZeroPage case 0x87: opcode_0x87(); break; // SAX - ZeroPage case 0x88: opcode_0x88(); break; // DEY case 0x89: opcode_0x89(); break; // NOP - Immediate case 0x8A: opcode_0x8A(); break; // TXA case 0x8B: opcode_0x8B(); break; // ANE - Immediate case 0x8C: opcode_0x8C(); break; // STY - Absolute case 0x8D: opcode_0x8D(); break; // STA - Absolute case 0x8E: opcode_0x8E(); break; // STX - Absolute case 0x8F: opcode_0x8F(); break; // SAX - Absolute case 0x90: opcode_0x90(); break; // BCC - Branch on Carry Clear case 0x91: opcode_0x91(); break; // STA - Indirect Indexed Y case 0x92: opcode_0x92(); break; // JAM case 0x93: opcode_0x93(); break; // SHA - ZeroPage , Y case 0x94: opcode_0x94(); break; // STY - ZeroPage , X case 0x95: opcode_0x95(); break; // STA - ZeroPage , X case 0x96: opcode_0x96(); break; // STX - ZeroPage , Y case 0x97: opcode_0x97(); break; // SAX - ZeroPage , Y case 0x98: opcode_0x98(); break; // TYA case 0x99: opcode_0x99(); break; // STA - Absolute , Y case 0x9A: opcode_0x9A(); break; // TXS case 0x9B: opcode_0x9B(); break; // TAS - Absolute , Y case 0x9C: opcode_0x9C(); break; // SHY - Absolute , X case 0x9D: opcode_0x9D(); break; // STA - Absolute , X case 0x9E: opcode_0x9E(); break; // SHX - Absolute , Y case 0x9F: opcode_0x9F(); break; // SHA - Absolute , Y case 0xA0: opcode_0xA0(); break; // LDY - Immediate case 0xA1: opcode_0xA1(); break; // LDA - Indexed Indirect X case 0xA2: opcode_0xA2(); break; // LDX - Immediate case 0xA3: opcode_0xA3(); break; // LAX - Indexed Indirect X case 0xA4: opcode_0xA4(); break; // LDY - ZeroPage case 0xA5: opcode_0xA5(); break; // LDA - ZeroPage case 0xA6: opcode_0xA6(); break; // LDX - ZeroPage case 0xA7: opcode_0xA7(); break; // LAX - ZeroPage case 0xA8: opcode_0xA8(); break; // TAY case 0xA9: opcode_0xA9(); break; // LDA - Immediate case 0xAA: opcode_0xAA(); break; // TAX case 0xAB: opcode_0xAB(); break; // LAX - Immediate case 0xAC: opcode_0xAC(); break; // LDY - Absolute case 0xAD: opcode_0xAD(); break; // LDA - Absolute case 0xAE: opcode_0xAE(); break; // LDX - Absolute case 0xAF: opcode_0xAF(); break; // LAX - Absolute case 0xB0: opcode_0xB0(); break; // BCS - Branch on Carry Set case 0xB1: opcode_0xB1(); break; // LDA - Indirect Indexed Y case 0xB2: opcode_0xB2(); break; // JAM case 0xB3: opcode_0xB3(); break; // LAX - Indirect Indexed Y case 0xB4: opcode_0xB4(); break; // LDY - ZeroPage , X case 0xB5: opcode_0xB5(); break; // LDA - ZeroPage , X case 0xB6: opcode_0xB6(); break; // LDX - ZeroPage , Y case 0xB7: opcode_0xB7(); break; // LAX - ZeroPage , Y case 0xB8: opcode_0xB8(); break; // CLV case 0xB9: opcode_0xB9(); break; // LDA - Absolute , Y case 0xBA: opcode_0xBA(); break; // TSX case 0xBB: opcode_0xBB(); break; // LAS - Absolute , Y case 0xBC: opcode_0xBC(); break; // LDY - Absolute , X case 0xBD: opcode_0xBD(); break; // LDA - Absolute , X case 0xBE: opcode_0xBE(); break; // LDX - Absolute , Y case 0xBF: opcode_0xBF(); break; // LAX - Absolute , Y case 0xC0: opcode_0xC0(); break; // CPY - Immediate case 0xC1: opcode_0xC1(); break; // CMP - Indexed Indirect X case 0xC2: opcode_0xC2(); break; // NOP - Immediate case 0xC3: opcode_0xC3(); break; // DCP - Indexed Indirect X case 0xC4: opcode_0xC4(); break; // CPY - ZeroPage case 0xC5: opcode_0xC5(); break; // CMP - ZeroPage case 0xC6: opcode_0xC6(); break; // DEC - ZeroPage case 0xC7: opcode_0xC7(); break; // DCP - ZeroPage case 0xC8: opcode_0xC8(); break; // INY case 0xC9: opcode_0xC9(); break; // CMP - Immediate case 0xCA: opcode_0xCA(); break; // DEX case 0xCB: opcode_0xCB(); break; // SBX - Immediate case 0xCC: opcode_0xCC(); break; // CPY - Absolute case 0xCD: opcode_0xCD(); break; // CMP - Absolute case 0xCE: opcode_0xCE(); break; // DEC - Absolute case 0xCF: opcode_0xCF(); break; // DCP - Absolute case 0xD0: opcode_0xD0(); break; // BNE - Branch on Zero Clear case 0xD1: opcode_0xD1(); break; // CMP - Indirect Indexed Y case 0xD2: opcode_0xD2(); break; // JAM case 0xD3: opcode_0xD3(); break; // DCP - Indirect Indexed Y case 0xD4: opcode_0xD4(); break; // NOP - ZeroPage , X case 0xD5: opcode_0xD5(); break; // CMP - ZeroPage , X case 0xD6: opcode_0xD6(); break; // DEC - ZeroPage , X case 0xD7: opcode_0xD7(); break; // DCP - ZeroPage , X case 0xD8: opcode_0xD8(); break; // CLD case 0xD9: opcode_0xD9(); break; // CMP - Absolute , Y case 0xDA: opcode_0xEA(); break; // NOP case 0xDB: opcode_0xDB(); break; // DCP - Absolute , Y case 0xDC: opcode_0xDC(); break; // NOP - Absolute , X case 0xDD: opcode_0xDD(); break; // CMP - Absolute , X case 0xDE: opcode_0xDE(); break; // DEC - Absolute , X case 0xDF: opcode_0xDF(); break; // DCP - Absolute , X case 0xE0: opcode_0xE0(); break; // CPX - Immediate case 0xE1: opcode_0xE1(); break; // SBC - Indexed Indirect X case 0xE2: opcode_0xE2(); break; // NOP - Immediate case 0xE3: opcode_0xE3(); break; // ISC - Indexed Indirect X case 0xE4: opcode_0xE4(); break; // CPX - ZeroPage case 0xE5: opcode_0xE5(); break; // SBC - ZeroPage case 0xE6: opcode_0xE6(); break; // INC - ZeroPage case 0xE7: opcode_0xE7(); break; // ISC - ZeroPage case 0xE8: opcode_0xE8(); break; // INX case 0xE9: opcode_0xE9(); break; // SBC - Immediate case 0xEA: opcode_0xEA(); break; // NOP case 0xEB: opcode_0xE9(); break; // SBC - Immediate case 0xEC: opcode_0xEC(); break; // CPX - Absolute case 0xED: opcode_0xED(); break; // SBC - Absolute case 0xEE: opcode_0xEE(); break; // INC - Absolute case 0xEF: opcode_0xEF(); break; // ISC - Absolute case 0xF0: opcode_0xF0(); break; // BEQ - Branch on Zero Set case 0xF1: opcode_0xF1(); break; // SBC - Indirect Indexed Y case 0xF2: opcode_0xF2(); break; // JAM case 0xF3: opcode_0xF3(); break; // ISC - Indirect Indexed Y case 0xF4: opcode_0xF4(); break; // NOP - ZeroPage , X case 0xF5: opcode_0xF5(); break; // SBC - ZeroPage , X case 0xF6: opcode_0xF6(); break; // INC - ZeroPage , X case 0xF7: opcode_0xF7(); break; // ISC - ZeroPage , X case 0xF8: opcode_0xF8(); break; // SED case 0xF9: opcode_0xF9(); break; // SBC - Absolute , Y case 0xFA: opcode_0xEA(); break; // NOP case 0xFB: opcode_0xFB(); break; // ISC - Absolute , Y case 0xFC: opcode_0xFC(); break; // NOP - Absolute , X case 0xFD: opcode_0xFD(); break; // SBC - Absolute , X case 0xFE: opcode_0xFE(); break; // INC - Absolute , X case 0xFF: opcode_0xFF(); break; // ISC - Absolute , X } } }