sm64

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

mem_error_screen.patch (10416B)


      1 diff --git a/Makefile b/Makefile
      2 index f50b7622..124c7ec6 100644
      3 --- a/Makefile
      4 +++ b/Makefile
      5 @@ -478,6 +478,7 @@ $(BUILD_DIR)/include/text_strings.h: $(BUILD_DIR)/include/text_menu_strings.h
      6  $(BUILD_DIR)/src/menu/file_select.o: $(BUILD_DIR)/include/text_strings.h
      7  $(BUILD_DIR)/src/menu/star_select.o: $(BUILD_DIR)/include/text_strings.h
      8  $(BUILD_DIR)/src/game/ingame_menu.o: $(BUILD_DIR)/include/text_strings.h
      9 +$(BUILD_DIR)/src/game/mem_error_screen.o: $(BUILD_DIR)/include/text_strings.h
     10  
     11  
     12  #==============================================================================#
     13 diff --git a/include/segments.h b/include/segments.h
     14 index a97d6ee8..186c968e 100644
     15 --- a/include/segments.h
     16 +++ b/include/segments.h
     17 @@ -3,6 +3,9 @@
     18  
     19  #include "config.h"
     20  
     21 +/* Use expansion pack RAM */
     22 +#define USE_EXT_RAM 1
     23 +
     24  /*
     25   * Memory addresses for segments. Ideally, this header file would not be
     26   * needed, and the addresses would be defined in sm64.ld and linker-inserted
     27 diff --git a/include/text_strings.h.in b/include/text_strings.h.in
     28 index 749179b1..2f6f7a3c 100644
     29 --- a/include/text_strings.h.in
     30 +++ b/include/text_strings.h.in
     31 @@ -25,6 +25,11 @@
     32  #define TEXT_PAUSE _("PAUSE") // Pause text, Castle Courses
     33  #define TEXT_HUD_CONGRATULATIONS _("CONGRATULATIONS") // Course Complete Text, Bowser Courses
     34  
     35 +// Memory Expansion Error Screen
     36 +#define TEXT_CONSOLE_8MB _("If you're using an N64 console, then you will need to buy an\nExpansion Pak to play this ROM hack.")
     37 +#define TEXT_PJ64 _("If you are using PJ64 1.6, go to:\nOptions > Settings > Rom Settings Tab > Memory Size\nthen select 8 MB from the drop-down box.")
     38 +#define TEXT_PJ64_2 _("If you are using PJ64 2.X, go to:\nOptions > Settings > Config: > Memory Size, select 8 MB")
     39 +
     40  #if defined(VERSION_JP) || defined(VERSION_SH) || defined(VERSION_CN)
     41  
     42  /**
     43 diff --git a/levels/entry.c b/levels/entry.c
     44 index 17c773ed..677a5ae9 100644
     45 --- a/levels/entry.c
     46 +++ b/levels/entry.c
     47 @@ -15,3 +15,12 @@ const LevelScript level_script_entry[] = {
     48      EXECUTE(/*seg*/ 0x14, /*script*/ _introSegmentRomStart, /*scriptEnd*/ _introSegmentRomEnd, /*entry*/ level_intro_splash_screen),
     49      JUMP(/*target*/ level_script_entry),
     50  };
     51 +
     52 +const LevelScript level_script_entry_error_screen[] = {
     53 +    INIT_LEVEL(),
     54 +    SLEEP(/*frames*/ 2),
     55 +    BLACKOUT(/*active*/ FALSE),
     56 +    SET_REG(/*value*/ 0),
     57 +    EXECUTE(/*seg*/ 0x14, /*script*/ _introSegmentRomStart, /*scriptEnd*/ _introSegmentRomEnd, /*entry*/ level_intro_entry_error_screen),
     58 +    JUMP(/*target*/ level_script_entry_error_screen),
     59 +};
     60 diff --git a/levels/intro/geo.c b/levels/intro/geo.c
     61 index 30a87806..6bf7b79a 100644
     62 --- a/levels/intro/geo.c
     63 +++ b/levels/intro/geo.c
     64 @@ -15,6 +15,24 @@
     65  
     66  #include "levels/intro/header.h"
     67  
     68 +const GeoLayout intro_geo_error_screen[] = {
     69 +     GEO_NODE_SCREEN_AREA(0, SCREEN_WIDTH/2, SCREEN_HEIGHT/2, SCREEN_WIDTH/2, SCREEN_HEIGHT/2),
     70 +     GEO_OPEN_NODE(),
     71 +          GEO_ZBUFFER(0),
     72 +          GEO_OPEN_NODE(),
     73 +                GEO_NODE_ORTHO(100),
     74 +                GEO_OPEN_NODE(),
     75 +                     GEO_BACKGROUND_COLOR(0x0001),
     76 +                GEO_CLOSE_NODE(),
     77 +          GEO_CLOSE_NODE(),
     78 +          GEO_ZBUFFER(0),
     79 +          GEO_OPEN_NODE(),
     80 +                GEO_ASM(0, geo18_display_error_message),
     81 +          GEO_CLOSE_NODE(),
     82 +     GEO_CLOSE_NODE(),
     83 +     GEO_END(),
     84 +};
     85 +
     86  // 0x0E0002D0
     87  const GeoLayout intro_geo_0002D0[] = {
     88      GEO_NODE_SCREEN_AREA(0, SCREEN_WIDTH/2, SCREEN_HEIGHT/2, SCREEN_WIDTH/2, SCREEN_HEIGHT/2),
     89 diff --git a/levels/intro/header.h b/levels/intro/header.h
     90 index 99277e86..04797cd7 100644
     91 --- a/levels/intro/header.h
     92 +++ b/levels/intro/header.h
     93 @@ -26,4 +26,8 @@ extern const LevelScript script_intro_L3[];
     94  extern const LevelScript script_intro_L4[];
     95  extern const LevelScript script_intro_L5[];
     96  
     97 +extern const GeoLayout intro_geo_error_screen[];
     98 +extern const LevelScript level_intro_entry_error_screen[];
     99 +extern Gfx *geo18_display_error_message(u32 run, UNUSED struct GraphNode *sp44, UNUSED u32 sp48);
    100 +
    101  #endif
    102 diff --git a/levels/intro/script.c b/levels/intro/script.c
    103 index 04b8fc4c..ca9058c4 100644
    104 --- a/levels/intro/script.c
    105 +++ b/levels/intro/script.c
    106 @@ -18,6 +18,21 @@
    107  #include "make_const_nonconst.h"
    108  #include "levels/intro/header.h"
    109  
    110 +const LevelScript level_intro_entry_error_screen[] = {
    111 +    INIT_LEVEL(),
    112 +    FIXED_LOAD(/*loadAddr*/ _goddardSegmentStart, /*romStart*/ _goddardSegmentRomStart, /*romEnd*/ _goddardSegmentRomEnd),
    113 +    LOAD_MIO0(/*seg*/ 0x07, _intro_segment_7SegmentRomStart, _intro_segment_7SegmentRomEnd),
    114 +    ALLOC_LEVEL_POOL(),
    115 +
    116 +    AREA(/*index*/ 1, intro_geo_error_screen),
    117 +    END_AREA(),
    118 +
    119 +    FREE_LEVEL_POOL(),
    120 +    LOAD_AREA(/*area*/ 1),
    121 +    SLEEP(/*frames*/ 32767),
    122 +    EXIT_AND_EXECUTE(/*seg*/ 0x14, _introSegmentRomStart, _introSegmentRomEnd, level_intro_entry_error_screen),
    123 +};
    124 +
    125  const LevelScript level_intro_splash_screen[] = {
    126      INIT_LEVEL(),
    127      FIXED_LOAD(/*loadAddr*/ _goddardSegmentStart, /*romStart*/ _goddardSegmentRomStart, /*romEnd*/ _goddardSegmentRomEnd),
    128 diff --git a/src/engine/level_script.h b/src/engine/level_script.h
    129 index d41a91c8..7d047236 100644
    130 --- a/src/engine/level_script.h
    131 +++ b/src/engine/level_script.h
    132 @@ -6,6 +6,7 @@
    133  struct LevelCommand;
    134  
    135  extern u8 level_script_entry[];
    136 +extern u8 level_script_entry_error_screen[];
    137  
    138  struct LevelCommand *level_script_execute(struct LevelCommand *cmd);
    139  
    140 diff --git a/src/game/main.c b/src/game/main.c
    141 index 1a9d9e7e..f4f7a9e5 100644
    142 --- a/src/game/main.c
    143 +++ b/src/game/main.c
    144 @@ -11,6 +11,7 @@
    145  #include "segments.h"
    146  #include "main.h"
    147  #include "rumble_init.h"
    148 +#include "mem_error_screen.h"
    149  
    150  // Message IDs
    151  #define MESG_SP_COMPLETE 100
    152 @@ -131,6 +132,10 @@ void alloc_pool(void) {
    153      void *start = (void *) SEG_POOL_START;
    154      void *end = (void *) SEG_POOL_END;
    155  
    156 +    // Detect memory size
    157 +    if (does_pool_end_lie_out_of_bounds(end))
    158 +        end = (void *)SEG_POOL_END_4MB;
    159 +
    160      main_pool_init(start, end);
    161      gEffectsMemoryPool = mem_pool_init(0x4000, MEMORY_POOL_LEFT);
    162  }
    163 @@ -336,7 +341,10 @@ void thread3_main(UNUSED void *arg) {
    164      create_thread(&gSoundThread, 4, thread4_sound, NULL, gThread4Stack + 0x2000, 20);
    165      osStartThread(&gSoundThread);
    166  
    167 -    create_thread(&gGameLoopThread, 5, thread5_game_loop, NULL, gThread5Stack + 0x2000, 10);
    168 +    if (!gNotEnoughMemory)
    169 +        create_thread(&gGameLoopThread, 5, thread5_game_loop, NULL, gThread5Stack + 0x2000, 10);
    170 +    else
    171 +        create_thread(&gGameLoopThread, 5, thread5_mem_error_message_loop, NULL, gThread5Stack + 0x2000, 10);
    172      osStartThread(&gGameLoopThread);
    173  
    174      while (TRUE) {
    175 diff --git a/src/game/mem_error_screen.c b/src/game/mem_error_screen.c
    176 new file mode 100644
    177 index 00000000..f432927c
    178 --- /dev/null
    179 +++ b/src/game/mem_error_screen.c
    180 @@ -0,0 +1,104 @@
    181 +/* clang-format off */
    182 +/*
    183 + * mem_error_screen.inc.c
    184 + *
    185 + * This enhancement should be used for ROM hacks that require the expansion pak.
    186 + *
    187 + */
    188 +/* clang-format on */
    189 +
    190 +#include <types.h>
    191 +#include "segments.h"
    192 +#include "text_strings.h"
    193 +#include "game_init.h"
    194 +#include "main.h"
    195 +#include "print.h"
    196 +#include "ingame_menu.h"
    197 +#include "segment2.h"
    198 +#include "../engine/level_script.h"
    199 +
    200 +// Ensure that USE_EXT_RAM is defined.
    201 +#ifndef USE_EXT_RAM
    202 +#error You have to define USE_EXT_RAM in 'include/segments.h'
    203 +#endif
    204 +
    205 +// Require 8 MB of RAM, even if the pool doesn't go into extended memory.
    206 +// Change the '8' to whatever MB limit you want.
    207 +// Note: only special emulators allow for RAM sizes above 8 MB.
    208 +#define REQUIRED_MIN_MEM_SIZE 1048576 * 8
    209 +
    210 +u8 gNotEnoughMemory = FALSE;
    211 +u8 gDelayForErrorMessage = 0;
    212 +
    213 +u8 does_pool_end_lie_out_of_bounds(void *end) {
    214 +    u32 endPhy = ((u32) end) & 0x1FFFFFFF;
    215 +    u32 memSize = *((u32 *) 0x80000318);
    216 +
    217 +    if (endPhy > memSize) {
    218 +        gNotEnoughMemory = TRUE;
    219 +        return TRUE;
    220 +    } else {
    221 +        if (memSize < REQUIRED_MIN_MEM_SIZE) {
    222 +            gNotEnoughMemory = TRUE;
    223 +        }
    224 +        return FALSE;
    225 +    }
    226 +}
    227 +
    228 +// If you're using an N64 console, then you will need to buy an\nexpansion pak to play this ROM hack.
    229 +u8 text_console_8mb[] = { TEXT_CONSOLE_8MB };
    230 +
    231 +// If you are using PJ64 1.6, go to: Options ► Settings ► Rom Settings Tab ► Memory Size then select 8
    232 +// MB from the drop-down box.
    233 +u8 text_pj64[] = { TEXT_PJ64 };
    234 +
    235 +// If you are using PJ64 2.X, go to: Options ► Settings ► Config: ► Memory Size, select 8 MB
    236 +u8 text_pj64_2[] = { TEXT_PJ64_2 };
    237 +
    238 +Gfx *geo18_display_error_message(u32 run, UNUSED struct GraphNode *sp44, UNUSED u32 sp48) {
    239 +    if (run) {
    240 +        if (gDelayForErrorMessage > 0) {
    241 +            // Draw color text title.
    242 +            print_text(10, 210, "ERROR    Need more memory");
    243 +
    244 +            // Init generic text rendering
    245 +            create_dl_ortho_matrix();
    246 +            gSPDisplayList(gDisplayListHead++,
    247 +                           dl_ia_text_begin); // Init rendering stuff for generic text
    248 +
    249 +            // Set text color to white
    250 +            gDPSetEnvColor(gDisplayListHead++, 255, 255, 255, 255);
    251 +
    252 +            print_generic_string(8, 170, text_console_8mb);
    253 +            print_generic_string(8, 120, text_pj64);
    254 +            print_generic_string(8, 54, text_pj64_2);
    255 +
    256 +            // Cleanup
    257 +            gSPDisplayList(gDisplayListHead++,
    258 +                           dl_ia_text_end); // Reset back to default render settings.
    259 +            gSPPopMatrix(gDisplayListHead++, G_MTX_MODELVIEW);
    260 +        } else {
    261 +            gDelayForErrorMessage += 1;
    262 +        }
    263 +    }
    264 +
    265 +    return 0;
    266 +}
    267 +
    268 +// Basic main loop for the error screen. Note that controllers are not enabled here.
    269 +void thread5_mem_error_message_loop(UNUSED void *arg) {
    270 +    struct LevelCommand *addr;
    271 +
    272 +    setup_game_memory();
    273 +    set_vblank_handler(2, &gGameVblankHandler, &gGameVblankQueue, (OSMesg) 1);
    274 +
    275 +    addr = segmented_to_virtual(level_script_entry_error_screen);
    276 +
    277 +    render_init();
    278 +
    279 +    while (1) {
    280 +        select_gfx_pool();
    281 +        addr = level_script_execute(addr);
    282 +        display_and_vsync();
    283 +    }
    284 +}
    285 diff --git a/src/game/mem_error_screen.h b/src/game/mem_error_screen.h
    286 new file mode 100644
    287 index 00000000..9fbff34c
    288 --- /dev/null
    289 +++ b/src/game/mem_error_screen.h
    290 @@ -0,0 +1,8 @@
    291 +#ifndef MEM_ERROR_SCREEN_H
    292 +#define MEM_ERROR_SCREEN_H
    293 +
    294 +extern u8 gNotEnoughMemory;
    295 +void thread5_mem_error_message_loop(UNUSED void *arg);
    296 +u8 does_pool_end_lie_out_of_bounds(void *end);
    297 +
    298 +#endif