From 21f4d0fa07e1b7fc1282b840336f4efc70413bb2 Mon Sep 17 00:00:00 2001 From: Kawe Mazidjatari <48657826+Mauler125@users.noreply.github.com> Date: Mon, 16 Jan 2023 21:09:21 +0100 Subject: [PATCH] Initial working bots implementation * Changed 'CClient' handle time to 'edict_t' which is an alias of 'uint16_t'. * Changed 'g_pServerGameDLL' and 'g_pServerGameClients' init (obtain from factory instead). * Use interface version macro's for obtaining factory pointers instead. * Added 'g_pServerGameEntities'. --- r5dev/engine/client/client.cpp | 4 +-- r5dev/engine/client/client.h | 7 +++-- r5dev/game/server/gameinterface.cpp | 3 +- r5dev/game/server/gameinterface.h | 30 ++++++++++++++------ r5dev/launcher/IApplication.cpp | 15 ++++++++-- r5dev/public/edict.h | 1 + r5dev/public/eiface.h | 40 +++++++++++++++++++++++++++ r5dev/server/vengineserver_impl.h | 10 ++----- r5dev/vpc/interfaces.h | 3 +- r5dev/vproj/clientsdk.vcxproj | 1 + r5dev/vproj/clientsdk.vcxproj.filters | 3 ++ r5dev/vproj/dedicated.vcxproj | 1 + r5dev/vproj/dedicated.vcxproj.filters | 3 ++ r5dev/vproj/gamesdk.vcxproj | 1 + r5dev/vproj/gamesdk.vcxproj.filters | 3 ++ r5dev/vstdlib/callback.cpp | 4 ++- 16 files changed, 104 insertions(+), 25 deletions(-) create mode 100644 r5dev/public/eiface.h diff --git a/r5dev/engine/client/client.cpp b/r5dev/engine/client/client.cpp index a24c7192..be2e0b50 100644 --- a/r5dev/engine/client/client.cpp +++ b/r5dev/engine/client/client.cpp @@ -25,7 +25,7 @@ CClient* CClient::GetClient(int nIndex) const //--------------------------------------------------------------------------------- // Purpose: gets the handle of this client //--------------------------------------------------------------------------------- -uint16_t CClient::GetHandle(void) const +edict_t CClient::GetHandle(void) const { return m_nHandle; } @@ -89,7 +89,7 @@ const char* CClient::GetClientName(void) const //--------------------------------------------------------------------------------- // Purpose: sets the handle of this client //--------------------------------------------------------------------------------- -void CClient::SetHandle(uint16_t nHandle) +void CClient::SetHandle(edict_t nHandle) { m_nHandle = nHandle; } diff --git a/r5dev/engine/client/client.h b/r5dev/engine/client/client.h index 66cff761..ff469d3d 100644 --- a/r5dev/engine/client/client.h +++ b/r5dev/engine/client/client.h @@ -2,6 +2,7 @@ #include "vpc/keyvalues.h" #include "common/protocol.h" #include "engine/net_chan.h" +#include "public/edict.h" //----------------------------------------------------------------------------- // Enumerations @@ -26,7 +27,7 @@ class CClient : IClientMessageHandler, INetChannelHandler { public: CClient* GetClient(int nIndex) const; - uint16_t GetHandle(void) const; + edict_t GetHandle(void) const; uint32_t GetUserID(void) const; uint64_t GetNucleusID(void) const; SIGNONSTATE GetSignonState(void) const; @@ -34,7 +35,7 @@ public: CNetChan* GetNetChan(void) const; const char* GetServerName(void) const; const char* GetClientName(void) const; - void SetHandle(uint16_t nHandle); + void SetHandle(edict_t nHandle); void SetUserID(uint32_t nUserID); void SetNucleusID(uint64_t nNucleusID); void SetSignonState(SIGNONSTATE nSignonState); @@ -56,7 +57,7 @@ public: private: uint32_t m_nUserID; //0x0010 - uint16_t m_nHandle; //0x0014 + edict_t m_nHandle; //0x0014 char m_szServerName[64]; //0x0016 int64_t m_nReputation; //0x0058 char pad_0014[182]; //0x0060 diff --git a/r5dev/game/server/gameinterface.cpp b/r5dev/game/server/gameinterface.cpp index c2078d56..acb67e01 100644 --- a/r5dev/game/server/gameinterface.cpp +++ b/r5dev/game/server/gameinterface.cpp @@ -9,6 +9,7 @@ #include "engine/server/sv_main.h" #include "game/server/gameinterface.h" #include "public/server_class.h" +#include "public/eiface.h" //----------------------------------------------------------------------------- // This is called when a new game is started. (restart, map) @@ -91,6 +92,6 @@ void CServerGameDLL_Detach() #endif } -// Pointer to CServerGameDLL virtual function table. CServerGameDLL* g_pServerGameDLL = nullptr; CServerGameClients* g_pServerGameClients = nullptr; +CServerGameEnts* g_pServerGameEntities = nullptr; diff --git a/r5dev/game/server/gameinterface.h b/r5dev/game/server/gameinterface.h index 54adbf71..f7994987 100644 --- a/r5dev/game/server/gameinterface.h +++ b/r5dev/game/server/gameinterface.h @@ -4,9 +4,16 @@ // // $NoKeywords: $ //=============================================================================// +#include "public/eiface.h" +//----------------------------------------------------------------------------- +// Forward declarations +//----------------------------------------------------------------------------- class ServerClass; +//----------------------------------------------------------------------------- +// +//----------------------------------------------------------------------------- class CServerGameDLL { public: @@ -19,7 +26,18 @@ public: static void __fastcall OnReceivedSayTextMessage(void* thisptr, int senderId, const char* text, bool isTeamChat); }; -class CServerGameClients // TODO: Reverse.. + +//----------------------------------------------------------------------------- +// +//----------------------------------------------------------------------------- +class CServerGameClients : public IServerGameClients +{ +}; + +//----------------------------------------------------------------------------- +// +//----------------------------------------------------------------------------- +class CServerGameEnts : public IServerGameEnts { }; @@ -28,6 +46,7 @@ inline auto CServerGameDLL__OnReceivedSayTextMessage = p_CServerGameDLL__OnRecei extern CServerGameDLL* g_pServerGameDLL; extern CServerGameClients* g_pServerGameClients; +extern CServerGameEnts* g_pServerGameEntities; void CServerGameDLL_Attach(); void CServerGameDLL_Detach(); @@ -40,22 +59,17 @@ class VServerGameDLL : public IDetour spdlog::debug("| FUN: OnReceivedSayTextMessage : {:#18x} |\n", p_CServerGameDLL__OnReceivedSayTextMessage.GetPtr()); spdlog::debug("| VAR: g_pServerGameDLL : {:#18x} |\n", reinterpret_cast(g_pServerGameDLL)); spdlog::debug("| VAR: g_pServerGameClients : {:#18x} |\n", reinterpret_cast(g_pServerGameClients)); + spdlog::debug("| VAR: g_pServerGameEntities : {:#18x} |\n", reinterpret_cast(g_pServerGameEntities)); spdlog::debug("+----------------------------------------------------------------+\n"); } virtual void GetFun(void) const { #if defined(GAMEDLL_S3) p_CServerGameDLL__OnReceivedSayTextMessage = g_GameDll.FindPatternSIMD("85 D2 0F 8E ?? ?? ?? ?? 4C 8B DC"); - CServerGameDLL__OnReceivedSayTextMessage = p_CServerGameDLL__OnReceivedSayTextMessage.RCast(); #endif } - virtual void GetVar(void) const - { - g_pServerGameDLL = p_SV_CreateBaseline.Offset(0x0).FindPatternSelf("48 8B", CMemory::Direction::DOWN).ResolveRelativeAddressSelf(0x3, 0x7).Deref().RCast(); - g_pServerGameClients = g_GameDll.FindPatternSIMD("48 89 5C 24 ?? 57 48 81 EC ?? ?? ?? ?? 0F B7 51 14"). - FindPatternSelf("48 8B 0D", CMemory::Direction::DOWN).ResolveRelativeAddressSelf(0x3, 0x7).RCast(); - } + virtual void GetVar(void) const { } virtual void GetCon(void) const { } virtual void Attach(void) const { } virtual void Detach(void) const { } diff --git a/r5dev/launcher/IApplication.cpp b/r5dev/launcher/IApplication.cpp index c4cf6452..92804d9e 100644 --- a/r5dev/launcher/IApplication.cpp +++ b/r5dev/launcher/IApplication.cpp @@ -21,6 +21,9 @@ #include "server/vengineserver_impl.h" #include "client/cdll_engine_int.h" #include "engine/enginetrace.h" +#ifndef CLIENT_DLL +#include "game/server/gameinterface.h" +#endif // !CLIENT_DLL #ifndef DEDICATED #include "gameui/IConsole.h" #endif // !DEDICATED @@ -68,9 +71,17 @@ bool CModAppSystemGroup::Create(CModAppSystemGroup* pModAppSystemGroup) //InitPluginSystem(pModAppSystemGroup); //CALL_PLUGIN_CALLBACKS(g_pPluginSystem->GetCreateCallbacks(), pModAppSystemGroup); +#ifndef CLIENT_DLL + g_pServerGameDLL = g_pFactory->GetFactoryPtr(INTERFACEVERSION_SERVERGAMEDLL, false).RCast(); + g_pServerGameClients = g_pFactory->GetFactoryPtr(INTERFACEVERSION_SERVERGAMECLIENTS_NEW, false).RCast(); + if (!g_pServerGameClients) + g_pServerGameClients = g_pFactory->GetFactoryPtr(INTERFACEVERSION_SERVERGAMECLIENTS, false).RCast(); + g_pServerGameEntities = g_pFactory->GetFactoryPtr(INTERFACEVERSION_SERVERGAMEENTS, false).RCast(); +#endif // !CLIENT_DLL + #ifndef DEDICATED - g_pClientEntityList = g_pFactory->GetFactoryPtr("VClientEntityList003", false).RCast(); - g_pEngineTrace = g_pFactory->GetFactoryPtr("EngineTraceClient004", false).RCast(); + g_pClientEntityList = g_pFactory->GetFactoryPtr(VCLIENTENTITYLIST_INTERFACE_VERSION, false).RCast(); + g_pEngineTrace = g_pFactory->GetFactoryPtr(INTERFACEVERSION_ENGINETRACE_CLIENT, false).RCast(); g_pImGuiConfig->Load(); // Load ImGui configs. for (auto& map : g_pCVar->DumpToMap()) diff --git a/r5dev/public/edict.h b/r5dev/public/edict.h index 06d10d5b..118acf91 100644 --- a/r5dev/public/edict.h +++ b/r5dev/public/edict.h @@ -7,6 +7,7 @@ #include "engine/server/sv_main.h" #endif // !CLIENT_DLL +typedef uint16_t edict_t; //----------------------------------------------------------------------------- // Purpose: Defines the ways that a map can be loaded. diff --git a/r5dev/public/eiface.h b/r5dev/public/eiface.h new file mode 100644 index 00000000..92ccd05e --- /dev/null +++ b/r5dev/public/eiface.h @@ -0,0 +1,40 @@ +//===== Copyright � 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: +// +// $NoKeywords: $ +// +//===========================================================================// +#ifndef EIFACE_H +#define EIFACE_H +#include "edict.h" + +//----------------------------------------------------------------------------- +// Purpose: Interface to get at server clients +//----------------------------------------------------------------------------- +abstract_class IServerGameClients +{ +public: + // Get server max players and lower bound for same + virtual void GetPlayerLimits(int& nMinPlayers, int& nMaxPlayers, int& nDefaultMaxPlayers) const = 0; + + // Client is connecting to server ( return false to reject the connection ) + // You can specify a rejection message by writing it into pszReject + virtual bool ClientConnect(edict_t nEntity, char const* pszName, char const* pszAddress, char* pszReject, int nMaxRejectLen) = 0; + + // Client is going active + // If bLoadGame is true, don't spawn the player because its state is already setup. + virtual void ClientActive(edict_t nEntity, bool bLoadGame) = 0; + virtual void ClientFullyConnect(edict_t nEntity, bool bRestore) = 0; +}; + +//----------------------------------------------------------------------------- +// Purpose: Interface to get at server entities +//----------------------------------------------------------------------------- +abstract_class IServerGameEnts +{ +public: + // !TODO +}; + +#endif // EIFACE_H diff --git a/r5dev/server/vengineserver_impl.h b/r5dev/server/vengineserver_impl.h index 2aeae575..548af34e 100644 --- a/r5dev/server/vengineserver_impl.h +++ b/r5dev/server/vengineserver_impl.h @@ -1,9 +1,5 @@ #pragma once - -//----------------------------------------------------------------------------- -// Forward declarations -//----------------------------------------------------------------------------- -class CClient; +#include "public/edict.h" /* ==== CVENGINESERVER ================================================================================================================================================== */ inline CMemory p_IVEngineServer__PersistenceAvailable; @@ -19,7 +15,7 @@ inline CMemory p_IVEngineServer__GetNumFakeClients; inline auto IVEngineServer__GetNumFakeClients = p_IVEngineServer__GetNumFakeClients.RCast(); inline CMemory p_IVEngineServer__CreateFakeClient; -inline auto IVEngineServer__CreateFakeClient = p_IVEngineServer__CreateFakeClient.RCast(); +inline auto IVEngineServer__CreateFakeClient = p_IVEngineServer__CreateFakeClient.RCast(); //inline CMemory p_RunFrameServer; //inline auto v_RunFrameServer = p_RunFrameServer.RCast(); @@ -88,7 +84,7 @@ class HVEngineServer : public IDetour 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__CreateFakeClient = p_IVEngineServer__CreateFakeClient.RCast();/*48 89 5C 24 ?? 48 89 74 24 ?? 57 48 83 EC 20 48 8B F2 41 8B F8*/ + IVEngineServer__CreateFakeClient = p_IVEngineServer__CreateFakeClient.RCast();/*48 89 5C 24 ?? 48 89 74 24 ?? 57 48 83 EC 20 48 8B F2 41 8B F8*/ // 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/vpc/interfaces.h b/r5dev/vpc/interfaces.h index 32bd57d0..9b9e3aff 100644 --- a/r5dev/vpc/interfaces.h +++ b/r5dev/vpc/interfaces.h @@ -14,7 +14,8 @@ #define MATERIALSYSTEM_CONFIG_VERSION "VMaterialSystemConfig004" #define SERVER_DLL_SHARED_APPSYSTEMS "VServerDllSharedAppSystems001" -#define INTERFACEVERSION_SERVERGAMECLIENTS "ServerGameClients004" +#define INTERFACEVERSION_SERVERGAMECLIENTS_NEW "ServerGameClients004" +#define INTERFACEVERSION_SERVERGAMECLIENTS "ServerGameClients003" #define INTERFACEVERSION_SERVERGAMEENTS "ServerGameEnts002" #define INTERFACEVERSION_SERVERGAMEDLL "ServerGameDLL005" diff --git a/r5dev/vproj/clientsdk.vcxproj b/r5dev/vproj/clientsdk.vcxproj index f930b00c..4ac86294 100644 --- a/r5dev/vproj/clientsdk.vcxproj +++ b/r5dev/vproj/clientsdk.vcxproj @@ -274,6 +274,7 @@ + diff --git a/r5dev/vproj/clientsdk.vcxproj.filters b/r5dev/vproj/clientsdk.vcxproj.filters index 0e46eed9..d6cfba05 100644 --- a/r5dev/vproj/clientsdk.vcxproj.filters +++ b/r5dev/vproj/clientsdk.vcxproj.filters @@ -1886,6 +1886,9 @@ sdk\public\utility + + sdk\public + diff --git a/r5dev/vproj/dedicated.vcxproj b/r5dev/vproj/dedicated.vcxproj index 6f73deaf..4c1fd527 100644 --- a/r5dev/vproj/dedicated.vcxproj +++ b/r5dev/vproj/dedicated.vcxproj @@ -221,6 +221,7 @@ + diff --git a/r5dev/vproj/dedicated.vcxproj.filters b/r5dev/vproj/dedicated.vcxproj.filters index 2f492f76..b46b8504 100644 --- a/r5dev/vproj/dedicated.vcxproj.filters +++ b/r5dev/vproj/dedicated.vcxproj.filters @@ -1290,6 +1290,9 @@ sdk\public\utility + + sdk\public + diff --git a/r5dev/vproj/gamesdk.vcxproj b/r5dev/vproj/gamesdk.vcxproj index b498b2cb..40725f59 100644 --- a/r5dev/vproj/gamesdk.vcxproj +++ b/r5dev/vproj/gamesdk.vcxproj @@ -302,6 +302,7 @@ + diff --git a/r5dev/vproj/gamesdk.vcxproj.filters b/r5dev/vproj/gamesdk.vcxproj.filters index b6c6a329..8c1a20f1 100644 --- a/r5dev/vproj/gamesdk.vcxproj.filters +++ b/r5dev/vproj/gamesdk.vcxproj.filters @@ -1988,6 +1988,9 @@ sdk\public\utility + + sdk\public + diff --git a/r5dev/vstdlib/callback.cpp b/r5dev/vstdlib/callback.cpp index 4af010eb..7a1f4094 100644 --- a/r5dev/vstdlib/callback.cpp +++ b/r5dev/vstdlib/callback.cpp @@ -55,6 +55,7 @@ #ifndef DEDICATED #include "game/client/viewrender.h" #endif // !DEDICATED +#include /* @@ -1247,6 +1248,7 @@ void CC_CreateFakePlayer_f(const CCommand& args) DevMsg(eDLL_T::SERVER, "usage: sv_addbot name teamid\n"); return; } - CClient* pPlayer = IVEngineServer__CreateFakeClient(nullptr, args.Arg(1), std::stoi(args.Arg(2))); + edict_t nHandle = IVEngineServer__CreateFakeClient(nullptr, args.Arg(1), std::stoi(args.Arg(2))); + g_pServerGameClients->ClientFullyConnect(nHandle, false); #endif // !CLIENT_DLL } \ No newline at end of file