Skip to content
Snippets Groups Projects
inst.rs 49.6 KiB
Newer Older
  • Learn to ignore specific revisions
  • use core::time;
    use std::thread;
    
    
    use crate::cpu::Cpu;
    
    pub const INSTRUCTIONS: [(fn(&mut Cpu), u8, &'static str); 256] = [
        // 0x0 opcodes
        (nop, 4, "NOP"),
        (ld_bc_u16, 12, "LD BC, u16"),
        (ld_mbc_a, 8, "LD [BC], A"),
        (inc_bc, 8, "INC BC"),
        (inc_b, 4, "INC B"),
        (dec_b, 4, "DEC B"),
        (ld_b_u8, 8, "LD B, u8"),
        (rlca, 4, "RLCA"),
        (ld_mu16_sp, 20, "LD [u16], SP"),
        (add_hl_bc, 8, "ADD HL, BC"),
    
        (ld_a_mbc, 8, "LD A, [BC]"),
    
        (dec_bc, 8, "DEC BC"),
        (inc_c, 4, "INC C"),
        (dec_c, 4, "DEC C"),
        (ld_c_u8, 8, "LD C, u8"),
    
        (rrca, 4, "RRCA"),
    
        // 0x1 opcodes
    
        (stop, 4, "STOP"),
    
        (ld_de_u16, 12, "LD DE, u16"),
    
        (ld_mde_a, 8, "LD [DE], A"),
    
        (inc_de, 8, "INC DE"),
    
        (inc_d, 4, "INC D"),
    
        (dec_d, 4, "DEC D"),
        (ld_d_u8, 8, "LD D, u8"),
        (rla, 4, "RLA"),
        (jr_i8, 12, "JR i8"),
    
        (add_hl_de, 8, "ADD HL, DE"),
    
        (ld_a_mde, 8, "LD A, [DE]"),
    
        (dec_de, 8, "DEC DE"),
    
        (inc_e, 4, "INC E"),
    
        (dec_e, 4, "DEC E"),
        (ld_e_u8, 8, "LD E, u8"),
    
        (rra, 4, "RRA"),
    
        // 0x2 opcodes
        (jr_nz_i8, 8, "JR NZ, i8"),
        (ld_hl_u16, 12, "LD HL, u16"),
        (ld_mhli_a, 8, "LD [HL+], A"),
        (inc_hl, 8, "INC HL"),
        (inc_h, 4, "INC H"),
    
        (dec_h, 4, "DEC H"),
        (ld_h_u8, 8, "LD H, u8"),
    
        (daa, 4, "DAA"),
    
        (jr_z_i8, 8, "JR Z, i8"),
    
        (add_hl_hl, 8, "ADD HL, HL"),
    
        (ld_a_mhli, 8, "LD A, [HL+] "),
    
        (dec_hl, 8, "DEC HL"),
    
        (inc_l, 4, "INC L"),
    
        (dec_l, 4, "DEC L"),
    
        (ld_l_u8, 8, "LD L, u8"),
        (cpl, 4, "CPL"),
        // 0x3 opcodes
        (jr_nc_i8, 8, "JR NC, i8"),
        (ld_sp_u16, 12, "LD SP, u16"),
        (ld_mhld_a, 8, "LD [HL-], A"),
    
        (inc_sp, 8, "INC SP"),
        (inc_mhl, 12, "INC [HL]"),
    
        (dec_mhl, 12, "DEC [HL]"),
    
        (ld_mhl_u8, 12, "LD [HL], u8 "),
        (scf, 4, "SCF"),
    
        (jr_c_i8, 8, "JR C, i8"),
    
        (add_hl_sp, 8, "ADD HL, SP"),
    
        (ld_a_mhld, 8, "LD A, [HL-]"),
    
        (dec_sp, 8, "DEC SP"),
    
        (inc_a, 4, "INC A"),
    
        (dec_a, 4, "DEC A"),
        (ld_a_u8, 8, "LD A, u8"),
    
        (ccf, 4, "CCF"),
    
        // 0x4 opcodes
    
    João Magalhães's avatar
    João Magalhães committed
        (ld_b_b, 4, "LD B, B"),
    
        (ld_b_c, 4, "LD B, C"),
        (ld_b_d, 4, "LD B, D"),
    
        (ld_b_e, 4, "LD B, E"),
    
        (ld_b_h, 4, "LD B, H"),
    
        (ld_b_l, 4, "LD B, L"),
    
        (ld_b_mhl, 8, "LD B, [HL]"),
    
        (ld_b_a, 4, "LD B, A"),
    
        (ld_c_b, 4, "LD C, B"),
        (ld_c_c, 4, "LD C, C"),
        (ld_c_d, 4, "LD C, D"),
        (ld_c_e, 4, "LD C, E"),
        (ld_c_h, 4, "LD C, H"),
        (ld_c_l, 4, "LD C, L"),
    
        (ld_c_mhl, 8, "LD C, [HL]"),
    
        (ld_c_a, 4, "LD C, A"),
        // 0x5 opcodes
    
        (ld_d_b, 4, "LD D, B"),
        (ld_d_c, 4, "LD D, C"),
        (ld_d_d, 4, "LD D, D"),
        (ld_d_e, 4, "LD D, E"),
        (ld_d_h, 4, "LD D, H"),
        (ld_d_l, 4, "LD D, L"),
    
        (ld_d_mhl, 8, "LD D, [HL]"),
    
        (ld_d_a, 4, "LD D, A"),
    
        (ld_e_b, 4, "LD E, B"),
        (ld_e_c, 4, "LD E, C"),
        (ld_e_d, 4, "LD E, D"),
        (ld_e_e, 4, "LD E, E"),
        (ld_e_h, 4, "LD E, H"),
    
        (ld_e_l, 4, "LD E, L"),
    
        (ld_e_mhl, 8, "LD E, [HL]"),
    
        (ld_e_a, 4, "LD E, A"),
    
        // 0x6 opcodes
    
    João Magalhães's avatar
    João Magalhães committed
        (ld_h_b, 4, "LD H, B"),
        (ld_h_c, 4, "LD H, C"),
        (ld_h_d, 4, "LD H, D"),
        (ld_h_e, 4, "LD H, E"),
        (ld_h_h, 4, "LD H, H"),
        (ld_h_l, 4, "LD H, L"),
        (ld_h_mhl, 8, "LD H, [HL]"),
    
        (ld_h_a, 4, "LD H, A"),
    
    João Magalhães's avatar
    João Magalhães committed
        (ld_l_b, 4, "LD L, B"),
        (ld_l_c, 4, "LD L, C"),
        (ld_l_d, 4, "LD L, D"),
        (ld_l_e, 4, "LD L, E"),
        (ld_l_h, 4, "LD L, H"),
        (ld_l_l, 4, "LD L, L"),
    
        (ld_l_mhl, 8, "LD L, [HL]"),
    
        (ld_l_a, 4, "LD L, A"),
    
        // 0x7 opcodes
    
        (ld_mhl_b, 8, "LD [HL], B"),
        (ld_mhl_c, 8, "LD [HL], C"),
        (ld_mhl_d, 8, "LD [HL], D"),
    
        (ld_mhl_e, 8, "LD [HL], E"),
    
    João Magalhães's avatar
    João Magalhães committed
        (ld_mhl_h, 8, "LD [HL], H"),
        (ld_mhl_l, 8, "LD [HL], L"),
    
        (halt, 4, "HALT"),
    
        (ld_mhl_a, 8, "LD [HL], A"),
        (ld_a_b, 4, "LD A, B"),
        (ld_a_c, 4, "LD A, C"),
    
        (ld_a_d, 4, "LD A, D"),
    
        (ld_a_e, 4, "LD A, E"),
        (ld_a_h, 4, "LD A, H"),
        (ld_a_l, 4, "LD A, L"),
    
        (ld_a_mhl, 8, "LD A, [HL]"),
    
    João Magalhães's avatar
    João Magalhães committed
        (ld_a_a, 4, "LD A, A"),
    
        // 0x8 opcodes
    
        (add_a_b, 4, "ADD A, B"),
    
        (add_a_c, 4, "ADD A, C"),
    
        (add_a_d, 4, "ADD A, D"),
    
        (add_a_e, 4, "ADD A, E"),
    
        (add_a_h, 4, "ADD A, H"),
        (add_a_l, 4, "ADD A, L"),
    
        (add_a_mhl, 8, "ADD A, [HL]"),
    
        (add_a_a, 4, "ADD A, A"),
    
        (adc_a_b, 4, "ADC A, B"),
        (adc_a_c, 4, "ADC A, C"),
        (adc_a_d, 4, "ADC A, D"),
        (adc_a_e, 4, "ADC A, E"),
        (adc_a_h, 4, "ADC A, H"),
        (adc_a_l, 4, "ADC A, L"),
        (adc_a_mhl, 8, "ADC A, [HL]"),
        (adc_a_a, 4, "ADC A, A"),
    
        // 0x9 opcodes
        (sub_a_b, 4, "SUB A, B"),
    
        (sub_a_c, 4, "SUB A, C"),
        (sub_a_d, 4, "SUB A, D"),
        (sub_a_e, 4, "SUB A, E"),
        (sub_a_h, 4, "SUB A, H"),
        (sub_a_l, 4, "SUB A, L"),
        (sub_a_mhl, 8, "SUB A, [HL]"),
    
        (sub_a_a, 4, "SUB A, A"),
    
        (sbc_a_b, 4, "SBC A, B"),
        (sbc_a_c, 4, "SBC A, C"),
        (sbc_a_d, 4, "SBC A, D"),
        (sbc_a_e, 4, "SBC A, E"),
        (sbc_a_h, 4, "SBC A, H"),
        (sbc_a_l, 4, "SBC A, L"),
        (sbc_a_mhl, 8, "SBC A, [HL]"),
        (sbc_a_a, 4, "SBC A, A"),
    
        // 0xa opcodes
    
        (and_a_b, 4, "AND A, B"),
    
        (and_a_c, 4, "AND A, C"),
    
        (and_a_d, 4, "AND A, D"),
        (and_a_e, 4, "AND A, E"),
        (and_a_h, 4, "AND A, H"),
        (and_a_l, 4, "AND A, L"),
        (and_a_mhl, 4, "AND A, [HL]"),
        (and_a_a, 4, "AND A, A"),
    
        (xor_a_b, 4, "XOR A, B"),
    
        (xor_a_c, 4, "XOR A, C"),
    
        (xor_a_d, 4, "XOR A, D"),
        (xor_a_e, 4, "XOR A, E"),
        (xor_a_h, 4, "XOR A, H"),
    
        (xor_a_l, 4, "XOR A, L"),
    
        (xor_a_mhl, 8, "XOR A, [HL]"),
    
        (xor_a_a, 4, "XOR A, A"),
        // 0xb opcodes
        (or_a_b, 4, "OR A, B"),
        (or_a_c, 4, "OR A, C"),
    
        (or_a_d, 4, "OR A, D"),
        (or_a_e, 4, "OR A, E"),
        (or_a_h, 4, "OR A, H"),
        (or_a_l, 4, "OR A, L"),
    
        (or_a_mhl, 8, "OR A, [HL]"),
    
        (or_a_a, 4, "OR A, A"),
    
        (cp_a_b, 4, "CP A, B"),
        (cp_a_c, 4, "CP A, C"),
        (cp_a_d, 4, "CP A, D"),
        (cp_a_e, 4, "CP A, E"),
        (cp_a_h, 4, "CP A, H"),
        (cp_a_l, 4, "CP A, L"),
    
        (cp_a_mhl, 8, "CP A, [HL]"),
    
        (cp_a_a, 4, "CP A, A"),
    
        // 0xc opcodes
        (ret_nz, 8, "RET NZ"),
        (pop_bc, 12, "POP BC"),
    
        (jp_nz_u16, 12, "JP NZ, u16"),
    
        (jp_u16, 16, "JP u16"),
        (call_nz_u16, 12, "CALL NZ, u16"),
        (push_bc, 16, "PUSH BC"),
    
    João Magalhães's avatar
    João Magalhães committed
        (add_a_u8, 8, "ADD A, u8"),
    
        (rst_00h, 16, "RST 00h"),
    
        (ret_z, 8, "RET Z"),
    
        (ret, 16, "RET"),
    
        (jp_z_u16, 12, "JP Z, u16"),
    
        (illegal, 4, "ILLEGAL"),
    
        (call_z_u16, 12, "CALL Z, u16"),
    
        (call_u16, 24, "CALL u16"),
    
        (adc_a_u8, 8, "ADC A, u8 "),
    
        (rst_08h, 16, "RST 08h"),
    
        // 0xd opcodes
    
    João Magalhães's avatar
    João Magalhães committed
        (ret_nc, 8, "RET NC"),
    
    João Magalhães's avatar
    João Magalhães committed
        (pop_de, 12, "POP DE"),
    
        (jp_nc_u16, 12, "JP NC, u16"),
    
        (illegal, 4, "ILLEGAL"),
    
        (call_nc_u16, 12, "CALL NC, u16 "),
    
        (push_de, 16, "PUSH DE"),
    
        (sub_a_u8, 8, "SUB A, u8"),
    
        (rst_10h, 16, "RST 10h"),
    
        (ret_c, 8, "RET C"),
    
        (reti, 16, "RETI"),
    
        (jp_c_u16, 12, "JP C, u16"),
    
        (illegal, 4, "ILLEGAL"),
    
        (call_c_u16, 12, "CALL C, u16"),
    
        (illegal, 4, "ILLEGAL"),
    
        (sbc_a_u8, 8, "SBC A, u8"),
    
        (rst_18h, 16, "RST 18h"),
    
        // 0xe opcodes
        (ld_mff00u8_a, 12, "LD [FF00+u8], A"),
    
        (pop_hl, 12, "POP HL"),
    
        (ld_mff00c_a, 8, "LD [FF00+C], A"),
    
        (illegal, 4, "ILLEGAL"),
        (illegal, 4, "ILLEGAL"),
    
        (push_hl, 16, "PUSH HL"),
    
        (and_a_u8, 8, "AND A, u8"),
    
        (rst_20h, 16, "RST 20h"),
    
        (add_sp_i8, 16, "ADD SP, i8"),
    
        (jp_hl, 4, "JP HL"),
    
        (ld_mu16_a, 16, "LD [u16], A"),
    
        (illegal, 4, "ILLEGAL"),
        (illegal, 4, "ILLEGAL"),
        (illegal, 4, "ILLEGAL"),
    
        (xor_a_u8, 8, "XOR A, u8"),
    
        (rst_28h, 16, "RST 28h"),
    
        // 0xf opcodes
        (ld_a_mff00u8, 12, "LD A, [FF00+u8]"),
    
        (pop_af, 12, "POP AF"),
    
        (ld_a_mff00c, 8, "LD A, [FF00+C]"),
    
        (di, 4, "DI"),
    
        (illegal, 4, "ILLEGAL"),
    
        (push_af, 16, "PUSH AF"),
    
        (or_a_u8, 8, "OR A, u8"),
    
        (rst_30h, 16, "RST 30h"),
    
        (ld_hl_spi8, 12, "LD HL, SP+i8"),
    
        (ld_sp_hl, 8, "LD SP, HL"),
    
        (ld_a_mu16, 16, "LD A [u16]"),
    
        (ei, 4, "EI"),
    
        (illegal, 4, "ILLEGAL"),
        (illegal, 4, "ILLEGAL"),
    
        (cp_a_u8, 8, "CP A, u8"),
        (rst_38h, 16, "RST 38h"),
    ];
    
    
    pub const EXTENDED: [(fn(&mut Cpu), u8, &'static str); 256] = [
    
        // 0x0 opcodes
    
        (rlc_b, 8, "RLC B"),
        (rlc_c, 8, "RLC C"),
        (rlc_d, 8, "RLC D"),
        (rlc_e, 8, "RLC E"),
        (rlc_h, 8, "RLC H"),
        (rlc_l, 8, "RLC L"),
    
        (rlc_mhl, 16, "RLC [HL]"),
    
        (rlc_a, 8, "RLC A"),
    
        (rrc_b, 8, "RRC B"),
        (rrc_c, 8, "RRC C"),
        (rrc_d, 8, "RRC D"),
        (rrc_e, 8, "RRC E"),
        (rrc_h, 8, "RRC H"),
        (rrc_l, 8, "RRC L"),
    
        (rrc_mhl, 16, "RRC [HL]"),
    
        (rrc_a, 8, "RRC A"),
    
        // 0x1 opcodes
    
        (rl_b, 8, "RL B"),
    
        (rl_c, 8, "RL C"),
    
        (rl_d, 8, "RL D"),
        (rl_e, 8, "RL E"),
        (rl_h, 8, "RL H"),
        (rl_l, 8, "RL L"),
        (rl_mhl, 16, "RL [HL]"),
        (rl_a, 8, "RL A"),
        (rr_b, 8, "RR B"),
    
        (rr_c, 8, "RR C"),
        (rr_d, 8, "RR D"),
    
        (rr_e, 8, "RR E"),
    
        (rr_h, 8, "RR H"),
        (rr_l, 8, "RR L"),
        (rr_mhl, 16, "RR [HL]"),
        (rr_a, 8, "RR A"),
    
        // 0x2 opcodes
    
        (sla_b, 8, "SLA B"),
    
        (sla_c, 8, "SLA C"),
        (sla_d, 8, "SLA D"),
        (sla_e, 8, "SLA E"),
        (sla_h, 8, "SLA H"),
        (sla_l, 8, "SLA L"),
        (sla_mhl, 16, "SLA [HL]"),
    
        (sla_a, 8, "SLA A"),
    
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        // 0x3 opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (swap_a, 8, "SWAP A"),
    
        (srl_b, 8, "SRL B"),
    
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        // 0x4 opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
    
        (bit_0_d, 8, "BIT 0, D"),
    
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        // 0x5 opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        // 0x6 opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        // 0x7 opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (bit_7_h, 8, "BIT 7, H"),
    
        (bit_7_l, 8, "BIT 7, L"),
    
        (noimpl, 4, "! UNIMP !"),
    
        (bit_7_a, 8, "BIT 7, A"),
    
        // 0x8 opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
    
        (res_0_a, 8, "RES 0, A"),
    
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        // 0x9 opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        // 0xa opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
    
        // 0xb opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
    
        (res_7_mhl, 16, "RES 7, [HL]"),
    
        (noimpl, 4, "! UNIMP !"),
        // 0xc opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        // 0xd opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
    
        (set_3_mhl, 16, "SET 3, [HL]"),
    
        (noimpl, 4, "! UNIMP !"),
        // 0xe opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (set_4_a, 8, "SET 4, A"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        // 0xf opcodes
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
        (noimpl, 4, "! UNIMP !"),
    
    ];
    
    fn nop(_cpu: &mut Cpu) {}
    
    fn noimpl(_cpu: &mut Cpu) {
    
        let ten_millis = time::Duration::from_millis(10000);
        thread::sleep(ten_millis); // @todo remove this hack
    
        todo!("Instruction not implemented");
    }
    
    
    fn illegal(_cpu: &mut Cpu) {
        panic!("Illegal instruction");
    }
    
    
    fn ld_bc_u16(cpu: &mut Cpu) {
        let word = cpu.read_u16();
        cpu.set_bc(word);
    }
    
    fn ld_mbc_a(cpu: &mut Cpu) {
        cpu.mmu.write(cpu.bc(), cpu.a);
    }
    
    fn inc_bc(cpu: &mut Cpu) {
        cpu.set_bc(cpu.bc().wrapping_add(1));
    }
    
    fn inc_b(cpu: &mut Cpu) {
    
        let b = cpu.b;
        let value = b.wrapping_add(1);
    
    
        cpu.set_sub(false);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((b & 0xf) == 0xf);
    
    
        cpu.b = value;
    }
    
    fn dec_b(cpu: &mut Cpu) {
        let b = cpu.b;
        let value = b.wrapping_sub(1);
    
        cpu.set_sub(true);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((b & 0xf) == 0x0);
    
    
        cpu.b = value;
    }
    
    fn ld_b_u8(cpu: &mut Cpu) {
        let byte = cpu.read_u8();
        cpu.b = byte;
    }
    
    fn rlca(cpu: &mut Cpu) {
        let carry = cpu.a >> 7;
    
        cpu.a = cpu.a << 1 | carry;
    
        cpu.set_sub(false);
        cpu.set_zero(false);
        cpu.set_half_carry(false);
        cpu.set_carry(carry == 1);
    }
    
    fn ld_mu16_sp(cpu: &mut Cpu) {
        let word = cpu.read_u16();
        cpu.mmu.write(word, cpu.sp as u8);
        cpu.mmu.write(word + 1, (cpu.sp >> 8) as u8);
    }
    
    fn add_hl_bc(cpu: &mut Cpu) {
        let value = add_u16_u16(cpu, cpu.hl(), cpu.bc());
        cpu.set_hl(value);
    }
    
    
    fn ld_a_mbc(cpu: &mut Cpu) {
        let byte = cpu.mmu.read(cpu.bc());
        cpu.a = byte;
    }
    
    
    fn dec_bc(cpu: &mut Cpu) {
        cpu.set_bc(cpu.bc().wrapping_sub(1));
    }
    
    fn inc_c(cpu: &mut Cpu) {
    
        let c = cpu.c;
    
        let value = cpu.c.wrapping_add(1);
    
        cpu.set_sub(false);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((c & 0xf) == 0xf);
    
    
        cpu.c = value;
    }
    
    fn dec_c(cpu: &mut Cpu) {
        let c = cpu.c;
        let value = c.wrapping_sub(1);
    
        cpu.set_sub(true);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((c & 0xf) == 0x0);
    
    
        cpu.c = value;
    }
    
    fn ld_c_u8(cpu: &mut Cpu) {
        let byte = cpu.read_u8();
        cpu.c = byte;
    }
    
    
    fn rrca(cpu: &mut Cpu) {
        let carry = cpu.a & 0x1;
        cpu.a = (cpu.a >> 1) | (carry << 7);
    
        cpu.set_sub(false);
        cpu.set_zero(false);
        cpu.set_half_carry(false);
        cpu.set_carry(carry != 0x0);
    }
    
    
    fn stop(cpu: &mut Cpu) {
        cpu.stop();
    }
    
    
    fn ld_de_u16(cpu: &mut Cpu) {
        let word = cpu.read_u16();
        cpu.set_de(word);
    }
    
    
    fn ld_mde_a(cpu: &mut Cpu) {
        cpu.mmu.write(cpu.de(), cpu.a);
    }
    
    
    fn inc_de(cpu: &mut Cpu) {
        cpu.set_de(cpu.de().wrapping_add(1));
    }
    
    
    fn inc_d(cpu: &mut Cpu) {
    
        let d = cpu.d;
        let value = d.wrapping_add(1);
    
    
        cpu.set_sub(false);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((d & 0xf) == 0xf);
    
    fn dec_d(cpu: &mut Cpu) {
        let d = cpu.d;
        let value = d.wrapping_sub(1);
    
        cpu.set_sub(true);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((d & 0xf) == 0x0);
    
    
        cpu.d = value;
    }
    
    fn ld_d_u8(cpu: &mut Cpu) {
        let byte = cpu.read_u8();
        cpu.d = byte;
    }
    
    fn rla(cpu: &mut Cpu) {
        let carry = cpu.get_carry();
    
        cpu.set_carry(cpu.a & 0x80 == 0x80);
    
        cpu.a = cpu.a << 1 | carry as u8;
    
        cpu.set_sub(false);
        cpu.set_zero(false);
        cpu.set_half_carry(false);
    }
    
    fn jr_i8(cpu: &mut Cpu) {
        let byte = cpu.read_u8() as i8;
        cpu.pc = (cpu.pc as i16).wrapping_add(byte as i16) as u16;
    }
    
    
    fn add_hl_de(cpu: &mut Cpu) {
        let value = add_u16_u16(cpu, cpu.hl(), cpu.de());
        cpu.set_hl(value);
    }
    
    
    fn ld_a_mde(cpu: &mut Cpu) {
        let byte = cpu.mmu.read(cpu.de());
        cpu.a = byte;
    }
    
    
    fn dec_de(cpu: &mut Cpu) {
        cpu.set_de(cpu.de().wrapping_sub(1));
    }
    
    
    fn inc_e(cpu: &mut Cpu) {
    
        let e = cpu.e;
    
        let value = cpu.e.wrapping_add(1);
    
        cpu.set_sub(false);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((e & 0xf) == 0xf);
    
    fn dec_e(cpu: &mut Cpu) {
        let e = cpu.e;
        let value = e.wrapping_sub(1);
    
        cpu.set_sub(true);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((e & 0xf) == 0x0);
    
    
        cpu.e = value;
    }
    
    fn ld_e_u8(cpu: &mut Cpu) {
        let byte = cpu.read_u8();
        cpu.e = byte;
    }
    
    
    fn rra(cpu: &mut Cpu) {
        let carry = cpu.get_carry();
    
        cpu.set_carry(cpu.a & 0x01 == 0x01);
    
        cpu.a = cpu.a >> 1 | ((carry as u8) << 7);
    
        cpu.set_sub(false);
        cpu.set_zero(false);
        cpu.set_half_carry(false);
    }
    
    
    fn jr_nz_i8(cpu: &mut Cpu) {
        let byte = cpu.read_u8() as i8;
    
        if cpu.get_zero() {
            return;
        }
    
        cpu.pc = (cpu.pc as i16).wrapping_add(byte as i16) as u16;
        cpu.ticks = cpu.ticks.wrapping_add(4);
    }
    
    fn ld_hl_u16(cpu: &mut Cpu) {
        let word = cpu.read_u16();
        cpu.set_hl(word);
    }
    
    fn ld_mhli_a(cpu: &mut Cpu) {
        cpu.mmu.write(cpu.hl(), cpu.a);
        cpu.set_hl(cpu.hl().wrapping_add(1));
    }
    
    fn inc_hl(cpu: &mut Cpu) {
        cpu.set_hl(cpu.hl().wrapping_add(1));
    }
    
    fn inc_h(cpu: &mut Cpu) {
    
        let h = cpu.h;
    
        let value = cpu.h.wrapping_add(1);
    
        cpu.set_sub(false);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((h & 0xf) == 0xf);
    
    fn dec_h(cpu: &mut Cpu) {
        let h = cpu.h;
        let value = h.wrapping_sub(1);
    
        cpu.set_sub(true);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((h & 0xf) == 0x0);
    
    
        cpu.h = value;
    }
    
    fn ld_h_u8(cpu: &mut Cpu) {
        let byte = cpu.read_u8();
        cpu.h = byte;
    }
    
    
    fn daa(cpu: &mut Cpu) {
        let a = cpu.a;
        let mut adjust = 0;
    
        if cpu.get_half_carry() {
            adjust |= 0x06;
        }
    
        if cpu.get_carry() {
            // Yes, we have to adjust it.
            adjust |= 0x60;
        }
    
        let res = if cpu.get_sub() {
            a.wrapping_sub(adjust)
        } else {
            if a & 0x0F > 0x09 {
                adjust |= 0x06;
            }
    
            if a > 0x99 {
                adjust |= 0x60;
            }
    
            a.wrapping_add(adjust)
        };
    
        cpu.a = res;
    
        cpu.set_zero(res == 0);
        cpu.set_half_carry(false);
        cpu.set_carry(adjust & 0x60 == 0x60);
    }
    
    
    fn jr_z_i8(cpu: &mut Cpu) {
        let byte = cpu.read_u8() as i8;
    
        if !cpu.get_zero() {
            return;
        }
    
        cpu.pc = (cpu.pc as i16).wrapping_add(byte as i16) as u16;
        cpu.ticks = cpu.ticks.wrapping_add(4);
    }
    
    
    fn add_hl_hl(cpu: &mut Cpu) {
        let value = add_u16_u16(cpu, cpu.hl(), cpu.hl());
        cpu.set_hl(value);
    }
    
    
    fn ld_a_mhli(cpu: &mut Cpu) {
        let byte = cpu.mmu.read(cpu.hl());
        cpu.a = byte;
        cpu.set_hl(cpu.hl().wrapping_add(1));
    }
    
    
    fn dec_hl(cpu: &mut Cpu) {
        cpu.set_hl(cpu.hl().wrapping_sub(1));
    }
    
    
    fn inc_l(cpu: &mut Cpu) {
    
        let l = cpu.l;
        let value = l.wrapping_add(1);
    
    
        cpu.set_sub(false);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((l & 0xf) == 0xf);
    
    fn dec_l(cpu: &mut Cpu) {
        let l = cpu.l;
        let value = l.wrapping_sub(1);
    
        cpu.set_sub(true);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((l & 0xf) == 0x0);
    
    fn ld_l_u8(cpu: &mut Cpu) {
        let byte = cpu.read_u8();
        cpu.l = byte;
    }
    
    fn cpl(cpu: &mut Cpu) {
        cpu.a = !cpu.a;
    
        cpu.set_sub(true);
        cpu.set_half_carry(true);
    }
    
    fn ld_sp_u16(cpu: &mut Cpu) {
        cpu.sp = cpu.read_u16();
    }
    
    fn jr_nc_i8(cpu: &mut Cpu) {
    
    João Magalhães's avatar
    João Magalhães committed
        let byte = cpu.read_u8() as i8;
    
    
        if cpu.get_carry() {
            return;
        }
    
        cpu.pc = (cpu.pc as i16).wrapping_add(byte as i16) as u16;
        cpu.ticks = cpu.ticks.wrapping_add(4);
    }
    
    fn ld_mhld_a(cpu: &mut Cpu) {
        cpu.mmu.write(cpu.hl(), cpu.a);
        cpu.set_hl(cpu.hl().wrapping_sub(1));
    }
    
    
    fn inc_sp(cpu: &mut Cpu) {
        cpu.sp = cpu.sp.wrapping_add(1);
    }
    
    fn inc_mhl(cpu: &mut Cpu) {
        let byte = cpu.mmu.read(cpu.hl());
        let value = byte.wrapping_add(1);
    
        cpu.set_sub(false);
        cpu.set_zero(value == 0);
        cpu.set_half_carry((byte & 0xf) == 0xf);
    
        cpu.mmu.write(cpu.hl(), value);
    }
    
    
    fn dec_mhl(cpu: &mut Cpu) {
        let byte = cpu.mmu.read(cpu.hl());
        let value = byte.wrapping_sub(1);
    
        cpu.set_sub(true);
        cpu.set_zero(value == 0);
    
        cpu.set_half_carry((byte & 0xf) == 0x0);
    
    
        cpu.mmu.write(cpu.hl(), value);
    }
    
    
    fn ld_mhl_u8(cpu: &mut Cpu) {
        let byte = cpu.read_u8();
        cpu.mmu.write(cpu.hl(), byte);
    }
    
    fn scf(cpu: &mut Cpu) {
        cpu.set_sub(false);
        cpu.set_half_carry(false);
        cpu.set_carry(true);
    }
    
    
    fn jr_c_i8(cpu: &mut Cpu) {
    
    João Magalhães's avatar
    João Magalhães committed
        let byte = cpu.read_u8() as i8;
    
    
        if !cpu.get_carry() {
            return;
        }
    
        cpu.pc = (cpu.pc as i16).wrapping_add(byte as i16) as u16;