sm64

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

crash.patch (12335B)


      1 diff --git a/asm/crash.s b/asm/crash.s
      2 new file mode 100644
      3 index 00000000..033bf952
      4 --- /dev/null
      5 +++ b/asm/crash.s
      6 @@ -0,0 +1,153 @@
      7 +// SM64 Crash Handler
      8 +// See Readme below.
      9 +
     10 +#include "macros.inc"
     11 +
     12 +/* ---------------------------------------------------------------
     13 + * IMPORTANT README:
     14 + * ---------------------------------------------------------------
     15 + * Frame buffer emulation is required. To enable it in GlideN64,
     16 + * check "Emulate frame buffer" and "Render frame buffer to output"
     17 + * in the "Frame buffer" tab.
     18 + *
     19 + * Your emulator's CPU core style should be set to interpreter for best results.
     20 + *
     21 + * See the DEBUG_ASSERT macro on how to call the crash screen for
     22 + * detected exceptions.
     23 + *
     24 + */
     25 +
     26 +.set noat
     27 +.set noreorder
     28 +.set gp=64
     29 +
     30 +.set COP0_CAUSE, $13
     31 +.set COP0_EPC, $14
     32 +.set COP0_BADVADDR, $8
     33 +
     34 +glabel crashFont
     35 +    .incbin "enhancements/crash_font.bin"
     36 +    .align 4
     37 +
     38 +glabel exceptionRegContext
     39 +    .fill 0x108
     40 +
     41 +glabel pAssertFile
     42 +    .dword 0
     43 +glabel nAssertLine
     44 +    .dword 0
     45 +glabel pAssertExpression
     46 +    .dword 0
     47 +glabel nAssertStopProgram
     48 +    .dword 0
     49 +
     50 +glabel _n64_assert
     51 +    lui  $at, %hi(pAssertFile)
     52 +    sw   $a0, %lo(pAssertFile)($at)
     53 +    lui  $at, %hi(nAssertLine)
     54 +    sw   $a1, %lo(nAssertLine)($at)
     55 +    lui  $at, %hi(pAssertExpression)
     56 +    sw   $a2, %lo(pAssertExpression)($at)
     57 +    lui  $at, %hi(nAssertStopProgram)
     58 +    sw   $a3, %lo(nAssertStopProgram)($at)
     59 +    beqz $a3, .end_2
     60 +     nop
     61 +    syscall // trigger crash screen
     62 +.end_2:
     63 +    jr $ra
     64 +     nop
     65 +
     66 +glabel cop0_get_cause
     67 +    jr   $ra
     68 +     mfc0 $v0, COP0_CAUSE
     69 +
     70 +glabel cop0_get_epc
     71 +    jr   $ra
     72 +     mfc0 $v0, COP0_EPC
     73 +
     74 +glabel cop0_get_badvaddr
     75 +    jr   $ra
     76 +     mfc0 $v0, COP0_BADVADDR
     77 +
     78 +// If the error code field of cop0's cause register is non-zero,
     79 +// draw crash details to the screen and hang
     80 +
     81 +// If there wasn't an error, continue to the original handler
     82 +
     83 +glabel __crash_handler_entry
     84 +    mfc0  $k1, COP0_CAUSE
     85 +    andi  $k1, $k1, (0x1F << 2)
     86 +    beqzl $k1, .end2  // exit if ExCode is 0
     87 +     lui   $k0, %hi(__osException)
     88 +    la    $k0, exceptionRegContext
     89 +    sd    $zero, 0x018 ($k0)
     90 +    sd    $at, 0x020 ($k0)
     91 +    sd    $v0, 0x028 ($k0)
     92 +    sd    $v1, 0x030 ($k0)
     93 +    sd    $a0, 0x038 ($k0)
     94 +    sd    $a1, 0x040 ($k0)
     95 +    sd    $a2, 0x048 ($k0)
     96 +    sd    $a3, 0x050 ($k0)
     97 +    sd    $t0, 0x058 ($k0)
     98 +    sd    $t1, 0x060 ($k0)
     99 +    sd    $t2, 0x068 ($k0)
    100 +    sd    $t3, 0x070 ($k0)
    101 +    sd    $t4, 0x078 ($k0)
    102 +    sd    $t5, 0x080 ($k0)
    103 +    sd    $t6, 0x088 ($k0)
    104 +    sd    $t7, 0x090 ($k0)
    105 +    sd    $s0, 0x098 ($k0)
    106 +    sd    $s1, 0x0A0 ($k0)
    107 +    sd    $s2, 0x0A8 ($k0)
    108 +    sd    $s3, 0x0B0 ($k0)
    109 +    sd    $s4, 0x0B8 ($k0)
    110 +    sd    $s5, 0x0C0 ($k0)
    111 +    sd    $s6, 0x0C8 ($k0)
    112 +    sd    $s7, 0x0D0 ($k0)
    113 +    sd    $t8, 0x0D8 ($k0)
    114 +    sd    $t9, 0x0E0 ($k0)
    115 +    sd    $gp, 0x0E8 ($k0)
    116 +    sd    $sp, 0x0F0 ($k0)
    117 +    sd    $fp, 0x0F8 ($k0)
    118 +    sd    $ra, 0x100 ($k0)
    119 +    // cop unusable exception fired twice on startup so we'll ignore it for now
    120 +    li    $t0, (0x0B << 2)
    121 +    beq   $k1, $t0, .end
    122 +     nop
    123 +    jal   show_crash_screen_and_hang
    124 +     nop
    125 +    .end:
    126 +    ld    $at, 0x020 ($k0)
    127 +    ld    $v0, 0x028 ($k0)
    128 +    ld    $v1, 0x030 ($k0)
    129 +    ld    $a0, 0x038 ($k0)
    130 +    ld    $a1, 0x040 ($k0)
    131 +    ld    $a2, 0x048 ($k0)
    132 +    ld    $a3, 0x050 ($k0)
    133 +    ld    $t0, 0x058 ($k0)
    134 +    ld    $t1, 0x060 ($k0)
    135 +    ld    $t2, 0x068 ($k0)
    136 +    ld    $t3, 0x070 ($k0)
    137 +    ld    $t4, 0x078 ($k0)
    138 +    ld    $t5, 0x080 ($k0)
    139 +    ld    $t6, 0x088 ($k0)
    140 +    ld    $t7, 0x090 ($k0)
    141 +    ld    $s0, 0x098 ($k0)
    142 +    ld    $s1, 0x0A0 ($k0)
    143 +    ld    $s2, 0x0A8 ($k0)
    144 +    ld    $s3, 0x0B0 ($k0)
    145 +    ld    $s4, 0x0B8 ($k0)
    146 +    ld    $s5, 0x0C0 ($k0)
    147 +    ld    $s6, 0x0C8 ($k0)
    148 +    ld    $s7, 0x0D0 ($k0)
    149 +    ld    $t8, 0x0D8 ($k0)
    150 +    ld    $t9, 0x0E0 ($k0)
    151 +    ld    $gp, 0x0E8 ($k0)
    152 +    ld    $sp, 0x0F0 ($k0)
    153 +    ld    $fp, 0x0F8 ($k0)
    154 +    ld    $ra, 0x100 ($k0)
    155 +    lui   $k0, %hi(__osException)
    156 +    .end2:
    157 +    addiu $k0, $k0, %lo(__osException)
    158 +    jr    $k0 // run the original handler
    159 +     nop
    160 diff --git a/lib/asm/__osExceptionPreamble.s b/lib/asm/__osExceptionPreamble.s
    161 index 4e841ea0..ab0c7d13 100644
    162 --- a/lib/asm/__osExceptionPreamble.s
    163 +++ b/lib/asm/__osExceptionPreamble.s
    164 @@ -11,8 +11,8 @@
    165  #endif
    166  
    167  glabel __osExceptionPreamble
    168 -    lui   $k0, %hi(__osException)
    169 -    addiu $k0, %lo(__osException)
    170 +    lui   $k0, %hi(__crash_handler_entry)
    171 +    addiu $k0, %lo(__crash_handler_entry)
    172      jr    $k0
    173       nop
    174  
    175 diff --git a/sm64.ld b/sm64.ld
    176 index da9bc4dd..25ac57e8 100755
    177 --- a/sm64.ld
    178 +++ b/sm64.ld
    179 @@ -117,6 +117,7 @@ SECTIONS
    180          BUILD_DIR/src/game/rendering_graph_node.o(.text);
    181          BUILD_DIR/src/game/profiler.o(.text);
    182          BUILD_DIR/asm/decompress.o(.text);
    183 +        BUILD_DIR/asm/crash.o(.text);
    184          BUILD_DIR/src/game/camera.o(.text);
    185          BUILD_DIR/src/game/debug_course.o(.text);
    186          BUILD_DIR/src/game/object_list_processor.o(.text);
    187 diff --git a/src/game/crash.c b/src/game/crash.c
    188 new file mode 100644
    189 index 00000000..716adfbd
    190 --- /dev/null
    191 +++ b/src/game/crash.c
    192 @@ -0,0 +1,260 @@
    193 +/* SM64 Crash Handler */
    194 +
    195 +#include <sm64.h>
    196 +
    197 +#include "crash.h"
    198 +
    199 +extern u32 exceptionRegContext[];
    200 +
    201 +extern char *pAssertFile;
    202 +extern int nAssertLine;
    203 +extern char *pAssertExpression;
    204 +extern int nAssertStopProgram;
    205 +
    206 +u16 fbFillColor = 0xFFFF;
    207 +u16 fbShadeColor = 0x0000;
    208 +u16 *fbAddress = NULL;
    209 +
    210 +extern u8 crashFont[];
    211 +
    212 +const char *szErrCodes[] = {
    213 +    "INTERRUPT",
    214 +    "TLB MOD",
    215 +    "UNMAPPED LOAD ADDR",
    216 +    "UNMAPPED STORE ADDR",
    217 +    "BAD LOAD ADDR",
    218 +    "BAD STORE ADDR",
    219 +    "BUS ERR ON INSTR FETCH",
    220 +    "BUS ERR ON LOADSTORE",
    221 +    "SYSCALL",
    222 +    "BREAKPOINT",
    223 +    "UNKNOWN INSTR",
    224 +    "COP UNUSABLE",
    225 +    "ARITHMETIC OVERFLOW",
    226 +    "TRAP EXC",
    227 +    "VIRTUAL COHERENCY INSTR",
    228 +    "FLOAT EXC",
    229 +};
    230 +
    231 +const char *szGPRegisters1[] = { "R0", "AT", "V0", "V1", "A0", "A1", "A2", "A3",
    232 +                                 "T0", "T1", "T2", "T3", "T4", "T5", "T6", NULL };
    233 +
    234 +const char *szGPRegisters2[] = { "T7", "S0", "S1", "S2", "S3", "S4",
    235 +                                 "S5", "S6", "S7", "T8", "T9", /*"K0", "K1",*/
    236 +                                 "GP", "SP", "FP", "RA", NULL };
    237 +
    238 +int crash_strlen(char *str) {
    239 +    int len = 0;
    240 +    while (*str++) {
    241 +        len++;
    242 +    }
    243 +    return len;
    244 +}
    245 +
    246 +void show_crash_screen_and_hang(void) {
    247 +    u32 cause;
    248 +    u32 epc;
    249 +    u8 errno;
    250 +
    251 +    fb_set_address((void *) (*(u32 *) 0xA4400004 | 0x80000000)); // replace me
    252 +
    253 +    cause = cop0_get_cause();
    254 +    epc = cop0_get_epc();
    255 +
    256 +    errno = (cause >> 2) & 0x1F;
    257 +
    258 +    if (nAssertStopProgram == 0) {
    259 +        fbFillColor = 0x6253;
    260 +        fb_fill(10, 10, 300, 220);
    261 +
    262 +        fb_print_str(80, 20, "AN ERROR HAS OCCURRED!");
    263 +        fb_print_int_hex(80, 30, errno, 8);
    264 +        fb_print_str(95, 30, szErrCodes[errno]);
    265 +
    266 +        if (errno >= 2 && errno <= 5) {
    267 +            /*
    268 +            2 UNMAPPED LOAD ADDR
    269 +            3 UNMAPPED STORE ADDR
    270 +            4 BAD LOAD ADDR
    271 +            5 BAD STORE ADDR
    272 +            */
    273 +            u32 badvaddr = cop0_get_badvaddr();
    274 +
    275 +            fb_print_str(145, 50, "VA");
    276 +            fb_print_int_hex(160, 50, badvaddr, 32);
    277 +        }
    278 +    } else {
    279 +        int afterFileX;
    280 +        int exprBoxWidth;
    281 +        fbFillColor = 0x5263;
    282 +        fb_fill(10, 10, 300, 220);
    283 +
    284 +        fb_print_str(80, 20, "ASSERTION FAILED!");
    285 +
    286 +        afterFileX = fb_print_str(80, 30, pAssertFile);
    287 +        fb_print_str(afterFileX, 30, ":");
    288 +        fb_print_uint(afterFileX + 5, 30, nAssertLine);
    289 +
    290 +        exprBoxWidth = (crash_strlen(pAssertExpression) * 5) + 2;
    291 +        fbFillColor = 0x0001;
    292 +        fb_fill(80 - 1, 40 - 1, exprBoxWidth, 10);
    293 +        fb_print_str(80, 40, pAssertExpression);
    294 +    }
    295 +
    296 +    fb_print_str(80, 50, "PC");
    297 +    fb_print_int_hex(95, 50, epc, 32);
    298 +
    299 +    fb_print_gpr_states(80, 70, szGPRegisters1, &exceptionRegContext[6 + 0]);
    300 +    fb_print_gpr_states(145, 70, szGPRegisters2, &exceptionRegContext[6 + 15 * 2]);
    301 +
    302 +    fb_swap();
    303 +    osWritebackDCacheAll();
    304 +
    305 +    while (1) // hang forever
    306 +    {
    307 +        UNUSED volatile int t = 0; // keep pj64 happy
    308 +    }
    309 +}
    310 +
    311 +u8 ascii_to_idx(char c) {
    312 +    return c - 0x20;
    313 +}
    314 +
    315 +void fb_set_address(void *address) {
    316 +    fbAddress = (u16 *) address;
    317 +}
    318 +
    319 +void fb_swap() {
    320 +    // update VI frame buffer register
    321 +    // todo other registers
    322 +    *(u32 *) (0xA4400004) = (u32) fbAddress & 0x00FFFFFF;
    323 +}
    324 +
    325 +void fb_fill(int baseX, int baseY, int width, int height) {
    326 +    int y, x;
    327 +
    328 +    for (y = baseY; y < baseY + height; y++) {
    329 +        for (x = baseX; x < baseX + width; x++) {
    330 +            fbAddress[y * 320 + x] = fbFillColor;
    331 +        }
    332 +    }
    333 +}
    334 +
    335 +void fb_draw_char(int x, int y, u8 idx) {
    336 +    u16 *out = &fbAddress[y * 320 + x];
    337 +    const u8 *in = &crashFont[idx * 3];
    338 +    int nbyte;
    339 +    int nrow;
    340 +    int ncol;
    341 +
    342 +    for (nbyte = 0; nbyte < 3; nbyte++) {
    343 +        u8 curbyte = in[nbyte];
    344 +        for (nrow = 0; nrow < 2; nrow++) {
    345 +            for (ncol = 0; ncol < 4; ncol++) {
    346 +                u8 px = curbyte & (1 << (7 - (nrow * 4 + ncol)));
    347 +                if (px != 0) {
    348 +                    out[ncol] = fbFillColor;
    349 +                }
    350 +            }
    351 +            out += 320;
    352 +        }
    353 +    }
    354 +}
    355 +
    356 +void fb_draw_char_shaded(int x, int y, u8 idx) {
    357 +    fbFillColor = 0x0001;
    358 +    fb_draw_char(x - 1, y + 1, idx);
    359 +
    360 +    fbFillColor = 0xFFFF;
    361 +    fb_draw_char(x, y, idx);
    362 +}
    363 +
    364 +int fb_print_str(int x, int y, const char *str) {
    365 +    while (1) {
    366 +        int yoffs = 0;
    367 +        u8 idx;
    368 +        char c = *str++;
    369 +
    370 +        if (c == '\0') {
    371 +            break;
    372 +        }
    373 +
    374 +        if (c == ' ') {
    375 +            x += 5;
    376 +            continue;
    377 +        }
    378 +
    379 +        switch (c) {
    380 +            case 'j':
    381 +            case 'g':
    382 +            case 'p':
    383 +            case 'q':
    384 +            case 'y':
    385 +            case 'Q':
    386 +                yoffs = 1;
    387 +                break;
    388 +            case ',':
    389 +                yoffs = 2;
    390 +                break;
    391 +        }
    392 +
    393 +        idx = ascii_to_idx(c);
    394 +        fb_draw_char_shaded(x, y + yoffs, idx);
    395 +        x += 5;
    396 +    }
    397 +
    398 +    return x;
    399 +}
    400 +
    401 +void fb_print_int_hex(int x, int y, u32 value, int nbits) {
    402 +    nbits -= 4;
    403 +
    404 +    while (nbits >= 0) {
    405 +        int nib = ((value >> nbits) & 0xF);
    406 +        u8 idx;
    407 +
    408 +        if (nib > 9) {
    409 +            idx = ('A' - 0x20) + (nib - 0xa);
    410 +        } else {
    411 +            idx = ('0' - 0x20) + nib;
    412 +        }
    413 +
    414 +        fb_draw_char_shaded(x, y, idx);
    415 +        x += 5;
    416 +
    417 +        nbits -= 4;
    418 +    }
    419 +}
    420 +
    421 +int fb_print_uint(int x, int y, u32 value) {
    422 +    int nchars = 0;
    423 +
    424 +    int v = value;
    425 +    int i;
    426 +    while (v /= 10) {
    427 +        nchars++;
    428 +    }
    429 +
    430 +    x += nchars * 5;
    431 +
    432 +    for (i = nchars; i >= 0; i--) {
    433 +        fb_draw_char_shaded(x, y, ('0' - 0x20) + (value % 10));
    434 +        value /= 10;
    435 +        x -= 5;
    436 +    }
    437 +
    438 +    return (x + nchars * 5);
    439 +}
    440 +
    441 +void fb_print_gpr_states(int x, int y, const char *regNames[], u32 *regContext) {
    442 +    int i;
    443 +    for (i = 0;; i++) {
    444 +        if (regNames[i] == NULL) {
    445 +            break;
    446 +        }
    447 +
    448 +        fb_print_str(x, y, regNames[i]);
    449 +        fb_print_int_hex(x + 15, y, regContext[i * 2 + 1], 32);
    450 +        y += 10;
    451 +    }
    452 +}
    453 diff --git a/src/game/crash.h b/src/game/crash.h
    454 new file mode 100644
    455 index 00000000..1386930d
    456 --- /dev/null
    457 +++ b/src/game/crash.h
    458 @@ -0,0 +1,28 @@
    459 +#ifndef _CRASH_H_
    460 +#define _CRASH_H_
    461 +
    462 +#include <types.h>
    463 +
    464 +#define CRASH_SCREEN_INCLUDED 1
    465 +
    466 +extern u32 cop0_get_cause(void);
    467 +extern u32 cop0_get_epc(void);
    468 +extern u32 cop0_get_badvaddr(void);
    469 +
    470 +extern void _n64_assert(const char* pFile, int nLine, const char *pExpression, int nStopProgram);
    471 +
    472 +extern u8 __crash_handler_entry[];
    473 +
    474 +void show_crash_screen_and_hang(void);
    475 +u8 ascii_to_idx(char c);
    476 +void fb_set_address(void *address);
    477 +void fb_swap(void);
    478 +void fb_fill(int baseX, int baseY, int width, int height);
    479 +void fb_draw_char(int x, int y, u8 idx);
    480 +void fb_draw_char_shaded(int x, int y, u8 idx);
    481 +int fb_print_str(int x, int y, const char *str);
    482 +int fb_print_uint(int x, int y, u32 value);
    483 +void fb_print_int_hex(int x, int y, u32 value, int nbits);
    484 +void fb_print_gpr_states(int x, int y, const char* regStrs[], u32 *regContext);
    485 +
    486 +#endif /* _CRASH_H_ */