sm64pc/src/goddard/debug_utils.c

865 lines
20 KiB
C
Raw Permalink Normal View History

2019-08-25 04:46:40 +00:00
#include <ultra64.h>
#include <stdarg.h>
#include <macros.h>
#include "gd_types.h"
#include "debug_utils.h"
#include "renderer.h"
// types
struct UnkBufThing {
/* 0x00 */ s32 size;
/* 0x04 */ char name[0x40];
}; /* sizeof = 0x44 */
// data
static s32 sNumRoutinesInStack = 0; // @ 801A8280
static s32 D_801A8284[7] = { // TODO: what is this array?
2, 1, 3, 4, 5, 8, 9
};
static s32 sNumActiveMemTrackers = 0; // @ 801A82A0
static u32 sPrimarySeed = 0x12345678; // @ 801A82A4
static u32 sSecondarySeed = 0x58374895; // @ 801A82A8
static char sHexNumerals[17] = { // @ 801A82AC
'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F', '\0'
};
static s32 sPadNumPrint = 0; // @ 801A82C0
// bss
u8 *gGdStreamBuffer; // @ 801BA190
static const char *sRoutineNames[64]; // @ 801BA198
static s32 sTimingActive; // @ 801BA298
static struct GdTimer sTimers[GD_NUM_TIMERS]; // @ 801BA2A0
static struct MemTracker sMemTrackers[GD_NUM_MEM_TRACKERS]; // @ 801BA720
static struct MemTracker *sActiveMemTrackers[16]; // @ 801BA920
/* @ 23A980 -> 23AA34; orig name: func_8018C1B0 */
struct MemTracker *new_memtracker(const char *name) {
s32 i;
struct MemTracker *memtrack = NULL;
for (i = 0; i < ARRAY_COUNT(sMemTrackers); i++) {
if (sMemTrackers[i].name == NULL) {
sMemTrackers[i].name = name;
memtrack = &sMemTrackers[i];
break;
}
}
if (memtrack != NULL) {
memtrack->total = 0.0f;
}
return memtrack;
}
/* @ 23AA34 -> 23AADC; orig name: func_8018C264 */
struct MemTracker *get_memtracker(const char *tracker) {
s32 i; // sp1C
for (i = 0; i < ARRAY_COUNT(sMemTrackers); i++) {
if (sMemTrackers[i].name != NULL) {
if (gd_str_not_equal(sMemTrackers[i].name, tracker) == FALSE) {
return &sMemTrackers[i];
}
}
}
return NULL;
}
/* @ 23AADC -> 23ABE0; orig name: func_8018C30C */
struct MemTracker *start_memtracker(const char *name) {
struct MemTracker *tracker;
tracker = get_memtracker(name);
if (tracker == NULL) {
tracker = new_memtracker(name);
if (tracker == NULL) {
fatal_printf("Unable to make memtracker '%s'", name);
}
}
tracker->begin = (f32) get_alloc_mem_amt();
if (sNumActiveMemTrackers >= ARRAY_COUNT(sActiveMemTrackers)) {
fatal_printf("too many memtracker calls");
}
sActiveMemTrackers[sNumActiveMemTrackers++] = tracker;
return tracker;
}
/* @ 23ABE0 -> 23AC28; not called; orig name: Unknown8018C410 */
void print_most_recent_memtracker(void) {
gd_printf("%s\n", sActiveMemTrackers[sNumActiveMemTrackers - 1]->name);
}
/* @ 23AC28 -> 23AD94; orig name: func_8018C458 */
u32 stop_memtracker(const char *name) {
struct MemTracker *tracker; // sp24
if (sNumActiveMemTrackers-- < 0) {
fatal_printf("bad mem tracker count");
}
tracker = get_memtracker(name);
if (tracker == NULL) {
fatal_printf("memtracker '%s' not found", name);
}
tracker->end = (f32) get_alloc_mem_amt();
tracker->total += (tracker->end - tracker->begin);
return (u32) tracker->total;
}
/* @ 23AD94 -> 23AE20; orig name: func_8018C5C4 */
void remove_all_memtrackers(void) {
s32 i;
for (i = 0; i < ARRAY_COUNT(sMemTrackers); i++) { // L8018C5CC
sMemTrackers[i].name = NULL;
sMemTrackers[i].begin = 0.0f;
sMemTrackers[i].end = 0.0f;
sMemTrackers[i].total = 0.0f;
}
2020-05-07 18:21:22 +00:00
#ifdef AVOID_UB
sNumActiveMemTrackers = 0;
#endif
2019-08-25 04:46:40 +00:00
}
/* 23AE20 -> 23AE44; orig name: func_8018C650 */
struct MemTracker *get_memtracker_by_id(s32 id) {
return &sMemTrackers[id];
}
/* 23AE44 -> 23AEFC; orig name: func_8018C674 */
void print_all_memtrackers(void) {
s32 i;
for (i = 0; i < ARRAY_COUNT(sMemTrackers); i++) {
if (sMemTrackers[i].name != NULL) {
gd_printf("'%s' = %dk\n", sMemTrackers[i].name, (s32)(sMemTrackers[i].total / 1024.0f));
}
}
}
/* 23AEFC -> 23AFB0; orig name: func_8018C72C */
void print_all_timers(void) {
s32 i;
gd_printf("\nTimers:\n");
for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
if (sTimers[i].name != NULL) {
gd_printf("'%s' = %f (%d)\n", sTimers[i].name, sTimers[i].scaledTotal,
sTimers[i].resetCount);
}
}
}
/* 23AFB0 -> 23AFC8; orig name: func_8018C7E0 */
void deactivate_timing(void) {
sTimingActive = FALSE;
}
/* 23AFC8 -> 23AFE4; orig name: func_8018C7F8 */
void activate_timing(void) {
sTimingActive = TRUE;
}
/* 23AFE4 -> 23B118; orig name: func_8018C814 */
void remove_all_timers(void) {
s32 i;
for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
sTimers[i].name = NULL;
sTimers[i].total = 0;
sTimers[i].unk0C = 0.0f;
sTimers[i].scaledTotal = 0.0f;
sTimers[i].prevScaledTotal = 0.0f;
sTimers[i].unk1C = D_801A8284[(u32) i % 7];
sTimers[i].resetCount = 0;
}
activate_timing();
}
/* 23B118 -> 23B1C4; orig name: func_8018C948 */
struct GdTimer *new_timer(const char *name) {
s32 i;
struct GdTimer *timer = NULL;
for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
if (sTimers[i].name == NULL) {
sTimers[i].name = name;
timer = &sTimers[i];
break;
}
}
return timer;
}
/* 23B1C4 -> 23B284; orig name: func_8018C9F4 */
struct GdTimer *get_timer(const char *timerName) {
s32 i;
for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
if (sTimers[i].name != NULL) {
if (gd_str_not_equal(sTimers[i].name, timerName) == FALSE) {
return &sTimers[i];
}
}
}
return NULL;
}
/* 23B284 -> 23B2E4; orig name: func_8018CAB4 */
struct GdTimer *get_timer_checked(const char *timerName) {
struct GdTimer *timer;
timer = get_timer(timerName);
if (timer == NULL) {
fatal_printf("Timer '%s' not found", timerName);
}
return timer;
}
/* 23B2E4 -> 23B350 */
struct GdTimer *get_timernum(s32 id) {
if (id >= ARRAY_COUNT(sTimers)) {
fatal_printf("get_timernum(): Timer number %d out of range (MAX %d)", id, ARRAY_COUNT(sTimers));
}
return &sTimers[id];
}
/* 23B350 -> 23B42C; orig name: func_8018CB80 */
void split_timer_ptr(struct GdTimer *timer) {
if (!sTimingActive) {
return;
}
timer->end = gd_get_ostime();
timer->total += timer->end - timer->start;
if (timer->total < 0) {
timer->total = 0;
}
timer->scaledTotal = ((f32) timer->total) / get_time_scale();
timer->start = timer->end;
}
/* 23B42C -> 23B49C; not called; orig name: Unknown8018CC5C */
void split_all_timers(void) {
s32 i;
struct GdTimer *timer;
for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
timer = get_timernum(i);
if (timer->name != NULL) {
split_timer_ptr(timer);
}
}
}
/* 23B49C -> 23B530; not called; orig name: Unknown8018CCCC */
void start_all_timers(void) {
s32 i;
struct GdTimer *timer;
if (!sTimingActive) {
return;
}
for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
timer = get_timernum(i);
if (timer->name != NULL) {
timer->start = gd_get_ostime();
}
}
}
/* 23B530 -> 23B600 */
void start_timer(const char *name) {
struct GdTimer *timer;
if (!sTimingActive) {
return;
}
timer = get_timer(name);
if (timer == NULL) {
timer = new_timer(name);
if (timer == NULL) {
fatal_printf("start_timer(): Unable to make timer '%s'", name);
}
}
timer->prevScaledTotal = timer->scaledTotal;
timer->start = gd_get_ostime();
timer->total = 0;
timer->resetCount = 1;
}
/* 23B600 -> 23B6BC */
void restart_timer(const char *name) {
struct GdTimer *timer;
if (!sTimingActive) {
return;
}
timer = get_timer(name);
if (timer == NULL) {
timer = new_timer(name);
if (timer == NULL) {
fatal_printf("restart_timer(): Unable to make timer '%s'", name);
}
}
timer->start = gd_get_ostime();
timer->resetCount++;
}
/* 23B6BC -> 23B718; orig name: func_8018CEEC */
void split_timer(const char *name) {
struct GdTimer *timer;
if (!sTimingActive) {
return;
}
timer = get_timer_checked(name);
split_timer_ptr(timer);
}
/* 23B718 -> 23B7F0; orig name: func_8018CF48 */
void stop_timer(const char *name) {
struct GdTimer *timer;
if (!sTimingActive) {
return;
}
timer = get_timer_checked(name);
timer->end = gd_get_ostime();
timer->total += timer->end - timer->start;
if (timer->total < 0) {
timer->total = 0;
}
timer->scaledTotal = ((f32) timer->total) / get_time_scale();
}
/* 23B7F0 -> 23B838; orig name: func_8018D020 */
f32 get_scaled_timer_total(const char *name) {
struct GdTimer *timer;
timer = get_timer_checked(name);
return timer->scaledTotal;
}
/* 23B838 -> 23B888; not called; orig name: Unknown8018D1A8 */
f32 get_timer_total(const char *name) {
struct GdTimer *timer;
timer = get_timer_checked(name);
return (f32) timer->total;
}
/* 23B888 -> 23B8B8; orig name: myPrint1 */
void fatal_print(const char *str) {
fatal_printf(str);
}
/* 23B8B8 -> 23B928; orig name: func_8018D0E8 */
void print_stack_trace(void) {
s32 i;
for (i = 0; i < sNumRoutinesInStack; i++) {
gd_printf("\tIn: '%s'\n", sRoutineNames[i]);
}
}
/* 23B928 -> 23BBF0; orig name: myPrintf */
void fatal_printf(const char *fmt, ...) {
char cur; // sp37 (sp34)
UNUSED u8 pad[4];
va_list vl; // sp2C
va_start(vl, fmt);
while ((cur = *fmt++)) {
switch (cur) {
case '%':
switch (cur = *fmt++) {
case 'd':
gd_printf("%d", va_arg(vl, s32));
break;
case 'f':
gd_printf("%f", va_arg(vl, double));
break;
case 's':
gd_printf("%s", va_arg(vl, char *));
break;
case 'c':
gd_printf("%c", va_arg(vl, char));
break;
case 'x':
gd_printf("%x", va_arg(vl, s32));
break;
default:
gd_printf("%c", cur);
}
break;
case '\\':
gd_printf("\\");
break;
case '\n':
gd_printf("\n");
break;
default:
gd_printf("%c", cur);
}
}
va_end(vl);
gd_printf("\n");
print_stack_trace();
gd_printf("\n");
gd_exit(-1);
}
/* 23BBF0 -> 23BC80; orig name: func_8018D420 */
void add_to_stacktrace(const char *routine) {
// Please check array bounds before writing to it
sRoutineNames[sNumRoutinesInStack++] = routine;
sRoutineNames[sNumRoutinesInStack] = NULL;
if (sNumRoutinesInStack >= ARRAY_COUNT(sRoutineNames)) {
fatal_printf("You're in too many routines");
}
}
/* 23BC80 -> 23BD30 */
// remove a routine from the stack trace routine buffer
void imout(void) {
s32 i;
if (--sNumRoutinesInStack < 0) {
for (i = 0; i < ARRAY_COUNT(sRoutineNames); i++) {
if (sRoutineNames[i] != NULL) {
gd_printf(" - %s\n", sRoutineNames[i]);
} else {
break;
}
}
fatal_printf("imout() - imout() called too many times");
}
}
/* 23BD30 -> 23BE78 */
// TODO: rename, figure out type of rng generator?
f32 func_8018D560(void) {
u32 temp;
u32 i;
f32 val;
for (i = 0; i < 4; i++) {
if (sPrimarySeed & 0x80000000) {
sPrimarySeed = sPrimarySeed << 1 | 1;
} else {
sPrimarySeed <<= 1;
}
}
sPrimarySeed += 4;
/* Seed Switch */
if ((sPrimarySeed ^= gd_get_ostime()) & 1) {
temp = sPrimarySeed;
sPrimarySeed = sSecondarySeed;
sSecondarySeed = temp;
}
val = (sPrimarySeed & 0xFFFF) / 65535.0; // 65535.0f
return val;
}
/* 23BE78 -> 23BFD8; orig name: func_8018D6A8 */
s32 gd_atoi(const char *str) {
char cur;
const char *origstr = str;
s32 curval;
s32 out = 0;
s32 isNegative = FALSE;
while (TRUE) {
cur = *str++;
if (cur < '0' || cur > '9') {
if (cur != '-') {
fatal_printf("gd_atoi() bad number '%s'", origstr);
}
}
if (cur == '-') {
isNegative = TRUE;
} else {
curval = cur - '0';
out += curval & 0xFF;
if (*str == '\0' || *str == '.' || *str < '0' || *str > '9') {
break;
}
out *= 10;
}
}
if (isNegative) {
out = -out;
}
return out;
}
/* 23BFD8 -> 23C018; orig name: func_8018D808 */
f64 gd_lazy_atof(const char *str, UNUSED u32 *unk) {
return gd_atoi(str);
}
/* 23C018 -> 23C078; orig name: func_8018D848 */
char *sprint_num_as_hex(char *str, s32 val) {
s32 shift;
for (shift = 28; shift > -4; shift -= 4) {
*str++ = sHexNumerals[(val >> shift) & 0xF];
}
*str = '\0';
return str;
}
/* 23C078 -> 23C174; orig name: func_8018D8A8 */
/* padnum = a decimal number with the max desired output width */
char *sprint_num(char *str, s32 val, s32 padnum) {
s32 i;
if (val == 0) {
*str++ = '0';
*str = '\0';
return str;
}
if (val < 0) {
val = -val;
*str++ = '-';
}
while (padnum > 0) {
if (padnum <= val) {
sPadNumPrint = TRUE;
for (i = 0; i < 9; i++) {
val -= padnum;
if (val < 0) {
val += padnum;
break;
}
}
*str++ = i + '0';
} else {
if (sPadNumPrint) {
*str++ = '0';
}
}
padnum /= 10;
}
*str = '\0';
return str;
}
/* 23C174 -> 23C1C8; orig name: func_8018D9A4 */
s32 int_sci_notation(s32 base, s32 significand) {
s32 i;
for (i = 1; i < significand; i++) {
base *= 10;
}
return base;
}
/* 23C1C8 -> 23C468; orig name: func_8018D9F8 */
char *sprint_val_withspecifiers(char *str, union PrintVal val, char *specifiers) {
s32 fracPart; // sp3C
s32 intPart; // sp38
s32 intPrec; // sp34
s32 fracPrec; // sp30
UNUSED u8 pad[4];
char cur; // sp2B
fracPrec = 6;
intPrec = 6;
while ((cur = *specifiers++)) {
if (cur == 'd') {
sPadNumPrint = FALSE;
str = sprint_num(str, val.i, 1000000000);
} else if (cur == 'x') {
sPadNumPrint = TRUE; /* doesn't affect hex printing, though... */
str = sprint_num_as_hex(str, val.i);
} else if (cur == 'f') {
intPart = (s32) val.f;
fracPart = (s32)((val.f - (f32) intPart) * (f32) int_sci_notation(10, fracPrec));
sPadNumPrint = FALSE;
str = sprint_num(str, intPart, int_sci_notation(10, intPrec));
*str++ = '.';
sPadNumPrint = TRUE;
str = sprint_num(str, fracPart, int_sci_notation(10, fracPrec - 1));
} else if (cur >= '0' && cur <= '9') {
cur = cur - '0';
intPrec = cur;
if (*specifiers++) {
fracPrec = (*specifiers++) - '0';
}
} else {
gd_strcpy(str, "<BAD TYPE>");
str += 10;
}
}
return str;
}
/* 23C468 -> 23C4AC; orig name: func_8018DC98 */
void gd_strcpy(char *dst, const char *src) {
2019-09-01 19:50:50 +00:00
while ((*dst++ = *src++)) {
2019-08-25 04:46:40 +00:00
;
2019-09-01 19:50:50 +00:00
}
2019-08-25 04:46:40 +00:00
}
/* 23C4AC -> 23C52C; not called; orig name: Unknown8018DCDC */
void ascii_to_uppercase(char *str) {
char c;
while ((c = *str)) {
if (c >= 'a' && c <= 'z') {
*str = c & 0xDF;
}
str++;
}
}
/* 23C52C -> 23C5A8; orig name: func_8018DD5C */
char *gd_strdup(const char *src) {
char *dst; // sp24
dst = gd_malloc_perm((gd_strlen(src) + 1) * sizeof(char));
if (dst == NULL) {
fatal_printf("gd_strdup(): out of memory");
}
gd_strcpy(dst, src);
return dst;
}
/* 23C5A8 -> 23C5FC; orig name: func_8018DDD8 */
u32 gd_strlen(const char *str) {
u32 len = 0;
while (*str++) {
len++;
}
return len;
}
/* 23C5FC -> 23C680; orig name: func_8018DE2C */
char *gd_strcat(char *dst, const char *src) {
2019-09-01 19:50:50 +00:00
while (*dst++) {
2019-08-25 04:46:40 +00:00
;
2019-09-01 19:50:50 +00:00
}
2019-08-25 04:46:40 +00:00
if (*src) {
dst--;
2019-09-01 19:50:50 +00:00
while ((*dst++ = *src++)) {
2019-08-25 04:46:40 +00:00
;
2019-09-01 19:50:50 +00:00
}
2019-08-25 04:46:40 +00:00
}
return --dst;
}
/* 23C67C -> 23C728; orig name: func_8018DEB0 */
/* Returns a bool, not the position of the mismatch */
s32 gd_str_not_equal(const char *str1, const char *str2) {
while (*str1 && *str2) {
if (*str1++ != *str2++) {
return TRUE;
}
}
return *str1 != '\0' || *str2 != '\0';
}
/* 23C728 -> 23C7B8; orig name; func_8018DF58 */
s32 gd_str_contains(const char *str1, const char *str2) {
const char *startsub = str2;
while (*str1 && *str2) {
if (*str1++ != *str2++) {
str2 = startsub;
}
}
return !*str2;
}
/* 23C7B8 -> 23C7DC; orig name: func_8018DFE8 */
s32 gd_feof(struct GdFile *f) {
return f->flags & 0x4;
}
/* 23C7DC -> 23C7FC; orig name: func_8018E00C */
void gd_set_feof(struct GdFile *f) {
f->flags |= 0x4;
}
/* 23C7FC -> 23CA0C */
struct GdFile *gd_fopen(const char *filename, const char *mode) {
struct GdFile *f; // sp74
char *loadedname; // sp70
u32 i; // sp6C
UNUSED u32 pad68;
struct UnkBufThing buf; // sp24
u8 *bufbytes; // sp20
u8 *fileposptr; // sp1C
s32 filecsr; // sp18
filecsr = 0;
while (TRUE) {
bufbytes = (u8 *) &buf;
for (i = 0; i < sizeof(struct UnkBufThing); i++) {
*bufbytes++ = gGdStreamBuffer[filecsr++];
}
func_801A59AC(&buf);
fileposptr = &gGdStreamBuffer[filecsr];
filecsr += buf.size;
loadedname = buf.name;
if (buf.size == 0) {
break;
}
if (!gd_str_not_equal(filename, loadedname)) {
break;
}
}
if (buf.size == 0) {
fatal_printf("gd_fopen() File not found '%s'", filename);
return NULL;
}
f = gd_malloc_perm(sizeof(struct GdFile));
if (f == NULL) {
fatal_printf("gd_fopen() Out of memory loading '%s'", filename);
return NULL;
}
f->stream = (s8 *) fileposptr;
f->size = buf.size;
f->pos = f->flags = 0;
if (gd_str_contains(mode, "w")) {
f->flags |= 0x1;
}
if (gd_str_contains(mode, "b")) {
f->flags |= 0x2;
}
return f;
}
/* 23CA0C -> 23CB38; orig name: func_8018E23C */
s32 gd_fread(s8 *buf, s32 bytes, UNUSED s32 count, struct GdFile *f) {
s32 bytesToRead = bytes;
s32 bytesread;
if (f->pos + bytesToRead > f->size) {
bytesToRead = f->size - f->pos;
}
if (bytesToRead == 0) {
gd_set_feof(f);
return -1;
}
bytesread = bytesToRead;
while (bytesread--) {
*buf++ = f->stream[f->pos++];
}
return bytesToRead;
}
/* 23CB38 -> 23CB54; orig name: func_8018E368 */
void gd_fclose(UNUSED struct GdFile *f) {
return;
}
/* 23CB54 -> 23CB70; orig name: func_8018E384 */
u32 gd_get_file_size(struct GdFile *f) {
return f->size;
}
/* 23CB70 -> 23CBA8; orig name: func_8018E3A0 */
s32 is_newline(char c) {
return c == '\r' || c == '\n';
}
/* 23CBA8 -> 23CCF0; orig name: func_8018E3D8 */
s32 gd_fread_line(char *buf, u32 size, struct GdFile *f) {
signed char c;
u32 pos = 0;
UNUSED u32 pad1c;
do {
if (gd_fread(&c, 1, 1, f) == -1) {
break;
}
} while (is_newline(c));
while (!is_newline(c)) {
if (c == -1) {
break;
}
if (pos > size) {
break;
}
buf[pos++] = c;
if (gd_fread(&c, 1, 1, f) == -1) {
break;
}
}
buf[pos++] = '\0';
return pos;
}