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