/* Hey EMACS -*- linux-c -*- */ /* $Id: newcpu.c 2783 2008-05-25 13:08:41Z roms $ */ /* * UAE - The Un*x Amiga Emulator * * MC68000 emulation * * (c) 1995 Bernd Schmidt */ #include "newcpu.h" #include "memory.h" #include "options.h" #include "sysconfig.h" #include "sysdeps.h" #include "uconfig.h" #include #include #include #include // tiemu begin #include "../dbg/bkpts.h" #include "../ti68k_def.h" #define write_log printf static const struct uae_prefs currprefs = { 0, 1, 1 }; #if defined(CYGNUS_SIM) && !defined(NO_GDB) extern const char *symfile; #endif /* CYGNUS_SIM */ #define FLOATFORMAT_H /* don't include glib.h in romcalls.h */ // tiemu end /* Opcode of faulting instruction */ static uint16_t last_op_for_exception_3; /* PC at fault time */ static uintptr_t last_addr_for_exception_3; /* Address that generated the exception */ static uintptr_t last_fault_for_exception_3; /* read (0) or write (1) access */ static int last_writeaccess_for_exception_3; /* instruction (1) or data (0) access */ static int last_instructionaccess_for_exception_3; int areg_byteinc[] = { 1, 1, 1, 1, 1, 1, 1, 2 }; int imm8_table[] = { 8, 1, 2, 3, 4, 5, 6, 7 }; int movem_index1[256]; int movem_index2[256]; int movem_next[256]; int fpp_movem_index1[256]; int fpp_movem_index2[256]; int fpp_movem_next[256]; cpuop_func *cpufunctbl[65536]; #define COUNT_INSTRS 0 #if COUNT_INSTRS static unsigned long int instrcount[65536]; static uint16_t opcodenums[65536]; static int compfn(const void *el1, const void *el2) { return instrcount[*(const uint16_t *)el1] < instrcount[*(const uint16_t *)el2]; } static char *icountfilename(void) { char *name = getenv("INSNCOUNT"); if(name) return name; return COUNT_INSTRS == 2 ? "frequent.68k" : "insncount"; } void dump_counts(void) { FILE *f = fopen(icountfilename(), "w"); unsigned long int total; int i; write_log("Writing instruction count file...\n"); for(i = 0; i < 65536; i++) { opcodenums[i] = i; total += instrcount[i]; } qsort(opcodenums, 65536, sizeof(uint16_t), compfn); fprintf(f, "Total: %lu\n", total); for(i = 0; i < 65536; i++) { unsigned long int cnt = instrcount[opcodenums[i]]; struct instr *dp; struct mnemolookup *lookup; if(!cnt) break; dp = table68k + opcodenums[i]; for(lookup = lookuptab; lookup->mnemo != dp->mnemo; lookup++) ; fprintf(f, "%04x: %lu %s\n", opcodenums[i], cnt, lookup->name); } fclose(f); } #else void dump_counts(void) { } #endif int currIntLev = -1; int broken_in; int delayTime = 0; static unsigned long op_illg_1(uint32_t opcode); static unsigned long op_illg_1(uint32_t opcode) { op_illg(opcode); return 4; } static void build_cpufunctbl(void) { int i; unsigned long opcode; struct cputbl *tbl = (currprefs.cpu_level == 4 ? op_smalltbl_0_ff : currprefs.cpu_level == 3 ? op_smalltbl_1_ff : currprefs.cpu_level == 2 ? op_smalltbl_2_ff : currprefs.cpu_level == 1 ? op_smalltbl_3_ff : op_smalltbl_5_ff); write_log("UAE: Building CPU function table (%d %d).\n", currprefs.cpu_level, currprefs.address_space_24); for(opcode = 0; opcode < 65536; opcode++) cpufunctbl[opcode] = op_illg_1; for(i = 0; tbl[i].handler != NULL; i++) { if(!tbl[i].specific) cpufunctbl[tbl[i].opcode] = tbl[i].handler; } for(opcode = 0; opcode < 65536; opcode++) { cpuop_func *f; if(table68k[opcode].mnemo == i_ILLG || table68k[opcode].clev > currprefs.cpu_level) continue; if(table68k[opcode].handler != -1) { f = cpufunctbl[table68k[opcode].handler]; if(f == op_illg_1) abort(); cpufunctbl[opcode] = f; } } for(i = 0; tbl[i].handler != NULL; i++) { if(tbl[i].specific) cpufunctbl[tbl[i].opcode] = tbl[i].handler; } } void fill_prefetch_slow(void) { #ifdef CPUEMU_6 if(currprefs.cpu_cycle_exact) { regs.ir = get_word_ce(m68k_getpc()); regs.irc = get_word_ce(m68k_getpc() + 2); } else { #endif regs.ir = get_word(m68k_getpc()); regs.irc = get_word(m68k_getpc() + 2); #ifdef CPUEMU_6 } #endif } #if 0 unsigned long cycles_mask, cycles_val; static void update_68k_cycles (void) { cycles_mask = 0; cycles_val = currprefs.m68k_speed; if (currprefs.m68k_speed < 1) { cycles_mask = 0xFFFFFFFF; cycles_val = 0; } } void check_prefs_changed_cpu (void) { if (currprefs.m68k_speed != changed_prefs.m68k_speed) { currprefs.m68k_speed = changed_prefs.m68k_speed; reset_frame_rate_hack (); update_68k_cycles (); } } #endif void init_m68k(void) { int i; printf("UAE: version 0.8.25\n"); // update_68k_cycles (); for(i = 0; i < 256; i++) { int j; for(j = 0; j < 8; j++) { if(i & (1 << j)) break; } movem_index1[i] = j; movem_index2[i] = 7 - j; movem_next[i] = i & (~(1 << j)); } for(i = 0; i < 256; i++) { int j; for(j = 7; j >= 0; j--) { if(i & (1 << j)) break; } fpp_movem_index1[i] = 7 - j; fpp_movem_index2[i] = j; fpp_movem_next[i] = i & (~(1 << j)); } #if COUNT_INSTRS { FILE *f = fopen(icountfilename(), "r"); memset(instrcount, 0, sizeof instrcount); if(f) { uint32_t opcode, count, total; char name[20]; write_log("Reading instruction count file...\n"); fscanf(f, "Total: %lu\n", &total); while(fscanf(f, "%lx: %lu %s\n", &opcode, &count, name) == 3) { instrcount[opcode] = count; } fclose(f); } } #endif write_log("UAE: Building CPU table for configuration: 68"); regs.address_space_mask = 0xffffffff; if(currprefs.address_space_24 && currprefs.cpu_level > 1) write_log("EC"); switch(currprefs.cpu_level) { case 1: write_log("010"); break; case 2: write_log("020"); break; case 3: write_log("020/881"); break; case 4: /* Who is going to miss the MMU anyway...? :-) */ write_log("040"); break; default: write_log("000"); break; } if(currprefs.address_space_24) { regs.address_space_mask = 0x00ffffff; write_log(" 24-bit addressing"); } write_log("\n"); read_table68k(); do_merges(); write_log("UAE: %d CPU functions\n", nr_cpuop_funcs); build_cpufunctbl(); } struct regstruct regs, lastint_regs; #ifndef NO_GDB static struct regstruct regs_backup[16]; static int backup_pointer = 0; #endif /* 0 */ #ifdef NO_GDB static long int m68kpc_offset; #endif /* NO_GDB */ int lastint_no; #define get_ibyte_1(o) get_byte(regs.pc + (regs.pc_p - regs.pc_oldp) + (o) + 1) #define get_iword_1(o) get_word(regs.pc + (regs.pc_p - regs.pc_oldp) + (o)) #define get_ilong_1(o) get_long(regs.pc + (regs.pc_p - regs.pc_oldp) + (o)) #ifdef NO_GDB #ifdef __WIN32__ #define snprintf _snprintf #endif char *sym_addr(uint32_t addr) { static char buf[256]; int rcid = romcalls_is_addr(addr); if(rcid == -1) snprintf(buf, sizeof(buf), "$%06lX", (unsigned long)addr); else snprintf(buf, sizeof(buf), "$%06lX -> tios::%s", (unsigned long)addr, romcalls_get_name(rcid)); return buf; } int32_t ShowEA(FILE *f, int reg, amodes mode, wordsizes size, char *buf) { uint16_t dp; int8_t disp8; int16_t disp16; int r; uint32_t dispreg; uintptr_t addr; int32_t offset = 0; char buffer[296]; switch(mode) { case Dreg: sprintf(buffer, "D%d", reg); break; case Areg: sprintf(buffer, "A%d", reg); break; case Aind: sprintf(buffer, "(A%d)", reg); break; case Aipi: sprintf(buffer, "(A%d)+", reg); break; case Apdi: sprintf(buffer, "-(A%d)", reg); break; case Ad16: disp16 = get_iword_1(m68kpc_offset); m68kpc_offset += 2; addr = m68k_areg(regs, reg) + (int16_t)disp16; if(disp16 & 0x8000) sprintf(buffer, "(-$%X,A%d) [%s]", (-(int16_t)(disp16)) & 0xffff, reg, sym_addr((uint32_t)addr)); else sprintf(buffer, "(+$%X,A%d) [%s]", disp16 & 0xffff, reg, sym_addr((uint32_t)addr)); break; case Ad8r: dp = get_iword_1(m68kpc_offset); m68kpc_offset += 2; disp8 = dp & 0xFF; r = (dp & 0x7000) >> 12; dispreg = dp & 0x8000 ? m68k_areg(regs, r) : m68k_dreg(regs, r); if(!(dp & 0x800)) dispreg = (int32_t)(int16_t)(dispreg); dispreg <<= (dp >> 9) & 3; if(dp & 0x100) { int32_t outer = 0, disp = 0; int32_t base = m68k_areg(regs, reg); char name[10]; sprintf(name, "A%d, ", reg); if(dp & 0x80) { base = 0; name[0] = 0; } if(dp & 0x40) dispreg = 0; if((dp & 0x30) == 0x20) { disp = (int32_t)(int16_t)get_iword_1(m68kpc_offset); m68kpc_offset += 2; } if((dp & 0x30) == 0x30) { disp = get_ilong_1(m68kpc_offset); m68kpc_offset += 4; } base += disp; if((dp & 0x3) == 0x2) { outer = (int32_t)(int16_t)get_iword_1(m68kpc_offset); m68kpc_offset += 2; } if((dp & 0x3) == 0x3) { outer = get_ilong_1(m68kpc_offset); m68kpc_offset += 4; } if(!(dp & 4)) base += dispreg; if(dp & 3) base = get_long(base); if(dp & 4) base += dispreg; addr = base + outer; sprintf(buffer, "(%s%c%d.%c+%ld)+%ld [$%06lX]", name, dp & 0x8000 ? 'A' : 'D', (int)r, dp & 0x800 ? 'L' : 'W', disp, outer, (unsigned long)addr); } else { addr = m68k_areg(regs, reg) + (int32_t)((int8_t)disp8) + dispreg; sprintf(buffer, "(A%d, %c%d.%c, $%02X) [$%06lX]", reg, dp & 0x8000 ? 'A' : 'D', (int)r, dp & 0x800 ? 'L' : 'W', disp8, (unsigned long)addr); } break; case PC16: addr = m68k_getpc() + m68kpc_offset; disp16 = get_iword_1(m68kpc_offset); m68kpc_offset += 2; addr += (int16_t)disp16; if(disp16 & 0x8000) sprintf(buffer, "(-$%X,PC) [%s]", (-(int16_t)(disp16)) & 0xffff, sym_addr((uint32_t)addr)); else sprintf(buffer, "(+$%X,PC) [%s]", disp16 & 0xffff, sym_addr((uint32_t)addr)); break; case PC8r: addr = m68k_getpc() + m68kpc_offset; dp = get_iword_1(m68kpc_offset); m68kpc_offset += 2; disp8 = dp & 0xFF; r = (dp & 0x7000) >> 12; dispreg = dp & 0x8000 ? m68k_areg(regs, r) : m68k_dreg(regs, r); if(!(dp & 0x800)) dispreg = (int32_t)(int16_t)(dispreg); dispreg <<= (dp >> 9) & 3; if(dp & 0x100) { int32_t outer = 0, disp = 0; int32_t base = addr; char name[10]; sprintf(name, "PC, "); if(dp & 0x80) { base = 0; name[0] = 0; } if(dp & 0x40) dispreg = 0; if((dp & 0x30) == 0x20) { disp = (int32_t)(int16_t)get_iword_1(m68kpc_offset); m68kpc_offset += 2; } if((dp & 0x30) == 0x30) { disp = get_ilong_1(m68kpc_offset); m68kpc_offset += 4; } base += disp; if((dp & 0x3) == 0x2) { outer = (int32_t)(int16_t)get_iword_1(m68kpc_offset); m68kpc_offset += 2; } if((dp & 0x3) == 0x3) { outer = get_ilong_1(m68kpc_offset); m68kpc_offset += 4; } if(!(dp & 4)) base += dispreg; if(dp & 3) base = get_long(base); if(dp & 4) base += dispreg; addr = base + outer; sprintf(buffer, "(%s%c%d.%c+%ld)+%ld [%s]", name, dp & 0x8000 ? 'A' : 'D', (int)r, dp & 0x800 ? 'L' : 'W', disp, outer, sym_addr((uint32_t)addr)); } else { addr += (int32_t)((int8_t)disp8) + dispreg; sprintf(buffer, "(PC, %c%d.%c, $%02X) [%s]", dp & 0x8000 ? 'A' : 'D', (int)r, dp & 0x800 ? 'L' : 'W', disp8, sym_addr((uint32_t)addr)); } break; case absw: sprintf(buffer, "$%lX", (unsigned long)(int32_t)(int16_t)get_iword_1(m68kpc_offset)); m68kpc_offset += 2; break; case absl: sprintf(buffer, "%s", sym_addr((uint32_t)get_ilong_1(m68kpc_offset))); m68kpc_offset += 4; break; case imm: switch(size) { case sz_byte: sprintf(buffer, "#$%X", (unsigned int)(get_iword_1(m68kpc_offset) & 0xff)); m68kpc_offset += 2; break; case sz_word: sprintf(buffer, "#$%X", (unsigned int)(get_iword_1(m68kpc_offset) & 0xffff)); m68kpc_offset += 2; break; case sz_long: sprintf(buffer, "#$%lX", (unsigned long)(get_ilong_1(m68kpc_offset))); m68kpc_offset += 4; break; default: break; } break; case imm0: offset = (int32_t)(int8_t)get_iword_1(m68kpc_offset); m68kpc_offset += 2; sprintf(buffer, "#$%X", (unsigned int)(offset & 0xff)); break; case imm1: offset = (int32_t)(int16_t)get_iword_1(m68kpc_offset); m68kpc_offset += 2; sprintf(buffer, "#$%X", (unsigned int)(offset & 0xffff)); break; case imm2: offset = (int32_t)get_ilong_1(m68kpc_offset); m68kpc_offset += 4; sprintf(buffer, "#$%lX", (unsigned long)offset); break; case immi: offset = (int32_t)(int8_t)(reg & 0xff); sprintf(buffer, "#$%lX", (unsigned long)offset); break; default: break; } if(buf == 0) fprintf(f, "%s", buffer); else strcat(buf, buffer); return offset; } #endif /* NO_GDB */ #if 0 /* The plan is that this will take over the job of exception 3 handling - * the CPU emulation functions will just do a longjmp to m68k_go whenever * they hit an odd address. */ static int verify_ea (int reg, amodes mode, wordsizes size, uint32_t *val) { uint16_t dp; int8_t disp8; int16_t disp16; int r; uint32_t dispreg; uintptr_t addr; switch (mode){ case Dreg: *val = m68k_dreg (regs, reg); return 1; case Areg: *val = m68k_areg (regs, reg); return 1; case Aind: case Aipi: addr = m68k_areg (regs, reg); break; case Apdi: addr = m68k_areg (regs, reg); break; case Ad16: disp16 = get_iword_1 (m68kpc_offset); m68kpc_offset += 2; addr = m68k_areg(regs,reg) + (int16_t)disp16; break; case Ad8r: addr = m68k_areg (regs, reg); d8r_common: dp = get_iword_1 (m68kpc_offset); m68kpc_offset += 2; disp8 = dp & 0xFF; r = (dp & 0x7000) >> 12; dispreg = dp & 0x8000 ? m68k_areg(regs,r) : m68k_dreg(regs,r); if (!(dp & 0x800)) dispreg = (int32_t)(int16_t)(dispreg); dispreg <<= (dp >> 9) & 3; if (dp & 0x100) { int32_t outer = 0, disp = 0; int32_t base = addr; if (dp & 0x80) base = 0; if (dp & 0x40) dispreg = 0; if ((dp & 0x30) == 0x20) { disp = (int32_t)(int16_t)get_iword_1 (m68kpc_offset); m68kpc_offset += 2; } if ((dp & 0x30) == 0x30) { disp = get_ilong_1 (m68kpc_offset); m68kpc_offset += 4; } base += disp; if ((dp & 0x3) == 0x2) { outer = (int32_t)(int16_t)get_iword_1 (m68kpc_offset); m68kpc_offset += 2; } if ((dp & 0x3) == 0x3) { outer = get_ilong_1 (m68kpc_offset); m68kpc_offset += 4; } if (!(dp & 4)) base += dispreg; if (dp & 3) base = get_long (base); if (dp & 4) base += dispreg; addr = base + outer; } else { addr += (int32_t)((int8_t)disp8) + dispreg; } break; case PC16: addr = m68k_getpc () + m68kpc_offset; disp16 = get_iword_1 (m68kpc_offset); m68kpc_offset += 2; addr += (int16_t)disp16; break; case PC8r: addr = m68k_getpc () + m68kpc_offset; goto d8r_common; case absw: addr = (int32_t)(int16_t)get_iword_1 (m68kpc_offset); m68kpc_offset += 2; break; case absl: addr = get_ilong_1 (m68kpc_offset); m68kpc_offset += 4; break; case imm: switch (size){ case sz_byte: *val = get_iword_1 (m68kpc_offset) & 0xff; m68kpc_offset += 2; break; case sz_word: *val = get_iword_1 (m68kpc_offset) & 0xffff; m68kpc_offset += 2; break; case sz_long: *val = get_ilong_1 (m68kpc_offset); m68kpc_offset += 4; break; default: break; } return 1; case imm0: *val = (int32_t)(int8_t)get_iword_1 (m68kpc_offset); m68kpc_offset += 2; return 1; case imm1: *val = (int32_t)(int16_t)get_iword_1 (m68kpc_offset); m68kpc_offset += 2; return 1; case imm2: *val = get_ilong_1 (m68kpc_offset); m68kpc_offset += 4; return 1; case immi: *val = (int32_t)(int8_t)(reg & 0xff); return 1; default: addr = 0; break; } if ((addr & 1) == 0) return 1; last_addr_for_exception_3 = m68k_getpc () + m68kpc_offset; last_fault_for_exception_3 = addr; last_writeaccess_for_exception_3 = 0; last_instructionaccess_for_exception_3 = 0; return 0; } #endif /* 0 */ uint32_t get_disp_ea_020(uint32_t base, uint32_t dp) { int reg = (dp >> 12) & 15; int32_t regd = regs.regs[reg]; if((dp & 0x800) == 0) regd = (int32_t)(int16_t)regd; regd <<= (dp >> 9) & 3; if(dp & 0x100) { int32_t outer = 0; if(dp & 0x80) base = 0; if(dp & 0x40) regd = 0; if((dp & 0x30) == 0x20) base += (int32_t)(int16_t)next_iword(); if((dp & 0x30) == 0x30) base += next_ilong(); if((dp & 0x3) == 0x2) outer = (int32_t)(int16_t)next_iword(); if((dp & 0x3) == 0x3) outer = next_ilong(); if((dp & 0x4) == 0) base += regd; if(dp & 0x3) base = get_long(base); if(dp & 0x4) base += regd; return base + outer; } else { return base + (int32_t)((int8_t)dp) + regd; } } uint32_t get_disp_ea_000(uint32_t base, uint32_t dp) { int reg = (dp >> 12) & 15; int32_t regd = regs.regs[reg]; #if 1 if((dp & 0x800) == 0) regd = (int32_t)(int16_t)regd; return base + (int8_t)dp + regd; #else /* Branch-free code... benchmark this again now that * things are no longer inline. */ int32_t regd16; uint32_t mask; mask = ((dp & 0x800) >> 11) - 1; regd16 = (int32_t)(int16_t)regd; regd16 &= mask; mask = ~mask; base += (int8_t)dp; regd &= mask; regd |= regd16; return base + regd; #endif } void MakeSR(void) { #if 0 assert((regs.t1 & 1) == regs.t1); assert((regs.t0 & 1) == regs.t0); assert((regs.s & 1) == regs.s); assert((regs.m & 1) == regs.m); assert((XFLG & 1) == XFLG); assert((NFLG & 1) == NFLG); assert((ZFLG & 1) == ZFLG); assert((VFLG & 1) == VFLG); assert((CFLG & 1) == CFLG); #endif regs.sr = ((regs.t1 << 15) /*| (regs.t0 << 14)*/ | (regs.s << 13) /*| (regs.m << 12)*/ | (regs.intmask << 8) | (GET_XFLG << 4) | (GET_NFLG << 3) | (GET_ZFLG << 2) | (GET_VFLG << 1) | GET_CFLG); } void MakeFromSR(void) { // int oldm = regs.m; int olds = regs.s; regs.t1 = (regs.sr >> 15) & 1; // regs.t0 = (regs.sr >> 14) & 1; regs.s = (regs.sr >> 13) & 1; // regs.m = (regs.sr >> 12) & 1; regs.intmask = (regs.sr >> 8) & 7; SET_XFLG((regs.sr >> 4) & 1); SET_NFLG((regs.sr >> 3) & 1); SET_ZFLG((regs.sr >> 2) & 1); SET_VFLG((regs.sr >> 1) & 1); SET_CFLG(regs.sr & 1); #if 0 if (currprefs.cpu_level >= 2) { if (olds != regs.s) { if (olds) { if (oldm) regs.msp = m68k_areg(regs, 7); else regs.isp = m68k_areg(regs, 7); m68k_areg(regs, 7) = regs.usp; } else { regs.usp = m68k_areg(regs, 7); m68k_areg(regs, 7) = regs.m ? regs.msp : regs.isp; } } else if (olds && oldm != regs.m) { if (oldm) { regs.msp = m68k_areg(regs, 7); m68k_areg(regs, 7) = regs.isp; } else { regs.isp = m68k_areg(regs, 7); m68k_areg(regs, 7) = regs.msp; } } } else { #endif if(olds != regs.s) { if(olds) { regs.isp = m68k_areg(regs, 7); m68k_areg(regs, 7) = regs.usp; } else { regs.usp = m68k_areg(regs, 7); m68k_areg(regs, 7) = regs.isp; } } // } set_special(SPCFLAG_INT); if(regs.t1 /*|| regs.t0*/) set_special(SPCFLAG_TRACE); else /* Keep SPCFLAG_DOTRACE, we still want a trace exception for SR-modifying instructions (including STOP). */ unset_special(SPCFLAG_TRACE); } int intlev(void) { int rc = currIntLev; currIntLev = -1; return rc; } static void exception_trace(int nr) { unset_special(SPCFLAG_TRACE | SPCFLAG_DOTRACE); if(regs.t1 /*&& !regs.t0*/) { /* trace stays pending if exception is div by zero, chk, * trapv or trap #x */ if(nr == 5 || nr == 6 || nr == 7 || (nr >= 32 && nr <= 47)) set_special(SPCFLAG_DOTRACE); } regs.t1 /*= regs.t0 = regs.m*/ = 0; } #if 0 static void exception_debug (int nr) { #ifdef DEBUGGER if (!exception_debugging) return; console_out ("Exception %d, PC=%08.8X\n", nr, m68k_getpc()); #endif } #endif /* 0 */ void Exception_normal(int nr, uintptr_t oldpc) { uint32_t currpc = m68k_getpc(), newpc; int sv = regs.s; // exception_debug (nr); MakeSR(); if(!regs.s) { regs.usp = m68k_areg(regs, 7); m68k_areg(regs, 7) = regs.isp; regs.s = 1; } if(nr == 2 || nr == 3) { uint16_t mode = (sv ? 4 : 0) | (last_instructionaccess_for_exception_3 ? 2 : 1); mode |= last_writeaccess_for_exception_3 ? 0 : 16; m68k_areg(regs, 7) -= 14; /* fixme: bit3=I/N */ put_word(m68k_areg(regs, 7) + 0, mode); put_long(m68k_areg(regs, 7) + 2, last_fault_for_exception_3); put_word(m68k_areg(regs, 7) + 6, last_op_for_exception_3); put_word(m68k_areg(regs, 7) + 8, regs.sr); put_long(m68k_areg(regs, 7) + 10, last_addr_for_exception_3); // write_log ("Exception %d (%x) at %x -> %x!\n", nr, oldpc, currpc, get_long (regs.vbr + 4*nr)); goto kludge_me_do; } m68k_areg(regs, 7) -= 4; put_long(m68k_areg(regs, 7), currpc); m68k_areg(regs, 7) -= 2; put_word(m68k_areg(regs, 7), regs.sr); kludge_me_do: newpc = get_long(regs.vbr + 4 * nr); if(newpc & 1) { exception3(regs.ir, m68k_getpc(), newpc); return; } m68k_setpc(newpc); fill_prefetch_slow(); exception_trace(nr); } void Exception(int nr, uintptr_t oldpc) { Exception_normal(nr, oldpc); } int uscycle = 0; void Interrupt(int nr) { regs.stopped = 0; unset_special(SPCFLAG_STOP); assert(nr < 8 && nr >= 0); lastint_regs = regs; lastint_no = nr; Exception(nr + 24, 0); regs.intmask = nr; set_special(SPCFLAG_INT); } static uint32_t caar, cacr, itt0, itt1, dtt0, dtt1, tc, mmusr, urp, srp; int m68k_move2c(int regno, uint32_t *regp) { if((currprefs.cpu_level == 1 && (regno & 0x7FF) > 1) || (currprefs.cpu_level < 4 && (regno & 0x7FF) > 2) || (currprefs.cpu_level == 4 && regno == 0x802)) { op_illg(0x4E7B); return 0; } else { switch(regno) { case 0: regs.sfc = *regp & 7; break; case 1: regs.dfc = *regp & 7; break; case 2: cacr = *regp & (currprefs.cpu_level < 4 ? 0x3 : 0x80008000); break; case 3: tc = *regp & 0xc000; break; /* Mask out fields that should be zero. */ case 4: itt0 = *regp & 0xffffe364; break; case 5: itt1 = *regp & 0xffffe364; break; case 6: dtt0 = *regp & 0xffffe364; break; case 7: dtt1 = *regp & 0xffffe364; break; case 0x800: regs.usp = *regp; break; case 0x801: regs.vbr = *regp; break; case 0x802: caar = *regp & 0xfc; break; // case 0x803: regs.msp = *regp; if (regs.m == 1) m68k_areg(regs, 7) = regs.msp; break; // case 0x804: regs.isp = *regp; if (regs.m == 0) m68k_areg(regs, 7) = regs.isp; break; case 0x805: mmusr = *regp; break; case 0x806: urp = *regp; break; case 0x807: srp = *regp; break; default: op_illg(0x4E7B); return 0; } } return 1; } int m68k_movec2(int regno, uint32_t *regp) { if((currprefs.cpu_level == 1 && (regno & 0x7FF) > 1) || (currprefs.cpu_level < 4 && (regno & 0x7FF) > 2) || (currprefs.cpu_level == 4 && regno == 0x802)) { op_illg(0x4E7A); return 0; } else { switch(regno) { case 0: *regp = regs.sfc; break; case 1: *regp = regs.dfc; break; case 2: *regp = cacr; break; case 3: *regp = tc; break; case 4: *regp = itt0; break; case 5: *regp = itt1; break; case 6: *regp = dtt0; break; case 7: *regp = dtt1; break; case 0x800: *regp = regs.usp; break; case 0x801: *regp = regs.vbr; break; case 0x802: *regp = caar; break; // case 0x803: *regp = regs.m == 1 ? m68k_areg(regs, 7) : regs.msp; break; // case 0x804: *regp = regs.m == 0 ? m68k_areg(regs, 7) : regs.isp; break; case 0x805: *regp = mmusr; break; case 0x806: *regp = urp; break; case 0x807: *regp = srp; break; default: op_illg(0x4E7A); return 0; } } return 1; } STATIC_INLINE int div_unsigned(uint32_t src_hi, uint32_t src_lo, uint32_t div, uint32_t *quot, uint32_t *rem) { uint32_t q = 0, cbit = 0; int i; if(div <= src_hi) { return 1; } for(i = 0; i < 32; i++) { cbit = src_hi & 0x80000000ul; src_hi <<= 1; if(src_lo & 0x80000000ul) src_hi++; src_lo <<= 1; q = q << 1; if(cbit || div <= src_hi) { q |= 1; src_hi -= div; } } *quot = q; *rem = src_hi; return 0; } void m68k_divl(uint32_t opcode, uint32_t src, uint16_t extra, uintptr_t oldpc) { #if defined(uae_s64) if(src == 0) { Exception(5, oldpc); return; } if(extra & 0x800) { /* signed variant */ uae_s64 a = (uae_s64)(int32_t)m68k_dreg(regs, (extra >> 12) & 7); uae_s64 quot, rem; if(extra & 0x400) { a &= 0xffffffffu; a |= (uae_s64)m68k_dreg(regs, extra & 7) << 32; } rem = a % (uae_s64)(int32_t)src; quot = a / (uae_s64)(int32_t)src; if((quot & UVAL64(0xffffffff80000000)) != 0 && (quot & UVAL64(0xffffffff80000000)) != UVAL64(0xffffffff80000000)) { SET_VFLG(1); SET_NFLG(1); SET_CFLG(0); } else { if(((int32_t)rem < 0) != ((uae_s64)a < 0)) rem = -rem; SET_VFLG(0); SET_CFLG(0); SET_ZFLG(((int32_t)quot) == 0); SET_NFLG(((int32_t)quot) < 0); m68k_dreg(regs, extra & 7) = rem; m68k_dreg(regs, (extra >> 12) & 7) = quot; } } else { /* unsigned */ uae_u64 a = (uae_u64)(uint32_t)m68k_dreg(regs, (extra >> 12) & 7); uae_u64 quot, rem; if(extra & 0x400) { a &= 0xffffffffu; a |= (uae_u64)m68k_dreg(regs, extra & 7) << 32; } rem = a % (uae_u64)src; quot = a / (uae_u64)src; if(quot > 0xffffffffu) { SET_VFLG(1); SET_NFLG(1); SET_CFLG(0); } else { SET_VFLG(0); SET_CFLG(0); SET_ZFLG(((int32_t)quot) == 0); SET_NFLG(((int32_t)quot) < 0); m68k_dreg(regs, extra & 7) = rem; m68k_dreg(regs, (extra >> 12) & 7) = quot; } } #else if(src == 0) { Exception(5, oldpc); return; } if(extra & 0x800) { /* signed variant */ int32_t lo = (int32_t)m68k_dreg(regs, (extra >> 12) & 7); int32_t hi = lo < 0 ? -1 : 0; int32_t save_high; uint32_t quot = 0, rem = 0; uint32_t sign; if(extra & 0x400) { hi = (int32_t)m68k_dreg(regs, extra & 7); } save_high = hi; sign = (hi ^ src); if(hi < 0) { hi = ~hi; lo = -lo; if(lo == 0) hi++; } if((int32_t)src < 0) src = -src; if(div_unsigned(hi, lo, src, ", &rem) || (sign & 0x80000000) ? quot > 0x80000000 : quot > 0x7fffffff) { SET_VFLG(1); SET_NFLG(1); SET_CFLG(0); } else { if(sign & 0x80000000) quot = -quot; if(((int32_t)rem < 0) != (save_high < 0)) rem = -rem; SET_VFLG(0); SET_CFLG(0); SET_ZFLG(((int32_t)quot) == 0); SET_NFLG(((int32_t)quot) < 0); m68k_dreg(regs, extra & 7) = rem; m68k_dreg(regs, (extra >> 12) & 7) = quot; } } else { /* unsigned */ uint32_t lo = (uint32_t)m68k_dreg(regs, (extra >> 12) & 7); uint32_t hi = 0; uint32_t quot, rem; if(extra & 0x400) { hi = (uint32_t)m68k_dreg(regs, extra & 7); } if(div_unsigned(hi, lo, src, ", &rem)) { SET_VFLG(1); SET_NFLG(1); SET_CFLG(0); } else { SET_VFLG(0); SET_CFLG(0); SET_ZFLG(((int32_t)quot) == 0); SET_NFLG(((int32_t)quot) < 0); m68k_dreg(regs, extra & 7) = rem; m68k_dreg(regs, (extra >> 12) & 7) = quot; } } #endif } STATIC_INLINE void mul_unsigned(uint32_t src1, uint32_t src2, uint32_t *dst_hi, uint32_t *dst_lo) { uint32_t r0 = (src1 & 0xffff) * (src2 & 0xffff); uint32_t r1 = ((src1 >> 16) & 0xffff) * (src2 & 0xffff); uint32_t r2 = (src1 & 0xffff) * ((src2 >> 16) & 0xffff); uint32_t r3 = ((src1 >> 16) & 0xffff) * ((src2 >> 16) & 0xffff); uint32_t lo; lo = r0 + ((r1 << 16) & 0xffff0000ul); if(lo < r0) r3++; r0 = lo; lo = r0 + ((r2 << 16) & 0xffff0000ul); if(lo < r0) r3++; r3 += ((r1 >> 16) & 0xffff) + ((r2 >> 16) & 0xffff); *dst_lo = lo; *dst_hi = r3; } void m68k_mull(uint32_t opcode, uint32_t src, uint16_t extra) { #if defined(uae_s64) if(extra & 0x800) { /* signed variant */ uae_s64 a = (uae_s64)(int32_t)m68k_dreg(regs, (extra >> 12) & 7); a *= (uae_s64)(int32_t)src; SET_VFLG(0); SET_CFLG(0); SET_ZFLG(a == 0); SET_NFLG(a < 0); if(extra & 0x400) m68k_dreg(regs, extra & 7) = a >> 32; else if((a & UVAL64(0xffffffff80000000)) != 0 && (a & UVAL64(0xffffffff80000000)) != UVAL64(0xffffffff80000000)) { SET_VFLG(1); } m68k_dreg(regs, (extra >> 12) & 7) = (uint32_t)a; } else { /* unsigned */ uae_u64 a = (uae_u64)(uint32_t)m68k_dreg(regs, (extra >> 12) & 7); a *= (uae_u64)src; SET_VFLG(0); SET_CFLG(0); SET_ZFLG(a == 0); SET_NFLG(((uae_s64)a) < 0); if(extra & 0x400) m68k_dreg(regs, extra & 7) = a >> 32; else if((a & UVAL64(0xffffffff00000000)) != 0) { SET_VFLG(1); } m68k_dreg(regs, (extra >> 12) & 7) = (uint32_t)a; } #else if(extra & 0x800) { /* signed variant */ int32_t src1, src2; uint32_t dst_lo, dst_hi; uint32_t sign; src1 = (int32_t)src; src2 = (int32_t)m68k_dreg(regs, (extra >> 12) & 7); sign = (src1 ^ src2); if(src1 < 0) src1 = -src1; if(src2 < 0) src2 = -src2; mul_unsigned((uint32_t)src1, (uint32_t)src2, &dst_hi, &dst_lo); if(sign & 0x80000000) { dst_hi = ~dst_hi; dst_lo = -dst_lo; if(dst_lo == 0) dst_hi++; } SET_VFLG(0); SET_CFLG(0); SET_ZFLG(dst_hi == 0 && dst_lo == 0); SET_NFLG(((int32_t)dst_hi) < 0); if(extra & 0x400) m68k_dreg(regs, extra & 7) = dst_hi; else if((dst_hi != 0 || (dst_lo & 0x80000000) != 0) && ((dst_hi & 0xffffffff) != 0xffffffff || (dst_lo & 0x80000000) != 0x80000000)) { SET_VFLG(1); } m68k_dreg(regs, (extra >> 12) & 7) = dst_lo; } else { /* unsigned */ uint32_t dst_lo, dst_hi; mul_unsigned(src, (uint32_t)m68k_dreg(regs, (extra >> 12) & 7), &dst_hi, &dst_lo); SET_VFLG(0); SET_CFLG(0); SET_ZFLG(dst_hi == 0 && dst_lo == 0); SET_NFLG(((int32_t)dst_hi) < 0); if(extra & 0x400) m68k_dreg(regs, extra & 7) = dst_hi; else if(dst_hi != 0) { SET_VFLG(1); } m68k_dreg(regs, (extra >> 12) & 7) = dst_lo; } #endif } #ifdef NO_GDB static char *ccnames[] = { "T", "F", "HI", "LS", "CC", "CS", "NE", "EQ", "VC", "VS", "PL", "MI", "GE", "LT", "GT", "LE" }; #endif /* NO_GDB */ void m68k_reset(void) { #if 0 /* The CPU is always the same in TiEmu. */ if (currprefs.cpu_level != changed_prefs.cpu_level || currprefs.address_space_24 != changed_prefs.address_space_24) { currprefs.address_space_24 = changed_prefs.address_space_24; currprefs.cpu_level = changed_prefs.cpu_level; build_cpufunctbl (); } #endif /* 0 */ // regs.kick_mask = 0x00F80000; regs.spcflags = 0; #if 0 if (savestate_state == STATE_RESTORE) { m68k_setpc (regs.pc); /* MakeFromSR() must not swap stack pointer */ regs.s = (regs.sr >> 13) & 1; MakeFromSR(); /* set stack pointer */ if (regs.s) m68k_areg(regs, 7) = regs.isp; else m68k_areg(regs, 7) = regs.usp; return; } #endif /* 0 */ m68k_areg(regs, 7) = tihw.initial_ssp; m68k_setpc(tihw.initial_pc); regs.s = 1; // regs.m = 0; regs.stopped = 0; regs.t1 = 0; // regs.t0 = 0; SET_ZFLG(0); SET_XFLG(0); SET_CFLG(0); SET_VFLG(0); SET_NFLG(0); regs.intmask = 7; regs.vbr = regs.sfc = regs.dfc = 0; regs.fpcr = regs.fpsr = regs.fpiar = 0; regs.irc = 0xffff; fill_prefetch_slow(); } unsigned long op_illg(uint32_t opcode) { #if 0 uintptr_t pc = m68k_getpc (); if (cloanto_rom && (opcode & 0xF100) == 0x7100) { m68k_dreg (regs, (opcode >> 9) & 7) = (int8_t)(opcode & 0xFF); m68k_incpc (2); fill_prefetch_slow (); return 4; } if (opcode == 0x4E7B && get_long (0x10) == 0 && (pc & 0xF80000) == 0xF80000) { write_log ("Your Kickstart requires a 68020 CPU. Giving up.\n"); broken_in = 1; set_special (SPCFLAG_BRK); quit_program = 1; } if (opcode == 0xFF0D) { if ((pc & 0xF80000) == 0xF80000) { /* This is from the dummy Kickstart replacement */ uint16_t arg = get_iword (2); m68k_incpc (4); ersatz_perform (arg); fill_prefetch_slow (); return 4; } else if ((pc & 0xFFFF0000) == RTAREA_BASE) { /* User-mode STOP replacement */ m68k_setstopped (1); return 4; } } if ((opcode & 0xF000) == 0xA000 && (pc & 0xFFFF0000) == RTAREA_BASE) { /* Calltrap. */ m68k_incpc(2); call_calltrap (opcode & 0xFFF); fill_prefetch_slow (); return 4; } #endif /* 0 */ if((opcode & 0xF000) == 0xF000) { Exception(0xB, 0); return 4; } if((opcode & 0xF000) == 0xA000) { #if 0 if ((pc & 0xFFFF0000) == RTAREA_BASE) { /* Calltrap. */ call_calltrap (opcode & 0xFFF); } #endif /* 0 */ #if defined(CYGNUS_SIM) && !defined(NO_GDB) /* Ignore ER_ASAP_TOO_LONG when running a file to debug. */ if(symfile && opcode == 0xA000 + 161) { m68k_incpc(2); fill_prefetch_slow(); return 4; } #endif /* CYGNUS_SIM */ Exception(0xA, 0); return 4; } #if 0 write_log ("Illegal instruction: %04x at %08lx\n", opcode, pc); #endif Exception(4, 0); return 4; } void mmu_op(uint32_t opcode, uint16_t extra) { if((opcode & 0xFE0) == 0x0500) { /* PFLUSH */ mmusr = 0; write_log("PFLUSH\n"); } else if((opcode & 0x0FD8) == 0x548) { /* PTEST */ write_log("PTEST\n"); } else op_illg(opcode); } #if 0 static uintptr_t last_trace_ad = 0; static void do_trace (void) { if (0/*regs.t0 && currprefs.cpu_level >= 2*/) { uint16_t opcode; /* should also include TRAP, CHK, SR modification FPcc */ /* probably never used so why bother */ /* We can afford this to be inefficient... */ m68k_setpc (m68k_getpc ()); fill_prefetch_slow (); opcode = get_word (regs.pc); if (opcode == 0x4e72 /* RTE */ || opcode == 0x4e74 /* RTD */ || opcode == 0x4e75 /* RTS */ || opcode == 0x4e77 /* RTR */ || opcode == 0x4e76 /* TRAPV */ || (opcode & 0xffc0) == 0x4e80 /* JSR */ || (opcode & 0xffc0) == 0x4ec0 /* JMP */ || (opcode & 0xff00) == 0x6100 /* BSR */ || ((opcode & 0xf000) == 0x6000 /* Bcc */ && cctrue((opcode >> 8) & 0xf)) || ((opcode & 0xf0f0) == 0x5050 /* DBcc */ && !cctrue((opcode >> 8) & 0xf) && (int16_t)m68k_dreg(regs, opcode & 7) != 0)) { last_trace_ad = m68k_getpc (); unset_special (SPCFLAG_TRACE); set_special (SPCFLAG_DOTRACE); } } else if (regs.t1) { last_trace_ad = m68k_getpc (); unset_special (SPCFLAG_TRACE); set_special (SPCFLAG_DOTRACE); } } static int do_specialties (int cycles) { if (regs.spcflags & SPCFLAG_COPPER) do_copper (); /*n_spcinsns++;*/ while ((regs.spcflags & SPCFLAG_BLTNASTY) && cycles > 0) { int c = blitnasty(); if (!c) { cycles -= 2 * CYCLE_UNIT; if (cycles < CYCLE_UNIT) cycles = 0; c = 1; } do_cycles (c * CYCLE_UNIT); if (regs.spcflags & SPCFLAG_COPPER) do_copper (); } if (regs.spcflags & SPCFLAG_DOTRACE) Exception (9,last_trace_ad); while (regs.spcflags & SPCFLAG_STOP) { do_cycles (4 * CYCLE_UNIT); if (regs.spcflags & SPCFLAG_COPPER) do_copper (); if (regs.spcflags & (SPCFLAG_INT | SPCFLAG_DOINT)) { int intr = intlev (); unset_special (SPCFLAG_INT | SPCFLAG_DOINT); if (intr != -1 && intr > regs.intmask) { Interrupt (intr); regs.stopped = 0; unset_special (SPCFLAG_STOP); } } } if (regs.spcflags & SPCFLAG_TRACE) do_trace (); if (regs.spcflags & SPCFLAG_DOINT) { int intr = intlev (); unset_special (SPCFLAG_DOINT); if (intr != -1 && intr > regs.intmask) { Interrupt (intr); regs.stopped = 0; } } if (regs.spcflags & SPCFLAG_INT) { unset_special (SPCFLAG_INT); set_special (SPCFLAG_DOINT); } if (regs.spcflags & (SPCFLAG_BRK | SPCFLAG_MODE_CHANGE)) { unset_special (SPCFLAG_BRK | SPCFLAG_MODE_CHANGE); return 1; } return 0; } /* It's really sad to have two almost identical functions for this, but we do it all for performance... :( */ static void m68k_run_1 (void) { for (;;) { int cycles; uint32_t opcode = regs.ir; /* assert (!regs.stopped && !(regs.spcflags & SPCFLAG_STOP)); */ /* regs_backup[backup_pointer = (backup_pointer + 1) % 16] = regs;*/ #if COUNT_INSTRS == 2 if (table68k[opcode].handler != -1) instrcount[table68k[opcode].handler]++; #elif COUNT_INSTRS == 1 instrcount[opcode]++; #endif cycles = (*cpufunctbl[opcode])(opcode); /*n_insns++;*/ cycles &= cycles_mask; cycles |= cycles_val; do_cycles (cycles); if (regs.spcflags) { if (do_specialties (cycles)) return; } } } #define DEBUG_PREFETCH /* Same thing, but don't use prefetch to get opcode. */ static void m68k_run_2 (void) { for (;;) { int cycles; uint32_t opcode = get_iword (0); /* assert (!regs.stopped && !(regs.spcflags & SPCFLAG_STOP)); */ /* regs_backup[backup_pointer = (backup_pointer + 1) % 16] = regs;*/ #if COUNT_INSTRS == 2 if (table68k[opcode].handler != -1) instrcount[table68k[opcode].handler]++; #elif COUNT_INSTRS == 1 instrcount[opcode]++; #endif cycles = (*cpufunctbl[opcode])(opcode); /*n_insns++;*/ cycles &= cycles_mask; cycles |= cycles_val; do_cycles (cycles); if (regs.spcflags) { if (do_specialties (cycles)) return; } } } #define m68k_run1(F) (F)() int in_m68k_go = 0; static void exception2_handle (uintptr_t addr, uintptr_t fault) { last_addr_for_exception_3 = addr; last_fault_for_exception_3 = fault; last_writeaccess_for_exception_3 = 0; last_instructionaccess_for_exception_3 = 0; Exception (2, addr); } void m68k_go (int may_quit) { if (in_m68k_go || !may_quit) { write_log ("Bug! m68k_go is not reentrant.\n"); abort (); } reset_frame_rate_hack (); update_68k_cycles (); in_m68k_go++; for (;;) { if (quit_program > 0) { if (quit_program == 1) break; quit_program = 0; if (savestate_state == STATE_RESTORE) { restore_state (savestate_filename); #if 0 activate_debugger (); #endif } m68k_reset (); reset_all_systems (); customreset (); /* We may have been restoring state, but we're done now. */ savestate_restore_finish (); fill_prefetch_slow (); handle_active_events (); if (regs.spcflags) do_specialties (0); m68k_setpc (regs.pc); } if (debugging) debug (); if (regs.panic) { regs.panic = 0; /* program jumped to non-existing memory and cpu was >= 68020 */ get_real_address (regs.isp); /* stack in no one's land? -> reboot */ if (regs.isp & 1) regs.panic = 1; if (!regs.panic) exception2_handle (regs.panic_pc, regs.panic_addr); if (regs.panic) { /* system is very badly confused */ write_log ("double bus error or corrupted stack, forcing reboot..\n"); regs.panic = 0; uae_reset (1); } } m68k_run1 (currprefs.cpu_level == 0 ? m68k_run_1 : m68k_run_2); } in_m68k_go--; } static void m68k_verify (uintptr_t addr, uintptr_t *nextpc) { uint32_t opcode, val; struct instr *dp; opcode = get_iword_1(0); last_op_for_exception_3 = opcode; m68kpc_offset = 2; if (cpufunctbl[opcode] == op_illg_1) { opcode = 0x4AFC; } dp = table68k + opcode; if (dp->suse) { if (!verify_ea (dp->sreg, dp->smode, dp->size, &val)) { Exception (3, 0); return; } } if (dp->duse) { if (!verify_ea (dp->dreg, dp->dmode, dp->size, &val)) { Exception (3, 0); return; } } } #endif /* 0 */ #ifdef NO_GDB int DasmFPU(uint16_t code, char *buf); int m68k_disasm(char *output, uintptr_t addr) { char buf[273]; uintptr_t newpc = 0; char instrname[20], *ccpt; uint32_t opcode; struct mnemolookup *lookup; struct instr *dp; uint32_t orig_opcode; uintptr_t nextpc; m68kpc_offset = addr - m68k_getpc(); output[0] = '\0'; sprintf(buf, "%06lx: ", m68k_getpc() + m68kpc_offset); strcat(output, buf); opcode = get_iword_1(m68kpc_offset); orig_opcode = opcode; m68kpc_offset += 2; if(cpufunctbl[opcode] == op_illg_1 || (orig_opcode & 0xf000) == 0xa000 || (orig_opcode & 0xf000) == 0xf000) { if(!((orig_opcode >= 0xf800) && (orig_opcode <= 0xfff2))) opcode = 0x4AFC; } dp = table68k + opcode; for(lookup = lookuptab; lookup->mnemo != dp->mnemo; lookup++) ; strcpy(instrname, lookup->name); ccpt = strstr(instrname, "cc"); if(ccpt != 0) { strncpy(ccpt, ccnames[dp->cc], 2); } strcat(output, instrname); switch(dp->size) { case sz_byte: strcat(output, ".B "); break; case sz_word: strcat(output, ".W "); break; case sz_long: strcat(output, ".L "); break; default: strcat(output, " "); break; } if(dp->suse) { newpc = m68k_getpc() + m68kpc_offset; newpc += ShowEA(0, dp->sreg, dp->smode, dp->size, output); } if(dp->suse && dp->duse) strcat(output, ","); if(dp->duse) { newpc = m68k_getpc() + m68kpc_offset; newpc += ShowEA(0, dp->dreg, dp->dmode, dp->size, output); } if(ccpt != 0) { if(cctrue(dp->cc)) { sprintf(buf, " [%06lX] (TRUE)", newpc); strcat(output, buf); } else { sprintf(buf, " [%06lX] (FALSE)", newpc); strcat(output, buf); } } else if((opcode & 0xff00) == 0x6100) { /* BSR */ sprintf(buf, " [%s]", sym_addr(newpc)); strcat(output, buf); } else if((orig_opcode >= 0xf800) && (orig_opcode <= 0xfff2)) { char *buffer = &(output[8]); unsigned long pm; uint32_t pc = m68k_getpc(); /* addr */ /* F-Line ROM calls (see KerNO doc and thanks to Lionel Debroux) */ switch(orig_opcode) { case 0xfff0: /* 6 byte bsr w/long word displacement */ pm = get_ilong_1(m68kpc_offset); m68kpc_offset += 6 - 2; if(pm & 0x8000) sprintf(buffer, "FLINE bsr.l *-$%lX [%lX]", (-(signed long)(int32_t)pm) - 2, pc + (signed long)(int32_t)pm + 2); else sprintf(buffer, "FLINE bsr.l *+$%lX [%lX]", pm + 2, pc + pm + 2); break; case 0xfff1: /* 6 byte bra w/long word displacement */ pm = get_ilong_1(m68kpc_offset); m68kpc_offset += 6 - 2; if(pm & 0x8000) sprintf(buffer, "FLINE bra.l *-$%lX [%lX]", (-(signed long)(int32_t)pm) - 2, pc + (signed long)(int32_t)pm + 2); else sprintf(buffer, "FLINE bra.l *+$%lX [%lX]", pm + 2, pc + pm + 2); break; case 0xfff2: /* 4 byte ROM CALL */ pm = get_iword_1(m68kpc_offset); m68kpc_offset += 4 - 2; sprintf(buffer, "FLINE $%04x.l [%s]", pm / 4, romcalls_get_name(pm / 4)); break; case 0xffee: /* jmp __ld_entry_point_plus_0x8000+word */ pm = get_iword_1(m68kpc_offset); m68kpc_offset += 4 - 2; { int handle; uint32_t addr; heap_search_for_address(pc + 2, &handle); if(handle > 0) heap_get_block_addr(handle, &addr); else addr = 0; sprintf(buffer, "FLINE jmp.w *+$%lX [%lX]", (signed long)(signed short)pm + 0x8000, addr + (signed long)(signed short)pm + 0x8000); } break; case 0xffef: /* jsr __ld_entry_point_plus_0x8000+word */ pm = get_iword_1(m68kpc_offset); m68kpc_offset += 4 - 2; { int handle; uint32_t addr; heap_search_for_address(pc + 2, &handle); if(handle > 0) heap_get_block_addr(handle, &addr); else addr = 0; sprintf(buffer, "FLINE jsr.w *+$%lX [%lX]", (signed long)(signed short)pm + 0x8000, addr + (signed long)(signed short)pm + 0x8000); } break; case 0xf8b5: /* 2 byte ROM call followed by an FPU opcode (special case: _bcd_math) */ { char tmp[64]; pm = get_iword_1(m68kpc_offset); m68kpc_offset += 4 - 2; DasmFPU(pm, tmp); sprintf(buffer, "FLINE _bcd_math (FPU: %s)", tmp); break; } default: /* 2 byte ROM CALL */ sprintf(buffer, "FLINE $%03x.w [%s]", opcode & 0x7ff, romcalls_get_name(opcode & 0x7ff)); break; } } else if((orig_opcode & 0xf000) == 0xa000) { /* ER_throw */ char *buffer = &(output[8]); sprintf(buffer, "ER_throw %d [%s]", opcode & 0xfff, ercodes_get_name(opcode & 0xfff)); } else if(opcode == 0x4AFC && orig_opcode != 0x4AFC) { /* illegal instruction, but not ILLEGAL */ sprintf(output, "%06lx: DC.W $%04X", addr, orig_opcode); } nextpc = m68k_getpc() + m68kpc_offset; return (nextpc - addr); } #endif /* NO_GDB */ #if 0 void m68k_dumpstate (FILE *f, uintptr_t *nextpc) { int i; for (i = 0; i < 8; i++){ fprintf (f, "D%d: %08lx ", i, m68k_dreg(regs, i)); if ((i & 3) == 3) fprintf (f, "\n"); } for (i = 0; i < 8; i++){ fprintf (f, "A%d: %08lx ", i, m68k_areg(regs, i)); if ((i & 3) == 3) fprintf (f, "\n"); } if (regs.s == 0) regs.usp = m68k_areg(regs, 7); if (regs.s && regs.m) regs.msp = m68k_areg(regs, 7); if (regs.s && regs.m == 0) regs.isp = m68k_areg(regs, 7); fprintf (f, "USP=%08lx ISP=%08lx MSP=%08lx VBR=%08lx\n", regs.usp,regs.isp,regs.msp,regs.vbr); fprintf (f, "T=%d%d S=%d M=%d X=%d N=%d Z=%d V=%d C=%d IMASK=%d\n", regs.t1, regs.t0, regs.s, regs.m, GET_XFLG, GET_NFLG, GET_ZFLG, GET_VFLG, GET_CFLG, regs.intmask); for (i = 0; i < 8; i++){ fprintf (f, "FP%d: %g ", i, regs.fp[i]); if ((i & 3) == 3) fprintf (f, "\n"); } fprintf (f, "N=%d Z=%d I=%d NAN=%d\n", (regs.fpsr & 0x8000000) != 0, (regs.fpsr & 0x4000000) != 0, (regs.fpsr & 0x2000000) != 0, (regs.fpsr & 0x1000000) != 0); m68k_disasm (f, m68k_getpc (), nextpc, 1); if (nextpc) fprintf (f, "next PC: %08lx\n", *nextpc); } /* CPU save/restore code */ #define CPUTYPE_EC 1 #define CPUMODE_HALT 1 uint8_t *restore_cpu (uint8_t *src) { int i,model,flags; uint32_t l; model = restore_u32(); switch (model) { case 68000: currprefs.cpu_level = 0; break; case 68010: currprefs.cpu_level = 1; break; case 68020: currprefs.cpu_level = 2; break; default: write_log ("Unknown cpu type %d\n", model); break; } flags = restore_u32(); currprefs.address_space_24 = 0; if (flags & CPUTYPE_EC) currprefs.address_space_24 = 1; for (i = 0; i < 15; i++) regs.regs[i] = restore_u32 (); regs.pc = restore_u32 (); regs.irc = restore_u16 (); regs.ir = restore_u16 (); regs.usp = restore_u32 (); regs.isp = restore_u32 (); regs.sr = restore_u16 (); l = restore_u32(); if (l & CPUMODE_HALT) { regs.stopped = 1; set_special (SPCFLAG_STOP); } else regs.stopped = 0; if (model >= 68010) { regs.dfc = restore_u32 (); regs.sfc = restore_u32 (); regs.vbr = restore_u32 (); } if (model >= 68020) { caar = restore_u32 (); cacr = restore_u32 (); regs.msp = restore_u32 (); } write_log ("CPU %d%s%03d, PC=%08.8X\n", model/1000, flags & 1 ? "EC" : "", model % 1000, regs.pc); return src; } static int cpumodel[] = { 68000, 68010, 68020, 68020 }; uint8_t *save_cpu (int *len, uint8_t *dstptr) { uint8_t *dstbak,*dst; int model,i; if (dstptr) dstbak = dst = dstptr; else dstbak = dst = malloc(4+4+15*4+4+4+4+4+2+4+4+4+4+4+4+4); model = cpumodel[currprefs.cpu_level]; save_u32 (model); /* MODEL */ save_u32 (currprefs.address_space_24 ? 1 : 0); /* FLAGS */ for(i = 0;i < 15; i++) save_u32 (regs.regs[i]); /* D0-D7 A0-A6 */ save_u32 (m68k_getpc ()); /* PC */ save_u16 (regs.irc); /* prefetch */ save_u16 (regs.ir); /* instruction prefetch */ MakeSR (); save_u32 (!regs.s ? regs.regs[15] : regs.usp); /* USP */ save_u32 (regs.s ? regs.regs[15] : regs.isp); /* ISP */ save_u16 (regs.sr); /* SR/CCR */ save_u32 (regs.stopped ? CPUMODE_HALT : 0); /* flags */ if(model >= 68010) { save_u32 (regs.dfc); /* DFC */ save_u32 (regs.sfc); /* SFC */ save_u32 (regs.vbr); /* VBR */ } if(model >= 68020) { save_u32 (caar); /* CAAR */ save_u32 (cacr); /* CACR */ save_u32 (regs.msp); /* MSP */ } *len = dst - dstbak; return dstbak; } #endif /* 0 */ static void exception3f(uint32_t opcode, uintptr_t addr, uintptr_t fault, int writeaccess, int instructionaccess) { last_addr_for_exception_3 = addr; last_fault_for_exception_3 = fault; last_op_for_exception_3 = opcode; last_writeaccess_for_exception_3 = writeaccess; last_instructionaccess_for_exception_3 = instructionaccess; Exception(3, fault); } void exception3(uint32_t opcode, uintptr_t addr, uintptr_t fault) { exception3f(opcode, addr, fault, 0, 0); } void exception3i(uint32_t opcode, uintptr_t addr, uintptr_t fault) { exception3f(opcode, addr, fault, 0, 1); } #if 0 void exception2 (uintptr_t addr, uintptr_t fault) { write_log ("delayed exception2!\n"); regs.panic_pc = m68k_getpc(); regs.panic_addr = addr; regs.panic = 2; set_special (SPCFLAG_BRK); m68k_setpc (0xf80000); fill_prefetch_slow (); } void cpureset (void) { customreset (); #if 0 uint16_t ins; if (currprefs.cpu_level == 0 && (currprefs.cpu_compatible || currprefs.cpu_cycle_exact)) { customreset (); return; } ins = get_word (m68k_getpc() + 2); if ((ins & ~7) == 0x4ed0) { int reg = ins & 7; uint32_t addr = m68k_areg (regs, reg); write_log ("reset/jmp (ax) combination emulated\n"); customreset (); if (addr < 0x80000) addr += 0xf80000; m68k_setpc (addr); } #endif } #endif