#include "global.h" #include "vt.h" #pragma GLOBAL_ASM("asm/non_matchings/boot/fault/D_800982B0.s") 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, fault_address_converter_func 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(sExceptionNames); i++) { if (value & flag) { FaultDrawer_Printf("(%s)", sExceptionNames[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(sExceptionNames); i++) { if (value & flag) { osSyncPrintf("(%s)\n", sExceptionNames[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, D_80096B80[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, D_80096B80[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(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 = graphOSThread.context.pc; graphRA = graphOSThread.context.ra; graphSP = graphOSThread.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); }