From ba252c8e98bd20a0156376b7b72d36957fed518c Mon Sep 17 00:00:00 2001 From: Kawe Mazidjatari <48657826+Mauler125@users.noreply.github.com> Date: Thu, 3 Nov 2022 10:44:59 +0100 Subject: [PATCH] Implement logic for hot swapping NavMesh The new command 'navmesh_reload' hot swaps the current NavMesh for all hulls in memory, with that of the 'new' files on the disk. The memory of the old NavMesh gets freed and destroyed. --- r5dev/game/server/ai_utility.cpp | 21 +++++++++++++++++++++ r5dev/game/server/detour_impl.h | 19 +++++++++++++++++++ r5dev/tier1/cmd.cpp | 2 ++ r5dev/vstdlib/callback.cpp | 14 ++++++++++++++ r5dev/vstdlib/callback.h | 1 + 5 files changed, 57 insertions(+) diff --git a/r5dev/game/server/ai_utility.cpp b/r5dev/game/server/ai_utility.cpp index 74b3f9f2..21cb922c 100644 --- a/r5dev/game/server/ai_utility.cpp +++ b/r5dev/game/server/ai_utility.cpp @@ -53,6 +53,27 @@ uint8_t IsGoalPolyReachable(dtNavMesh* nav, dtPolyRef fromRef, dtPolyRef goalRef return v_dtNavMesh__isPolyReachable(nav, fromRef, goalRef, hullId); } +//----------------------------------------------------------------------------- +// Purpose: hot swaps the NavMesh with the current files on the disk +// (All hulls will be reloaded! If NavMesh for hull no longer exist, it will be empty!!!) +//----------------------------------------------------------------------------- +void Detour_Reload() +{ + // Destroy and free the memory for all NavMesh hulls. + for (int i = 0; i < MAX_HULLS; i++) + { + dtNavMesh* nav = GetNavMeshForHull(i); + if (nav) + { + v_Detour_FreeNavMesh(nav); + MemAllocSingleton()->Free(nav); + } + } + + // Reload NavMesh for current level. + v_Detour_LevelInit(); +} + /////////////////////////////////////////////////////////////////////////////// void CAI_Utility_Attach() { diff --git a/r5dev/game/server/detour_impl.h b/r5dev/game/server/detour_impl.h index 81119d28..eb1fb9a9 100644 --- a/r5dev/game/server/detour_impl.h +++ b/r5dev/game/server/detour_impl.h @@ -6,6 +6,12 @@ //------------------------------------------------------------------------- // RUNTIME: DETOUR //------------------------------------------------------------------------- +inline CMemory p_Detour_LevelInit; +inline auto v_Detour_LevelInit = p_Detour_LevelInit.RCast(); + +inline CMemory p_Detour_FreeNavMesh; +inline auto v_Detour_FreeNavMesh = p_Detour_FreeNavMesh.RCast(); + inline CMemory p_dtNavMesh__Init; inline auto v_dtNavMesh__Init = p_dtNavMesh__Init.RCast(); @@ -38,11 +44,14 @@ inline dtNavMeshQuery* g_pNavMeshQuery = nullptr; dtNavMesh* GetNavMeshForHull(int hullSize); uint32_t GetHullMaskById(int hullId); +void Detour_Reload(); /////////////////////////////////////////////////////////////////////////////// class VRecast : public IDetour { virtual void GetAdr(void) const { + spdlog::debug("| FUN: Detour_LevelInit : {:#18x} |\n", p_Detour_LevelInit.GetPtr()); + spdlog::debug("| FUN: p_Detour_FreeNavMesh : {:#18x} |\n", p_Detour_FreeNavMesh.GetPtr()); spdlog::debug("| FUN: dtNavMesh::Init : {:#18x} |\n", p_dtNavMesh__Init.GetPtr()); spdlog::debug("| FUN: dtNavMesh::addTile : {:#18x} |\n", p_dtNavMesh__addTile.GetPtr()); spdlog::debug("| FUN: dtNavMesh::isPolyReachable : {:#18x} |\n", p_dtNavMesh__isPolyReachable.GetPtr()); @@ -52,10 +61,20 @@ class VRecast : public IDetour } virtual void GetFun(void) const { + +#if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) + p_Detour_LevelInit = g_GameDll.FindPatternSIMD(reinterpret_cast("\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\x45\x33\xF6\x48\x8D\x3D\x00\x00\x00\x00"), "xxxx?xxxx?xxxx?xxxxxxxxxxxx????xxxxxx????"); + p_Detour_FreeNavMesh = g_GameDll.FindPatternSIMD(reinterpret_cast("\x40\x53\x48\x83\xEC\x30\x48\x89\x6C\x24\x00\x48\x89\x74\x24\x00"), "xxxxxxxxxx?xxxx?"); +#elif defined (GAMEDLL_S2) || defined (GAMEDLL_S3) + p_Detour_LevelInit = g_GameDll.FindPatternSIMD(reinterpret_cast("\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\x48\x81\xEC\x00\x00\x00\x00\x45\x33\xE4"), "xxxx?xxxx?xxxx?xxxxxxxxxxxxx????xxx????xxx"); + p_Detour_FreeNavMesh = g_GameDll.FindPatternSIMD(reinterpret_cast("\x40\x53\x48\x83\xEC\x30\x48\x89\x6C\x24\x00\x48\x8B\xD9"), "xxxxxxxxxx?xxx"); +#endif p_dtNavMesh__Init = g_GameDll.FindPatternSIMD(reinterpret_cast("\x4C\x89\x44\x24\x00\x53\x41\x56\x48\x81\xEC\x00\x00\x00\x00\x0F\x10\x11"), "xxxx?xxxxxx????xxx"); p_dtNavMesh__addTile = g_GameDll.FindPatternSIMD(reinterpret_cast("\x44\x89\x4C\x24\x00\x41\x55"), "xxxx?xx"); p_dtNavMesh__isPolyReachable = g_GameDll.FindPatternSIMD(reinterpret_cast("\x48\x89\x6C\x24\x00\x48\x89\x74\x24\x00\x48\x89\x7C\x24\x00\x41\x56\x49\x63\xF1"), "xxxx?xxxx?xxxx?xxxxx"); + v_Detour_LevelInit = p_Detour_LevelInit.RCast(); /*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 ?? ?? ?? ?? 48 81 EC ?? ?? ?? ?? 45 33 E4*/ + v_Detour_FreeNavMesh = p_Detour_FreeNavMesh.RCast(); v_dtNavMesh__Init = p_dtNavMesh__Init.RCast(); /*4C 89 44 24 ?? 53 41 56 48 81 EC ?? ?? ?? ?? 0F 10 11*/ v_dtNavMesh__addTile = p_dtNavMesh__addTile.RCast(); /*44 89 4C 24 ?? 41 55*/ v_dtNavMesh__isPolyReachable = p_dtNavMesh__isPolyReachable.RCast(); /*48 89 6C 24 ?? 48 89 74 24 ?? 48 89 7C 24 ?? 41 56 49 63 F1*/ diff --git a/r5dev/tier1/cmd.cpp b/r5dev/tier1/cmd.cpp index a1c53ec7..67048e1d 100644 --- a/r5dev/tier1/cmd.cpp +++ b/r5dev/tier1/cmd.cpp @@ -343,6 +343,8 @@ void ConCommand::Init(void) ConCommand::Create("sv_banid", "Bans a client from the server by handle, nucleus id or ip address | Usage: sv_banid \"\"/\"/\".", FCVAR_RELEASE, Host_BanID_f, nullptr); ConCommand::Create("sv_unban", "Unbans a client from the server by nucleus id or ip address | Usage: sv_unban \"\"/\"\".", FCVAR_RELEASE, Host_Unban_f, nullptr); ConCommand::Create("sv_reloadbanlist", "Reloads the banned list.", FCVAR_RELEASE, Host_ReloadBanList_f, nullptr); + + ConCommand::Create("navmesh_reload", "Reloads the NavMesh for all hulls.", FCVAR_RELEASE, Detour_Reload_f, nullptr); #endif // !CLIENT_DLL #ifndef DEDICATED //------------------------------------------------------------------------- diff --git a/r5dev/vstdlib/callback.cpp b/r5dev/vstdlib/callback.cpp index e141bc37..a8c2eac6 100644 --- a/r5dev/vstdlib/callback.cpp +++ b/r5dev/vstdlib/callback.cpp @@ -48,6 +48,9 @@ #include "materialsystem/cmaterialglue.h" #include "public/idebugoverlay.h" #endif // !DEDICATED +#ifndef CLIENT_DLL +#include "game/server/detour_impl.h" +#endif // !CLIENT_DLL #ifndef DEDICATED #include "game/client/view.h" #endif // !DEDICATED @@ -210,6 +213,17 @@ void Host_Changelevel_f(const CCommand& args) } } +/* +===================== +Detour_Reload_f + + Hot swaps the NavMesh +===================== +*/ +void Detour_Reload_f(const CCommand& args) +{ + Detour_Reload(); +} #endif // !CLIENT_DLL /* ===================== diff --git a/r5dev/vstdlib/callback.h b/r5dev/vstdlib/callback.h index 4cdd1f0a..7ae5456b 100644 --- a/r5dev/vstdlib/callback.h +++ b/r5dev/vstdlib/callback.h @@ -24,6 +24,7 @@ void Host_Unban_f(const CCommand& args); void Host_ReloadBanList_f(const CCommand& args); void Host_ReloadPlaylists_f(const CCommand& args); void Host_Changelevel_f(const CCommand& args); +void Detour_Reload_f(const CCommand& args); #endif // !CLIENT_DLL void Pak_ListPaks_f(const CCommand& args); void Pak_RequestUnload_f(const CCommand& args);