From 6583e174de4d4d5c1d482607b66866fdbc0a7f28 Mon Sep 17 00:00:00 2001 From: Amos <48657826+Mauler125@users.noreply.github.com> Date: Tue, 1 Mar 2022 02:39:46 +0100 Subject: [PATCH] Implement custom paks loading per level Game will check scripts/levels/mapname.json and load all rpaks in the rpak field. Game now also unloads pak files loaded by the SDK preventing crashes and unnecessary memory usage. VPK's and STBSP's are planned to be added as well. --- r5dev/common/opcodes.cpp | 53 ++++++------- r5dev/common/opcodes.h | 7 -- r5dev/core/init.cpp | 2 + r5dev/engine/cmodel_bsp.cpp | 83 ++++++++++++++++++++ r5dev/engine/cmodel_bsp.h | 14 +++- r5dev/engine/host_cmd.h | 7 ++ r5dev/engine/host_state.cpp | 61 ++++++++++---- r5dev/engine/host_state.h | 4 +- r5dev/engine/modelloader.cpp | 26 ++++++ r5dev/engine/modelloader.h | 40 ++++++---- r5dev/engine/sys_utils.cpp | 20 +++++ r5dev/rtech/rtech_game.cpp | 28 ++++++- r5dev/rtech/rtech_game.h | 42 ++++++++-- r5dev/thirdparty/imgui/src/imgui_utility.cpp | 2 +- r5dev/vproj/dedicated.vcxproj | 2 + r5dev/vproj/dedicated.vcxproj.filters | 6 ++ r5dev/vproj/gamesdk.vcxproj | 2 + r5dev/vproj/gamesdk.vcxproj.filters | 6 ++ 18 files changed, 323 insertions(+), 82 deletions(-) create mode 100644 r5dev/engine/cmodel_bsp.cpp create mode 100644 r5dev/engine/modelloader.cpp diff --git a/r5dev/common/opcodes.cpp b/r5dev/common/opcodes.cpp index 74c096ce..0b009545 100644 --- a/r5dev/common/opcodes.cpp +++ b/r5dev/common/opcodes.cpp @@ -103,13 +103,28 @@ void Dedicated_Init() // CMODELLOADER //------------------------------------------------------------------------- { - CModelLoader__LoadModel.Offset(0x462).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Prevent call to 'CStudioRenderContext::LoadMaterials'. - CModelLoader__Studio_LoadModel.Offset(0x325).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CMaterialSystem::FindMaterialEx' fails as RAX is nullptr. - CModelLoader__Studio_LoadModel.Offset(0x33D).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CMaterialGlue' class method fails as RAX is nullptr. - CModelLoader__Studio_LoadModel.Offset(0x359).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CMaterialGlue' class method fails as RAX is nullptr. - CModelLoader__Studio_LoadModel.Offset(0x374).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CMaterialGlue' class method fails as RAX is nullptr. - CModelLoader__Studio_LoadModel.Offset(0x38D).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'ReturnZero' fails as RAX is nullptr. - CModelLoader__Studio_LoadModel.Offset(0x3A4).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CMaterialGlue' class method fails as RAX is nullptr. + p_CModelLoader__LoadModel.Offset(0x462).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Prevent call to 'CStudioRenderContext::LoadMaterials'. + p_CModelLoader__Studio_LoadModel.Offset(0x325).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CMaterialSystem::FindMaterialEx' fails as RAX is nullptr. + p_CModelLoader__Studio_LoadModel.Offset(0x33D).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CMaterialGlue' class method fails as RAX is nullptr. + p_CModelLoader__Studio_LoadModel.Offset(0x359).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CMaterialGlue' class method fails as RAX is nullptr. + p_CModelLoader__Studio_LoadModel.Offset(0x374).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CMaterialGlue' class method fails as RAX is nullptr. + p_CModelLoader__Studio_LoadModel.Offset(0x38D).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'ReturnZero' fails as RAX is nullptr. + p_CModelLoader__Studio_LoadModel.Offset(0x3A4).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CMaterialGlue' class method fails as RAX is nullptr. + + p_CModelLoader__Map_LoadModelGuts.Offset(0x41).Patch({ 0xE9, 0x4F, 0x04, 0x00, 0x00 }); // JNE --> NOP | SKYLIGHTS. + p_CModelLoader__Map_LoadModelGuts.Offset(0x974).Patch({ 0x90, 0x90 }); // JE --> NOP | VERTNORMALS. + p_CModelLoader__Map_LoadModelGuts.Offset(0xA55).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | MATERIALSORTS. + p_CModelLoader__Map_LoadModelGuts.Offset(0xA62).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | MESHBOUNDS. + p_CModelLoader__Map_LoadModelGuts.Offset(0xA83).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | MESHVERTS. + p_CModelLoader__Map_LoadModelGuts.Offset(0xAC0).Patch({ 0x90, 0x90 }); // JE --> NOP | INDICES. + p_CModelLoader__Map_LoadModelGuts.Offset(0xBF2).Patch({ 0x90, 0x90 }); // JE --> NOP | WORLDLIGHTS. + p_CModelLoader__Map_LoadModelGuts.Offset(0xDA9).Patch({ 0x90, 0x90 }); // JE --> NOP | TWEAKLIGHTS. + p_CModelLoader__Map_LoadModelGuts.Offset(0xEEB).Patch({ 0xE9, 0x3D, 0x01, 0x00, 0x00 }); // JLE --> JMP | Exception 0x57 in while trying to dereference [R15 + R14 *8 + 0x10]. + p_CModelLoader__Map_LoadModelGuts.Offset(0x61B).Patch({ 0xE9, 0xE2, 0x02, 0x00, 0x00 }); // JZ --> JMP | Prevent call to 'CMod_LoadTextures()'. + p_CModelLoader__Map_LoadModelGuts.Offset(0x1045).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Prevent call to 'Mod_LoadCubemapSamples()'. + + p_CModelLoader__Map_LoadModelGuts.Offset(0x129).Patch({ 0x90, 0x90, 0x90 }); // MOV --> NOP | RCX is nullptr during dereference since shadersystem isn't initialized. Exception 'C0000005'. + p_CModelLoader__Map_LoadModelGuts.Offset(0x12C).Patch({ 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CTexture' class member in RAX + 0x78 fails. Previous instruction could not dereference. } //------------------------------------------------------------------------- @@ -184,8 +199,8 @@ void Dedicated_Init() // RUNTIME: HOST_NEWGAME //------------------------------------------------------------------------- { - Host_NewGame.Offset(0x4E0).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); - Host_NewGame.Offset(0x637).Patch({ 0xE9, 0xC1, 0x00, 0x00, 0x00 }); // JNE --> JMP | Prevent connect localhost from being executed in Host_NewGame. + p_Host_NewGame.Offset(0x4E0).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); + p_Host_NewGame.Offset(0x637).Patch({ 0xE9, 0xC1, 0x00, 0x00, 0x00 }); // JNE --> JMP | Prevent connect localhost from being executed in Host_NewGame. } //------------------------------------------------------------------------- @@ -228,26 +243,6 @@ void Dedicated_Init() FairFight_Init.Offset(0x0).FindPatternSelf("0F 87", ADDRESS::Direction::DOWN, 200).Patch({ 0x0F, 0x85 }); // JA --> JNZ | Prevent 'FairFight' anti-cheat from initializing on the server by comparing RAX against 0x0 instead. Init will crash since the plugins aren't shipped. } - //------------------------------------------------------------------------- - // RUNTIME: BSP_LUMP - //------------------------------------------------------------------------- - { - CollisionBSPData_LoadAllLumps.Offset(0x41).Patch({ 0xE9, 0x4F, 0x04, 0x00, 0x00 }); // JNE --> NOP | SKYLIGHTS. - CollisionBSPData_LoadAllLumps.Offset(0x974).Patch({ 0x90, 0x90 }); // JE --> NOP | VERTNORMALS. - CollisionBSPData_LoadAllLumps.Offset(0xA55).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | MATERIALSORTS. - CollisionBSPData_LoadAllLumps.Offset(0xA62).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | MESHBOUNDS. - CollisionBSPData_LoadAllLumps.Offset(0xA83).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | MESHVERTS. - CollisionBSPData_LoadAllLumps.Offset(0xAC0).Patch({ 0x90, 0x90 }); // JE --> NOP | INDICES. - CollisionBSPData_LoadAllLumps.Offset(0xBF2).Patch({ 0x90, 0x90 }); // JE --> NOP | WORLDLIGHTS. - CollisionBSPData_LoadAllLumps.Offset(0xDA9).Patch({ 0x90, 0x90 }); // JE --> NOP | TWEAKLIGHTS. - CollisionBSPData_LoadAllLumps.Offset(0xEEB).Patch({ 0xE9, 0x3D, 0x01, 0x00, 0x00 }); // JLE --> JMP | Exception 0x57 in while trying to dereference [R15 + R14 *8 + 0x10]. - CollisionBSPData_LoadAllLumps.Offset(0x61B).Patch({ 0xE9, 0xE2, 0x02, 0x00, 0x00 }); // JZ --> JMP | Prevent call to 'CMod_LoadTextures()'. - CollisionBSPData_LoadAllLumps.Offset(0x1045).Patch({ 0x90, 0x90, 0x90, 0x90, 0x90 }); // CAL --> NOP | Prevent call to 'Mod_LoadCubemapSamples()'. - - CollisionBSPData_LinkPhysics.Offset(0x129).Patch({ 0x90, 0x90, 0x90 }); // MOV --> NOP | RCX is nullptr during dereference since shadersystem isn't initialized. Exception 'C0000005'. - CollisionBSPData_LinkPhysics.Offset(0x12C).Patch({ 0x90, 0x90, 0x90 }); // CAL --> NOP | Virtual call to 'CTexture' class member in RAX + 0x78 fails. Previous instruction could not dereference. - } - //------------------------------------------------------------------------- // RUNTIME: PROP_STATIC //------------------------------------------------------------------------- diff --git a/r5dev/common/opcodes.h b/r5dev/common/opcodes.h index ee1aacd6..2cdf94cc 100644 --- a/r5dev/common/opcodes.h +++ b/r5dev/common/opcodes.h @@ -75,12 +75,6 @@ namespace ADDRESS Host_Shutdown = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x8B\xC4\x48\x83\xEC\x00\x80\x3D\x00\x00\x00\x00\x00\x0F\x85\x00\x00\x00\x00\x8B\x15\x00\x00\x00\x00", "xxxxxx?xx?????xx????xx????"); // 0x140239620 // 48 8B C4 48 83 EC ?? 80 3D ? ? ? ? ? 0F 85 ? ? ? ? 8B 15 ? ? ? ? // - //------------------------------------------------------------------------- - // RUNTIME: HOST_NEWGAME - //------------------------------------------------------------------------- - ADDRESS Host_NewGame = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x8B\xC4\x00\x41\x54\x41\x00\x48\x81\xEC\x00\x00\x00\x00\xF2", "xxx?xxx?xxx??xxx"); - // 0x140238DA0 // 48 8B C4 ?? 41 54 41 ?? 48 81 EC ?? ?? 00 00 F2 // - //------------------------------------------------------------------------- // RUNTIME: HOST_DISCONNECT //------------------------------------------------------------------------- @@ -131,7 +125,6 @@ class HOpcodes : public IDetour std::cout << "| FUN: Host_Init_0 : 0x" << std::hex << std::uppercase << gHost_Init_0.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "| FUN: Host_Init_1 : 0x" << std::hex << std::uppercase << gHost_Init_1.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "| FUN: Host_Init_2 : 0x" << std::hex << std::uppercase << gHost_Init_2.GetPtr() << std::setw(npad) << " |" << std::endl; - std::cout << "| FUN: Host_NewGame : 0x" << std::hex << std::uppercase << Host_NewGame.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "| FUN: Host_Disconnect : 0x" << std::hex << std::uppercase << Host_Disconnect.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "| FUN: _Host_RunFrame : 0x" << std::hex << std::uppercase << _Host_RunFrame.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "+----------------------------------------------------------------+" << std::endl; diff --git a/r5dev/core/init.cpp b/r5dev/core/init.cpp index a64dfda3..21b6234c 100644 --- a/r5dev/core/init.cpp +++ b/r5dev/core/init.cpp @@ -124,6 +124,7 @@ void Systems_Init() #if defined (DEDICATED) || defined (GAMEDLL_S3) CHostState_Attach(); #endif // DEDICATED || GAMEDLL_S3 + //CModelLoader_Attach(); CNetChan_Attach(); ConCommand_Attach(); @@ -208,6 +209,7 @@ void Systems_Shutdown() #if defined (DEDICATED) || defined (GAMEDLL_S3) CHostState_Detach(); // Dedicated only for now until backwards compatible with S1. #endif // DEDICATED || GAMEDLL_S3 + //CModelLoader_Detach(); CNetChan_Detach(); ConCommand_Detach(); diff --git a/r5dev/engine/cmodel_bsp.cpp b/r5dev/engine/cmodel_bsp.cpp new file mode 100644 index 00000000..528c382d --- /dev/null +++ b/r5dev/engine/cmodel_bsp.cpp @@ -0,0 +1,83 @@ +//=============================================================================// +// +// Purpose: +// +// $NoKeywords: $ +//=============================================================================// + +#include "core/stdafx.h" +#include "engine/host_cmd.h" +#include "engine/host_state.h" +#include "engine/sys_utils.h" +#include "engine/cmodel_bsp.h" +#include "rtech/rtech_game.h" + +int g_nLoadedPakFileId[256]{}; + +//----------------------------------------------------------------------------- +// Purpose: loads required assets for specified level +// Todo : STBSP and VPK loading +// Todo : Unload all assets loaded by this functions using 'nPakId' +//----------------------------------------------------------------------------- +void MOD_LoadDependencies(eBspRes_t resourceType) +{ + std::ostringstream ostream; + ostream << "platform\\scripts\\levels\\" << g_pHostState->m_levelName << ".json"; + + std::filesystem::path fsPath = std::filesystem::current_path() /= ostream.str(); + DevMsg(eDLL_T::ENGINE, "Loading level prerequisites file: '%s'\n", fsPath.string().c_str()); + + if (FileExists(fsPath.string().c_str())) + { + nlohmann::json jsIn; + try + { + std::ifstream iPakLoadDefFile(fsPath, std::ios::binary); // Parse prerequisites file. + iPakLoadDefFile >> jsIn; + iPakLoadDefFile.close(); + + switch (resourceType) + { + case eBspRes_t::RES_RPAK: + { + if (!jsIn.is_null()) + { + if (!jsIn["rpak"].is_null()) + { + int iPakIdx{}; + for (auto it = jsIn["rpak"].begin(); it != jsIn["rpak"].end(); ++it) + { + if (it.value().is_string()) + { + std::string svToLoad = it.value().get() + ".rpak"; + unsigned int nPakId = 0; + nPakId = RTech_AsyncLoad((void*)svToLoad.c_str(), g_pMallocPool.GetPtr(), 4, 0); + + if (nPakId == -1) + { + DevMsg(eDLL_T::RTECH, "RTech AsyncLoad failed read '%s' results '%u'\n", fsPath.string().c_str(), nPakId); + } + else + { + g_nLoadedPakFileId[iPakIdx] = nPakId; + iPakIdx++; + } + } + } + } + } + break; + } + default: + { + break; + } + } + } + catch (const std::exception& ex) + { + DevMsg(eDLL_T::ENGINE, "Error parsing level prerequisites file: '%s'\n", fsPath.string().c_str(), ex.what()); + return; + } + } +} diff --git a/r5dev/engine/cmodel_bsp.h b/r5dev/engine/cmodel_bsp.h index a929eaa3..466f4c10 100644 --- a/r5dev/engine/cmodel_bsp.h +++ b/r5dev/engine/cmodel_bsp.h @@ -3,19 +3,25 @@ // !TODO: BUILD AGNOSTIC! // namespace { - ADDRESS CollisionBSPData_LoadAllLumps = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x54\x24\x00\x48\x89\x4C\x24\x00\x55\x53\x56\x57\x41\x54\x41\x55\x41\x57", "xxxx?xxxx?xxxxxxxxxx"); // BSP. - // 0x1402546F0 // 48 89 54 24 ? 48 89 4C 24 ? 55 53 56 57 41 54 41 55 41 57 // - ADDRESS CollisionBSPData_LinkPhysics = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x5C\x24\x00\x48\x89\x6C\x24\x00\x57\x48\x81\xEC\x00\x00\x00\x00\x48\x8B\xF9\x33\xED", "xxxx?xxxx?xxxx????xxxxx"); // case 1: only gets called on changelevel, needs more research, function gets called by CModelLoader virtual function. // 0x140256480 // 48 89 5C 24 ? 48 89 6C 24 ? 57 48 81 EC ? ? ? ? 48 8B F9 33 ED // } +enum class eBspRes_t +{ + RES_RPAK = 0, + RES_VPK, + RES_STBSP +}; + +extern int g_nLoadedPakFileId[256]; + +void MOD_LoadDependencies(eBspRes_t resourceType); /////////////////////////////////////////////////////////////////////////////// class HModel_BSP : public IDetour { virtual void debugp() { - std::cout << "| FUN: CollisionBSPData_LoadAllLumps : 0x" << std::hex << std::uppercase << CollisionBSPData_LoadAllLumps.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "| FUN: CollisionBSPData_LinkPhysics : 0x" << std::hex << std::uppercase << CollisionBSPData_LinkPhysics.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "+----------------------------------------------------------------+" << std::endl; } diff --git a/r5dev/engine/host_cmd.h b/r5dev/engine/host_cmd.h index 29d872bd..77e407f9 100644 --- a/r5dev/engine/host_cmd.h +++ b/r5dev/engine/host_cmd.h @@ -16,9 +16,15 @@ namespace #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 ? ? ? ?*/ + + ADDRESS p_Host_NewGame = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x8B\xC4\x56\x41\x54\x41\x57\x48\x81\xEC\x00\x00\x00\x00\xF2\x0F\x10\x05\x00\x00\x00\x00", "xxxxxxxxxxx????xxxx????"); /*48 8B C4 56 41 54 41 57 48 81 EC ? ? ? ? F2 0F 10 05 ? ? ? ?*/ + bool (*Host_NewGame)(char* pszMapName, char* pszMapGroup, bool bLoadGame, char bBackground, LARGE_INTEGER PerformanceCount) = (bool (*)(char*, char*, bool, char, LARGE_INTEGER))p_Host_NewGame.GetPtr(); #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*/ + + ADDRESS p_Host_NewGame = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x8B\xC4\x00\x41\x54\x41\x55\x48\x81\xEC\x70\x04\x00\x00\xF2\x0F\x10\x05\x00\x00\x00\x0B", "xxx?xxxxxxxxxxxxxxx???x"); /*48 8B C4 ?? 41 54 41 55 48 81 EC 70 04 00 00 F2 0F 10 05 ?? ?? ?? 0B*/ + bool (*Host_NewGame)(char* pszMapName, char* pszMapGroup, bool bLoadGame, char bBackground, LARGE_INTEGER PerformanceCount) = (bool (*)(char*, char*, bool, char, LARGE_INTEGER))p_Host_NewGame.GetPtr(); #endif ADDRESS p_malloc_internal = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\xE9\x00\x00\x00\x00\xCC\xCC\xCC\x40\x53\x48\x83\xEC\x20\x48\x8D\x05\x00\x00\x00\x00", "x????xxxxxxxxxxxx????"); void* (*malloc_internal)(void* pPool, int64_t size) = (void* (*)(void*, int64_t))p_malloc_internal.GetPtr(); /*E9 ? ? ? ? CC CC CC 40 53 48 83 EC 20 48 8D 05 ? ? ? ?*/ @@ -41,6 +47,7 @@ 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 << "| FUN: Host_NewGame : 0x" << std::hex << std::uppercase << p_Host_NewGame.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "| FUN: malloc_internal : 0x" << std::hex << std::uppercase << p_malloc_internal.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "| VAR: g_pEngineParms : 0x" << std::hex << std::uppercase << g_pEngineParmsBuffer.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "| VAR: g_pMallocPool : 0x" << std::hex << std::uppercase << g_pMallocPool.GetPtr() << std::setw(npad) << " |" << std::endl; diff --git a/r5dev/engine/host_state.cpp b/r5dev/engine/host_state.cpp index 19c41754..bfed983b 100644 --- a/r5dev/engine/host_state.cpp +++ b/r5dev/engine/host_state.cpp @@ -15,11 +15,13 @@ #else // #include "engine/cl_rcon.h" #endif // DEDICATED -#include "engine/sys_engine.h" -#include "engine/sys_utils.h" +#include "engine/gl_screen.h" #include "engine/host_state.h" #include "engine/net_chan.h" -#include "engine/gl_screen.h" +#include "engine/sys_engine.h" +#include "engine/sys_utils.h" +#include "engine/cmodel_bsp.h" +#include "rtech/rtech_game.h" #ifndef DEDICATED #include "vgui/vgui_baseui_interface.h" #endif // DEDICATED @@ -30,6 +32,7 @@ std::chrono::time_point tpPylonStartClock = std::chrono::steady_clock::now(); std::chrono::time_point tpBanListStartClock = std::chrono::steady_clock::now(); +bool g_bLevelResourceInitialized = false; //----------------------------------------------------------------------------- // Purpose: state machine's main processing loop @@ -68,7 +71,6 @@ FORCEINLINE void CHostState::FrameUpdate(void* rcx, void* rdx, float time) case HostStates_t::HS_NEW_GAME: { DevMsg(eDLL_T::ENGINE, "Loading level: '%s'\n", g_pHostState->m_levelName); - g_pHostState->State_NewGame(); break; } @@ -91,12 +93,15 @@ FORCEINLINE void CHostState::FrameUpdate(void* rcx, void* rdx, float time) case HostStates_t::HS_GAME_SHUTDOWN: { DevMsg(eDLL_T::ENGINE, "Shutdown host game\n"); + + g_bLevelResourceInitialized = false; Host_Game_ShutdownFn(g_pHostState); break; } case HostStates_t::HS_RESTART: { DevMsg(eDLL_T::ENGINE, "Restarting state machine\n"); + g_bLevelResourceInitialized = false; #ifndef DEDICATED CL_EndMovieFn(); #endif // !DEDICATED @@ -107,6 +112,7 @@ FORCEINLINE void CHostState::FrameUpdate(void* rcx, void* rdx, float time) case HostStates_t::HS_SHUTDOWN: { DevMsg(eDLL_T::ENGINE, "Shutdown state machine\n"); + g_bLevelResourceInitialized = false; #ifndef DEDICATED CL_EndMovieFn(); #endif // !DEDICATED @@ -197,11 +203,44 @@ FORCEINLINE void CHostState::LoadConfig(void) const } } +//----------------------------------------------------------------------------- +// Purpose: shutdown active game +//----------------------------------------------------------------------------- +FORCEINLINE void CHostState::GameShutDown(void) +{ + g_bLevelResourceInitialized = false; + if (m_bActiveGame) + { + g_pServerGameDLL->GameShutdown(); + m_bActiveGame = 0; + } +} + +//----------------------------------------------------------------------------- +// Purpose: unloads all pakfiles loaded by the SDK +//----------------------------------------------------------------------------- +FORCEINLINE void CHostState::UnloadPakFile(void) +{ + for (int i = 0; i < sizeof(g_nLoadedPakFileId); i++) + { + if (g_nLoadedPakFileId[i] > 0) + { + RTech_UnloadPak(g_nLoadedPakFileId[i]); + } + else + { + memset(g_nLoadedPakFileId, '\0', sizeof(g_nLoadedPakFileId)); + break; + } + } +} + //----------------------------------------------------------------------------- // Purpose: initialize new game //----------------------------------------------------------------------------- FORCEINLINE void CHostState::State_NewGame(void) { + g_bLevelResourceInitialized = false; m_bSplitScreenConnect = false; if (!g_ServerGameClients) // Init Game if it ain't valid. { @@ -227,18 +266,6 @@ FORCEINLINE void CHostState::State_NewGame(void) } } -//----------------------------------------------------------------------------- -// Purpose: shutdown active game -//----------------------------------------------------------------------------- -FORCEINLINE void CHostState::GameShutDown(void) -{ - if (m_bActiveGame) - { - g_pServerGameDLL->GameShutdown(); - m_bActiveGame = 0; - } -} - //----------------------------------------------------------------------------- // Purpose: change singleplayer level //----------------------------------------------------------------------------- @@ -246,6 +273,7 @@ FORCEINLINE void CHostState::State_ChangeLevelSP(void) { DevMsg(eDLL_T::ENGINE, "Changing singleplayer level to: '%s'\n", m_levelName); m_flShortFrameTime = 1.5; // Set frame time. + g_bLevelResourceInitialized = false; if (CModelLoader_Map_IsValidFn(g_CModelLoader, m_levelName)) // Check if map is valid and if we can start a new game. { @@ -272,6 +300,7 @@ FORCEINLINE void CHostState::State_ChangeLevelMP(void) { DevMsg(eDLL_T::ENGINE, "Changing multiplayer level to: '%s'\n", m_levelName); m_flShortFrameTime = 0.5; // Set frame time. + g_bLevelResourceInitialized = false; g_pServerGameDLL->LevelShutdown(); if (CModelLoader_Map_IsValidFn(g_CModelLoader, m_levelName)) // Check if map is valid and if we can start a new game. diff --git a/r5dev/engine/host_state.h b/r5dev/engine/host_state.h index de2ad21a..b79098c1 100644 --- a/r5dev/engine/host_state.h +++ b/r5dev/engine/host_state.h @@ -22,9 +22,10 @@ public: FORCEINLINE void Setup(void) const; FORCEINLINE void Think(void) const; - FORCEINLINE void State_NewGame(void); FORCEINLINE void GameShutDown(void); + FORCEINLINE void UnloadPakFile(void); + FORCEINLINE void State_NewGame(void); FORCEINLINE void State_ChangeLevelSP(void); FORCEINLINE void State_ChangeLevelMP(void); @@ -71,6 +72,7 @@ namespace // !TEMP static auto SendOfflineRequestToStryderFn = ADDRESS(0x14033D380).RCast(); } +extern bool g_bLevelResourceInitialized; /////////////////////////////////////////////////////////////////////////////// void CHostState_Attach(); void CHostState_Detach(); diff --git a/r5dev/engine/modelloader.cpp b/r5dev/engine/modelloader.cpp new file mode 100644 index 00000000..6eded16a --- /dev/null +++ b/r5dev/engine/modelloader.cpp @@ -0,0 +1,26 @@ +//===========================================================================// +// +// Purpose: Model loading / unloading interface +// +// $NoKeywords: $ +//===========================================================================// + +#include "core/stdafx.h" +#include "engine/cmodel_bsp.h" +#include "engine/modelloader.h" + +uint64_t HCModelLoader__Map_LoadModelGuts(void* thisptr, void* mod) +{ + return CModelLoader__Map_LoadModelGuts(thisptr, mod); +} + +/////////////////////////////////////////////////////////////////////////////// +void CModelLoader_Attach() +{ + DetourAttach((LPVOID*)&CModelLoader__Map_LoadModelGuts, &HCModelLoader__Map_LoadModelGuts); +} + +void CModelLoader_Detach() +{ + DetourDetach((LPVOID*)&CModelLoader__Map_LoadModelGuts, &HCModelLoader__Map_LoadModelGuts); +} \ No newline at end of file diff --git a/r5dev/engine/modelloader.h b/r5dev/engine/modelloader.h index 9cfcd834..8da9f598 100644 --- a/r5dev/engine/modelloader.h +++ b/r5dev/engine/modelloader.h @@ -3,34 +3,44 @@ namespace { #if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) - ADDRESS CModelLoader__FindModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x55\x41\x55\x41\x56\x48\x8D\xAC\x24\x00\x00\x00\x00", "xxxxxxxxxx????"); - // 0x1402A1F10 // 40 55 41 55 41 56 48 8D AC 24 ? ? ? ? // + ADDRESS p_CModelLoader__FindModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x55\x41\x55\x41\x56\x48\x8D\xAC\x24\x00\x00\x00\x00", "xxxxxxxxxx????"); + void*(*CModelLoader__FindModel)(void* thisptr, const char* pszModelName) = (void*(*)(void*, const char*))p_CModelLoader__FindModel.GetPtr(); /*40 55 41 55 41 56 48 8D AC 24 ? ? ? ?*/ - ADDRESS CModelLoader__LoadModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x53\x57\x41\x56\x48\x81\xEC\x00\x00\x00\x00\x48\x8B\xFA", "xxxxxxxx????xxx"); - // 0x1402A23B0 // 40 53 57 41 56 48 81 EC ? ? ? ? 48 8B FA // + ADDRESS p_CModelLoader__LoadModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x53\x57\x41\x56\x48\x81\xEC\x00\x00\x00\x00\x48\x8B\xFA", "xxxxxxxx????xxx"); + void(*CModelLoader__LoadModel)(void* thisptr, void* mod) = (void(*)(void*, void*))p_CModelLoader__LoadModel.GetPtr(); /*40 53 57 41 56 48 81 EC ? ? ? ? 48 8B FA*/ - ADDRESS CModelLoader__Studio_LoadModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x5C\x24\x00\x55\x56\x57\x41\x54\x41\x56\x48\x8D\xAC\x24\x00\x00\x00\x00", "xxxx?xxxxxxxxxxx????"); - // 0x140252F10 // 48 89 5C 24 ? 55 56 57 41 54 41 57 48 81 EC ? ? ? ? // + ADDRESS p_CModelLoader__Studio_LoadModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x5C\x24\x00\x55\x56\x57\x41\x54\x41\x56\x48\x8D\xAC\x24\x00\x00\x00\x00", "xxxx?xxxxxxxxxxx????"); + void* (*CModelLoader__Studio_LoadModel)(void* thisptr) = (void* (*)(void*))p_CModelLoader__Studio_LoadModel.GetPtr(); /*48 89 5C 24 ? 55 56 57 41 54 41 56 48 8D AC 24 ? ? ? ?*/ + + ADDRESS p_CModelLoader__Map_LoadModelGuts = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x54\x24\x00\x48\x89\x4C\x24\x00\x55\x53\x41\x54\x41\x55\x48\x8D\xAC\x24\x00\x00\x00\x00\x48\x81\xEC\x00\x00\x00\x00\xFF\x05\x00\x00\x00\x00", "xxxx?xxxx?xxxxxxxxxx????xxx????xx????"); // BSP. + uint64_t(*CModelLoader__Map_LoadModelGuts)(void* thisptr, void* mod) = (uint64_t(*)(void*, void*))p_CModelLoader__Map_LoadModelGuts.GetPtr(); /*48 89 54 24 ? 48 89 4C 24 ? 55 53 41 54 41 55 48 8D AC 24 ? ? ? ? 48 81 EC ? ? ? ? FF 05 ? ? ? ? */ #elif defined (GAMEDLL_S2) || defined (GAMEDLL_S3) - ADDRESS CModelLoader__FindModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x55\x41\x57\x48\x83\xEC\x48\x80\x3A\x2A", "xxxxxxxxxxx"); - // 0x140253530 // 40 55 41 57 48 83 EC 48 80 3A 2A // + ADDRESS p_CModelLoader__FindModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x55\x41\x57\x48\x83\xEC\x48\x80\x3A\x2A", "xxxxxxxxxxx"); + void*(*CModelLoader__FindModel)(void* thisptr, const char* pszModelName) = (void* (*)(void*, const char*))p_CModelLoader__FindModel.GetPtr(); /*40 55 41 57 48 83 EC 48 80 3A 2A*/ - ADDRESS CModelLoader__LoadModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x53\x57\x41\x57\x48\x81\xEC\x00\x00\x00\x00\x48\x8B\x05\x00\x00\x00\x00", "xxxxxxxx????xxx????"); - // 0x140253810 // 40 53 57 41 57 48 81 EC ? ? ? ? 48 8B 05 ? ? ? ? // + ADDRESS p_CModelLoader__LoadModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x40\x53\x57\x41\x57\x48\x81\xEC\x00\x00\x00\x00\x48\x8B\x05\x00\x00\x00\x00", "xxxxxxxx????xxx????"); + void(*CModelLoader__LoadModel)(void* thisptr, void* mod) = (void(*)(void*, void*))p_CModelLoader__LoadModel.GetPtr(); /*40 53 57 41 57 48 81 EC ? ? ? ? 48 8B 05 ? ? ? ?*/ - ADDRESS CModelLoader__Studio_LoadModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x5C\x24\x00\x55\x56\x57\x41\x54\x41\x57\x48\x81\xEC\x00\x00\x00\x00", "xxxx?xxxxxxxxxx????"); - // 0x140252F10 // 48 89 5C 24 ? 55 56 57 41 54 41 57 48 81 EC ? ? ? ? // + ADDRESS p_CModelLoader__Studio_LoadModel = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x5C\x24\x00\x55\x56\x57\x41\x54\x41\x57\x48\x81\xEC\x00\x00\x00\x00", "xxxx?xxxxxxxxxx????"); + void*(*CModelLoader__Studio_LoadModel)(void* thisptr) = (void*(*)(void*))p_CModelLoader__Studio_LoadModel.GetPtr(); /*48 89 5C 24 ? 55 56 57 41 54 41 57 48 81 EC ? ? ? ?*/ + + ADDRESS p_CModelLoader__Map_LoadModelGuts = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x54\x24\x00\x48\x89\x4C\x24\x00\x55\x53\x56\x57\x41\x54\x41\x55\x41\x57", "xxxx?xxxx?xxxxxxxxxx"); // BSP. + uint64_t(*CModelLoader__Map_LoadModelGuts)(void* thisptr, void* mod) = (uint64_t(*)(void*, void*))p_CModelLoader__Map_LoadModelGuts.GetPtr(); /*48 89 54 24 ? 48 89 4C 24 ? 55 53 56 57 41 54 41 55 41 57*/ #endif } +void CModelLoader_Attach(); +void CModelLoader_Detach(); + /////////////////////////////////////////////////////////////////////////////// class HModelLoader : public IDetour { virtual void debugp() { - std::cout << "| FUN: CModelLoader::FindModel : 0x" << std::hex << std::uppercase << CModelLoader__FindModel.GetPtr() << std::setw(npad) << " |" << std::endl; - std::cout << "| FUN: CModelLoader::LoadModel : 0x" << std::hex << std::uppercase << CModelLoader__LoadModel.GetPtr() << std::setw(npad) << " |" << std::endl; - std::cout << "| FUN: CModelLoader::Studio_LoadModel : 0x" << std::hex << std::uppercase << CModelLoader__Studio_LoadModel.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "| FUN: CModelLoader::FindModel : 0x" << std::hex << std::uppercase << p_CModelLoader__FindModel.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "| FUN: CModelLoader::LoadModel : 0x" << std::hex << std::uppercase << p_CModelLoader__LoadModel.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "| FUN: CModelLoader::Map_LoadModelGuts : 0x" << std::hex << std::uppercase << p_CModelLoader__Map_LoadModelGuts.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "| FUN: CModelLoader::Studio_LoadModel : 0x" << std::hex << std::uppercase << p_CModelLoader__Studio_LoadModel.GetPtr() << std::setw(npad) << " |" << std::endl; std::cout << "+----------------------------------------------------------------+" << std::endl; } }; diff --git a/r5dev/engine/sys_utils.cpp b/r5dev/engine/sys_utils.cpp index 2cff4077..0d973dc0 100644 --- a/r5dev/engine/sys_utils.cpp +++ b/r5dev/engine/sys_utils.cpp @@ -9,7 +9,9 @@ #include "tier0/cvar.h" #include "tier0/commandline.h" #include "engine/common.h" +#include "engine/host_state.h" #include "engine/sys_utils.h" +#include "engine/cmodel_bsp.h" #ifdef DEDICATED #include "engine/sv_rcon.h" #else @@ -180,6 +182,24 @@ void* HSys_LoadAssetHelper(const CHAR* lpFileName, std::int64_t a2, LARGE_INTEGE std::string base_file = lpFileName; const std::string mod_dir = "paks\\Win32\\"; const std::string base_dir = "paks\\Win64\\"; + static bool bBasePaksLoaded = false; + + if (g_pHostState) + { + std::string svLevelName = g_pHostState->m_levelName; + std::string svMapPakName = svLevelName + ".rpak"; + + if (!g_bLevelResourceInitialized && !g_pHostState->m_bActiveGame && + bBasePaksLoaded || !strcmp(std::string(lpFileName).erase(0, 11).c_str(), "mp_lobby.rpak")) + { + // Attempt to load level dependencies if they exist. + MOD_LoadDependencies(eBspRes_t::RES_RPAK); + + // By the time mp_lobby.rpak is loaded, all the base paks are loaded as well and we can load anything else. + bBasePaksLoaded = true; + g_bLevelResourceInitialized = true; + } + } if (strstr(lpFileName, base_dir.c_str())) { diff --git a/r5dev/rtech/rtech_game.cpp b/r5dev/rtech/rtech_game.cpp index 793521e4..37e101d6 100644 --- a/r5dev/rtech/rtech_game.cpp +++ b/r5dev/rtech/rtech_game.cpp @@ -35,11 +35,11 @@ void HRtech_AsyncLoad(std::string svPakFileName) if (FileExists(svPakFilePathMod.c_str()) || FileExists(svPakFilePathBase.c_str())) { - unsigned int results = RTech_AsyncLoad((void*)svPakFileName.c_str(), g_pMallocPool.GetPtr(), NULL, NULL); + int nPakId = RTech_AsyncLoad((void*)svPakFileName.c_str(), g_pMallocPool.GetPtr(), NULL, NULL); - if (results == 0xFFFFFFFF) + if (nPakId == 0xFFFFFFFF) { - DevMsg(eDLL_T::RTECH, "RTech AsyncLoad failed read '%s' results '%u'\n", svPakFileName.c_str(), results); + DevMsg(eDLL_T::RTECH, "RTech AsyncLoad failed read '%s' results '%u'\n", svPakFileName.c_str(), nPakId); } } else @@ -48,12 +48,34 @@ void HRtech_AsyncLoad(std::string svPakFileName) } } +//----------------------------------------------------------------------------- +// Purpose: +//----------------------------------------------------------------------------- +void* HJT_HelpWithAnything(bool bShouldLoadPak) +{ +#if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) + static void* retaddr = reinterpret_cast(p_Host_NewGame.Offset(0x400).FindPatternSelf("48 8B ?? ?? ?? ?? 01", ADDRESS::Direction::DOWN).GetPtr()); +#elif defined (GAMEDLL_S2) || defined (GAMEDLL_S3) + static void* retaddr = reinterpret_cast(p_Host_NewGame.Offset(0x4A0).FindPatternSelf("48 8B ?? ?? ?? ?? 01", ADDRESS::Direction::DOWN).GetPtr()); +#endif + void* results = JT_HelpWithAnything(bShouldLoadPak); + + if (retaddr != _ReturnAddress()) // Check if this is called after 'PakFile_Init()'. + { + return results; + } + // Do stuff here after 'PakFile_Init()'. + return results; +} + void RTech_Game_Attach() { //DetourAttach((LPVOID*)&RTech_UnloadAsset, &HRTech_UnloadAsset); + //DetourAttach((LPVOID*)&JT_HelpWithAnything, &HJT_HelpWithAnything); } void RTech_Game_Detach() { //DetourAttach((LPVOID*)&RTech_UnloadAsset, &HRTech_UnloadAsset); + //DetourAttach((LPVOID*)&JT_HelpWithAnything, &HJT_HelpWithAnything); } diff --git a/r5dev/rtech/rtech_game.h b/r5dev/rtech/rtech_game.h index 9729d0de..172cc475 100644 --- a/r5dev/rtech/rtech_game.h +++ b/r5dev/rtech/rtech_game.h @@ -1,5 +1,25 @@ #pragma once +enum class ePakStatus : int +{ + PAK_STATUS_FREED = 0, + PAK_STATUS_LOAD_PENDING, + PAK_STATUS_REPAK_RUNNING, + PAK_STATUS_REPAK_DONE, + PAK_STATUS_LOAD_STARTING, + PAK_STATUS_LOAD_PAKHDR, + PAK_STATUS_LOAD_PATCH_INIT, + PAK_STATUS_LOAD_PATCH_EDIT_STREAM, + PAK_STATUS_LOAD_ASSETS, + PAK_STATUS_LOADED, // 9 + PAK_STATUS_UNLOAD_PENDING, + PAK_STATUS_FREE_PENDING, + PAK_STATUS_CANCELING, + PAK_STATUS_ERROR, // 13 + PAK_STATUS_INVALID_PAKHANDLE, + PAK_STATUS_BUSY +}; + namespace { /* ==== RTECH_GAME ====================================================================================================================================================== */ @@ -11,16 +31,23 @@ namespace 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 ? ? ? ?*/ #endif - + ADDRESS p_RTech_UnloadPak = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x5C\x24\x00\x48\x89\x74\x24\x00\x57\x48\x83\xEC\x30\x8B\xC1", "xxxx?xxxx?xxxxxxx"); + void*(*RTech_UnloadPak)(int nPakId) = (void*(*)(int nPakId))p_RTech_UnloadPak.GetPtr();/*48 89 5C 24 ? 48 89 74 24 ? 57 48 83 EC 30 8B C1*/ #if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) || defined (GAMEDLL_S2) 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"); /*40 53 48 83 EC 40 48 89 6C 24 ? 41 8B E8*/ - unsigned int (*RTech_AsyncLoad)(void* Src, std::uintptr_t pMalloc, int nIdx, char szPakFile) = (unsigned int (*)(void*, std::uintptr_t, int, char))p_RTech_AsyncLoad.GetPtr(); + int (*RTech_AsyncLoad)(void* Src, std::uintptr_t pMalloc, int nIdx, char szPakFile) = (int (*)(void*, std::uintptr_t, int, char))p_RTech_AsyncLoad.GetPtr(); #elif defined (GAMEDLL_S2) || defined (GAMEDLL_S3) 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"); /*40 53 48 83 EC 40 48 89 6C 24 ? 41 0F B6 E9*/ - unsigned int (*RTech_AsyncLoad)(void* Src, std::uintptr_t pMalloc, int nIdx, char szPakFile) = (unsigned int (*)(void*, std::uintptr_t, int, char))p_RTech_AsyncLoad.GetPtr(); + int (*RTech_AsyncLoad)(void* Src, std::uintptr_t pMalloc, int nIdx, char szPakFile) = (int (*)(void*, std::uintptr_t, int, char))p_RTech_AsyncLoad.GetPtr(); ADDRESS p_RTech_LoadPak = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x4C\x24\x00\x56\x41\x55", "xxxx?xxx"); /*48 89 4C 24 ? 56 41 55*/ unsigned int (*RTech_LoadPak)(void* thisptr, void* a2, std::uint64_t a3) = (unsigned int (*)(void*, void*, std::uint64_t))p_RTech_LoadPak.GetPtr(); + + ADDRESS p_RTech_LoadMapPak = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x81\xEC\x00\x00\x00\x00\x0F\xB6\x05\x00\x00\x00\x00\x4C\x8D\x05\x00\x00\x00\x00\x84\xC0", "xxx????xxx????xxx????xx"); + bool (*RTech_LoadMapPak)(const char* szPakFile) = (bool (*)(const char*))p_RTech_LoadMapPak.GetPtr(); /*48 81 EC ? ? ? ? 0F B6 05 ? ? ? ? 4C 8D 05 ? ? ? ? 84 C0*/ + + ADDRESS p_JT_HelpWithAnything = g_mGameDll.FindPatternSIMD((std::uint8_t*)"\x48\x89\x5C\x24\x00\x48\x89\x74\x24\x00\x57\x48\x83\xEC\x30\x80\x3D\x00\x00\x00\x00\x00", "xxxx?xxxx?xxxxxxx?????"); + void* (*JT_HelpWithAnything)(bool bShouldLoadPak) = (void* (*)(bool))p_JT_HelpWithAnything.GetPtr(); /*48 89 5C 24 ? 48 89 74 24 ? 57 48 83 EC 30 80 3D ? ? ? ? ?*/ #endif } void HRTech_UnloadAsset(std::int64_t a1, std::int64_t a2); @@ -34,10 +61,13 @@ 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 << "| 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; #if defined (GAMEDLL_S2) || defined (GAMEDLL_S3) - std::cout << "| FUN: RTech_LoadPak : 0x" << std::hex << std::uppercase << p_RTech_LoadPak.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "| FUN: RTech_LoadPak : 0x" << std::hex << std::uppercase << p_RTech_LoadPak.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "| FUN: RTech_LoadMapPak : 0x" << std::hex << std::uppercase << p_RTech_LoadMapPak.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "| FUN: RTech_UnloadPak : 0x" << std::hex << std::uppercase << p_RTech_UnloadPak.GetPtr() << std::setw(npad) << " |" << std::endl; + std::cout << "| FUN: JT_HelpWithAnything : 0x" << std::hex << std::uppercase << p_JT_HelpWithAnything.GetPtr() << std::setw(npad) << " |" << std::endl; #endif // GAMEDLL_S2 || GAMEDLL_S3 std::cout << "+----------------------------------------------------------------+" << std::endl; } diff --git a/r5dev/thirdparty/imgui/src/imgui_utility.cpp b/r5dev/thirdparty/imgui/src/imgui_utility.cpp index 7925141a..4f9219a4 100644 --- a/r5dev/thirdparty/imgui/src/imgui_utility.cpp +++ b/r5dev/thirdparty/imgui/src/imgui_utility.cpp @@ -70,7 +70,7 @@ void ImGuiConfig::Load() } catch (const std::exception& ex) { - DevMsg(eDLL_T::MS, "ImGui config file '%s' not found. Changing the settings in the console or server browser options re-create's it. Exception: '%s'\n", fsPath.string().c_str(), ex.what()); + DevMsg(eDLL_T::MS, "ImGui config file '%s' not found\n Changing the settings in the console or server browser options re-create's it\n Exception: '%s'\n", fsPath.string().c_str(), ex.what()); return; } } diff --git a/r5dev/vproj/dedicated.vcxproj b/r5dev/vproj/dedicated.vcxproj index 1a6d6755..cb37e1d0 100644 --- a/r5dev/vproj/dedicated.vcxproj +++ b/r5dev/vproj/dedicated.vcxproj @@ -414,9 +414,11 @@ + + diff --git a/r5dev/vproj/dedicated.vcxproj.filters b/r5dev/vproj/dedicated.vcxproj.filters index e1f2192f..ea45b490 100644 --- a/r5dev/vproj/dedicated.vcxproj.filters +++ b/r5dev/vproj/dedicated.vcxproj.filters @@ -968,6 +968,12 @@ sdk\game\server + + sdk\engine + + + sdk\engine + diff --git a/r5dev/vproj/gamesdk.vcxproj b/r5dev/vproj/gamesdk.vcxproj index 7fc02d64..cccc1163 100644 --- a/r5dev/vproj/gamesdk.vcxproj +++ b/r5dev/vproj/gamesdk.vcxproj @@ -36,10 +36,12 @@ + + diff --git a/r5dev/vproj/gamesdk.vcxproj.filters b/r5dev/vproj/gamesdk.vcxproj.filters index edd2bc5e..147c5ff7 100644 --- a/r5dev/vproj/gamesdk.vcxproj.filters +++ b/r5dev/vproj/gamesdk.vcxproj.filters @@ -384,6 +384,12 @@ sdk\engine + + sdk\engine + + + sdk\engine +