From fdd74aa622aee8fbb624a2470b23234e34d89c1f Mon Sep 17 00:00:00 2001 From: Kawe Mazidjatari <48657826+Mauler125@users.noreply.github.com> Date: Sun, 18 Sep 2022 23:19:50 +0200 Subject: [PATCH] Implement net message processing budget Implement net message process budget (channel gets removed if value is exceeded). Use 'net_processLimit' to enable the implementation on the server. It will get enabled by default after testing and some cleanup. This helps against people trying to slow the server down by spamming net messages with a higher rate, e.g. using 'bind "mousewheel_up" "status"'. --- r5dev/core/init.cpp | 4 +- r5dev/engine/client/client.cpp | 21 +++++++++ r5dev/engine/client/client.h | 23 +++++++--- r5dev/engine/net.cpp | 6 +-- r5dev/engine/net.h | 2 +- r5dev/engine/net_chan.cpp | 44 ++++++++++++++++++- r5dev/networksystem/bansystem.cpp | 14 +++--- .../resource/playlist/playlists_r5_patch.txt | 4 ++ r5dev/server/vengineserver_impl.cpp | 6 +-- r5dev/server/vengineserver_impl.h | 31 ++++++++++--- r5dev/tier1/IConVar.cpp | 1 + r5dev/tier1/cvar.cpp | 1 + r5dev/tier1/cvar.h | 1 + 13 files changed, 130 insertions(+), 28 deletions(-) diff --git a/r5dev/core/init.cpp b/r5dev/core/init.cpp index 053eda63..77a65ea1 100644 --- a/r5dev/core/init.cpp +++ b/r5dev/core/init.cpp @@ -182,7 +182,7 @@ void Systems_Init() #endif // !DEDICATED && GAMEDLL_S3 NET_Attach(); - //NetChan_Attach(); + NetChan_Attach(); ConCommand_Attach(); IConVar_Attach(); @@ -309,7 +309,7 @@ void Systems_Shutdown() #endif // !DEDICATED && GAMEDLL_S3 NET_Detach(); - //NetChan_Detach(); + NetChan_Detach(); ConCommand_Detach(); IConVar_Detach(); diff --git a/r5dev/engine/client/client.cpp b/r5dev/engine/client/client.cpp index 1a6c7cb8..775e26e6 100644 --- a/r5dev/engine/client/client.cpp +++ b/r5dev/engine/client/client.cpp @@ -250,6 +250,27 @@ bool CClient::VConnect(CClient* pClient, const char* szName, void* pNetChannel, return v_CClient_Connect(pClient, szName, pNetChannel, bFakePlayer, a5, szMessage, nMessageSize); } +//--------------------------------------------------------------------------------- +// Purpose: disconnect client +// Input : nBadRep - +// *szReason - +// ... - +//--------------------------------------------------------------------------------- +void CClient::Disconnect(int nBadRep/*!!ENUM!!*/, const char* szReason, ...) +{ + char szBuf[1024]; + {///////////////////////////// + va_list vArgs{}; + va_start(vArgs, szReason); + + vsnprintf(szBuf, sizeof(szBuf), szReason, vArgs); + + szBuf[sizeof(szBuf) - 1] = '\0'; + va_end(vArgs); + }///////////////////////////// + v_CClient_Disconnect(this, nBadRep, szBuf); +} + /////////////////////////////////////////////////////////////////////////////////// void CBaseClient_Attach() { diff --git a/r5dev/engine/client/client.h b/r5dev/engine/client/client.h index 8e6cacc2..9f29a9c5 100644 --- a/r5dev/engine/client/client.h +++ b/r5dev/engine/client/client.h @@ -38,6 +38,7 @@ public: bool IsFakeClient(void) const; bool IsHumanPlayer(void) const; bool Connect(const char* szName, void* pNetChannel, bool bFakePlayer, void* a5, char* szMessage, int nMessageSize); + void Disconnect(int nBadRep /*!!ENUM!!*/, const char* szReason, ...); static bool VConnect(CClient* pClient, const char* szName, void* pNetChannel, bool bFakePlayer, void* a5, char* szMessage, int nMessageSize); void Clear(void); static void VClear(CClient* pBaseClient); @@ -87,10 +88,13 @@ static_assert(sizeof(CClient) == 0x4A4C0); /* ==== CBASECLIENT ===================================================================================================================================================== */ inline CMemory p_CClient_Connect; -inline auto v_CClient_Connect = p_CClient_Connect.RCast(); +inline auto v_CClient_Connect = p_CClient_Connect.RCast(); + +inline CMemory p_CClient_Disconnect; +inline auto v_CClient_Disconnect = p_CClient_Disconnect.RCast(); inline CMemory p_CClient_Clear; -inline auto v_CClient_Clear = p_CClient_Clear.RCast(); +inline auto v_CClient_Clear = p_CClient_Clear.RCast(); /////////////////////////////////////////////////////////////////////////////// void CBaseClient_Attach(); @@ -102,17 +106,24 @@ class VClient : public IDetour virtual void GetAdr(void) const { spdlog::debug("| FUN: CClient::Connect : {:#18x} |\n", p_CClient_Connect.GetPtr()); + spdlog::debug("| FUN: CClient::Disconnect : {:#18x} |\n", p_CClient_Disconnect.GetPtr()); spdlog::debug("| FUN: CClient::Clear : {:#18x} |\n", p_CClient_Clear.GetPtr()); spdlog::debug("| VAR: g_pClient[128] : {:#18x} |\n", reinterpret_cast(g_pClient)); spdlog::debug("+----------------------------------------------------------------+\n"); } virtual void GetFun(void) const { - p_CClient_Connect = g_GameDll.FindPatternSIMD(reinterpret_cast("\x48\x89\x5C\x24\x00\x48\x89\x6C\x24\x00\x56\x57\x41\x56\x48\x83\xEC\x20\x41\x0F\xB6\xE9"), "xxxx?xxxx?xxxxxxxxxxxx"); - p_CClient_Clear = g_GameDll.FindPatternSIMD(reinterpret_cast("\x40\x53\x41\x56\x41\x57\x48\x83\xEC\x20\x48\x8B\xD9\x48\x89\x74"), "xxxxxxxxxxxxxxxx"); + p_CClient_Connect = g_GameDll.FindPatternSIMD(reinterpret_cast("\x48\x89\x5C\x24\x00\x48\x89\x6C\x24\x00\x56\x57\x41\x56\x48\x83\xEC\x20\x41\x0F\xB6\xE9"), "xxxx?xxxx?xxxxxxxxxxxx"); +#if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) || defined (GAMEDLL_S2) + p_CClient_Disconnect = g_GameDll.FindPatternSIMD(reinterpret_cast("\x48\x8B\xC4\x4C\x89\x40\x18\x4C\x89\x48\x20\x53\x56\x57\x48\x81\xEC\x00\x00\x00\x00\x83\xB9\x00\x00\x00\x00\x00\x49\x8B\xF8\x0F\xB6\xF2"), "xxxxxxxxxxxxxxxxx????xx?????xxxxxx"); +#else // !GAMEDLL_S0 || !GAMEDLL_S1 || !GAMEDLL_S2 + p_CClient_Disconnect = g_GameDll.FindPatternSIMD(reinterpret_cast("\x48\x8B\xC4\x4C\x89\x40\x18\x4C\x89\x48\x20\x53\x56\x57\x48\x81\xEC\x00\x00\x00\x00\x83\xB9\x00\x00\x00\x00\x00\x49\x8B\xF8\x8B\xF2"), "xxxxxxxxxxxxxxxxx????xx?????xxxxx"); +#endif + p_CClient_Clear = g_GameDll.FindPatternSIMD(reinterpret_cast("\x40\x53\x41\x56\x41\x57\x48\x83\xEC\x20\x48\x8B\xD9\x48\x89\x74"), "xxxxxxxxxxxxxxxx"); - v_CClient_Connect = p_CClient_Connect.RCast(); /*48 89 5C 24 ?? 48 89 6C 24 ?? 56 57 41 56 48 83 EC 20 41 0F B6 E9*/ - v_CClient_Clear = p_CClient_Clear.RCast(); /*40 53 41 56 41 57 48 83 EC 20 48 8B D9 48 89 74*/ + v_CClient_Connect = p_CClient_Connect.RCast(); /*48 89 5C 24 ?? 48 89 6C 24 ?? 56 57 41 56 48 83 EC 20 41 0F B6 E9*/ + v_CClient_Disconnect = p_CClient_Disconnect.RCast(); /*48 8B C4 4C 89 40 18 4C 89 48 20 53 56 57 48 81 EC ?? ?? ?? ?? 83 B9 ?? ?? ?? ?? ?? 49 8B F8 8B F2*/ + v_CClient_Clear = p_CClient_Clear.RCast(); /*40 53 41 56 41 57 48 83 EC 20 48 8B D9 48 89 74*/ } virtual void GetVar(void) const { diff --git a/r5dev/engine/net.cpp b/r5dev/engine/net.cpp index 2504e1a6..d7d29dea 100644 --- a/r5dev/engine/net.cpp +++ b/r5dev/engine/net.cpp @@ -162,7 +162,7 @@ void NET_Shutdown(void* thisptr, const char* szReason, uint8_t bBadRep, bool bRe // bBadRep - // bRemoveNow - //----------------------------------------------------------------------------- -void NET_DisconnectClient(CClient* pClient, int nIndex, const char* szReason, uint8_t bBadRep, bool bRemoveNow) +void NET_RemoveChannel(CClient* pClient, int nIndex, const char* szReason, uint8_t bBadRep, bool bRemoveNow) { #ifndef CLIENT_DLL if (!pClient || std::strlen(szReason) == NULL || !pClient->GetNetChan()) @@ -170,9 +170,9 @@ void NET_DisconnectClient(CClient* pClient, int nIndex, const char* szReason, ui return; } - v_NET_Shutdown(pClient->GetNetChan(), szReason, bBadRep, bRemoveNow); // Shutdown netchan. + v_NET_Shutdown(pClient->GetNetChan(), szReason, bBadRep, bRemoveNow); // Shutdown NetChannel. pClient->Clear(); // Reset CClient instance for client. - g_bIsPersistenceVarSet[nIndex] = false; // Reset Persistence var. + g_ServerPlayer[nIndex].Reset(); // Reset ServerPlayer slot. #endif // !CLIENT_DLL } #endif // !NETCONSOLE diff --git a/r5dev/engine/net.h b/r5dev/engine/net.h index 02fc0129..ce6d1c40 100644 --- a/r5dev/engine/net.h +++ b/r5dev/engine/net.h @@ -43,7 +43,7 @@ void NET_SetKey(string svNetKey); void NET_GenerateKey(); void NET_PrintFunc(const char* fmt, ...); void NET_Shutdown(void* thisptr, const char* szReason, uint8_t bBadRep, bool bRemoveNow); -void NET_DisconnectClient(CClient* pClient, int nIndex, const char* szReason, uint8_t bBadRep, bool bRemoveNow); +void NET_RemoveChannel(CClient* pClient, int nIndex, const char* szReason, uint8_t bBadRep, bool bRemoveNow); void NET_Attach(); void NET_Detach(); diff --git a/r5dev/engine/net_chan.cpp b/r5dev/engine/net_chan.cpp index d95fd30b..d308d7c3 100644 --- a/r5dev/engine/net_chan.cpp +++ b/r5dev/engine/net_chan.cpp @@ -8,6 +8,12 @@ #include "tier1/cvar.h" #include "engine/net.h" #include "engine/net_chan.h" +#ifndef CLIENT_DLL +#include "engine/server/server.h" +#include "engine/client/client.h" +#include "server/vengineserver_impl.h" +#endif // !CLIENT_DLL + //----------------------------------------------------------------------------- // Purpose: gets the netchannel name @@ -201,9 +207,45 @@ void CNetChan::Clear(bool bStopProcessing) v_NetChan_Clear(this, bStopProcessing); } +//----------------------------------------------------------------------------- +// Purpose: process message +// Input : *pChan - +// *pMsg - +// Output : true on success, false on failure +//----------------------------------------------------------------------------- bool CNetChan::ProcessMessages(CNetChan* pChan, bf_read* pMsg) { +#ifndef CLIENT_DLL + if (!ThreadInServerFrameThread() || !net_processLimit->GetInt()) + return v_NetChan_ProcessMessages(pChan, pMsg); + + const double flStartTime = Plat_FloatTime(); + const bool bResult = v_NetChan_ProcessMessages(pChan, pMsg); + + CClient* pClient = reinterpret_cast(pChan->m_MessageHandler); + uint16_t nSlot = pClient->GetUserID(); + ServerPlayer_t* pSlot = &g_ServerPlayer[nSlot]; + + if (flStartTime - pSlot->m_flLastNetProcessTime >= 1.0 || + pSlot->m_flLastNetProcessTime == -1.0) + { + pSlot->m_flLastNetProcessTime = flStartTime; + pSlot->m_flCurrentNetProcessTime = 0.0; + } + pSlot->m_flCurrentNetProcessTime += + (Plat_FloatTime() * 1000) - (flStartTime * 1000); + + if (pSlot->m_flCurrentNetProcessTime >= + net_processLimit->GetDouble()) + { + pClient->Disconnect(2, "#DISCONNECT_NETCHAN_OVERFLOW"); + return false; + } + + return bResult; +#else // !CLIENT_DLL return v_NetChan_ProcessMessages(pChan, pMsg); +#endif } /////////////////////////////////////////////////////////////////////////////// @@ -214,4 +256,4 @@ void NetChan_Attach() void NetChan_Detach() { DetourDetach((LPVOID*)&v_NetChan_ProcessMessages, &CNetChan::ProcessMessages); -} \ No newline at end of file +} diff --git a/r5dev/networksystem/bansystem.cpp b/r5dev/networksystem/bansystem.cpp index 53f7b656..7b8ce776 100644 --- a/r5dev/networksystem/bansystem.cpp +++ b/r5dev/networksystem/bansystem.cpp @@ -227,7 +227,7 @@ void CBanSystem::BanListCheck(void) if (AddEntry(svIpAddress, pClient->GetNucleusID()) && !bSave) bSave = true; - NET_DisconnectClient(pClient, c, m_vRefuseList[i].first.c_str(), 0, true); + NET_RemoveChannel(pClient, c, m_vRefuseList[i].first.c_str(), 0, true); } } @@ -299,7 +299,7 @@ void CBanSystem::KickPlayerByName(const string& svPlayerName) if (strlen(pNetChan->GetName()) > 0) { if (svPlayerName.compare(pNetChan->GetName()) == NULL) // Our wanted name? - NET_DisconnectClient(pClient, i, "Kicked from server", 0, true); + NET_RemoveChannel(pClient, i, "Kicked from server", 0, true); } } } @@ -344,14 +344,14 @@ void CBanSystem::KickPlayerById(const string& svHandle) continue; } - NET_DisconnectClient(pClient, i, "Kicked from server", 0, true); + NET_RemoveChannel(pClient, i, "Kicked from server", 0, true); } else { if (svHandle.compare(pNetChan->GetAddress()) != NULL) continue; - NET_DisconnectClient(pClient, i, "Kicked from server", 0, true); + NET_RemoveChannel(pClient, i, "Kicked from server", 0, true); } } } @@ -386,7 +386,7 @@ void CBanSystem::BanPlayerByName(const string& svPlayerName) if (AddEntry(pNetChan->GetAddress(), pClient->GetNucleusID()) && !bSave) bSave = true; - NET_DisconnectClient(pClient, i, "Banned from server", 0, true); + NET_RemoveChannel(pClient, i, "Banned from server", 0, true); } } } @@ -441,7 +441,7 @@ void CBanSystem::BanPlayerById(const string& svHandle) bSave = true; Save(); - NET_DisconnectClient(pClient, i, "Banned from server", 0, true); + NET_RemoveChannel(pClient, i, "Banned from server", 0, true); } else { @@ -452,7 +452,7 @@ void CBanSystem::BanPlayerById(const string& svHandle) bSave = true; Save(); - NET_DisconnectClient(pClient, i, "Banned from server", 0, true); + NET_RemoveChannel(pClient, i, "Banned from server", 0, true); } } diff --git a/r5dev/resource/playlist/playlists_r5_patch.txt b/r5dev/resource/playlist/playlists_r5_patch.txt index ffcb4d58..c21d2f3c 100644 --- a/r5dev/resource/playlist/playlists_r5_patch.txt +++ b/r5dev/resource/playlist/playlists_r5_patch.txt @@ -1067,6 +1067,10 @@ playlists "DISCONNECT_BANNED" "The client's game account has been banned: Banned" "VALVE_REJECT_BANNED" "You have been banned from this server." + + "DISCONNECT_SEND_RELIABLEOVERFLOW" "Connection to server overflowed (code:river)." + "DISCONNECT_SEND_OVERFLOW" "Connection to server overflowed (code:dam)." + "DISCONNECT_NETCHAN_OVERFLOW" "Connection to server overflowed (code:vulkan)." } } } diff --git a/r5dev/server/vengineserver_impl.cpp b/r5dev/server/vengineserver_impl.cpp index 26e62f39..dc64b5e3 100644 --- a/r5dev/server/vengineserver_impl.cpp +++ b/r5dev/server/vengineserver_impl.cpp @@ -18,7 +18,7 @@ bool HIVEngineServer__PersistenceAvailable(void* entidx, int clienthandle) CClient* pClient = g_pClient->GetClient(clienthandle); // Get client instance. pClient->SetPersistenceState(PERSISTENCE::PERSISTENCE_READY); // Set the client instance to 'ready'. - if (!g_bIsPersistenceVarSet[clienthandle] && sv_showconnecting->GetBool()) + if (!g_ServerPlayer[clienthandle].m_bPersistenceEnabled && sv_showconnecting->GetBool()) { CNetChan* pNetChan = pClient->GetNetChan(); @@ -34,7 +34,7 @@ bool HIVEngineServer__PersistenceAvailable(void* entidx, int clienthandle) DevMsg(eDLL_T::SERVER, " |- ADR : | '%s'\n", svIpAddress.c_str()); DevMsg(eDLL_T::SERVER, " -------------------------------------------------------------\n"); - g_bIsPersistenceVarSet[clienthandle] = true; + g_ServerPlayer[clienthandle].m_bPersistenceEnabled = true; } /////////////////////////////////////////////////////////////////////////// return IVEngineServer__PersistenceAvailable(entidx, clienthandle); @@ -51,4 +51,4 @@ void IVEngineServer_Detach() } /////////////////////////////////////////////////////////////////////////////// -bool g_bIsPersistenceVarSet[MAX_PLAYERS]; +ServerPlayer_t g_ServerPlayer[MAX_PLAYERS]; diff --git a/r5dev/server/vengineserver_impl.h b/r5dev/server/vengineserver_impl.h index e11fe8c0..ab6010c1 100644 --- a/r5dev/server/vengineserver_impl.h +++ b/r5dev/server/vengineserver_impl.h @@ -13,6 +13,9 @@ inline auto IVEngineServer__GetNumHumanPlayers = p_IVEngineServer__GetNumHumanPl inline CMemory p_IVEngineServer__GetNumFakeClients; inline auto IVEngineServer__GetNumFakeClients = p_IVEngineServer__GetNumFakeClients.RCast(); +//inline CMemory p_RunFrameServer; +//inline auto v_RunFrameServer = p_RunFrameServer.RCast(); + inline bool* g_bDedicated = nullptr; /////////////////////////////////////////////////////////////////////////////// @@ -22,7 +25,22 @@ void IVEngineServer_Attach(); void IVEngineServer_Detach(); /////////////////////////////////////////////////////////////////////////////// -extern bool g_bIsPersistenceVarSet[MAX_PLAYERS]; + +struct ServerPlayer_t +{ + inline void Reset(void) + { + m_flCurrentNetProcessTime = 0.0; + m_flLastNetProcessTime = 0.0; + m_bPersistenceEnabled = false; + } + + double m_flCurrentNetProcessTime; + double m_flLastNetProcessTime; + bool m_bPersistenceEnabled; +}; + +extern ServerPlayer_t g_ServerPlayer[MAX_PLAYERS]; /////////////////////////////////////////////////////////////////////////////// class HVEngineServer : public IDetour @@ -33,6 +51,7 @@ class HVEngineServer : public IDetour spdlog::debug("| FUN: IVEngineServer::IsDedicatedServer : {:#18x} |\n", p_IVEngineServer__IsDedicatedServer.GetPtr()); spdlog::debug("| FUN: IVEngineServer::GetNumHumanPlayers : {:#18x} |\n", p_IVEngineServer__GetNumHumanPlayers.GetPtr()); spdlog::debug("| FUN: IVEngineServer::GetNumFakeClients : {:#18x} |\n", p_IVEngineServer__GetNumFakeClients.GetPtr()); + //spdlog::debug("| FUN: RunFrameServer : {:#18x} |\n", p_RunFrameServer.GetPtr()); spdlog::debug("| VAR: g_bDedicated : {:#18x} |\n", reinterpret_cast(g_bDedicated)); spdlog::debug("+----------------------------------------------------------------+\n"); } @@ -42,11 +61,13 @@ class HVEngineServer : public IDetour p_IVEngineServer__IsDedicatedServer = g_GameDll.FindPatternSIMD(reinterpret_cast("\x0F\xB6\x05\x00\x00\x00\x00\xC3\xCC\xCC\xCC\xCC\xCC\xCC\xCC\xCC\x48\x8B\x05\x00\x00\x00\x00\xC3\xCC\xCC\xCC\xCC\xCC\xCC\xCC\xCC\x40\x53"), "xxx????xxxxxxxxxxxx????xxxxxxxxxxx"); p_IVEngineServer__GetNumHumanPlayers = g_GameDll.FindPatternSIMD(reinterpret_cast("\x8B\x15\x00\x00\x00\x00\x33\xC0\x85\xD2\x7E\x24"), "xx????xxxxxx"); p_IVEngineServer__GetNumFakeClients = g_GameDll.FindPatternSIMD(reinterpret_cast("\x8B\x05\x00\x00\x00\x00\x33\xC9\x85\xC0\x7E\x2D"), "xx????xxxxxx"); +// p_RunFrameServer = g_GameDll.FindPatternSIMD(reinterpret_cast("\x48\x89\x5C\x24\x00\x57\x48\x83\xEC\x30\x0F\x29\x74\x24\x00\x48\x8D\x0D\x00\x00\x00\x00"), "xxxx?xxxxxxxxx?xxx????"); - IVEngineServer__PersistenceAvailable = p_IVEngineServer__PersistenceAvailable.RCast(); /*3B 15 ?? ?? ?? ?? 7D 33*/ - IVEngineServer__IsDedicatedServer = p_IVEngineServer__IsDedicatedServer.RCast(); /*0F B6 05 ?? ?? ?? ?? C3 CC CC CC CC CC CC CC CC 48 8B 05 ?? ?? ?? ?? C3 CC CC CC CC CC CC CC CC 40 53*/ - IVEngineServer__GetNumHumanPlayers = p_IVEngineServer__GetNumHumanPlayers.RCast(); /*8B 15 ?? ?? ?? ?? 33 C0 85 D2 7E 24*/ - IVEngineServer__GetNumFakeClients = p_IVEngineServer__GetNumFakeClients.RCast(); /*8B 05 ?? ?? ?? ?? 33 C9 85 C0 7E 2D*/ + IVEngineServer__PersistenceAvailable = p_IVEngineServer__PersistenceAvailable.RCast(); /*3B 15 ?? ?? ?? ?? 7D 33*/ + IVEngineServer__IsDedicatedServer = p_IVEngineServer__IsDedicatedServer.RCast(); /*0F B6 05 ?? ?? ?? ?? C3 CC CC CC CC CC CC CC CC 48 8B 05 ?? ?? ?? ?? C3 CC CC CC CC CC CC CC CC 40 53*/ + IVEngineServer__GetNumHumanPlayers = p_IVEngineServer__GetNumHumanPlayers.RCast(); /*8B 15 ?? ?? ?? ?? 33 C0 85 D2 7E 24*/ + IVEngineServer__GetNumFakeClients = p_IVEngineServer__GetNumFakeClients.RCast(); /*8B 05 ?? ?? ?? ?? 33 C9 85 C0 7E 2D*/ +// v_RunFrameServer = p_RunFrameServer.RCast(); /*48 89 5C 24 ?? 57 48 83 EC 30 0F 29 74 24 ?? 48 8D 0D ?? ?? ?? ??*/ } virtual void GetVar(void) const { diff --git a/r5dev/tier1/IConVar.cpp b/r5dev/tier1/IConVar.cpp index 85c59b56..4d339371 100644 --- a/r5dev/tier1/IConVar.cpp +++ b/r5dev/tier1/IConVar.cpp @@ -203,6 +203,7 @@ void ConVar::Init(void) const net_tracePayload = ConVar::Create("net_tracePayload" , "0", FCVAR_DEVELOPMENTONLY , "Log the payload of the send/recv datagram to a file on the disk.", false, 0.f, false, 0.f, nullptr, nullptr); net_encryptionEnable = ConVar::Create("net_encryptionEnable" , "1", FCVAR_DEVELOPMENTONLY | FCVAR_REPLICATED , "Use AES encryption on game packets.", false, 0.f, false, 0.f, nullptr, nullptr); net_useRandomKey = ConVar::Create("net_useRandomKey" , "1" , FCVAR_RELEASE , "Use random AES encryption key for game packets.", false, 0.f, false, 0.f, &NET_UseRandomKeyChanged_f, nullptr); + net_processLimit = ConVar::Create("net_processLimit" , "0" , FCVAR_RELEASE , "Net message process budget in milliseconds (removing netchannel if exceeded).", true, 0.f, false, 0.f, nullptr, "0 = disabled."); //------------------------------------------------------------------------- // NETWORKSYSTEM | pylon_matchmaking_hostname = ConVar::Create("pylon_matchmaking_hostname", "ms.r5reloaded.com", FCVAR_RELEASE , "Holds the pylon matchmaking hostname.", false, 0.f, false, 0.f, &MP_HostName_Changed_f, nullptr); diff --git a/r5dev/tier1/cvar.cpp b/r5dev/tier1/cvar.cpp index ccbc61f3..13963f3d 100644 --- a/r5dev/tier1/cvar.cpp +++ b/r5dev/tier1/cvar.cpp @@ -172,6 +172,7 @@ ConVar* net_tracePayload = nullptr; ConVar* net_encryptionEnable = nullptr; ConVar* net_useRandomKey = nullptr; ConVar* net_usesocketsforloopback = nullptr; +ConVar* net_processLimit = nullptr; ConVar* pylon_matchmaking_hostname = nullptr; ConVar* pylon_host_update_interval = nullptr; ConVar* pylon_showdebuginfo = nullptr; diff --git a/r5dev/tier1/cvar.h b/r5dev/tier1/cvar.h index a4798f07..6ba32f97 100644 --- a/r5dev/tier1/cvar.h +++ b/r5dev/tier1/cvar.h @@ -167,6 +167,7 @@ extern ConVar* net_tracePayload; extern ConVar* net_encryptionEnable; extern ConVar* net_useRandomKey; extern ConVar* net_usesocketsforloopback; +extern ConVar* net_processLimit; extern ConVar* pylon_matchmaking_hostname; extern ConVar* pylon_host_update_interval; extern ConVar* pylon_showdebuginfo;