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