From fd924b56e860e3499a881338ecb9f1742f9ae81c Mon Sep 17 00:00:00 2001
From: Amos <48657826+Mauler125@users.noreply.github.com>
Date: Fri, 18 Feb 2022 14:00:58 +0100
Subject: [PATCH] Shutdown dedicated dll properly
Systems where not getting shutdown properly. For dedicated 'ExitProcess()' in the GameDLL caused 'abort()' to get called even when systems where shutdown properly. We call TerminateProcess after all systems have shutdown properly in the SDK and GameDLL.
---
r5dev/core/dllmain.cpp | 8 ++++++++
r5dev/core/init.cpp | 7 +++++++
r5dev/core/init.h | 3 +++
r5dev/dedicated.vcxproj | 2 ++
r5dev/dedicated.vcxproj.filters | 6 ++++++
r5dev/launcher/prx.cpp | 24 ++++++++++++++++++++++++
r5dev/launcher/prx.h | 24 ++++++++++++++++++++++++
7 files changed, 74 insertions(+)
create mode 100644 r5dev/launcher/prx.cpp
create mode 100644 r5dev/launcher/prx.h
diff --git a/r5dev/core/dllmain.cpp b/r5dev/core/dllmain.cpp
index fd7aa57e..1144a9c0 100644
--- a/r5dev/core/dllmain.cpp
+++ b/r5dev/core/dllmain.cpp
@@ -46,6 +46,14 @@ void R5Dev_Init()
void R5Dev_Shutdown()
{
+ static bool bShutDown = false;
+ if (bShutDown)
+ {
+ spdlog::error("Recursive shutdown!\n");
+ return;
+ }
+ bShutDown = true;
+
Systems_Shutdown();
WinSys_Detach();
diff --git a/r5dev/core/init.cpp b/r5dev/core/init.cpp
index da7cb126..9ec94195 100644
--- a/r5dev/core/init.cpp
+++ b/r5dev/core/init.cpp
@@ -17,6 +17,7 @@
#include "vpc/interfaces.h"
#include "common/opcodes.h"
#include "launcher/IApplication.h"
+#include "launcher/prx.h"
#include "ebisusdk/EbisuSDK.h"
#ifndef DEDICATED
#include "milessdk/win64_rrthreads.h"
@@ -83,6 +84,9 @@ void Systems_Init()
// Hook functions
IApplication_Attach();
+#ifdef DEDICATED
+ PRX_Attach();
+#endif // DEDICATED
CBaseClient_Attach();
CBaseFileSystem_Attach();
@@ -159,6 +163,9 @@ void Systems_Shutdown()
// Unhook functions
IApplication_Detach();
+#ifdef DEDICATED
+ PRX_Detach();
+#endif // DEDICATED
CBaseClient_Detach();
CBaseFileSystem_Detach();
diff --git a/r5dev/core/init.h b/r5dev/core/init.h
index 8ddabd80..d18b756c 100644
--- a/r5dev/core/init.h
+++ b/r5dev/core/init.h
@@ -5,6 +5,9 @@ namespace
/* ==== ------- ========================================================================================================================================================= */
}
+void R5Dev_Init();
+void R5Dev_Shutdown();
+
void Systems_Init();
void Systems_Shutdown();
void PrintHAddress();
diff --git a/r5dev/dedicated.vcxproj b/r5dev/dedicated.vcxproj
index 05babb6f..41f59e58 100644
--- a/r5dev/dedicated.vcxproj
+++ b/r5dev/dedicated.vcxproj
@@ -207,6 +207,7 @@
+
@@ -406,6 +407,7 @@
+
diff --git a/r5dev/dedicated.vcxproj.filters b/r5dev/dedicated.vcxproj.filters
index e47064ea..cff77e39 100644
--- a/r5dev/dedicated.vcxproj.filters
+++ b/r5dev/dedicated.vcxproj.filters
@@ -714,6 +714,9 @@
thirdparty\protobuf
+
+ sdk\launcher
+
@@ -956,6 +959,9 @@
thirdparty\protobuf
+
+ sdk\launcher
+
diff --git a/r5dev/launcher/prx.cpp b/r5dev/launcher/prx.cpp
new file mode 100644
index 00000000..750117a2
--- /dev/null
+++ b/r5dev/launcher/prx.cpp
@@ -0,0 +1,24 @@
+#include
+#include
+#include
+
+//-----------------------------------------------------------------------------
+// Purpose: shutdown and unload SDK
+//-----------------------------------------------------------------------------
+void h_exit_or_terminate_process(UINT uExitCode)
+{
+ R5Dev_Shutdown();
+
+ HANDLE h = GetCurrentProcess();
+ TerminateProcess(h, uExitCode);
+}
+
+void PRX_Attach()
+{
+ DetourAttach((LPVOID*)&exit_or_terminate_process, &h_exit_or_terminate_process);
+}
+
+void PRX_Detach()
+{
+ DetourAttach((LPVOID*)&exit_or_terminate_process, &h_exit_or_terminate_process);
+}
\ No newline at end of file
diff --git a/r5dev/launcher/prx.h b/r5dev/launcher/prx.h
new file mode 100644
index 00000000..c2127843
--- /dev/null
+++ b/r5dev/launcher/prx.h
@@ -0,0 +1,24 @@
+#pragma once
+
+namespace
+{
+ /* ==== PRX ============================================================================================================================================================= */
+ ADDRESS p_exit_or_terminate_process = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x53\x48\x83\xEC\x20\x8B\xD9\xE8\x00\x00\x00\x00\x84\xC0", "xxxxxxxxx????xx");
+ void (*exit_or_terminate_process)(UINT uExitCode) = (void (*)(UINT))p_exit_or_terminate_process.GetPtr(); /*40 53 48 83 EC 20 8B D9 E8 ? ? ? ? 84 C0 */
+}
+
+void PRX_Attach();
+void PRX_Detach();
+
+///////////////////////////////////////////////////////////////////////////////
+class HPRX : public IDetour
+{
+ virtual void debugp()
+ {
+ std::cout << "| FUN: exit_or_terminate_process : 0x" << std::hex << std::uppercase << p_exit_or_terminate_process.GetPtr() << std::setw(npad) << " |" << std::endl;
+ std::cout << "+----------------------------------------------------------------+" << std::endl;
+ }
+};
+///////////////////////////////////////////////////////////////////////////////
+
+REGISTER(HPRX);