diff --git a/dist/popnhax/popnhax.xml b/dist/popnhax/popnhax.xml
index 506e6c2..63921fd 100644
--- a/dist/popnhax/popnhax.xml
+++ b/dist/popnhax/popnhax.xml
@@ -26,7 +26,7 @@
0
-
+
0
@@ -61,9 +61,13 @@
-
-
+
+
+
0
+
+
+ 0
@@ -91,7 +95,7 @@
THREAD_PRIORITY_ABOVE_NORMAL 1
THREAD_PRIORITY_HIGHEST 2
THREAD_PRIORITY_TIME_CRITICAL 15 -->
- 0
+ 1
diff --git a/popnhax/config.h b/popnhax/config.h
index f1ae5ab..77567fc 100644
--- a/popnhax/config.h
+++ b/popnhax/config.h
@@ -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;
diff --git a/popnhax/dllmain.cc b/popnhax/dllmain.cc
index 6ff1a48..38f9547 100644
--- a/popnhax/dllmain.cc
+++ b/popnhax/dllmain.cc
@@ -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");