#include "global.h" #include "terminal.h" StackEntry* sStackInfoListStart = NULL; StackEntry* sStackInfoListEnd = NULL; void StackCheck_Init(StackEntry* entry, void* stackBottom, void* stackTop, u32 initValue, s32 minSpace, const char* name) { StackEntry* iter; u32* addr; if (entry == NULL) { sStackInfoListStart = NULL; } else { entry->head = stackBottom; entry->tail = stackTop; entry->initValue = initValue; entry->minSpace = minSpace; entry->name = name; iter = sStackInfoListStart; while (iter) { if (iter == entry) { PRINTF(VT_COL(RED, WHITE) "stackcheck_init: %08x は既にリスト中にある\n" VT_RST, entry); return; } iter = iter->next; } entry->prev = sStackInfoListEnd; entry->next = NULL; if (sStackInfoListEnd) { sStackInfoListEnd->next = entry; } sStackInfoListEnd = entry; if (!sStackInfoListStart) { sStackInfoListStart = entry; } if (entry->minSpace != -1) { addr = entry->head; while (addr < entry->tail) { *addr++ = entry->initValue; } } } } void StackCheck_Cleanup(StackEntry* entry) { u32 inconsistency = false; if (!entry->prev) { if (entry == sStackInfoListStart) { sStackInfoListStart = entry->next; } else { inconsistency = true; } } else { entry->prev->next = entry->next; } if (!entry->next) { if (entry == sStackInfoListEnd) { sStackInfoListEnd = entry->prev; } else { inconsistency = true; } } if (inconsistency) { PRINTF(VT_COL(RED, WHITE) "stackcheck_cleanup: %08x リスト不整合です\n" VT_RST, entry); } } u32 StackCheck_GetState(StackEntry* entry) { u32* last; u32 used; u32 free; u32 ret; for (last = entry->head; last < entry->tail; last++) { if (entry->initValue != *last) { break; } } used = (uintptr_t)entry->tail - (uintptr_t)last; free = (uintptr_t)last - (uintptr_t)entry->head; if (free == 0) { ret = STACK_STATUS_OVERFLOW; PRINTF(VT_FGCOL(RED)); } else if (free < (u32)entry->minSpace && entry->minSpace != -1) { ret = STACK_STATUS_WARNING; PRINTF(VT_FGCOL(YELLOW)); } else { PRINTF(VT_FGCOL(GREEN)); ret = STACK_STATUS_OK; } #if !OOT_DEBUG // This string is still in .rodata for retail builds (void)"(null)"; #endif PRINTF("head=%08x tail=%08x last=%08x used=%08x free=%08x [%s]\n", entry->head, entry->tail, last, used, free, entry->name != NULL ? entry->name : "(null)"); PRINTF(VT_RST); #if OOT_DEBUG if (ret != STACK_STATUS_OK) { LogUtils_LogHexDump(entry->head, (uintptr_t)entry->tail - (uintptr_t)entry->head); } #endif return ret; } u32 StackCheck_CheckAll(void) { u32 ret = 0; StackEntry* iter = sStackInfoListStart; while (iter) { u32 state = StackCheck_GetState(iter); if (state != STACK_STATUS_OK) { ret = 1; } iter = iter->next; } return ret; } u32 StackCheck_Check(StackEntry* entry) { if (entry == NULL) { return StackCheck_CheckAll(); } else { return StackCheck_GetState(entry); } }