From 69161ae6c0db8b646a6d4939632a57111870466e Mon Sep 17 00:00:00 2001 From: Amos <48657826+Mauler125@users.noreply.github.com> Date: Mon, 27 Dec 2021 16:53:35 +0100 Subject: [PATCH] Implement RTech_AsyncLoad hook and concommand The ConCommand 'rtech_asyncload' allows the user to load specific pak files on-demand. --- r5dev/common/opcodes.cpp | 16 ------ r5dev/common/opcodes.h | 11 ---- r5dev/core/init.cpp | 11 ++-- r5dev/dedicated.vcxproj | 8 ++- r5dev/dedicated.vcxproj.filters | 16 +++++- r5dev/engine/host_cmd.cpp | 5 ++ r5dev/engine/host_cmd.h | 36 +++++++++++++ r5dev/r5dev.vcxproj | 8 ++- r5dev/r5dev.vcxproj.filters | 16 +++++- r5dev/rtech/rtech_game.cpp | 60 ++++++++++++++++++++++ r5dev/rtech/rtech_game.h | 38 ++++++++++++++ r5dev/rtech/{rtech.cpp => rtech_utils.cpp} | 2 +- r5dev/rtech/{rtech.h => rtech_utils.h} | 0 r5dev/tier0/ConCommand.cpp | 3 +- r5dev/tier0/completion.cpp | 13 ++++- r5dev/tier0/completion.h | 3 +- r5dev/tier0/cvar.cpp | 2 +- r5dev/windows/console.cpp | 2 +- 18 files changed, 205 insertions(+), 45 deletions(-) create mode 100644 r5dev/engine/host_cmd.cpp create mode 100644 r5dev/engine/host_cmd.h create mode 100644 r5dev/rtech/rtech_game.cpp create mode 100644 r5dev/rtech/rtech_game.h rename r5dev/rtech/{rtech.cpp => rtech_utils.cpp} (99%) rename r5dev/rtech/{rtech.h => rtech_utils.h} (100%) diff --git a/r5dev/common/opcodes.cpp b/r5dev/common/opcodes.cpp index 65d71602..5f2423e1 100644 --- a/r5dev/common/opcodes.cpp +++ b/r5dev/common/opcodes.cpp @@ -5,22 +5,6 @@ * _opcodes.cpp *-----------------------------------------------------------------------------*/ - - // TODO: Move to RTech::.. -void HRTech_UnloadAsset(int64_t a1, int64_t a2) -{ - // Return early if address is out of scope. - if (a2 <= 0x0000000000 || a2 >= 0xFFFFFFFFFF) - { - return; - } - return RTech_UnloadAsset(a1, a2); -} - -void Opcodes_Hook() -{ - DetourAttach((LPVOID*)&RTech_UnloadAsset, &HRTech_UnloadAsset); -} #ifdef DEDICATED void Dedicated_Init() { diff --git a/r5dev/common/opcodes.h b/r5dev/common/opcodes.h index cf421447..db361103 100644 --- a/r5dev/common/opcodes.h +++ b/r5dev/common/opcodes.h @@ -7,9 +7,6 @@ void Dedicated_Init(); void RuntimePtc_Init(); void RuntimePtc_Toggle(); -// TEMP -void Opcodes_Hook(); - namespace { ///* -------------- ORIGIN ------------------------------------------------------------------------------------------------------------------------------------------------ */ @@ -100,14 +97,6 @@ namespace ADDRESS SCR_BeginLoadingPlaque = 0x000000014023E870; } - - // TODO: Move to RTech::.. - ADDRESS p_RTech_UnloadAsset = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x83\xEC\x28\x48\x85\xD2\x74\x40\x48\x8B\x05\x00\x00\x00\x00", "xxxxxxxxxxxx????"); - void (*RTech_UnloadAsset)(int64_t a1, int64_t a2) = (void (*)(int64_t, int64_t))p_RTech_UnloadAsset.GetPtr(); /*48 83 EC 28 48 85 D2 74 40 48 8B 05 ? ? ? ?*/ - - ADDRESS p_RTech_AsyncLoad = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x53\x48\x83\xEC\x40\x48\x89\x6C\x24\x00\x41\x0F\xB6\xE9", "xxxxxxxxxx?xxxx"); - unsigned int (*RTech_AsyncLoad)(void* Src, __int64 a2, int a3, char pakfile) = (unsigned int (*)(void*, __int64, int, char))p_RTech_AsyncLoad.GetPtr(); /*40 53 48 83 EC 40 48 89 6C 24 ? 41 0F B6 E9*/ - void PrintOAddress() // Test the sigscan results { std::cout << "+--------------------------------------------------------+" << std::endl; diff --git a/r5dev/core/init.cpp b/r5dev/core/init.cpp index 5ffee716..1d99b247 100644 --- a/r5dev/core/init.cpp +++ b/r5dev/core/init.cpp @@ -32,8 +32,10 @@ #include "squirrel/sqinit.h" #include "squirrel/sqapi.h" #include "squirrel/sqvm.h" +#include "rtech/rtech_game.h" #include "rtech/stryder.h" #include "engine/baseclient.h" +#include "engine/host_cmd.h" #include "engine/host_state.h" #include "engine/net_chan.h" #include "engine/sys_dll.h" @@ -63,9 +65,6 @@ void Systems_Init() DetourTransactionBegin(); DetourUpdateThread(GetCurrentThread()); - // TEMP - Opcodes_Hook(); - // Hook functions IApplication_Attach(); CBaseClient_Attach(); @@ -93,6 +92,9 @@ void Systems_Init() IVEngineServer_Attach(); SQAPI_Attach(); SQVM_Attach(); + + RTech_Game_Attach(); + SysDll_Attach(); SysUtils_Attach(); @@ -157,6 +159,9 @@ void Systems_Shutdown() IVEngineServer_Detach(); SQAPI_Detach(); SQVM_Detach(); + + RTech_Game_Detach(); + SysDll_Detach(); SysUtils_Detach(); diff --git a/r5dev/dedicated.vcxproj b/r5dev/dedicated.vcxproj index b31728a8..a5aa1bae 100644 --- a/r5dev/dedicated.vcxproj +++ b/r5dev/dedicated.vcxproj @@ -182,6 +182,7 @@ + @@ -202,7 +203,8 @@ - + + @@ -337,6 +339,7 @@ + @@ -350,7 +353,8 @@ - + + diff --git a/r5dev/dedicated.vcxproj.filters b/r5dev/dedicated.vcxproj.filters index a1381cad..b4c53d81 100644 --- a/r5dev/dedicated.vcxproj.filters +++ b/r5dev/dedicated.vcxproj.filters @@ -177,7 +177,7 @@ sdk\rtech - + sdk\rtech @@ -558,6 +558,12 @@ sdk\common + + sdk\engine + + + sdk\rtech + @@ -614,7 +620,7 @@ sdk\rtech - + sdk\rtech @@ -680,6 +686,12 @@ sdk\mathlib + + sdk\engine + + + sdk\rtech + diff --git a/r5dev/engine/host_cmd.cpp b/r5dev/engine/host_cmd.cpp new file mode 100644 index 00000000..cb442d7e --- /dev/null +++ b/r5dev/engine/host_cmd.cpp @@ -0,0 +1,5 @@ +#include "core/stdafx.h" +#include "tier0/basetypes.h" +#include "engine/host_cmd.h" + +// TODO: this file is for when dedicated is stable, to move hardcoded patches in Host_Init for a more dynamic solution. diff --git a/r5dev/engine/host_cmd.h b/r5dev/engine/host_cmd.h new file mode 100644 index 00000000..828c0f59 --- /dev/null +++ b/r5dev/engine/host_cmd.h @@ -0,0 +1,36 @@ +#pragma once + +namespace +{ + /* ==== HOST ============================================================================================================================================================ */ +#if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) + ADDRESS p_Host_Init = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x5C\x24\x00\x48\x89\x6C\x24\x00\x48\x89\x74\x24\x00\x57\x41\x54\x41\x55\x41\x56\x41\x57\x48\x81\xEC\x00\x00\x00\x00\x48\x8B\xD9\xFF\x15\x00\x00\x00\x00", "xxxx?xxxx?xxxx?xxxxxxxxxxxx????xxxxx????"); + void* (*Host_Init)(bool* bDedicated) = (void* (*)(bool*))p_Host_Init.GetPtr(); /*48 89 5C 24 ? 48 89 6C 24 ? 48 89 74 24 ? 57 41 54 41 55 41 56 41 57 48 81 EC ? ? ? ? 48 8B D9 FF 15 ? ? ? ?*/ +#elif defined (GAMEDLL_S2) || defined (GAMEDLL_S3) + ADDRESS p_Host_Init = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x5C\x24\x00\x48\x89\x74\x24\x00\x48\x89\x7C\x24\x00\x55\x41\x54\x41\x55\x41\x56\x41\x57\x48\x8D\xAC\x24\x00\x00\x00\x00\xB8\x00\x00\x00\x00\xE8\x00\x00\x00\x00\x48\x2B\xE0\x48\x8B\xD9", "xxxx?xxxx?xxxx?xxxxxxxxxxxxx????x????x????xxxxxx"); + void* (*Host_Init)(bool* bDedicated) = (void* (*)(bool*))p_Host_Init.GetPtr(); /*48 89 5C 24 ? 48 89 74 24 ? 48 89 7C 24 ? 55 41 54 41 55 41 56 41 57 48 8D AC 24 ? ? ? ? B8 ? ? ? ? E8 ? ? ? ? 48 2B E0 48 8B D9*/ +#endif +} + +namespace +{ +#if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) + ADDRESS g_pMallocPool = p_Host_Init.Offset(0x600).FindPatternSelf("48 8D 15 ?? ?? ?? 01", ADDRESS::Direction::DOWN, 100).ResolveRelativeAddressSelf(0x3, 0x7); +#elif defined (GAMEDLL_S2) || defined (GAMEDLL_S3) + ADDRESS g_pMallocPool = p_Host_Init.Offset(0x130).FindPatternSelf("48 8D 15 ?? ?? ?? 01", ADDRESS::Direction::DOWN, 100).ResolveRelativeAddressSelf(0x3, 0x7); +#endif +} + +/////////////////////////////////////////////////////////////////////////////// +class HHostCmd : public IDetour +{ + virtual void debugp() + { + std::cout << "| FUN: Host_Init : 0x" << std::hex << std::uppercase << p_Host_Init.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "| VAR: g_pMallocPool : 0x" << std::hex << std::uppercase << g_pMallocPool.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "+----------------------------------------------------------------+" << std::endl; + } +}; +/////////////////////////////////////////////////////////////////////////////// + +REGISTER(HHostCmd); diff --git a/r5dev/r5dev.vcxproj b/r5dev/r5dev.vcxproj index 1e8dbcec..58d193dd 100644 --- a/r5dev/r5dev.vcxproj +++ b/r5dev/r5dev.vcxproj @@ -33,6 +33,7 @@ + @@ -49,7 +50,8 @@ - + + @@ -123,6 +125,7 @@ + @@ -149,7 +152,8 @@ - + + diff --git a/r5dev/r5dev.vcxproj.filters b/r5dev/r5dev.vcxproj.filters index 20dc7022..78e557f1 100644 --- a/r5dev/r5dev.vcxproj.filters +++ b/r5dev/r5dev.vcxproj.filters @@ -168,7 +168,7 @@ sdk\public - + sdk\rtech @@ -297,6 +297,12 @@ sdk\mathlib + + sdk\rtech + + + sdk\engine + @@ -356,7 +362,7 @@ sdk\public\include - + sdk\rtech @@ -827,6 +833,12 @@ sdk\common + + sdk\rtech + + + sdk\engine + diff --git a/r5dev/rtech/rtech_game.cpp b/r5dev/rtech/rtech_game.cpp new file mode 100644 index 00000000..321a7d61 --- /dev/null +++ b/r5dev/rtech/rtech_game.cpp @@ -0,0 +1,60 @@ +#include "core/stdafx.h" +#include "tier0/basetypes.h" +#include "engine/host_cmd.h" +#include "engine/sys_utils.h" +#include "rtech/rtech_game.h" + +//----------------------------------------------------------------------------- +// Purpose: unloads asset files from the memory pool +//----------------------------------------------------------------------------- +void HRTech_UnloadAsset(std::int64_t a1, std::int64_t a2) +{ +#if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) + std::int64_t pAsset = a1; +#elif defined (GAMEDLL_S2) || defined (GAMEDLL_S3) + std::int64_t pAsset = a2; +#endif + // Return early if address is out of scope. + if (pAsset <= 0x0000000000 || pAsset >= 0xFFFFFFFFFF) + { + return; + } +#if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) + return RTech_UnloadAsset(a1); +#elif defined (GAMEDLL_S2) || defined (GAMEDLL_S3) + return RTech_UnloadAsset(a1, a2); +#endif +} + +//----------------------------------------------------------------------------- +// Purpose: load user-requested pak files on-demand +//----------------------------------------------------------------------------- +void HRtech_AsyncLoad(std::string svPakFileName) +{ + std::string svPakFilePathMod = "paks\\Win32\\" + svPakFileName; + std::string svPakFilePathBase = "paks\\Win64\\" + svPakFileName; + + if (FileExists(svPakFilePathMod.c_str()) || FileExists(svPakFilePathBase.c_str())) + { + unsigned int results = RTech_AsyncLoad((void*)svPakFileName.c_str(), g_pMallocPool.GetPtr(), NULL, NULL); + + if (results == 0xFFFFFFFF) + { + DevMsg(eDLL_T::RTECH, "RTech AsyncLoad failed read '%s' results '%u'\n", svPakFileName, results); + } + } + else + { + DevMsg(eDLL_T::RTECH, "RTech AsyncLoad failed. File '%s' doesn't exist\n", svPakFileName.c_str()); + } +} + +void RTech_Game_Attach() +{ + DetourAttach((LPVOID*)&RTech_UnloadAsset, &HRTech_UnloadAsset); +} + +void RTech_Game_Detach() +{ + DetourAttach((LPVOID*)&RTech_UnloadAsset, &HRTech_UnloadAsset); +} diff --git a/r5dev/rtech/rtech_game.h b/r5dev/rtech/rtech_game.h new file mode 100644 index 00000000..ddedd3e1 --- /dev/null +++ b/r5dev/rtech/rtech_game.h @@ -0,0 +1,38 @@ +#pragma once + +namespace +{ + /* ==== RTECH_GAME ====================================================================================================================================================== */ +#if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) + ADDRESS p_RTech_UnloadAsset = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x83\xEC\x28\x48\x85\xC9\x0F\x84\x00\x00\x00\x00\x48\x8B\x05\x00\x00\x00\x00", "xxxxxxxxx????xxx????"); + void (*RTech_UnloadAsset)(std::int64_t a1) = (void (*)(std::int64_t))p_RTech_UnloadAsset.GetPtr(); /*48 83 EC 28 48 85 C9 0F 84 ? ? ? ? 48 8B 05 ? ? ? ? */ + + ADDRESS p_RTech_AsyncLoad = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x53\x48\x83\xEC\x40\x48\x89\x6C\x24\x00\x41\x8B\xE8", "xxxxxxxxxx?xxx"); + unsigned int (*RTech_AsyncLoad)(void* Src, __int64 a2, int a3, char pakfile) = (unsigned int (*)(void*, __int64, int, char))p_RTech_AsyncLoad.GetPtr(); /*40 53 48 83 EC 40 48 89 6C 24 ? 41 8B E8*/ +#elif defined (GAMEDLL_S2) || defined (GAMEDLL_S3) + ADDRESS p_RTech_UnloadAsset = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x83\xEC\x28\x48\x85\xD2\x74\x40\x48\x8B\x05\x00\x00\x00\x00", "xxxxxxxxxxxx????"); + void (*RTech_UnloadAsset)(std::int64_t a1, std::int64_t a2) = (void (*)(std::int64_t, std::int64_t))p_RTech_UnloadAsset.GetPtr(); /*48 83 EC 28 48 85 D2 74 40 48 8B 05 ? ? ? ?*/ + + ADDRESS p_RTech_AsyncLoad = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x53\x48\x83\xEC\x40\x48\x89\x6C\x24\x00\x41\x0F\xB6\xE9", "xxxxxxxxxx?xxxx"); + unsigned int (*RTech_AsyncLoad)(void* Src, __int64 a2, int a3, char pakfile) = (unsigned int (*)(void*, __int64, int, char))p_RTech_AsyncLoad.GetPtr(); /*40 53 48 83 EC 40 48 89 6C 24 ? 41 0F B6 E9*/ +#endif +} +void HRTech_UnloadAsset(std::int64_t a1, std::int64_t a2); +void HRtech_AsyncLoad(std::string svPakFileName); + +void RTech_Game_Attach(); +void RTech_Game_Detach(); + +/////////////////////////////////////////////////////////////////////////////// +class HRTechGame : public IDetour +{ + virtual void debugp() + { + std::cout << "| FUN: RTech_UnloadAsset : 0x" << std::hex << std::uppercase << p_RTech_UnloadAsset.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "| FUN: RTech_AsyncLoad : 0x" << std::hex << std::uppercase << p_RTech_AsyncLoad.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "+----------------------------------------------------------------+" << std::endl; + } +}; +/////////////////////////////////////////////////////////////////////////////// + +REGISTER(HRTechGame); diff --git a/r5dev/rtech/rtech.cpp b/r5dev/rtech/rtech_utils.cpp similarity index 99% rename from r5dev/rtech/rtech.cpp rename to r5dev/rtech/rtech_utils.cpp index a6fcd506..6164c356 100644 --- a/r5dev/rtech/rtech.cpp +++ b/r5dev/rtech/rtech_utils.cpp @@ -1,6 +1,6 @@ #include "core/stdafx.h" -#include "rtech/rtech.h" #include "engine/sys_utils.h" +#include "rtech/rtech_utils.h" /****************************************************************************** ------------------------------------------------------------------------------- diff --git a/r5dev/rtech/rtech.h b/r5dev/rtech/rtech_utils.h similarity index 100% rename from r5dev/rtech/rtech.h rename to r5dev/rtech/rtech_utils.h diff --git a/r5dev/tier0/ConCommand.cpp b/r5dev/tier0/ConCommand.cpp index 98b568a9..2b0739f0 100644 --- a/r5dev/tier0/ConCommand.cpp +++ b/r5dev/tier0/ConCommand.cpp @@ -127,7 +127,8 @@ void ConCommand_InitConCommand() void* fs_decompress_pak = ConCommand_RegisterCommand("fs_decompress_pak", "Decompresses user specified 'vpk_dir' file.", 0, _VPK_Decompress_f_CompletionFunc, nullptr); //------------------------------------------------------------------------- // RTECH API | - void* rtech_toguid = ConCommand_RegisterCommand("rtech_toguid", "Calculates the GUID from input data.", 0, _RTech_GenerateGUID_f_CompletionFunc, nullptr); + void* rtech_strtoguid = ConCommand_RegisterCommand("rtech_strtoguid", "Calculates the GUID from input data.", 0, _RTech_StringToGUID_f_CompletionFunc, nullptr); + void* rtech_asyncload = ConCommand_RegisterCommand("rtech_asyncload", "Loads user specified 'RPak' file.", 0, _RTech_AsyncLoad_f_CompletionFunc, nullptr); void* rtech_decompress = ConCommand_RegisterCommand("rtech_decompress", "Decompresses user specified 'RPak' file.", 0, _RTech_Decompress_f_CompletionFunc, nullptr); //------------------------------------------------------------------------- // NETCHANNEL | diff --git a/r5dev/tier0/completion.cpp b/r5dev/tier0/completion.cpp index d978612c..d29a06e5 100644 --- a/r5dev/tier0/completion.cpp +++ b/r5dev/tier0/completion.cpp @@ -12,7 +12,8 @@ #include "tier0/completion.h" #include "engine/net_chan.h" #include "engine/sys_utils.h" -#include "rtech/rtech.h" +#include "rtech/rtech_game.h" +#include "rtech/rtech_utils.h" #include "vpklib/packedstore.h" #include "gameui/IConsole.h" #include "public/include/bansystem.h" @@ -363,7 +364,7 @@ void _ReloadBanList_f_CompletionFunc(CCommand* cmd) g_pBanSystem->Load(); // Reload banlist. } -void _RTech_GenerateGUID_f_CompletionFunc(CCommand* cmd) +void _RTech_StringToGUID_f_CompletionFunc(CCommand* cmd) { std::int32_t argSize = *(std::int32_t*)((std::uintptr_t)cmd + 0x4); @@ -381,6 +382,14 @@ void _RTech_GenerateGUID_f_CompletionFunc(CCommand* cmd) DevMsg(eDLL_T::RTECH, "] GUID: '0x%llX'\n", guid); } +void _RTech_AsyncLoad_f_CompletionFunc(CCommand* cmd) +{ + CCommand& args = *cmd; // Get reference. + std::string firstArg = args[1]; // Get first arg. + + HRtech_AsyncLoad(firstArg); +} + void _RTech_Decompress_f_CompletionFunc(CCommand* cmd) { std::int32_t argSize = *(std::int32_t*)((std::uintptr_t)cmd + 0x4); diff --git a/r5dev/tier0/completion.h b/r5dev/tier0/completion.h index 5a01962b..839f5928 100644 --- a/r5dev/tier0/completion.h +++ b/r5dev/tier0/completion.h @@ -26,7 +26,8 @@ void _Ban_f_CompletionFunc(CCommand* cmd); void _BanID_f_CompletionFunc(CCommand* cmd); void _Unban_f_CompletionFunc(CCommand* cmd); void _ReloadBanList_f_CompletionFunc(CCommand* cmd); -void _RTech_GenerateGUID_f_CompletionFunc(CCommand* cmd); +void _RTech_StringToGUID_f_CompletionFunc(CCommand* cmd); +void _RTech_AsyncLoad_f_CompletionFunc(CCommand* cmd); void _RTech_Decompress_f_CompletionFunc(CCommand* cmd); void _VPK_Decompress_f_CompletionFunc(CCommand* cmd); void _NET_TraceNetChan_f_CompletionFunc(CCommand* cmd); diff --git a/r5dev/tier0/cvar.cpp b/r5dev/tier0/cvar.cpp index cac32474..e7ab5658 100644 --- a/r5dev/tier0/cvar.cpp +++ b/r5dev/tier0/cvar.cpp @@ -23,7 +23,7 @@ ConVar* fs_warning_level_native = new ConVar(); ConVar* fs_packedstore_entryblock_stats = new ConVar(); //------------------------------------------------------------------------- // MATERIALSYSTEM | -ConVar* mat_showdxoutput = new ConVar(); +ConVar* mat_showdxoutput = new ConVar(); //------------------------------------------------------------------------- // SQUIRREL | ConVar* sq_showrsonloading = new ConVar(); diff --git a/r5dev/windows/console.cpp b/r5dev/windows/console.cpp index 96927d05..41af100d 100644 --- a/r5dev/windows/console.cpp +++ b/r5dev/windows/console.cpp @@ -1,6 +1,6 @@ #include "core/stdafx.h" #include "core/init.h" -#include "rtech/rtech.h" +#include "rtech/rtech_utils.h" #ifndef DEDICATED #include "windows/id3dx.h" #endif // !DEDICATED