#include "ultra64.h" #include "global.h" #include "vt.h" // data const char* sCpuExceptions[] = { "Interrupt", "TLB modification", "TLB exception on load", "TLB exception on store", "Address error on load", "Address error on store", "Bus error on inst.", "Bus error on data", "System call exception", "Breakpoint exception", "Reserved instruction", "Coprocessor unusable", "Arithmetic overflow", "Trap exception", "Virtual coherency on inst.", "Floating point exception", "Watchpoint exception", "Virtual coherency on data", }; const char* sFpuExceptions[] = { "Unimplemented operation", "Invalid operation", "Division by zero", "Overflow", "Underflow", "Inexact operation", }; void Fault_SleepImpl(u32 duration) { u64 value = (duration * OS_CPU_COUNTER) / 1000ull; Sleep_Cycles(value); } void Fault_AddClient(FaultClient* client, fault_client_func callback, void* param0, void* param1) { OSIntMask mask; u32 alreadyExists = 0; mask = osSetIntMask(1); { FaultClient* iter = sFaultContext->clients; while (iter) { if (iter == client) { alreadyExists = 1; goto end; } iter = iter->next; } } client->callback = callback; client->param0 = param0; client->param1 = param1; client->next = sFaultContext->clients; sFaultContext->clients = client; end: osSetIntMask(mask); if (alreadyExists) { osSyncPrintf(VT_COL(RED, WHITE) "fault_AddClient: %08x は既にリスト中にある\n" VT_RST, client); } } void Fault_RemoveClient(FaultClient* client) { FaultClient* iter; FaultClient* lastIter; OSIntMask mask; u32 listIsEmpty; iter = sFaultContext->clients; listIsEmpty = 0; lastIter = NULL; mask = osSetIntMask(1); while (iter) { if (iter == client) { if (lastIter) { lastIter->next = client->next; } else { sFaultContext->clients = client; if (sFaultContext->clients) { sFaultContext->clients = client->next; } else { listIsEmpty = 1; } } break; } lastIter = iter; iter = iter->next; } osSetIntMask(mask); if (listIsEmpty) { osSyncPrintf(VT_COL(RED, WHITE) "fault_RemoveClient: %08x リスト不整合です\n" VT_RST, client); } } void Fault_AddAddrConvClient(FaultAddrConvClient* client, FaultAddrConvFunc callback, void* param) { OSIntMask mask; u32 alreadyExists = 0; mask = osSetIntMask(1); { FaultAddrConvClient* iter = sFaultContext->addrConvClients; while (iter) { if (iter == client) { alreadyExists = 1; goto end; } iter = iter->next; } } client->callback = callback; client->param = param; client->next = sFaultContext->addrConvClients; sFaultContext->addrConvClients = client; end: osSetIntMask(mask); if (alreadyExists) { osSyncPrintf(VT_COL(RED, WHITE) "fault_AddressConverterAddClient: %08x は既にリスト中にある\n" VT_RST, client); } } void Fault_RemoveAddrConvClient(FaultAddrConvClient* client) { FaultAddrConvClient* iter; FaultAddrConvClient* lastIter; OSIntMask mask; u32 listIsEmpty; iter = sFaultContext->addrConvClients; listIsEmpty = 0; lastIter = NULL; mask = osSetIntMask(1); while (iter) { if (iter == client) { if (lastIter) { lastIter->next = client->next; } else { sFaultContext->addrConvClients = client; if (sFaultContext->addrConvClients) { sFaultContext->addrConvClients = client->next; } else { listIsEmpty = 1; } } break; } lastIter = iter; iter = iter->next; } osSetIntMask(mask); if (listIsEmpty) { osSyncPrintf(VT_COL(RED, WHITE) "fault_AddressConverterRemoveClient: %08x は既にリスト中にある\n" VT_RST, client); } } void* Fault_ConvertAddress(void* addr) { void* ret; FaultAddrConvClient* iter = sFaultContext->addrConvClients; while (iter) { if (iter->callback) { ret = iter->callback(addr, iter->param); if (ret != NULL) { return ret; } } iter = iter->next; } return NULL; } void Fault_Sleep(u32 duration) { Fault_SleepImpl(duration); } void Fault_PadCallback(Input* input) { Padmgr_GetInput2(input, 0); } void Fault_UpdatePadImpl() { sFaultContext->padCallback(sFaultContext->padInput); } u32 Fault_WaitForInputImpl() { Input* curInput = &sFaultContext->padInput[0]; s32 count = 600; u32 kDown; while (1) { Fault_Sleep(0x10); Fault_UpdatePadImpl(); kDown = curInput->press.button; if (kDown == BTN_L) { sFaultContext->faultActive = !sFaultContext->faultActive; } if (sFaultContext->faultActive) { if (count-- < 1) { return 0; } } else { if (kDown == BTN_A || kDown == BTN_DRIGHT) { return 0; } if (kDown == BTN_DLEFT) { return 1; } if (kDown == BTN_DUP) { FaultDrawer_SetOsSyncPrintfEnabled(1); } if (kDown == BTN_DDOWN) { FaultDrawer_SetOsSyncPrintfEnabled(0); } } } } void Fault_WaitForInput() { Fault_WaitForInputImpl(); } void Fault_DrawRec(s32 x, s32 y, s32 w, s32 h, u16 color) { FaultDrawer_DrawRecImpl(x, y, x + w - 1, y + h - 1, color); } void Fault_FillScreenBlack() { FaultDrawer_SetForeColor(0xFFFF); FaultDrawer_SetBackColor(1); FaultDrawer_FillScreen(); FaultDrawer_SetBackColor(0); } void Fault_FillScreenRed() { FaultDrawer_SetForeColor(0xFFFF); FaultDrawer_SetBackColor(0xF001); FaultDrawer_FillScreen(); FaultDrawer_SetBackColor(0); } void Fault_DrawCornerRec(u16 color) { Fault_DrawRec(0x16, 0x10, 8, 1, color); } void Fault_PrintFReg(s32 idx, f32* value) { u32 raw = *(u32*)value; s32 v0 = ((raw & 0x7f800000) >> 0x17) - 0x7f; if ((v0 >= -0x7e && v0 < 0x80) || raw == 0) { FaultDrawer_Printf("F%02d:%14.7e ", idx, *value); } else { FaultDrawer_Printf("F%02d: %08x(16) ", idx, raw); } } void osSyncPrintfFReg(s32 idx, f32* value) { u32 raw = *(u32*)value; s32 v0 = ((raw & 0x7F800000) >> 0x17) - 0x7F; if ((v0 >= -0x7E && v0 < 0x80) || raw == 0) { osSyncPrintf("F%02d:%14.7e ", idx, *value); } else { osSyncPrintf("F%02d: %08x(16) ", idx, *(u32*)value); } } void Fault_PrintFPCR(u32 value) { s32 i; u32 flag = 0x20000; FaultDrawer_Printf("FPCSR:%08xH ", value); for (i = 0; i < ARRAY_COUNT(sFpuExceptions); i++) { if (value & flag) { FaultDrawer_Printf("(%s)", sFpuExceptions[i]); break; } flag >>= 1; } FaultDrawer_Printf("\n"); } void osSyncPrintfFPCR(u32 value) { s32 i; u32 flag = 0x20000; osSyncPrintf("FPCSR:%08xH ", value); for (i = 0; i < ARRAY_COUNT(sFpuExceptions); i++) { if (value & flag) { osSyncPrintf("(%s)\n", sFpuExceptions[i]); break; } flag >>= 1; } } void Fault_PrintThreadContext(OSThread* t) { __OSThreadContext* ctx; s32 causeStrIdx = (s32)((((u32)t->context.cause >> 2) & 0x1f) << 0x10) >> 0x10; if (causeStrIdx == 0x17) { causeStrIdx = 0x10; } if (causeStrIdx == 0x1F) { causeStrIdx = 0x11; } FaultDrawer_FillScreen(); FaultDrawer_SetCharPad(-2, 4); FaultDrawer_SetCursor(0x16, 0x14); ctx = &t->context; FaultDrawer_Printf("THREAD:%d (%d:%s)\n", t->id, causeStrIdx, sCpuExceptions[causeStrIdx]); FaultDrawer_SetCharPad(-1, 0); FaultDrawer_Printf("PC:%08xH SR:%08xH VA:%08xH\n", (u32)ctx->pc, (u32)ctx->sr, (u32)ctx->badvaddr); FaultDrawer_Printf("AT:%08xH V0:%08xH V1:%08xH\n", (u32)ctx->at, (u32)ctx->v0, (u32)ctx->v1); FaultDrawer_Printf("A0:%08xH A1:%08xH A2:%08xH\n", (u32)ctx->a0, (u32)ctx->a1, (u32)ctx->a2); FaultDrawer_Printf("A3:%08xH T0:%08xH T1:%08xH\n", (u32)ctx->a3, (u32)ctx->t0, (u32)ctx->t1); FaultDrawer_Printf("T2:%08xH T3:%08xH T4:%08xH\n", (u32)ctx->t2, (u32)ctx->t3, (u32)ctx->t4); FaultDrawer_Printf("T5:%08xH T6:%08xH T7:%08xH\n", (u32)ctx->t5, (u32)ctx->t6, (u32)ctx->t7); FaultDrawer_Printf("S0:%08xH S1:%08xH S2:%08xH\n", (u32)ctx->s0, (u32)ctx->s1, (u32)ctx->s2); FaultDrawer_Printf("S3:%08xH S4:%08xH S5:%08xH\n", (u32)ctx->s3, (u32)ctx->s4, (u32)ctx->s5); FaultDrawer_Printf("S6:%08xH S7:%08xH T8:%08xH\n", (u32)ctx->s6, (u32)ctx->s7, (u32)ctx->t8); FaultDrawer_Printf("T9:%08xH GP:%08xH SP:%08xH\n", (u32)ctx->t9, (u32)ctx->gp, (u32)ctx->sp); FaultDrawer_Printf("S8:%08xH RA:%08xH LO:%08xH\n\n", (u32)ctx->s8, (u32)ctx->ra, (u32)ctx->lo); Fault_PrintFPCR(ctx->fpcsr); FaultDrawer_Printf("\n"); Fault_PrintFReg(0, &ctx->fp0.f.f_even); Fault_PrintFReg(2, &ctx->fp2.f.f_even); FaultDrawer_Printf("\n"); Fault_PrintFReg(4, &ctx->fp4.f.f_even); Fault_PrintFReg(6, &ctx->fp6.f.f_even); FaultDrawer_Printf("\n"); Fault_PrintFReg(8, &ctx->fp8.f.f_even); Fault_PrintFReg(0xA, &ctx->fp10.f.f_even); FaultDrawer_Printf("\n"); Fault_PrintFReg(0xC, &ctx->fp12.f.f_even); Fault_PrintFReg(0xE, &ctx->fp14.f.f_even); FaultDrawer_Printf("\n"); Fault_PrintFReg(0x10, &ctx->fp16.f.f_even); Fault_PrintFReg(0x12, &ctx->fp18.f.f_even); FaultDrawer_Printf("\n"); Fault_PrintFReg(0x14, &ctx->fp20.f.f_even); Fault_PrintFReg(0x16, &ctx->fp22.f.f_even); FaultDrawer_Printf("\n"); Fault_PrintFReg(0x18, &ctx->fp24.f.f_even); Fault_PrintFReg(0x1A, &ctx->fp26.f.f_even); FaultDrawer_Printf("\n"); Fault_PrintFReg(0x1C, &ctx->fp28.f.f_even); Fault_PrintFReg(0x1E, &ctx->fp30.f.f_even); FaultDrawer_Printf("\n"); FaultDrawer_SetCharPad(0, 0); if (D_8009BE54 != 0) { FaultDrawer_DrawText(0xA0, 0xD8, "%5.2f sec\n", D_8009BE54); } } void osSyncPrintfThreadContext(OSThread* t) { __OSThreadContext* ctx; s32 causeStrIdx = (s32)((((u32)t->context.cause >> 2) & 0x1f) << 0x10) >> 0x10; if (causeStrIdx == 0x17) { causeStrIdx = 0x10; } if (causeStrIdx == 0x1f) { causeStrIdx = 0x11; } ctx = &t->context; osSyncPrintf("\n"); osSyncPrintf("THREAD ID:%d (%d:%s)\n", t->id, causeStrIdx, sCpuExceptions[causeStrIdx]); osSyncPrintf("PC:%08xH SR:%08xH VA:%08xH\n", (u32)ctx->pc, (u32)ctx->sr, (u32)ctx->badvaddr); osSyncPrintf("AT:%08xH V0:%08xH V1:%08xH\n", (u32)ctx->at, (u32)ctx->v0, (u32)ctx->v1); osSyncPrintf("A0:%08xH A1:%08xH A2:%08xH\n", (u32)ctx->a0, (u32)ctx->a1, (u32)ctx->a2); osSyncPrintf("A3:%08xH T0:%08xH T1:%08xH\n", (u32)ctx->a3, (u32)ctx->t0, (u32)ctx->t1); osSyncPrintf("T2:%08xH T3:%08xH T4:%08xH\n", (u32)ctx->t2, (u32)ctx->t3, (u32)ctx->t4); osSyncPrintf("T5:%08xH T6:%08xH T7:%08xH\n", (u32)ctx->t5, (u32)ctx->t6, (u32)ctx->t7); osSyncPrintf("S0:%08xH S1:%08xH S2:%08xH\n", (u32)ctx->s0, (u32)ctx->s1, (u32)ctx->s2); osSyncPrintf("S3:%08xH S4:%08xH S5:%08xH\n", (u32)ctx->s3, (u32)ctx->s4, (u32)ctx->s5); osSyncPrintf("S6:%08xH S7:%08xH T8:%08xH\n", (u32)ctx->s6, (u32)ctx->s7, (u32)ctx->t8); osSyncPrintf("T9:%08xH GP:%08xH SP:%08xH\n", (u32)ctx->t9, (u32)ctx->gp, (u32)ctx->sp); osSyncPrintf("S8:%08xH RA:%08xH LO:%08xH\n", (u32)ctx->s8, (u32)ctx->ra, (u32)ctx->lo); osSyncPrintf("\n"); osSyncPrintfFPCR(ctx->fpcsr); osSyncPrintf("\n"); osSyncPrintfFReg(0, &ctx->fp0.f.f_even); osSyncPrintfFReg(2, &ctx->fp2.f.f_even); osSyncPrintf("\n"); osSyncPrintfFReg(4, &ctx->fp4.f.f_even); osSyncPrintfFReg(6, &ctx->fp6.f.f_even); osSyncPrintf("\n"); osSyncPrintfFReg(8, &ctx->fp8.f.f_even); osSyncPrintfFReg(0xa, &ctx->fp10.f.f_even); osSyncPrintf("\n"); osSyncPrintfFReg(0xc, &ctx->fp12.f.f_even); osSyncPrintfFReg(0xe, &ctx->fp14.f.f_even); osSyncPrintf("\n"); osSyncPrintfFReg(0x10, &ctx->fp16.f.f_even); osSyncPrintfFReg(0x12, &ctx->fp18.f.f_even); osSyncPrintf("\n"); osSyncPrintfFReg(0x14, &ctx->fp20.f.f_even); osSyncPrintfFReg(0x16, &ctx->fp22.f.f_even); osSyncPrintf("\n"); osSyncPrintfFReg(0x18, &ctx->fp24.f.f_even); osSyncPrintfFReg(0x1a, &ctx->fp26.f.f_even); osSyncPrintf("\n"); osSyncPrintfFReg(0x1c, &ctx->fp28.f.f_even); osSyncPrintfFReg(0x1e, &ctx->fp30.f.f_even); osSyncPrintf("\n"); } OSThread* Fault_FindFaultedThread() { OSThread* iter = __osGetActiveQueue(); while (iter->priority != -1) { if (iter->priority > 0 && iter->priority < 0x7f && (iter->flags & 3)) { return iter; } iter = iter->tlnext; } return NULL; } void Fault_Wait5Seconds(void) { u32 pad; OSTime start; start = osGetTime(); do { Fault_Sleep(0x10); } while ((osGetTime() - start) <= OS_USEC_TO_CYCLES(5000000)); sFaultContext->faultActive = 1; } void Fault_WaitForButtonCombo(void) { Input* input = &sFaultContext->padInput[0]; FaultDrawer_SetForeColor(0xffff); FaultDrawer_SetBackColor(1); do { do { Fault_Sleep(0x10); Fault_UpdatePadImpl(); } while (!CHECK_BTN_ALL(input->press.button, 0x80)); } while (!CHECK_BTN_ALL(input->cur.button, BTN_DLEFT | BTN_L | BTN_R | BTN_CRIGHT)); } void Fault_DrawMemDumpPage(const char* title, u32* addr, u32 param_3) { u32* alignedAddr; u32* writeAddr; s32 y; s32 x; alignedAddr = addr; if (alignedAddr < (u32*)0x80000000) { alignedAddr = (u32*)0x80000000; } if (alignedAddr > (u32*)0x807fff00) { alignedAddr = (u32*)0x807fff00; } alignedAddr = (u32*)((u32)alignedAddr & ~3); writeAddr = alignedAddr; Fault_FillScreenBlack(); FaultDrawer_SetCharPad(-2, 0); FaultDrawer_DrawText(0x24, 0x12, "%s %08x", title ? title : "PrintDump", alignedAddr); if (alignedAddr >= (u32*)0x80000000 && alignedAddr < (u32*)0xC0000000) { for (y = 0x1C; y != 0xE2; y += 9) { FaultDrawer_DrawText(0x18, y, "%06x", writeAddr); for (x = 0x52; x != 0x122; x += 0x34) { FaultDrawer_DrawText(x, y, "%08x", *writeAddr++); } } } FaultDrawer_SetCharPad(0, 0); } void Fault_DrawMemDump(u32 pc, u32 sp, u32 unk0, u32 unk1) { s32 count; s32 off; Input* input = &sFaultContext->padInput[0]; u32 addr = pc; do { count = 0; if (addr < 0x80000000) { addr = 0x80000000; } if (addr > 0x807fff00) { addr = 0x807fff00; } addr &= ~0xF; Fault_DrawMemDumpPage("Dump", (u32*)addr, 0); count = 600; while (sFaultContext->faultActive) { if (count == 0) { return; } count--; Fault_Sleep(0x10); Fault_UpdatePadImpl(); if (CHECK_BTN_ALL(input->press.button, BTN_L)) { sFaultContext->faultActive = 0; } } do { Fault_Sleep(0x10); Fault_UpdatePadImpl(); } while (input->press.button == 0); if (CHECK_BTN_ALL(input->press.button, BTN_START)) { return; } off = 0x10; if (CHECK_BTN_ALL(input->cur.button, BTN_A)) { off = 0x100; } if (CHECK_BTN_ALL(input->cur.button, BTN_B)) { off <<= 8; } if (CHECK_BTN_ALL(input->press.button, BTN_DUP)) { addr -= off; } if (CHECK_BTN_ALL(input->press.button, BTN_DDOWN)) { addr += off; } if (CHECK_BTN_ALL(input->press.button, BTN_CUP)) { addr = pc; } if (CHECK_BTN_ALL(input->press.button, BTN_CDOWN)) { addr = sp; } if (CHECK_BTN_ALL(input->press.button, BTN_CLEFT)) { addr = unk0; } if (CHECK_BTN_ALL(input->press.button, BTN_CRIGHT)) { addr = unk1; } } while (!CHECK_BTN_ALL(input->press.button, BTN_L)); sFaultContext->faultActive = 1; } #ifdef NON_MATCHING // This function still needs a bit of work void Fault_FindNextStackCall(u32** sp, u32** pc, u32** ra) { u32* currentSp; u32* currentPc; u32* currentRa; u32 lastInst; u32 currInst; currentSp = *sp; currentPc = *pc; currentRa = *ra; if ((((u32)currentSp & 3) != 0) || (currentSp < (u32*)0x80000000) || (currentSp >= (u32*)0xC0000000) || (((u32)currentRa & 3) != 0) || (currentRa < (u32*)0x80000000) || (currentRa >= (u32*)0xC0000000)) { *sp = NULL; *pc = NULL; *ra = NULL; return; } if ((((u32)currentPc & 3) != 0) || (currentPc < (u32*)0x80000000) || (currentPc >= (u32*)0xC0000000)) { *pc = currentRa; return; } lastInst = 0; while (1) { currInst = *currentPc; if (((currInst >> 0x10) & 0xFFFF) == 0x8FBF) { currentRa = *(u32**)((u32)currentSp + (s16)currInst); } else if (((currInst >> 0x10) & 0xFFFF) == 0x27BD) { currentSp = (u32*)((u32)currentSp + (s16)currInst); } else if (currInst == 0x42000018) { currentSp = NULL; currentPc = NULL; currentRa = NULL; break; } if (lastInst == 0x03E00008) { break; } if ((lastInst >> 0x1A) == 2) { currentPc = (u32*)((((u32)currentPc >> 0x1C) << 0x1C) | ((lastInst << 6) >> 4)); break; } lastInst = currInst; currentPc++; } *sp = currentSp; *pc = currentPc; *ra = currentRa; } #else #pragma GLOBAL_ASM("asm/non_matchings/boot/fault/Fault_FindNextStackCall.s") #endif void Fault_DrawStackTrace(OSThread* t, u32 flags) { s32 y; u32 sp; u32 ra; u32 pc; u32 pad; u32 convertedPc; sp = t->context.sp; ra = t->context.ra; pc = t->context.pc; Fault_FillScreenBlack(); FaultDrawer_DrawText(0x78, 0x10, "STACK TRACE"); FaultDrawer_DrawText(0x24, 0x18, "SP PC (VPC)"); for (y = 1; (y < 22) && (((ra != 0) || (sp != 0)) && (pc != (u32)__osCleanupThread)); y++) { FaultDrawer_DrawText(0x24, y * 8 + 24, "%08x %08x", sp, pc); if (flags & 1) { convertedPc = (u32)Fault_ConvertAddress((void*)pc); if (convertedPc != 0) { FaultDrawer_Printf(" -> %08x", convertedPc); } } else { FaultDrawer_Printf(" -> ????????"); } Fault_FindNextStackCall((u32**)&sp, (u32**)&pc, (u32**)&ra); } } void osSyncPrintfStackTrace(OSThread* t, u32 flags) { s32 y; u32 sp; u32 ra; u32 pc; u32 convertedPc; sp = t->context.sp; ra = t->context.ra; pc = t->context.pc; osSyncPrintf("STACK TRACE"); osSyncPrintf("SP PC (VPC)\n"); for (y = 1; (y < 22) && (((ra != 0) || (sp != 0)) && (pc != (u32)__osCleanupThread)); y++) { osSyncPrintf("%08x %08x", sp, pc); if (flags & 1) { convertedPc = (u32)Fault_ConvertAddress((void*)pc); if (convertedPc != 0) { osSyncPrintf(" -> %08x", convertedPc); } } else { osSyncPrintf(" -> ????????"); } osSyncPrintf("\n"); Fault_FindNextStackCall((u32**)&sp, (u32**)&pc, (u32**)&ra); } } void Fault_ResumeThread(OSThread* t) { t->context.cause = 0; t->context.fpcsr = 0; t->context.pc += 4; *(u32*)t->context.pc = 0xd; osWritebackDCache((void*)t->context.pc, 4); osInvalICache((void*)t->context.pc, 4); osStartThread(t); } void Fault_CommitFB() { u16* fb; osViSetYScale(1.0f); osViSetMode(&osViModeNtscLan1); osViSetSpecialFeatures(0x42); // gama_disable|dither_fliter_enable_aa_mode3_disable osViBlack(0); if (sFaultContext->fb) { fb = sFaultContext->fb; } else { fb = (u16*)osViGetNextFramebuffer(); if ((u32)fb == 0x80000000) { fb = (u16*)((osMemSize | 0x80000000) - 0x25800); } } osViSwapBuffer(fb); FaultDrawer_SetDrawerFB(fb, SCREEN_WIDTH, SCREEN_HEIGHT); } void Fault_ProcessClients(void) { FaultClient* iter = sFaultContext->clients; s32 idx = 0; while (iter != NULL) { if (iter->callback) { Fault_FillScreenBlack(); FaultDrawer_SetCharPad(-2, 0); FaultDrawer_Printf("\x1A\x38" "CallBack (%d) %08x %08x %08x\n" "\x1A\x37", idx++, iter, iter->param0, iter->param1); FaultDrawer_SetCharPad(0, 0); iter->callback(iter->param0, iter->param1); Fault_WaitForInput(); Fault_CommitFB(); } iter = iter->next; } } #ifdef NON_MATCHING // regalloc and ordering differences around the two bool variables (faultCustomOptions and faultCopyToLog) void Fault_SetOptionsFromController3(void) { Input* input3; u32 pad; u32 graphPC; u32 graphRA; u32 graphSP; input3 = &sFaultContext->padInput[3]; if (CHECK_BTN_ALL(input3->press.button, 0x80)) { faultCustomOptions = faultCustomOptions == 0; } if (faultCustomOptions) { graphPC = sGraphThread.context.pc; graphRA = sGraphThread.context.ra; graphSP = sGraphThread.context.sp; if (CHECK_BTN_ALL(input3->press.button, BTN_R)) { faultCopyToLog = !faultCopyToLog; FaultDrawer_SetOsSyncPrintfEnabled(faultCopyToLog); } if (CHECK_BTN_ALL(input3->press.button, BTN_A)) { osSyncPrintf("GRAPH PC=%08x RA=%08x STACK=%08x\n", graphPC, graphRA, graphSP); } if (CHECK_BTN_ALL(input3->press.button, BTN_B)) { FaultDrawer_SetDrawerFB(osViGetNextFramebuffer(), 0x140, 0xF0); Fault_DrawRec(0, 0xD7, 0x140, 9, 1); FaultDrawer_SetCharPad(-2, 0); FaultDrawer_DrawText(0x20, 0xD8, "GRAPH PC %08x RA %08x SP %08x", graphPC, graphRA, graphSP); } } } #else #pragma GLOBAL_ASM("asm/non_matchings/boot/fault/Fault_SetOptionsFromController3.s") #endif void Fault_SetOptions(void) { Fault_UpdatePadImpl(); Fault_SetOptionsFromController3(); } void Fault_ThreadEntry(void* arg) { OSMesg msg; u32 pad; OSThread* faultedThread; osSetEventMesg(10, &sFaultContext->queue, (OSMesg)1); osSetEventMesg(12, &sFaultContext->queue, (OSMesg)2); while (1) { do { osRecvMesg(&sFaultContext->queue, &msg, 1); if (msg == (OSMesg)1) { sFaultContext->msgId = 1; osSyncPrintf("フォルトマネージャ:OS_EVENT_CPU_BREAKを受信しました\n"); } else if (msg == (OSMesg)2) { sFaultContext->msgId = 2; osSyncPrintf("フォルトマネージャ:OS_EVENT_FAULTを受信しました\n"); } else if (msg == (OSMesg)3) { Fault_SetOptions(); faultedThread = NULL; continue; } else { sFaultContext->msgId = 3; osSyncPrintf("フォルトマネージャ:不明なメッセージを受信しました\n"); } faultedThread = __osGetCurrFaultedThread(); osSyncPrintf("__osGetCurrFaultedThread()=%08x\n", faultedThread); if (!faultedThread) { faultedThread = Fault_FindFaultedThread(); osSyncPrintf("FindFaultedThread()=%08x\n", faultedThread); } } while (faultedThread == NULL); __osSetFpcCsr(__osGetFpcCsr() & 0xFFFFF07F); sFaultContext->faultedThread = faultedThread; while (!sFaultContext->faultHandlerEnabled) { Fault_Sleep(1000); } Fault_Sleep(500); Fault_CommitFB(); if (sFaultContext->faultActive) { Fault_Wait5Seconds(); } else { Fault_DrawCornerRec(0xF801); Fault_WaitForButtonCombo(); } sFaultContext->faultActive = 1; FaultDrawer_SetForeColor(0xFFFF); FaultDrawer_SetBackColor(0); do { Fault_PrintThreadContext(faultedThread); osSyncPrintfThreadContext(faultedThread); Fault_WaitForInput(); Fault_DrawStackTrace(faultedThread, 0); osSyncPrintfStackTrace(faultedThread, 0); Fault_WaitForInput(); Fault_ProcessClients(); Fault_DrawMemDump((u32)(faultedThread->context.pc - 0x100), (u32)faultedThread->context.sp, 0, 0); Fault_DrawStackTrace(faultedThread, 1); osSyncPrintfStackTrace(faultedThread, 1); Fault_WaitForInput(); Fault_FillScreenRed(); FaultDrawer_DrawText(0x40, 0x50, " CONGRATURATIONS! "); FaultDrawer_DrawText(0x40, 0x5A, "All Pages are displayed."); FaultDrawer_DrawText(0x40, 0x64, " THANK YOU! "); FaultDrawer_DrawText(0x40, 0x6E, " You are great debugger!"); Fault_WaitForInput(); } while (!sFaultContext->exitDebugger); while (!sFaultContext->exitDebugger) { ; } Fault_ResumeThread(faultedThread); } } void Fault_SetFB(void* fb, u16 w, u16 h) { sFaultContext->fb = fb; FaultDrawer_SetDrawerFB(fb, w, h); } void Fault_Start(void) { sFaultContext = &gFaultStruct; bzero(sFaultContext, sizeof(FaultThreadStruct)); FaultDrawer_Init(); FaultDrawer_SetInputCallback(Fault_WaitForInput); sFaultContext->exitDebugger = 0; sFaultContext->msgId = 0; sFaultContext->faultHandlerEnabled = 0; sFaultContext->faultedThread = NULL; sFaultContext->padCallback = &Fault_PadCallback; sFaultContext->clients = NULL; sFaultContext->faultActive = 0; gFaultStruct.faultHandlerEnabled = 1; osCreateMesgQueue(&sFaultContext->queue, sFaultContext->msg, 1); StackCheck_Init(&sFaultThreadInfo, sFaultStack, sFaultStack + sizeof(sFaultStack), 0, 0x100, "fault"); osCreateThread(&sFaultContext->thread, 2, Fault_ThreadEntry, NULL, sFaultStack + sizeof(sFaultStack), 0x7F); osStartThread(&sFaultContext->thread); } void Fault_HangupFaultClient(const char* arg0, char* arg1) { osSyncPrintf("HungUp on Thread %d\n", osGetThreadId(NULL)); osSyncPrintf("%s\n", arg0 ? arg0 : "(NULL)"); osSyncPrintf("%s\n", arg1 ? arg1 : "(NULL)"); FaultDrawer_Printf("HungUp on Thread %d\n", osGetThreadId(NULL)); FaultDrawer_Printf("%s\n", arg0 ? arg0 : "(NULL)"); FaultDrawer_Printf("%s\n", arg1 ? arg1 : "(NULL)"); } void Fault_AddHungupAndCrashImpl(const char* arg0, char* arg1) { FaultClient client; char padd[4]; Fault_AddClient(&client, (fault_client_func)Fault_HangupFaultClient, (void*)arg0, arg1); *(u32*)0x11111111 = 0; // trigger an exception } void Fault_AddHungupAndCrash(const char* filename, u32 line) { char msg[256]; sprintf(msg, "HungUp %s:%d", filename, line); Fault_AddHungupAndCrashImpl(msg, NULL); }