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_ */