retry from results, and back to song select

This commit is contained in:
CrazyRedMachine 2023-07-30 00:50:42 +02:00
parent 7b6b31fa13
commit 2e68d79f64
3 changed files with 136 additions and 31 deletions

View File

@ -26,7 +26,7 @@
<!-- Premium free (unlimited stages per credit) -->
<pfree __type="bool">0</pfree>
<!-- Press numpad 9 in song to retire instantly, or on result screen to leave game session (thank you for playing) -->
<!-- quick_retire with pfree also enables quick retry: hold numpad 7 to retry -->
<!-- quick_retire with pfree also enables quick retry: press numpad 8 during song or on result screen to retry (keep holding during option select to skip it) -->
<quick_retire __type="bool">0</quick_retire>
<!-- Network features -->
@ -61,9 +61,13 @@
<!-- END OF USER SETTINGS -->
<!-- EXPERIMENTAL PATCHES (USE AT OWN RISK, PREFERABLY OFFLINE, MIGHT INTERFERE WITH OTHER OPTIONS) -->
<!-- Enable practice mode menu (r-ran, constant chart speed, slow motion, etc...) -->
<!-- EXPERIMENTAL PATCHES -->
<!-- Enable practice mode menu, (r-ran, constant chart speed, slow motion, etc...) -->
<!-- PLEASE USE OFFLINE ONLY -->
<practice_mode __type="bool">0</practice_mode>
<!-- Press 9 on option select screen to go back to song selection (requires pfree and quick_retire) -->
<!-- Note: causes issues with sounds on song select -->
<back_to_song_select __type="bool">0</back_to_song_select>
<!-- DEBUG OPTIONS FOLLOW (GENERALLY SHOULD NOT BE CHANGED UNLESS YOU KNOW WHAT YOU'RE DOING) -->
@ -91,7 +95,7 @@
THREAD_PRIORITY_ABOVE_NORMAL 1
THREAD_PRIORITY_HIGHEST 2
THREAD_PRIORITY_TIME_CRITICAL 15 -->
<enhanced_polling_priority __type="s8">0</enhanced_polling_priority>
<enhanced_polling_priority __type="s8">1</enhanced_polling_priority>
<!-- Song db patches -->
<!-- Auto select patch file from data_mods folder (will detect datecode from ea3-config or force_datecode option) -->

View File

@ -10,6 +10,7 @@ struct popnhax_config {
bool show_fast_slow;
bool pfree;
bool quick_retire;
bool back_to_song_select;
bool score_challenge;
bool force_hd_timing;
uint8_t force_hd_resolution;

View File

@ -98,6 +98,8 @@ PSMAP_MEMBER_REQ(PSMAP_PROPERTY_TYPE_BOOL, struct popnhax_config, pfree,
"/popnhax/pfree")
PSMAP_MEMBER_REQ(PSMAP_PROPERTY_TYPE_BOOL, struct popnhax_config, quick_retire,
"/popnhax/quick_retire")
PSMAP_MEMBER_REQ(PSMAP_PROPERTY_TYPE_BOOL, struct popnhax_config, back_to_song_select,
"/popnhax/back_to_song_select")
PSMAP_MEMBER_REQ(PSMAP_PROPERTY_TYPE_BOOL, struct popnhax_config, score_challenge,
"/popnhax/score_challenge")
PSMAP_MEMBER_REQ(PSMAP_PROPERTY_TYPE_BOOL, struct popnhax_config, force_hd_timing,
@ -263,19 +265,29 @@ void omnimix_patch_jbx() {
real_omnimix_patch_jbx();
}
uint32_t g_startsong_addr = 0;
uint32_t g_transition_addr = 0;
uint32_t g_stage_addr = 0;
uint32_t g_score_addr = 0;
bool g_pfree_mode = false;
bool g_return_to_options = false;
bool g_return_to_song_select = false;
void (*real_screen_transition)();
void quickexit_screen_transition()
{
if (g_return_to_options)
{
__asm("mov dword ptr [esi+0x30], 0x1C\n");
__asm("mov dword ptr [edi+0x30], 0x1C\n");
//flag is set back to false in the option select screen after score cleanup
}
else if (g_return_to_song_select)
{
__asm("mov dword ptr [edi+0x30], 0x17\n");
g_return_to_song_select = false;
}
real_screen_transition();
}
uint32_t g_score_addr = 0;
void (*real_retrieve_score)();
void quickretry_retrieve_score()
{
@ -293,7 +305,6 @@ void quickretry_retrieve_score()
real_retrieve_score();
}
uint32_t g_startsong_addr = 0;
void quickexit_option_screen_cleanup()
{
if (g_return_to_options)
@ -330,12 +341,12 @@ void quickexit_option_screen()
__asm("add ebx, 0xAC\n");
__asm("mov ebx, [ebx]\n");
__asm("shr ebx, 24\n");
__asm("shr ebx, 28\n"); // numpad8: 80 00 00 00
__asm("cmp bl, 8\n");
__asm("pop ebx\n");
__asm("jne real_option_screen\n");
/* numpad 7 is held, rewrite transition pointer */
/* numpad 8 is held, rewrite transition pointer */
__asm("pop edi\n");
__asm("pop ecx\n");
__asm("pop edx\n");
@ -349,6 +360,52 @@ void quickexit_option_screen()
real_option_screen();
}
void (*real_option_screen_later)();
void backtosongselect_option_screen()
{
__asm("push ecx\n");
__asm("mov ecx, %0\n": :"m"(g_addr_icca));
__asm("mov ebx, [ecx]\n");
__asm("pop ecx\n");
__asm("add ebx, 0xAC\n");
__asm("mov ebx, [ebx]\n");
__asm("shr ebx, 16\n"); // numpad9: 00 08 00 00
__asm("cmp bl, 8\n");
__asm("jne exit_back_select\n");
if (g_pfree_mode)
{
g_return_to_song_select = true;
}
__asm("exit_back_select:\n");
real_option_screen_later();
}
void (*real_option_screen_yellow)();
void backtosongselect_option_yellow()
{
__asm("push ecx\n");
__asm("mov ecx, %0\n": :"m"(g_addr_icca));
__asm("mov ebx, [ecx]\n");
__asm("pop ecx\n");
__asm("add ebx, 0xAC\n");
__asm("mov ebx, [ebx]\n");
__asm("shr ebx, 16\n"); // numpad9: 00 08 00 00
__asm("cmp bl, 8\n");
__asm("jne exit_back_select_yellow\n");
if (g_pfree_mode)
{
g_return_to_song_select = true;
}
__asm("exit_back_select_yellow:\n");
real_option_screen_yellow();
}
static bool r_ran;
static bool regul_flg;
@ -373,8 +430,19 @@ void restore_plop()
}
/*
numpad values:
bool g_pfree_mode = false;
| <<24 <<28 <<16
---+---------------
8 | 7 8 9
4 | 4 5 6
2 | 1 2 3
1 | 0 00
e.g. numpad 9 = 8<<16 = 00 08 00 00
numpad 2 = 2<<28 = 20 00 00 00
*/
void (*real_game_loop)();
void quickexit_game_loop()
{
@ -392,11 +460,11 @@ void quickexit_game_loop()
__asm("cmp bl, 8\n");
__asm("je leave_song\n");
__asm("shr ebx, 8\n"); // numpad7: 08 00 00 00
__asm("shr ebx, 12\n"); // (adds to the previous shr 16) numpad8: 08 00 00 00
__asm("cmp bl, 8\n");
__asm("jne call_real\n");
/* numpad 7 is pressed: quick retry if pfree is active */
/* numpad 8 is pressed: quick retry if pfree is active */
use_sp_flg = 0;
if (!g_pfree_mode)
@ -429,12 +497,9 @@ void patch_eval_timing() {
real_eval_timing();
}
uint32_t g_transition_addr = 0;
uint32_t g_stage_addr = 0;
void (*real_result_loop)();
void quickexit_result_loop()
{
//__asm("push ebx\n"); //handled by :"b"
__asm("push ecx\n");
__asm("mov ecx, %0\n": :"m"(g_addr_icca));
__asm("mov ebx, [ecx]\n");
@ -442,31 +507,34 @@ void quickexit_result_loop()
__asm("add ebx, 0xAC\n");
__asm("mov ebx, [ebx]\n");
__asm("shr ebx, 16\n");
__asm("shr ebx, 16\n"); // numpad9: 00 08 00 00
__asm("cmp bl, 8\n");
__asm("je quit_session\n");
__asm("shr ebx, 12\n"); // (adds to the previous shr 16) numpad8: 08 00 00 00
__asm("cmp bl, 8\n");
__asm("jne call_real_result\n");
/* set value 5 in g_stage_addr and -4 in g_transition_addr */
if (!g_pfree_mode)
__asm("jmp call_real_result\n");
g_return_to_options = true; //transition screen hook will catch it
__asm("jmp call_real_result\n");
__asm("quit_session:\n");
g_return_to_options = false;
/* set value 5 in g_stage_addr and -4 in g_transition_addr (to get fade to black transition) */
__asm("mov ebx, %0\n": :"b"(g_stage_addr));
__asm("mov dword ptr[ebx], 5\n");
__asm("mov ebx, %0\n": :"b"(g_transition_addr));
__asm("mov dword ptr[ebx], 0xFFFFFFFC\n");
__asm("mov dword ptr[ebx], 0xFFFFFFFC\n"); //quit session
disp = 0;// 7.9 message off
use_sp_flg = 0;
disp = 0;// 7.9 message off
__asm("call_real_result:\n");
//__asm("pop ebx\n"); //handled by :"b"
// r2nk226#1109 ついかテスト
if (use_sp_flg){
g_return_to_options = 1;
quickexit_option_screen_cleanup();
}
ex_res_flg = 1;
real_result_loop();
}
uint32_t g_timing_addr = 0;
@ -1824,7 +1892,7 @@ static bool patch_quick_retire(bool pfree)
g_startsong_addr = *(uint32_t*)(patch_addr);
}
/* instant retry (go back to option select) with numpad 7 */
/* instant retry (go back to option select) with numpad 8 */
{
/* retrieve current stage score addr for cleanup (also used to fix quick retire medal) */
int64_t pattern_offset = search(data, dllSize, "\xF3\xA5\x5F\x5E\x5B\xC2\x04\x00", 8, 0);
@ -1840,7 +1908,7 @@ static bool patch_quick_retire(bool pfree)
}
{
/* hook quick retire transition to go back to option select instead */
int64_t pattern_offset = search(data, dllSize, "\x6A\x01\x8B\xCE\xFF\xD0\xE8", 7, 0);
int64_t pattern_offset = search(data, dllSize, "\x8B\xE8\x8B\x47\x30\x83\xF8\x17", 8, 0);
if (pattern_offset == -1) {
LOG("popnhax: quick retry: cannot retrieve screen transition function\n");
@ -1852,7 +1920,7 @@ static bool patch_quick_retire(bool pfree)
(void **)&real_screen_transition);
}
/* instant launch song with numpad 7 on option select (hold 7 during song for quick retry) */
/* instant launch song with numpad 8 on option select (hold 8 during song for quick retry) */
{
int64_t pattern_offset = search(data, dllSize, "\x51\x50\x8B\x83\x0C\x0A\x00\x00\xEB\x09\x33\xD2", 12, 0);
@ -1872,6 +1940,38 @@ static bool patch_quick_retire(bool pfree)
(void **)&real_option_screen);
}
if (config.back_to_song_select)
{
/* go back to song select with numpad 9 on song option screen (before pressing yellow) */
{
int64_t pattern_offset = search(data, dllSize, "\x8B\x85\x0C\x0A\x00\x00\x83\x78\x34\x00\x75", 11, 0);
if (pattern_offset == -1) {
LOG("popnhax: quick retry: cannot retrieve option screen loop function\n");
return false;
}
uint64_t patch_addr = (int64_t)data + pattern_offset;
MH_CreateHook((LPVOID)patch_addr, (LPVOID)backtosongselect_option_screen,
(void **)&real_option_screen_later);
}
/* go back to song select with numpad 9 on song option screen (after pressing yellow) */
{
int64_t pattern_offset = search(data, dllSize, "\x8B\x85\x0C\x0A\x00\x00\x83\x78\x38\x00\x75", 11, 0);
if (pattern_offset == -1) {
LOG("popnhax: quick retry: cannot retrieve option screen loop function\n");
return false;
}
uint64_t patch_addr = (int64_t)data + pattern_offset;
MH_CreateHook((LPVOID)patch_addr, (LPVOID)backtosongselect_option_yellow,
(void **)&real_option_screen_yellow);
}
LOG("popnhax: quick retry: return to song select enabled\n");
}
if (pfree)
LOG("popnhax: quick retry enabled\n");