#![allow(unused_variables, dead_code, non_snake_case)] use super::{R6502, Bus}; #[derive(Clone, Copy, PartialEq)] pub enum ModeID { IMP, // Implied ACM, // Accumulator - Not using, IMP might cover this IMM, // Immediate ZP0, // Zero Page ZPX, // Zero Page, X ZPY, // Zero Page, Y REL, // Relative ABS, // Absolute ABX, // Absolute, X ABY, // Aboslute, Y IND, // Indirect IZX, // Indirect, X IZY, // Indirect, Y } // Instruction decoding: // https://llx.com/Neil/a2/opcodes.html // GROUP ONE ADDRESS MODES // 000 (zero page,X) IZX // 001 zero page ZP0 // 010 #immediate IMM // 011 absolute ABS // 100 (zero page),Y IZY // 101 zero page,X ZPX // 110 absolute,Y ABY // 111 absolute,X ABX // GROUP TWO ADDRESS MODES // 000 #immediate IMM // 001 zero page ZP0 // 010 accumulator IMP // 011 absolute ABS // 101 zero page,X ZPX // 111 absolute,X ABX pub struct AddressingModes; impl AddressingModes { pub const GROUP_ONE_ADDRS: [fn(&mut R6502, &mut dyn Bus) -> ModeID; 8] = [ AddressingModes::IZX, AddressingModes::ZP0, AddressingModes::IMM, AddressingModes::ABS, AddressingModes::IZY, AddressingModes::ZPX, AddressingModes::ABY, AddressingModes::ABX, ]; } impl AddressingModes { // This is also the accumulator mode pub fn IMP(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { cpu.working_data = cpu.a as u16; ModeID::IMP } pub fn IMM(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { cpu.working_data = bus.read(cpu.pc) as u16; cpu.pc += 1; ModeID::IMM } pub fn ZP0(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { cpu.working_addr = bus.read(cpu.pc) as u16 & 0x00FF; cpu.pc += 1; cpu.working_data = bus.read(cpu.working_addr) as u16; ModeID::ZP0 } pub fn ZPX(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { cpu.working_addr = bus.read(cpu.pc) as u16 & 0x00FF; cpu.working_addr += cpu.x as u16; cpu.pc += 1; cpu.working_data = bus.read(cpu.working_addr) as u16; ModeID::ZPX } pub fn ZPY(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { cpu.working_addr = bus.read(cpu.pc) as u16 & 0x00FF; cpu.working_addr += cpu.y as u16; cpu.pc += 1; cpu.working_data = bus.read(cpu.working_addr) as u16; ModeID::ZPY } pub fn REL(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { // NOTE: Not sure if we can use the working_data variable for this. // if any instruction using this address mode needs extra data read // then we need another variable to store this address // // Use working_addr to just like the other modes cpu.working_data = bus.read(cpu.pc) as u16; cpu.pc += 1; ModeID::REL } pub fn ABS(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { cpu.working_addr = bus.read(cpu.pc) as u16; cpu.pc += 1; cpu.working_addr |= (bus.read(cpu.pc) as u16) << 8; cpu.pc += 1; cpu.working_data = bus.read(cpu.working_addr) as u16 & 0x00FF; ModeID::ABS } pub fn ABX(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { cpu.working_addr = bus.read(cpu.pc) as u16; cpu.pc += 1; cpu.working_addr |= (bus.read(cpu.pc) as u16) << 8; cpu.pc += 1; cpu.working_addr += cpu.x as u16; cpu.working_data = bus.read(cpu.working_addr) as u16 & 0x00FF; ModeID::ABX } pub fn ABY(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { cpu.working_addr = bus.read(cpu.pc) as u16; cpu.pc += 1; cpu.working_addr |= (bus.read(cpu.pc) as u16) << 8; cpu.pc += 1; cpu.working_addr += cpu.y as u16; cpu.working_data = bus.read(cpu.working_addr) as u16 & 0x00FF; ModeID::ABY } pub fn IND(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { ModeID::IND } pub fn IZX(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { ModeID::IZX } pub fn IZY(cpu: &mut R6502, bus: &mut dyn Bus) -> ModeID { ModeID::IZY } }