sm64

A Super Mario 64 decompilation
Log | Files | Refs | README | LICENSE

debug_utils.c (22363B)


      1 #include <PR/ultratypes.h>
      2 #include <stdarg.h>
      3 
      4 #include "debug_utils.h"
      5 #include "gd_types.h"
      6 #include "macros.h"
      7 #include "renderer.h"
      8 #include "draw_objects.h"
      9 
     10 // types
     11 struct UnkBufThing {
     12     /* 0x00 */ s32 size;
     13     /* 0x04 */ char name[0x40];
     14 }; /* sizeof = 0x44 */
     15 
     16 // data
     17 static s32 sNumRoutinesInStack = 0; // @ 801A8280
     18 static s32 sTimerGadgetColours[7] = {
     19     COLOUR_RED,
     20     COLOUR_WHITE,
     21     COLOUR_GREEN,
     22     COLOUR_BLUE,
     23     COLOUR_GRAY,
     24     COLOUR_YELLOW,
     25     COLOUR_PINK
     26 };
     27 static s32 sNumActiveMemTrackers = 0;   // @ 801A82A0
     28 static u32 sPrimarySeed = 0x12345678;   // @ 801A82A4
     29 static u32 sSecondarySeed = 0x58374895; // @ 801A82A8
     30 
     31 // bss
     32 u8 *gGdStreamBuffer;                                        // @ 801BA190
     33 static const char *sRoutineNames[64];                       // @ 801BA198
     34 static s32 sTimingActive;                                   // @ 801BA298
     35 static struct GdTimer sTimers[GD_NUM_TIMERS];               // @ 801BA2A0
     36 static struct MemTracker sMemTrackers[GD_NUM_MEM_TRACKERS]; // @ 801BA720
     37 static struct MemTracker *sActiveMemTrackers[16];           // @ 801BA920
     38 
     39 /*
     40  * Memtrackers
     41  *
     42  * These are used to monitor how much heap memory is being used by certain
     43  * operations.
     44  * To create a memtracker, call new_memtracker with a unique name.
     45  * To record the amount of memory used by a certain allocation, call
     46  * start_memtracker before allocating memory, and call stop_memtracker after
     47  * allocating memory.
     48  * The memtracker keeps track of the memory allocated between a single
     49  * start_memtracker/stop_memtracker pair as well as the total memory allocated
     50  * of all start_memtracker/stop_memtracker pairs.
     51  */
     52 
     53 /**
     54  * Creates a new memtracker with the specified name
     55  */
     56 struct MemTracker *new_memtracker(const char *name) {
     57     s32 i;
     58     struct MemTracker *tracker = NULL;
     59 
     60     for (i = 0; i < ARRAY_COUNT(sMemTrackers); i++) {
     61         if (sMemTrackers[i].name == NULL) {
     62             sMemTrackers[i].name = name;
     63             tracker = &sMemTrackers[i];
     64             break;
     65         }
     66     }
     67 
     68     if (tracker != NULL) {
     69         tracker->total = 0.0f;
     70     }
     71 
     72     return tracker;
     73 }
     74 
     75 /**
     76  * Returns the memtracker with the specified name, or NULL if it
     77  * does not exist
     78  */
     79 struct MemTracker *get_memtracker(const char *name) {
     80     s32 i;
     81 
     82     for (i = 0; i < ARRAY_COUNT(sMemTrackers); i++) {
     83         if (sMemTrackers[i].name != NULL) {
     84             if (gd_str_not_equal(sMemTrackers[i].name, name) == FALSE) {
     85                 return &sMemTrackers[i];
     86             }
     87         }
     88     }
     89 
     90     return NULL;
     91 }
     92 
     93 /**
     94  * Records the amount of heap usage before allocating memory.
     95  */
     96 struct MemTracker *start_memtracker(const char *name) {
     97     struct MemTracker *tracker = get_memtracker(name);
     98 
     99     // Create one if it doesn't exist
    100     if (tracker == NULL) {
    101         tracker = new_memtracker(name);
    102         if (tracker == NULL) {
    103             fatal_printf("Unable to make memtracker '%s'", name);
    104         }
    105     }
    106 
    107     tracker->begin = (f32) get_alloc_mem_amt();
    108     if (sNumActiveMemTrackers >= ARRAY_COUNT(sActiveMemTrackers)) {
    109         fatal_printf("too many memtracker calls");
    110     }
    111 
    112     sActiveMemTrackers[sNumActiveMemTrackers++] = tracker;
    113 
    114     return tracker;
    115 }
    116 
    117 /* @ 23ABE0 -> 23AC28; not called; orig name: Unknown8018C410 */
    118 void print_most_recent_memtracker_name(void) {
    119     gd_printf("%s\n", sActiveMemTrackers[sNumActiveMemTrackers - 1]->name);
    120 }
    121 
    122 /**
    123  * Records the amount of heap usage after allocating memory.
    124  */
    125 u32 stop_memtracker(const char *name) {
    126     struct MemTracker *tracker;
    127 
    128     if (sNumActiveMemTrackers-- < 0) {
    129         fatal_printf("bad mem tracker count");
    130     }
    131 
    132     tracker = get_memtracker(name);
    133     if (tracker == NULL) {
    134         fatal_printf("memtracker '%s' not found", name);
    135     }
    136 
    137     tracker->end = get_alloc_mem_amt();
    138     tracker->total += (tracker->end - tracker->begin);
    139 
    140     return (u32) tracker->total;
    141 }
    142 
    143 /**
    144  * Destroys all memtrackers
    145  */
    146 void remove_all_memtrackers(void) {
    147     s32 i;
    148 
    149     for (i = 0; i < ARRAY_COUNT(sMemTrackers); i++) {
    150         sMemTrackers[i].name = NULL;
    151         sMemTrackers[i].begin = 0.0f;
    152         sMemTrackers[i].end = 0.0f;
    153         sMemTrackers[i].total = 0.0f;
    154     }
    155 
    156 #ifdef AVOID_UB
    157     sNumActiveMemTrackers = 0;
    158 #endif
    159 }
    160 
    161 /**
    162  * Returns a memtracker by index rather than name
    163  */
    164 struct MemTracker *get_memtracker_by_index(s32 index) {
    165     return &sMemTrackers[index];
    166 }
    167 
    168 /**
    169  * Prints the total memory allocated for each memtracker
    170  */
    171 void print_all_memtrackers(void) {
    172     s32 i;
    173 
    174     for (i = 0; i < ARRAY_COUNT(sMemTrackers); i++) {
    175         if (sMemTrackers[i].name != NULL) {
    176             gd_printf("'%s' = %dk\n", sMemTrackers[i].name, (s32)(sMemTrackers[i].total / 1024.0f));
    177         }
    178     }
    179 }
    180 
    181 /*
    182  * Timers
    183  *
    184  * These are used to profile the code by measuring the time it takes to perform
    185  * operations.
    186  * To record elapsed time, call start_timer, perform some operations, then call stop_timer.
    187  * You can also use restart_timer/split_timer instead of start_timer/stop_timer
    188  * to keep a running total.
    189  */
    190 
    191 /* 23AEFC -> 23AFB0; orig name: func_8018C72C */
    192 void print_all_timers(void) {
    193     s32 i;
    194 
    195     gd_printf("\nTimers:\n");
    196     for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
    197         if (sTimers[i].name != NULL) {
    198             gd_printf("'%s' = %f (%d)\n", sTimers[i].name, sTimers[i].scaledTotal,
    199                       sTimers[i].resetCount);
    200         }
    201     }
    202 }
    203 
    204 /* 23AFB0 -> 23AFC8; orig name: func_8018C7E0 */
    205 void deactivate_timing(void) {
    206     sTimingActive = FALSE;
    207 }
    208 
    209 /* 23AFC8 -> 23AFE4; orig name: func_8018C7F8 */
    210 void activate_timing(void) {
    211     sTimingActive = TRUE;
    212 }
    213 
    214 /**
    215  * Destroys all timers
    216  */
    217 void remove_all_timers(void) {
    218     s32 i;
    219 
    220     for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
    221         sTimers[i].name = NULL;
    222         sTimers[i].total = 0;
    223         sTimers[i].unused = 0.0f;
    224         sTimers[i].scaledTotal = 0.0f;
    225         sTimers[i].prevScaledTotal = 0.0f;
    226         sTimers[i].gadgetColourNum = sTimerGadgetColours[(u32) i % 7];
    227         sTimers[i].resetCount = 0;
    228     }
    229     activate_timing();
    230 }
    231 
    232 /**
    233  * Creates a new timer with the specified name
    234  */
    235 static struct GdTimer *new_timer(const char *name) {
    236     s32 i;
    237     struct GdTimer *timer = NULL;
    238 
    239     for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
    240         if (sTimers[i].name == NULL) {
    241             sTimers[i].name = name;
    242             timer = &sTimers[i];
    243             break;
    244         }
    245     }
    246 
    247     return timer;
    248 }
    249 
    250 /**
    251  * Returns the timer with the specified name, or NULL if it does not exist.
    252  */
    253 struct GdTimer *get_timer(const char *timerName) {
    254     s32 i;
    255 
    256     for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
    257         if (sTimers[i].name != NULL) {
    258             if (gd_str_not_equal(sTimers[i].name, timerName) == FALSE) {
    259                 return &sTimers[i];
    260             }
    261         }
    262     }
    263 
    264     return NULL;
    265 }
    266 
    267 /**
    268  * Returns the timer with the specified name, or aborts the program if it does
    269  * not exist.
    270  */
    271 static struct GdTimer *get_timer_checked(const char *timerName) {
    272     struct GdTimer *timer;
    273 
    274     timer = get_timer(timerName);
    275     if (timer == NULL) {
    276         fatal_printf("Timer '%s' not found", timerName);
    277     }
    278 
    279     return timer;
    280 }
    281 
    282 /**
    283  * Returns a timer by index rather than name
    284  */
    285 struct GdTimer *get_timernum(s32 index) {
    286     if (index >= ARRAY_COUNT(sTimers)) {
    287         fatal_printf("get_timernum(): Timer number %d out of range (MAX %d)", index, ARRAY_COUNT(sTimers));
    288     }
    289 
    290     return &sTimers[index];
    291 }
    292 
    293 /* 23B350 -> 23B42C; orig name: func_8018CB80 */
    294 void split_timer_ptr(struct GdTimer *timer) {
    295     if (!sTimingActive) {
    296         return;
    297     }
    298 
    299     timer->end = gd_get_ostime();
    300     timer->total += timer->end - timer->start;
    301 
    302     if (timer->total < 0) {
    303         timer->total = 0;
    304     }
    305 
    306     timer->scaledTotal = ((f32) timer->total) / get_time_scale();
    307     timer->start = timer->end;
    308 }
    309 
    310 /* 23B42C -> 23B49C; not called; orig name: Unknown8018CC5C */
    311 void split_all_timers(void) {
    312     s32 i;
    313     struct GdTimer *timer;
    314 
    315     for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
    316         timer = get_timernum(i);
    317         if (timer->name != NULL) {
    318             split_timer_ptr(timer);
    319         }
    320     }
    321 }
    322 
    323 /**
    324  * Unused - records the start time for all timers
    325  */
    326 void start_all_timers(void) {
    327     s32 i;
    328     struct GdTimer *timer;
    329 
    330     if (!sTimingActive) {
    331         return;
    332     }
    333 
    334     for (i = 0; i < ARRAY_COUNT(sTimers); i++) {
    335         timer = get_timernum(i);
    336 
    337         if (timer->name != NULL) {
    338             timer->start = gd_get_ostime();
    339         }
    340     }
    341 }
    342 
    343 /**
    344  * Records the current time before performing an operation
    345  */
    346 void start_timer(const char *name) {
    347     struct GdTimer *timer;
    348 
    349     if (!sTimingActive) {
    350         return;
    351     }
    352 
    353     // Create timer if it does not exist.
    354     timer = get_timer(name);
    355     if (timer == NULL) {
    356         timer = new_timer(name);
    357         if (timer == NULL) {
    358             fatal_printf("start_timer(): Unable to make timer '%s'", name);
    359         }
    360     }
    361 
    362     timer->prevScaledTotal = timer->scaledTotal;
    363     timer->start = gd_get_ostime();
    364     timer->total = 0;
    365     timer->resetCount = 1;
    366 }
    367 
    368 /**
    369  * Records the current time before performing an operation
    370  */
    371 void restart_timer(const char *name) {
    372     struct GdTimer *timer;
    373 
    374     if (!sTimingActive) {
    375         return;
    376     }
    377 
    378     // Create timer if it does not exist.
    379     timer = get_timer(name);
    380     if (timer == NULL) {
    381         timer = new_timer(name);
    382         if (timer == NULL) {
    383             fatal_printf("restart_timer(): Unable to make timer '%s'", name);
    384         }
    385     }
    386 
    387     timer->start = gd_get_ostime();
    388     timer->resetCount++;
    389 }
    390 
    391 /**
    392  * Records the current time after performing an operation, adds the elapsed time
    393  * to the total, then restarts the timer
    394  */
    395 void split_timer(const char *name) {
    396     struct GdTimer *timer;
    397 
    398     if (!sTimingActive) {
    399         return;
    400     }
    401 
    402     timer = get_timer_checked(name);
    403     split_timer_ptr(timer);
    404 }
    405 
    406 /**
    407  * Records the current time after performing an operation
    408  */
    409 void stop_timer(const char *name) {
    410     struct GdTimer *timer;
    411 
    412     if (!sTimingActive) {
    413         return;
    414     }
    415 
    416     timer = get_timer_checked(name);
    417     timer->end = gd_get_ostime();
    418     timer->total += timer->end - timer->start;
    419     if (timer->total < 0) {
    420         timer->total = 0;
    421     }
    422 
    423     timer->scaledTotal = ((f32) timer->total) / get_time_scale();
    424 }
    425 
    426 /**
    427  * Returns the scaled total for the specified timer
    428  */
    429 f32 get_scaled_timer_total(const char *name) {
    430     struct GdTimer *timer = get_timer_checked(name);
    431 
    432     return timer->scaledTotal;
    433 }
    434 
    435 /**
    436  * Unused - returns the raw total for the specified timer
    437  */
    438 f32 get_timer_total(const char *name) {
    439     struct GdTimer *timer = get_timer_checked(name);
    440 
    441     return (f32) timer->total;
    442 }
    443 
    444 
    445 /*
    446  * Miscellaneous debug functions
    447  */
    448 
    449 
    450 /**
    451  * Prints the given string, prints the stack trace, and exits the program
    452  */
    453 void fatal_print(const char *str) {
    454     fatal_printf(str);
    455 }
    456 
    457 /**
    458  * Prints the stack trace registered by callng imin()/imout()
    459  */
    460 void print_stack_trace(void) {
    461     s32 i;
    462 
    463     for (i = 0; i < sNumRoutinesInStack; i++) {
    464         gd_printf("\tIn: '%s'\n", sRoutineNames[i]);
    465     }
    466 }
    467 
    468 /**
    469  * Prints the formatted string, prints the stack trace, and exits the program
    470  */
    471 void fatal_printf(const char *fmt, ...) {
    472     char cur;
    473     UNUSED u8 filler[4];
    474     va_list vl;
    475 
    476     va_start(vl, fmt);
    477     while ((cur = *fmt++)) {
    478         switch (cur) {
    479             case '%':
    480                 switch (cur = *fmt++) {
    481                     case 'd':
    482                         gd_printf("%d", va_arg(vl, s32));
    483                         break;
    484                     case 'f':
    485                         gd_printf("%f", va_arg(vl, double));
    486                         break;
    487                     case 's':
    488                         gd_printf("%s", va_arg(vl, char *));
    489                         break;
    490                     case 'c':
    491 #ifdef AVOID_UB
    492                         gd_printf("%c", (char)va_arg(vl, int));
    493 #else
    494                         gd_printf("%c", va_arg(vl, char));
    495 #endif
    496                         break;
    497                     case 'x':
    498                         gd_printf("%x", va_arg(vl, s32));
    499                         break;
    500                     default:
    501                         gd_printf("%c", cur);
    502                 }
    503                 break;
    504             case '\\':
    505                 gd_printf("\\");
    506                 break;
    507             case '\n':
    508                 gd_printf("\n");
    509                 break;
    510             default:
    511                 gd_printf("%c", cur);
    512         }
    513     }
    514     va_end(vl);
    515 
    516     gd_printf("\n");
    517     print_stack_trace();
    518     gd_printf("\n");
    519     gd_exit(-1);
    520 }
    521 
    522 /**
    523  * "I'm in"
    524  * Adds the function name to the stack trace
    525  */
    526 void imin(const char *routine) {
    527     sRoutineNames[sNumRoutinesInStack++] = routine;
    528     sRoutineNames[sNumRoutinesInStack] = NULL;  //! array bounds is checked after writing this.
    529 
    530     if (sNumRoutinesInStack >= ARRAY_COUNT(sRoutineNames)) {
    531         fatal_printf("You're in too many routines");
    532     }
    533 }
    534 
    535 /**
    536  * "I'm out"
    537  * Removes the function name from the stack trace
    538  */
    539 void imout(void) {
    540     s32 i;
    541 
    542     if (--sNumRoutinesInStack < 0) {
    543         for (i = 0; i < ARRAY_COUNT(sRoutineNames); i++) {
    544             if (sRoutineNames[i] != NULL) {
    545                 gd_printf(" - %s\n", sRoutineNames[i]);
    546             } else {
    547                 break;
    548             }
    549         }
    550 
    551         fatal_printf("imout() - imout() called too many times");
    552     }
    553 }
    554 
    555 /**
    556  * Returns a random floating point number between 0 and 1 (inclusive)
    557  * TODO: figure out type of rng generator?
    558  */
    559 f32 gd_rand_float(void) {
    560     u32 temp;
    561     u32 i;
    562     f32 val;
    563 
    564     for (i = 0; i < 4; i++) {
    565         if (sPrimarySeed & 0x80000000) {
    566             sPrimarySeed = sPrimarySeed << 1 | 1;
    567         } else {
    568             sPrimarySeed <<= 1;
    569         }
    570     }
    571     sPrimarySeed += 4;
    572 
    573     /* Seed Switch */
    574     if ((sPrimarySeed ^= gd_get_ostime()) & 1) {
    575         temp = sPrimarySeed;
    576         sPrimarySeed = sSecondarySeed;
    577         sSecondarySeed = temp;
    578     }
    579 
    580     val = (sPrimarySeed & 0xFFFF) / 65535.0; // 65535.0f
    581 
    582     return val;
    583 }
    584 
    585 /**
    586  * Reimplementation of the standard "atoi" function
    587  */
    588 s32 gd_atoi(const char *str) {
    589     char cur;
    590     const char *origstr = str;
    591     s32 curval;
    592     s32 out = 0;
    593     s32 isNegative = FALSE;
    594 
    595     while (TRUE) {
    596         cur = *str++;
    597 
    598         // Each character must be either a digit or a minus sign
    599         if ((cur < '0' || cur > '9') && (cur != '-'))
    600             fatal_printf("gd_atoi() bad number '%s'", origstr);
    601 
    602         if (cur == '-') {
    603             isNegative = TRUE;
    604         } else {
    605             curval = cur - '0';
    606             out += curval & 0xFF;
    607 
    608             if (*str == '\0' || *str == '.' || *str < '0' || *str > '9') {
    609                 break;
    610             }
    611 
    612             out *= 10;
    613         }
    614     }
    615 
    616     if (isNegative) {
    617         out = -out;
    618     }
    619 
    620     return out;
    621 }
    622 
    623 /**
    624  * Like the standard "atof" function, but only supports integer values
    625  */
    626 f64 gd_lazy_atof(const char *str, UNUSED u32 *unk) {
    627     return gd_atoi(str);
    628 }
    629 
    630 static char sHexNumerals[] = {"0123456789ABCDEF"};
    631 
    632 /* 23C018 -> 23C078; orig name: func_8018D848 */
    633 char *format_number_hex(char *str, s32 val) {
    634     s32 shift;
    635 
    636     for (shift = 28; shift > -4; shift -= 4) {
    637         *str++ = sHexNumerals[(val >> shift) & 0xF];
    638     }
    639 
    640     *str = '\0';
    641 
    642     return str;
    643 }
    644 
    645 static s32 sPadNumPrint = 0; // @ 801A82C0
    646 
    647 /* 23C078 -> 23C174; orig name: func_8018D8A8 */
    648 /* padnum = a decimal number with the max desired output width */
    649 char *format_number_decimal(char *str, s32 val, s32 padnum) {
    650     s32 i;
    651 
    652     if (val == 0) {
    653         *str++ = '0';
    654         *str = '\0';
    655         return str;
    656     }
    657 
    658     if (val < 0) {
    659         val = -val;
    660         *str++ = '-';
    661     }
    662 
    663     while (padnum > 0) {
    664         if (padnum <= val) {
    665             sPadNumPrint = TRUE;
    666 
    667             for (i = 0; i < 9; i++) {
    668                 val -= padnum;
    669                 if (val < 0) {
    670                     val += padnum;
    671                     break;
    672                 }
    673             }
    674 
    675             *str++ = i + '0';
    676         } else {
    677             if (sPadNumPrint) {
    678                 *str++ = '0';
    679             }
    680         }
    681 
    682         padnum /= 10;
    683     }
    684 
    685     *str = '\0';
    686 
    687     return str;
    688 }
    689 
    690 /* 23C174 -> 23C1C8; orig name: func_8018D9A4 */
    691 static s32 int_sci_notation(s32 base, s32 significand) {
    692     s32 i;
    693 
    694     for (i = 1; i < significand; i++) {
    695         base *= 10;
    696     }
    697 
    698     return base;
    699 }
    700 
    701 /* 23C1C8 -> 23C468; orig name: func_8018D9F8 */
    702 char *sprint_val_withspecifiers(char *str, union PrintVal val, char *specifiers) {
    703     s32 fracPart; // sp3C
    704     s32 intPart;  // sp38
    705     s32 intPrec;  // sp34
    706     s32 fracPrec; // sp30
    707     UNUSED u8 filler[4];
    708     char cur; // sp2B
    709 
    710     fracPrec = 6;
    711     intPrec = 6;
    712 
    713     while ((cur = *specifiers++)) {
    714         if (cur == 'd') {
    715             sPadNumPrint = FALSE;
    716             str = format_number_decimal(str, val.i, 1000000000);
    717         } else if (cur == 'x') {
    718             sPadNumPrint = TRUE; /* doesn't affect hex printing, though... */
    719             str = format_number_hex(str, val.i);
    720         } else if (cur == 'f') {
    721             intPart = (s32) val.f;
    722             fracPart = (s32)((val.f - (f32) intPart) * (f32) int_sci_notation(10, fracPrec));
    723             sPadNumPrint = FALSE;
    724             str = format_number_decimal(str, intPart, int_sci_notation(10, intPrec));
    725             *str++ = '.';
    726             sPadNumPrint = TRUE;
    727             str = format_number_decimal(str, fracPart, int_sci_notation(10, fracPrec - 1));
    728         } else if (cur >= '0' && cur <= '9') {
    729             cur = cur - '0';
    730             intPrec = cur;
    731             if (*specifiers++) {
    732                 fracPrec = (*specifiers++) - '0';
    733             }
    734         } else {
    735             gd_strcpy(str, "<BAD TYPE>");
    736             str += 10;
    737         }
    738     }
    739 
    740     return str;
    741 }
    742 
    743 /* 23C468 -> 23C4AC; orig name: func_8018DC98 */
    744 void gd_strcpy(char *dst, const char *src) {
    745     while ((*dst++ = *src++)) {
    746         ;
    747     }
    748 }
    749 
    750 /* 23C4AC -> 23C52C; not called; orig name: Unknown8018DCDC */
    751 void ascii_to_uppercase(char *str) {
    752     char c;
    753 
    754     while ((c = *str)) {
    755         if (c >= 'a' && c <= 'z') {
    756             *str = c & 0xDF;
    757         }
    758         str++;
    759     }
    760 }
    761 
    762 /* 23C52C -> 23C5A8; orig name: func_8018DD5C */
    763 char *gd_strdup(const char *src) {
    764     char *dst; // sp24
    765 
    766     dst = gd_malloc_perm((gd_strlen(src) + 1) * sizeof(char));
    767 
    768     if (dst == NULL) {
    769         fatal_printf("gd_strdup(): out of memory");
    770     }
    771     gd_strcpy(dst, src);
    772 
    773     return dst;
    774 }
    775 
    776 /* 23C5A8 -> 23C5FC; orig name: func_8018DDD8 */
    777 u32 gd_strlen(const char *str) {
    778     u32 len = 0;
    779 
    780     while (*str++) {
    781         len++;
    782     }
    783 
    784     return len;
    785 }
    786 
    787 /* 23C5FC -> 23C680; orig name: func_8018DE2C */
    788 char *gd_strcat(char *dst, const char *src) {
    789     while (*dst++) {
    790         ;
    791     }
    792 
    793     if (*src) {
    794         dst--;
    795         while ((*dst++ = *src++)) {
    796             ;
    797         }
    798     }
    799 
    800     return --dst;
    801 }
    802 
    803 /* 23C67C -> 23C728; orig name: func_8018DEB0 */
    804 /* Returns a bool, not the position of the mismatch */
    805 s32 gd_str_not_equal(const char *str1, const char *str2) {
    806     while (*str1 && *str2) {
    807         if (*str1++ != *str2++) {
    808             return TRUE;
    809         }
    810     }
    811 
    812     return *str1 != '\0' || *str2 != '\0';
    813 }
    814 
    815 /* 23C728 -> 23C7B8; orig name; func_8018DF58 */
    816 s32 gd_str_contains(const char *str1, const char *str2) {
    817     const char *startsub = str2;
    818 
    819     while (*str1 && *str2) {
    820         if (*str1++ != *str2++) {
    821             str2 = startsub;
    822         }
    823     }
    824 
    825     return !*str2;
    826 }
    827 
    828 /* 23C7B8 -> 23C7DC; orig name: func_8018DFE8 */
    829 s32 gd_feof(struct GdFile *f) {
    830     return f->flags & 0x4;
    831 }
    832 
    833 /* 23C7DC -> 23C7FC; orig name: func_8018E00C */
    834 void gd_set_feof(struct GdFile *f) {
    835     f->flags |= 0x4;
    836 }
    837 
    838 /* 23C7FC -> 23CA0C */
    839 struct GdFile *gd_fopen(const char *filename, const char *mode) {
    840     struct GdFile *f; // sp74
    841     char *loadedname; // sp70
    842     u32 i;            // sp6C
    843     UNUSED u8 filler[4];
    844     struct UnkBufThing buf; // sp24
    845     u8 *bufbytes;           // sp20
    846     u8 *fileposptr;         // sp1C
    847     s32 filecsr;            // sp18
    848 
    849     filecsr = 0;
    850 
    851     while (TRUE) {
    852         bufbytes = (u8 *) &buf;
    853         for (i = 0; i < sizeof(struct UnkBufThing); i++) {
    854             *bufbytes++ = gGdStreamBuffer[filecsr++];
    855         }
    856         stub_renderer_13(&buf);
    857         fileposptr = &gGdStreamBuffer[filecsr];
    858         filecsr += buf.size;
    859 
    860         loadedname = buf.name;
    861 
    862         if (buf.size == 0) {
    863             break;
    864         }
    865         if (!gd_str_not_equal(filename, loadedname)) {
    866             break;
    867         }
    868     }
    869 
    870     if (buf.size == 0) {
    871         fatal_printf("gd_fopen() File not found '%s'", filename);
    872         return NULL;
    873     }
    874 
    875     f = gd_malloc_perm(sizeof(struct GdFile));
    876     if (f == NULL) {
    877         fatal_printf("gd_fopen() Out of memory loading '%s'", filename);
    878         return NULL;
    879     }
    880 
    881     f->stream = (s8 *) fileposptr;
    882     f->size = buf.size;
    883     f->pos = f->flags = 0;
    884     if (gd_str_contains(mode, "w")) {
    885         f->flags |= 0x1;
    886     }
    887     if (gd_str_contains(mode, "b")) {
    888         f->flags |= 0x2;
    889     }
    890 
    891     return f;
    892 }
    893 
    894 /* 23CA0C -> 23CB38; orig name: func_8018E23C */
    895 s32 gd_fread(s8 *buf, s32 bytes, UNUSED s32 count, struct GdFile *f) {
    896     s32 bytesToRead = bytes;
    897     s32 bytesread;
    898 
    899     if (f->pos + bytesToRead > f->size) {
    900         bytesToRead = f->size - f->pos;
    901     }
    902 
    903     if (bytesToRead == 0) {
    904         gd_set_feof(f);
    905         return -1;
    906     }
    907 
    908     bytesread = bytesToRead;
    909     while (bytesread--) {
    910         *buf++ = f->stream[f->pos++];
    911     }
    912 
    913     return bytesToRead;
    914 }
    915 
    916 /* 23CB38 -> 23CB54; orig name: func_8018E368 */
    917 void gd_fclose(UNUSED struct GdFile *f) {
    918     return;
    919 }
    920 
    921 /* 23CB54 -> 23CB70; orig name: func_8018E384 */
    922 u32 gd_get_file_size(struct GdFile *f) {
    923     return f->size;
    924 }
    925 
    926 /* 23CB70 -> 23CBA8; orig name: func_8018E3A0 */
    927 s32 is_newline(char c) {
    928     return c == '\r' || c == '\n';
    929 }
    930 
    931 /* 23CBA8 -> 23CCF0; orig name: func_8018E3D8 */
    932 s32 gd_fread_line(char *buf, u32 size, struct GdFile *f) {
    933     signed char c;
    934     u32 pos = 0;
    935     UNUSED u8 filler[4];
    936 
    937     do {
    938         if (gd_fread(&c, 1, 1, f) == -1) {
    939             break;
    940         }
    941     } while (is_newline(c));
    942 
    943     while (!is_newline(c)) {
    944         if (c == -1) {
    945             break;
    946         }
    947         if (pos > size) {
    948             break;
    949         }
    950         buf[pos++] = c;
    951         if (gd_fread(&c, 1, 1, f) == -1) {
    952             break;
    953         }
    954     }
    955     buf[pos++] = '\0';
    956 
    957     return pos;
    958 }