#include #include #include #include #include #include #include cpu_context ctx = {0}; #define CPU_DEBUG 0 #define FILE_LOG 0 #if FILE_LOG == 1 FILE *log; #endif void cpu_init() { ctx.regs.a = 0x01; ctx.regs.f = 0xB0; ctx.regs.b = 0x00; ctx.regs.c = 0x13; ctx.regs.d = 0x00; ctx.regs.e = 0xD8; ctx.regs.h = 0x01; ctx.regs.l = 0x4D; ctx.regs.sp = 0xFFFE; ctx.regs.pc = 0x100; ctx.ie_register = 0; ctx.int_flags = 0; ctx.int_master_enabled = false; ctx.enabling_ime = false; timer_get_context()->div = 0xABCC; #if FILE_LOG == 1 log = fopen("log.txt", "w"); #endif } static void fetch_instruction() { ctx.cur_opcode = bus_read(ctx.regs.pc++); ctx.cur_inst = instruction_by_opcode(ctx.cur_opcode); } static void execute() { IN_PROC proc = inst_get_processor(ctx.cur_inst->type); if (!proc) { printf("no implementation(%02X)\n", ctx.cur_opcode); NO_IMPL } proc(&ctx); } bool cpu_step() { if(!ctx.halted) { u16 pc = ctx.regs.pc; fetch_instruction(); emu_cycles(1); #if FILE_LOG == 1 fprintf(log, "A:%02X F:%02X B:%02X C:%02X D:%02X E:%02X H:%02X L:%02X SP:%04X PC:%04X PCMEM:%02X,%02X,%02X,%02X\n", ctx.regs.a, ctx.regs.f, ctx.regs.b, ctx.regs.c, ctx.regs.d, ctx.regs.e, ctx.regs.h, ctx.regs.l, ctx.regs.sp, pc, bus_read(pc), bus_read(pc+1), bus_read(pc+2), bus_read(pc+3) ); #endif fetch_data(); #if CPU_DEBUG == 1 char flags[16]; sprintf(flags, "%c%c%c%c", ctx.regs.f & (1 << 7) ? 'Z' : '-', ctx.regs.f & (1 << 6) ? 'N' : '-', ctx.regs.f & (1 << 5) ? 'H' : '-', ctx.regs.f & (1 << 4) ? 'C' : '-' ); char inst[16]; inst_to_str(&ctx, inst); printf("%08lX - %04X: %-12s (%02X %02X %02X) A: %02X F: %s BC: %02X%02X DE: %02X%02X HL: %02X%02X SP: %04X PC: %04X\n", emu_get_context()->ticks, pc, inst, ctx.cur_opcode, bus_read(pc+1), bus_read(pc+2), ctx.regs.a, flags, ctx.regs.b, ctx.regs.c, ctx.regs.d, ctx.regs.e, ctx.regs.h, ctx.regs.l, ctx.regs.sp, ctx.regs.pc); if(ctx.cur_inst == NULL){ printf("Unknown Instruction! %02X\n", ctx.cur_opcode); exit(-7); } dbg_update(); dbg_print(); #endif execute(); } else { //is halted emu_cycles(1); if(ctx.int_flags) { ctx.halted = false; } } if(ctx.int_master_enabled) { cpu_handle_interrupts(&ctx); ctx.enabling_ime = false; } if(ctx.enabling_ime) { ctx.int_master_enabled = true; } return true; } u8 cpu_get_ie_register(){ return ctx.ie_register; } void cpu_set_ie_register(u8 ie){ ctx.ie_register = ie; } void cpu_request_interrupt(interrupt_type t) { ctx.int_flags |= t; } cpu_context *cpu_get_context() { return &ctx; }