From fad1a092e48ade73dfacc1496be42490e3ecf41d Mon Sep 17 00:00:00 2001 From: Kawe Mazidjatari <48657826+Mauler125@users.noreply.github.com> Date: Sat, 13 Aug 2022 19:41:45 +0200 Subject: [PATCH] More CCvar reversing Still work-in-progress --- r5dev/mathlib/swap.h | 6 + r5dev/public/icvar.h | 19 + r5dev/tier0/basetypes.h | 15 +- r5dev/tier1/cmd.h | 2 +- r5dev/tier1/cvar.cpp | 261 +++++- r5dev/tier1/cvar.h | 60 +- r5dev/tier1/generichash.cpp | 430 ++++++++++ r5dev/tier1/utllinkedlist.h | 1095 +++++++++++++++++++++++++ r5dev/vproj/clientsdk.vcxproj | 3 + r5dev/vproj/clientsdk.vcxproj.filters | 9 + r5dev/vproj/dedicated.vcxproj | 3 + r5dev/vproj/dedicated.vcxproj.filters | 9 + r5dev/vproj/gamesdk.vcxproj | 3 + r5dev/vproj/gamesdk.vcxproj.filters | 9 + r5dev/vstdlib/concommandhash.h | 214 +++++ 15 files changed, 2091 insertions(+), 47 deletions(-) create mode 100644 r5dev/tier1/generichash.cpp create mode 100644 r5dev/tier1/utllinkedlist.h create mode 100644 r5dev/vstdlib/concommandhash.h diff --git a/r5dev/mathlib/swap.h b/r5dev/mathlib/swap.h index cf4c52d9..df5d8c09 100644 --- a/r5dev/mathlib/swap.h +++ b/r5dev/mathlib/swap.h @@ -5,6 +5,12 @@ //===========================================================================// #pragma once +#define LittleShort( val ) ( val ) +#define LittleWord( val ) ( val ) +#define LittleLong( val ) ( val ) +#define LittleDWord( val ) ( val ) +#define LittleQWord( val ) ( val ) + template inline T WordSwapC(T w) { diff --git a/r5dev/public/icvar.h b/r5dev/public/icvar.h index f58cfab3..34827c84 100644 --- a/r5dev/public/icvar.h +++ b/r5dev/public/icvar.h @@ -1,9 +1,28 @@ #ifndef ICVAR_H #define ICVAR_H +//----------------------------------------------------------------------------- +// Forward declarations +//----------------------------------------------------------------------------- +class ConCommandBase; + //----------------------------------------------------------------------------- // ConVars/ComCommands are marked as having a particular DLL identifier //----------------------------------------------------------------------------- typedef int CVarDLLIdentifier_t; +//----------------------------------------------------------------------------- +// ConVars/ComCommands are marked as having a particular DLL identifier +//----------------------------------------------------------------------------- +typedef int CVarDLLIdentifier_t; + +abstract_class ICVarIteratorInternal +{ +public: + virtual void SetFirst(void) = 0; + virtual void Next(void) = 0; + virtual bool IsValid(void) = 0; + virtual ConCommandBase* Get(void) = 0; +}; + #endif // ICVAR_H diff --git a/r5dev/tier0/basetypes.h b/r5dev/tier0/basetypes.h index c0d51a09..cff90e42 100644 --- a/r5dev/tier0/basetypes.h +++ b/r5dev/tier0/basetypes.h @@ -342,4 +342,17 @@ protected: #define FORWARD_DECLARE_HANDLE(name) typedef struct name##__ *name #define DECLARE_DERIVED_POINTER_HANDLE( _name, _basehandle ) struct _name##__ : public _basehandle##__ {}; typedef struct _name##__ *_name -#define DECLARE_ALIASED_POINTER_HANDLE( _name, _alias ) typedef struct _alias##__ *name \ No newline at end of file +#define DECLARE_ALIASED_POINTER_HANDLE( _name, _alias ) typedef struct _alias##__ *name + +#define ExecuteNTimes( nTimes, x ) \ + { \ + static int __executeCount=0;\ + if ( __executeCount < nTimes )\ + { \ + ++__executeCount; \ + x; \ + } \ + } + + +#define ExecuteOnce( x ) ExecuteNTimes( 1, x ) \ No newline at end of file diff --git a/r5dev/tier1/cmd.h b/r5dev/tier1/cmd.h index b88ba1a8..5d071973 100644 --- a/r5dev/tier1/cmd.h +++ b/r5dev/tier1/cmd.h @@ -124,7 +124,7 @@ public: //----------------------------------------------------------------------------- class ConCommand : public ConCommandBase { - friend class CCVar; + friend class CCvar; public: static ConCommand* Create(const char* szName, const char* szHelpString, int nFlags, FnCommandCallback_t pCallback, FnCommandCompletionCallback pCommandCompletionCallback); diff --git a/r5dev/tier1/cvar.cpp b/r5dev/tier1/cvar.cpp index 3d094706..731b95dc 100644 --- a/r5dev/tier1/cvar.cpp +++ b/r5dev/tier1/cvar.cpp @@ -4,6 +4,7 @@ #include "tier1/IConVar.h" #include "engine/sys_dll2.h" #include "filesystem/filesystem.h" +#include "vstdlib/concommandhash.h" //----------------------------------------------------------------------------- // ENGINE | @@ -343,7 +344,7 @@ int CCvarUtilities::CountVariablesWithFlags(int flags) ConCommandBase* var; // Loop through cvars... - CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); + CCvar::CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); pFactory->SetFirst(); while (pFactory->IsValid()) @@ -369,7 +370,7 @@ int CCvarUtilities::CountVariablesWithFlags(int flags) void CCvarUtilities::EnableDevCvars() { // Loop through cvars... - CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); + CCvar::CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); pFactory->SetFirst(); while (pFactory->IsValid()) @@ -382,6 +383,27 @@ void CCvarUtilities::EnableDevCvars() } } +//----------------------------------------------------------------------------- +// Purpose: Removes the FCVAR_DEVELOPMENTONLY flag from all cvars, making them accessible +//----------------------------------------------------------------------------- +void CCvarUtilities::EnableHiddenCvars() +{ + // Loop through cvars... + CCvar::CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); + pFactory->SetFirst(); + + while (pFactory->IsValid()) + { + // remove flag from all cvars + ConCommandBase* pCommandBase = pFactory->Get(); + pCommandBase->RemoveFlags(FCVAR_HIDDEN); + + pFactory->Next(); + } + + MemAllocSingleton()->Free(pFactory); +} + //----------------------------------------------------------------------------- // Purpose: // Output : void CCvar::CvarList_f @@ -439,7 +461,7 @@ void CCvarUtilities::CvarList(const CCommand& args) CUtlRBTree< ConCommandBase* > sorted(0, 0, ConCommandBaseLessFunc); // Loop through cvars... - CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); + CCvar::CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); pFactory->SetFirst(); while (pFactory->IsValid()) @@ -539,7 +561,7 @@ void CCvarUtilities::CvarHelp(const CCommand& args) //----------------------------------------------------------------------------- void CCvarUtilities::CvarDifferences(const CCommand& args) { - CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); + CCvar::CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); pFactory->SetFirst(); int i = 0; @@ -588,7 +610,7 @@ void CCvarUtilities::CvarFindFlags_f(const CCommand& args) ConCommandBase* var; // Loop through vars and print out findings - CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); + CCvar::CCVarIteratorInternal* pFactory = g_pCVar->FactoryInternalIterator(); pFactory->SetFirst(); while (pFactory->IsValid()) @@ -656,7 +678,7 @@ int CCvarUtilities::CvarFindFlagsCompletionCallback(const char* partial, char co // Purpose: registers input commands. // Input : *pszCommandName - //----------------------------------------------------------------------------- -ConCommandBase* CCVar::RegisterConCommand(ConCommandBase* pCommandToRemove) +ConCommandBase* CCvar::RegisterConCommand(ConCommandBase* pCommandToRemove) { const int index = 9; return CallVFunc(index, this, pCommandToRemove); @@ -666,7 +688,7 @@ ConCommandBase* CCVar::RegisterConCommand(ConCommandBase* pCommandToRemove) // Purpose: unregisters input commands. // Input : *pszCommandName - //----------------------------------------------------------------------------- -ConCommandBase* CCVar::UnregisterConCommand(ConCommandBase* pCommandToRemove) +ConCommandBase* CCvar::UnregisterConCommand(ConCommandBase* pCommandToRemove) { const int index = 10; return CallVFunc(index, this, pCommandToRemove); @@ -676,7 +698,7 @@ ConCommandBase* CCVar::UnregisterConCommand(ConCommandBase* pCommandToRemove) // Purpose: finds base commands. // Input : *pszCommandName - //----------------------------------------------------------------------------- -ConCommandBase* CCVar::FindCommandBase(const char* pszCommandName) +ConCommandBase* CCvar::FindCommandBase(const char* pszCommandName) { const int index = 14; return CallVFunc(index, this, pszCommandName); @@ -686,7 +708,7 @@ ConCommandBase* CCVar::FindCommandBase(const char* pszCommandName) // Purpose: finds ConVars. // Input : *pszVarName - //----------------------------------------------------------------------------- -ConVar* CCVar::FindVar(const char* pszVarName) +ConVar* CCvar::FindVar(const char* pszVarName) { const int index = 16; return CallVFunc(index, this, pszVarName); @@ -696,7 +718,7 @@ ConVar* CCVar::FindVar(const char* pszVarName) // Purpose: finds ConCommands. // Input : *pszCommandName - //----------------------------------------------------------------------------- -ConCommand* CCVar::FindCommand(const char* pszCommandName) +ConCommand* CCvar::FindCommand(const char* pszCommandName) { const int index = 18; return CallVFunc(index, this, pszCommandName); @@ -705,7 +727,7 @@ ConCommand* CCVar::FindCommand(const char* pszCommandName) //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- -void CCVar::CallGlobalChangeCallbacks(ConVar* pConVar, const char* pOldString) +void CCvar::CallGlobalChangeCallbacks(ConVar* pConVar, const char* pOldString) { const int index = 23; CallVFunc(index, this, pConVar, pOldString); @@ -714,22 +736,22 @@ void CCVar::CallGlobalChangeCallbacks(ConVar* pConVar, const char* pOldString) //----------------------------------------------------------------------------- // Purpose: deal with queued material system ConVars //----------------------------------------------------------------------------- -bool CCVar::IsMaterialThreadSetAllowed(void) +bool CCvar::IsMaterialThreadSetAllowed(void) { const int index = 35; return CallVFunc(index, this); } -void CCVar::QueueMaterialThreadSetValue(ConVar* pConVar, float flValue) +void CCvar::QueueMaterialThreadSetValue(ConVar* pConVar, float flValue) { const int index = 36; CallVFunc(index, this, pConVar, flValue); } -void CCVar::QueueMaterialThreadSetValue(ConVar* pConVar, int nValue) +void CCvar::QueueMaterialThreadSetValue(ConVar* pConVar, int nValue) { const int index = 37; CallVFunc(index, this, pConVar, nValue); } -void CCVar::QueueMaterialThreadSetValue(ConVar* pConVar, const char* pValue) +void CCvar::QueueMaterialThreadSetValue(ConVar* pConVar, const char* pValue) { const int index = 38; CallVFunc(index, this, pConVar, pValue); @@ -738,7 +760,7 @@ void CCVar::QueueMaterialThreadSetValue(ConVar* pConVar, const char* pValue) //----------------------------------------------------------------------------- // Purpose: iterates over all ConVars //----------------------------------------------------------------------------- -CCVarIteratorInternal* CCVar::FactoryInternalIterator(void) +CCvar::CCVarIteratorInternal* CCvar::FactoryInternalIterator(void) { const int index = 41; return CallVFunc(index, this); @@ -747,7 +769,7 @@ CCVarIteratorInternal* CCVar::FactoryInternalIterator(void) //----------------------------------------------------------------------------- // Purpose: returns all ConVars //----------------------------------------------------------------------------- -unordered_map CCVar::DumpToMap(void) +unordered_map CCvar::DumpToMap(void) { stringstream ss; CCVarIteratorInternal* itint = FactoryInternalIterator(); // Allocate new InternalIterator. @@ -765,4 +787,207 @@ unordered_map CCVar::DumpToMap(void) } /////////////////////////////////////////////////////////////////////////////// -CCVar* g_pCVar = nullptr; +CCvar* g_pCVar = nullptr; + + +//----------------------------------------------------------------------------- +// Console command hash data structure +//----------------------------------------------------------------------------- +CConCommandHash::CConCommandHash() +{ + Purge(true); +} + +CConCommandHash::~CConCommandHash() +{ + Purge(false); +} + +void CConCommandHash::Purge(bool bReinitialize) +{ + m_aBuckets.Purge(); + m_aDataPool.Purge(); + if (bReinitialize) + { + Init(); + } +} + +// Initialize. +void CConCommandHash::Init(void) +{ + // kNUM_BUCKETS must be a power of two. + COMPILE_TIME_ASSERT((kNUM_BUCKETS & (kNUM_BUCKETS - 1)) == 0); + + // Set the bucket size. + m_aBuckets.SetSize(kNUM_BUCKETS); + for (int iBucket = 0; iBucket < kNUM_BUCKETS; ++iBucket) + { + m_aBuckets[iBucket] = m_aDataPool.InvalidIndex(); + } + + // Calculate the grow size. + int nGrowSize = 4 * kNUM_BUCKETS; + m_aDataPool.SetGrowSize(nGrowSize); +} + +//----------------------------------------------------------------------------- +// Purpose: Insert data into the hash table given its key (unsigned int), +// WITH a check to see if the element already exists within the hash. +//----------------------------------------------------------------------------- +CConCommandHash::CCommandHashHandle_t CConCommandHash::Insert(ConCommandBase* cmd) +{ + // Check to see if that key already exists in the buckets (should be unique). + CCommandHashHandle_t hHash = Find(cmd); + if (hHash != InvalidHandle()) + return hHash; + + return FastInsert(cmd); +} +//----------------------------------------------------------------------------- +// Purpose: Insert data into the hash table given its key (unsigned int), +// WITHOUT a check to see if the element already exists within the hash. +//----------------------------------------------------------------------------- +CConCommandHash::CCommandHashHandle_t CConCommandHash::FastInsert(ConCommandBase* cmd) +{ + // Get a new element from the pool. + intptr_t iHashData = m_aDataPool.Alloc(true); + HashEntry_t* RESTRICT pHashData = &m_aDataPool[iHashData]; + if (!pHashData) + return InvalidHandle(); + + HashKey_t key = Hash(cmd); + + // Add data to new element. + pHashData->m_uiKey = key; + pHashData->m_Data = cmd; + + // Link element. + int iBucket = key & kBUCKETMASK; // HashFuncs::Hash( uiKey, m_uiBucketMask ); + m_aDataPool.LinkBefore(m_aBuckets[iBucket], iHashData); + m_aBuckets[iBucket] = iHashData; + + return iHashData; +} + +//----------------------------------------------------------------------------- +// Purpose: Remove a given element from the hash. +//----------------------------------------------------------------------------- +void CConCommandHash::Remove(CCommandHashHandle_t hHash) RESTRICT +{ + HashEntry_t* RESTRICT entry = &m_aDataPool[hHash]; + HashKey_t iBucket = entry->m_uiKey & kBUCKETMASK; + if (m_aBuckets[iBucket] == hHash) + { + // It is a bucket head. + m_aBuckets[iBucket] = m_aDataPool.Next(hHash); + } + else + { + // Not a bucket head. + m_aDataPool.Unlink(hHash); + } + + // Remove the element. + m_aDataPool.Remove(hHash); +} + +//----------------------------------------------------------------------------- +// Purpose: Remove all elements from the hash +//----------------------------------------------------------------------------- +void CConCommandHash::RemoveAll(void) +{ + m_aBuckets.RemoveAll(); + m_aDataPool.RemoveAll(); +} + +//----------------------------------------------------------------------------- +// Find hash entry corresponding to a string name +//----------------------------------------------------------------------------- +CConCommandHash::CCommandHashHandle_t CConCommandHash::Find(const char* name, HashKey_t hashkey) const RESTRICT +{ + // hash the "key" - get the correct hash table "bucket" + int iBucket = hashkey & kBUCKETMASK; + + for (datapool_t::IndexLocalType_t iElement = m_aBuckets[iBucket]; iElement != m_aDataPool.InvalidIndex(); iElement = m_aDataPool.Next(iElement)) + { + const HashEntry_t& element = m_aDataPool[iElement]; + if (element.m_uiKey == hashkey && // if hashes of strings match, + Q_stricmp(name, element.m_Data->GetName()) == 0) // then test the actual strings + { + return iElement; + } + } + + // found nuffink + return InvalidHandle(); +} + +//----------------------------------------------------------------------------- +// Find a command in the hash. +//----------------------------------------------------------------------------- +CConCommandHash::CCommandHashHandle_t CConCommandHash::Find(const ConCommandBase* cmd) const RESTRICT +{ + // Set this #if to 1 if the assert at bottom starts whining -- + // that indicates that a console command is being double-registered, + // or something similarly nonfatally bad. With this #if 1, we'll search + // by name instead of by pointer, which is more robust in the face + // of double registered commands, but obviously slower. +#if 0 + return Find(cmd->GetName()); +#else + HashKey_t hashkey = Hash(cmd); + int iBucket = hashkey & kBUCKETMASK; + + // hunt through all entries in that bucket + for (datapool_t::IndexLocalType_t iElement = m_aBuckets[iBucket]; iElement != m_aDataPool.InvalidIndex(); iElement = m_aDataPool.Next(iElement)) + { + const HashEntry_t& element = m_aDataPool[iElement]; + if (element.m_uiKey == hashkey && // if the hashes match... + element.m_Data == cmd) // and the pointers... + { + // in debug, test to make sure we don't have commands under the same name + // or something goofy like that + Assert(iElement == Find(cmd->GetName()), + "ConCommand %s had two entries in the hash!", cmd->GetName()); + + // return this element + return iElement; + } + } + + // found nothing. +#ifdef DBGFLAG_ASSERT // double check against search by name + CCommandHashHandle_t dbghand = Find(cmd->GetName()); + + AssertMsg1(InvalidHandle() == dbghand, + "ConCommand %s couldn't be found by pointer, but was found by name!", cmd->GetName()); +#endif + return InvalidHandle(); +#endif +} + + +//#ifdef _DEBUG +// Dump a report to MSG +void CConCommandHash::Report(void) +{ + DevMsg(eDLL_T::ENGINE, "Console command hash bucket load:\n"); + int total = 0; + for (int iBucket = 0; iBucket < kNUM_BUCKETS; ++iBucket) + { + int count = 0; + CCommandHashHandle_t iElement = m_aBuckets[iBucket]; // get the head of the bucket + while (iElement != m_aDataPool.InvalidIndex()) + { + ++count; + iElement = m_aDataPool.Next(iElement); + } + + DevMsg(eDLL_T::ENGINE, "%d: %d\n", iBucket, count); + total += count; + } + + DevMsg(eDLL_T::ENGINE, "\tAverage: %.1f\n", total / ((float)(kNUM_BUCKETS))); +} +//#endif diff --git a/r5dev/tier1/cvar.h b/r5dev/tier1/cvar.h index 495c3974..01cb85d6 100644 --- a/r5dev/tier1/cvar.h +++ b/r5dev/tier1/cvar.h @@ -1,5 +1,6 @@ #pragma once #include "tier1/IConVar.h" +#include //------------------------------------------------------------------------- // ENGINE | @@ -180,6 +181,7 @@ public: // Enable cvars marked with FCVAR_DEVELOPMENTONLY void EnableDevCvars(); + void EnableHiddenCvars(); // Lists cvars to console void CvarList(const CCommand& args); @@ -211,16 +213,7 @@ private: extern CCvarUtilities* cv; -class CCVarIteratorInternal // Fully reversed table, just look at the virtual function table and rename the function. -{ -public: - virtual void SetFirst(void) = 0; //0 - virtual void Next(void) = 0; //1 - virtual bool IsValid(void) = 0; //2 - virtual ConCommandBase* Get(void) = 0; //3 -}; - -class CCVar +class CCvar { public: ConCommandBase* RegisterConCommand(ConCommandBase* pCommandToAdd); @@ -236,12 +229,24 @@ public: void QueueMaterialThreadSetValue(ConVar* pConVar, int nValue); void QueueMaterialThreadSetValue(ConVar* pConVar, const char* pValue); + class CCVarIteratorInternal : public ICVarIteratorInternal + { + public: + virtual void SetFirst(void) = 0; //0 + virtual void Next(void) = 0; //1 + virtual bool IsValid(void) = 0; //2 + virtual ConCommandBase* Get(void) = 0; //3 + + CCvar* const m_pOuter; + CConCommandHash* const m_pHash; + CConCommandHash::CCommandHashIterator_t m_hashIter; + }; + CCVarIteratorInternal* FactoryInternalIterator(void); unordered_map DumpToMap(void); - -protected: friend class CCVarIteratorInternal; +protected: enum ConVarSetType_t { CONVAR_SET_STRING = 0, @@ -264,48 +269,49 @@ private: char pad0[22]; //!TODO: int m_nNextDLLIdentifier; ConCommandBase* m_pConCommandList; - char m_CommandHash[208]; //!TODO: + CConCommandHash m_CommandHash; + char pad1[96]; CUtlVector< QueuedConVarSet_t > m_QueuedConVarSets; bool m_bMaterialSystemThreadSetAllowed; }; /////////////////////////////////////////////////////////////////////////////// -extern CCVar* g_pCVar; +extern CCvar* g_pCVar; /* ==== CCVAR =========================================================================================================================================================== */ -inline CMemory p_CCVar_Disconnect; -inline auto CCVar_Disconnect = p_CCVar_Disconnect.RCast(); +inline CMemory p_CCvar_Disconnect; +inline auto CCvar_Disconnect = p_CCvar_Disconnect.RCast(); -inline CMemory p_CCVar_GetCommandLineValue; -inline auto CCVar_GetCommandLineValue = p_CCVar_GetCommandLineValue.RCast(); +inline CMemory p_CCvar_GetCommandLineValue; +inline auto CCvar_GetCommandLineValue = p_CCvar_GetCommandLineValue.RCast(); /////////////////////////////////////////////////////////////////////////////// class VCVar : public IDetour { virtual void GetAdr(void) const { - spdlog::debug("| FUN: CCVar::Disconnect : {:#18x} |\n", p_CCVar_Disconnect.GetPtr()); - spdlog::debug("| FUN: CCVar::GetCommandLineValue : {:#18x} |\n", p_CCVar_GetCommandLineValue.GetPtr()); + spdlog::debug("| FUN: CCvar::Disconnect : {:#18x} |\n", p_CCvar_Disconnect.GetPtr()); + spdlog::debug("| FUN: CCvar::GetCommandLineValue : {:#18x} |\n", p_CCvar_GetCommandLineValue.GetPtr()); spdlog::debug("| VAR: g_pCVar : {:#18x} |\n", reinterpret_cast(g_pCVar)); spdlog::debug("+----------------------------------------------------------------+\n"); } virtual void GetFun(void) const { #if defined (GAMEDLL_S0) || defined (GAMEDLL_S1) - p_CCVar_Disconnect = g_GameDll.FindPatternSIMD(reinterpret_cast("\x40\x57\x41\x56\x48\x83\xEC\x38\x4C\x8B\x35"), "xxxxxxxxxxx"); - CCVar_Disconnect = p_CCVar_Disconnect.RCast(); /*40 57 41 56 48 83 EC 38 4C 8B 35 ? ? ? ?*/ + p_CCvar_Disconnect = g_GameDll.FindPatternSIMD(reinterpret_cast("\x40\x57\x41\x56\x48\x83\xEC\x38\x4C\x8B\x35"), "xxxxxxxxxxx"); + CCvar_Disconnect = p_CCvar_Disconnect.RCast(); /*40 57 41 56 48 83 EC 38 4C 8B 35 ? ? ? ?*/ #elif defined (GAMEDLL_S2) || defined (GAMEDLL_S3) - p_CCVar_Disconnect = g_GameDll.FindPatternSIMD(reinterpret_cast("\x48\x83\xEC\x28\x48\x8B\x0D\x00\x00\x00\x00\x48\x85\xC9\x74\x26\x80\x3D\x00\x00\x00\x00\x00\x74\x1D\x48\x8B\x01\x8B\x15\x00\x00\x00\x00\xFF\x50\x58\xC7\x05\x00\x00\x00\x00\x00\x00\x00\x00\xC6\x05\x00\x00\x00\x00\x00\x48\xC7\x05\x00\x00\x00"), "xxxxxxx????xxxxxxx?????xxxxxxx????xxxxx????????xx"); - CCVar_Disconnect = p_CCVar_Disconnect.RCast(); /*48 83 EC 28 48 8B 0D ? ? ? ? 48 85 C9 74 26 80 3D ? ? ? ? ? 74 1D 48 8B 01 8B 15 ? ? ? ? FF 50 58 C7 05 ? ? ? ? ? ? ? ? C6 05 ? ? ? ? ? 48 C7 05 ? ? ? ? ? ? ? ?*/ + p_CCvar_Disconnect = g_GameDll.FindPatternSIMD(reinterpret_cast("\x48\x83\xEC\x28\x48\x8B\x0D\x00\x00\x00\x00\x48\x85\xC9\x74\x26\x80\x3D\x00\x00\x00\x00\x00\x74\x1D\x48\x8B\x01\x8B\x15\x00\x00\x00\x00\xFF\x50\x58\xC7\x05\x00\x00\x00\x00\x00\x00\x00\x00\xC6\x05\x00\x00\x00\x00\x00\x48\xC7\x05\x00\x00\x00"), "xxxxxxx????xxxxxxx?????xxxxxxx????xxxxx????????xx"); + CCvar_Disconnect = p_CCvar_Disconnect.RCast(); /*48 83 EC 28 48 8B 0D ? ? ? ? 48 85 C9 74 26 80 3D ? ? ? ? ? 74 1D 48 8B 01 8B 15 ? ? ? ? FF 50 58 C7 05 ? ? ? ? ? ? ? ? C6 05 ? ? ? ? ? 48 C7 05 ? ? ? ? ? ? ? ?*/ #endif - p_CCVar_GetCommandLineValue = g_GameDll.FindPatternSIMD(reinterpret_cast("\x40\x55\x48\x83\xEC\x20\x48\x8D\x6C\x24\x00\x48\x89\x5D\x10\x49\xC7\xC0\x00\x00\x00\x00"), "xxxxxxxxxx?xxxxxxx????"); - CCVar_GetCommandLineValue = p_CCVar_GetCommandLineValue.RCast(); /*40 55 48 83 EC 20 48 8D 6C 24 ? 48 89 5D 10 49 C7 C0 ? ? ? ?*/ + p_CCvar_GetCommandLineValue = g_GameDll.FindPatternSIMD(reinterpret_cast("\x40\x55\x48\x83\xEC\x20\x48\x8D\x6C\x24\x00\x48\x89\x5D\x10\x49\xC7\xC0\x00\x00\x00\x00"), "xxxxxxxxxx?xxxxxxx????"); + CCvar_GetCommandLineValue = p_CCvar_GetCommandLineValue.RCast(); /*40 55 48 83 EC 20 48 8D 6C 24 ? 48 89 5D 10 49 C7 C0 ? ? ? ?*/ } virtual void GetVar(void) const { g_pCVar = g_GameDll.FindPatternSIMD(reinterpret_cast( "\x48\x83\xEC\x28\x48\x8B\x05\x00\x00\x00\x00\x48\x8D\x0D\x00\x00\x00\x00\x48\x85\xC0\x48\x89\x15"), - "xxxxxxx????xxx????xxxxxx").FindPatternSelf("48 8D 0D", CMemory::Direction::DOWN, 40).ResolveRelativeAddressSelf(0x3, 0x7).RCast(); + "xxxxxxx????xxx????xxxxxx").FindPatternSelf("48 8D 0D", CMemory::Direction::DOWN, 40).ResolveRelativeAddressSelf(0x3, 0x7).RCast(); } virtual void GetCon(void) const { } virtual void Attach(void) const { } diff --git a/r5dev/tier1/generichash.cpp b/r5dev/tier1/generichash.cpp new file mode 100644 index 00000000..5f8602f8 --- /dev/null +++ b/r5dev/tier1/generichash.cpp @@ -0,0 +1,430 @@ +//======= Copyright � 2005, , Valve Corporation, All rights reserved. ========= +// +// Purpose: Variant Pearson Hash general purpose hashing algorithm described +// by Cargill in C++ Report 1994. Generates a 16-bit result. +// +//============================================================================= + +#include "core/stdafx.h" +#include "tier0/dbg.h" +#include "tier0/basetypes.h" +#include "tier0/platform.h" +#include "tier1/strtools.h" +#include "tier1/generichash.h" +#include "mathlib/swap.h" + +// NOTE: This has to be the last file included! +//#include "tier0/memdbgon.h" + + +//----------------------------------------------------------------------------- +// +// Table of randomly shuffled values from 0-255 generated by: +// +//----------------------------------------------------------------------------- +/* +void MakeRandomValues() +{ + int i, j, r; + unsigned t; + srand( 0xdeadbeef ); + + for ( i = 0; i < 256; i++ ) + { + g_nRandomValues[i] = (unsigned )i; + } + + for (j = 0; j < 8; j++) + { + for (i = 0; i < 256; i++) + { + r = rand() & 0xff; + t = g_nRandomValues[i]; + g_nRandomValues[i] = g_nRandomValues[r]; + g_nRandomValues[r] = t; + } + } + + printf("static unsigned g_nRandomValues[256] =\n{\n"); + + for (i = 0; i < 256; i += 16) + { + printf("\t"); + for (j = 0; j < 16; j++) + printf(" %3d,", g_nRandomValues[i+j]); + printf("\n"); + } + printf("};\n"); +} +*/ + +static unsigned g_nRandomValues[256] = +{ + 238, 164, 191, 168, 115, 16, 142, 11, 213, 214, 57, 151, 248, 252, 26, 198, + 13, 105, 102, 25, 43, 42, 227, 107, 210, 251, 86, 66, 83, 193, 126, 108, + 131, 3, 64, 186, 192, 81, 37, 158, 39, 244, 14, 254, 75, 30, 2, 88, + 172, 176, 255, 69, 0, 45, 116, 139, 23, 65, 183, 148, 33, 46, 203, 20, + 143, 205, 60, 197, 118, 9, 171, 51, 233, 135, 220, 49, 71, 184, 82, 109, + 36, 161, 169, 150, 63, 96, 173, 125, 113, 67, 224, 78, 232, 215, 35, 219, + 79, 181, 41, 229, 149, 153, 111, 217, 21, 72, 120, 163, 133, 40, 122, 140, + 208, 231, 211, 200, 160, 182, 104, 110, 178, 237, 15, 101, 27, 50, 24, 189, + 177, 130, 187, 92, 253, 136, 100, 212, 19, 174, 70, 22, 170, 206, 162, 74, + 247, 5, 47, 32, 179, 117, 132, 195, 124, 123, 245, 128, 236, 223, 12, 84, + 54, 218, 146, 228, 157, 94, 106, 31, 17, 29, 194, 34, 56, 134, 239, 246, + 241, 216, 127, 98, 7, 204, 154, 152, 209, 188, 48, 61, 87, 97, 225, 85, + 90, 167, 155, 112, 145, 114, 141, 93, 250, 4, 201, 156, 38, 89, 226, 196, + 1, 235, 44, 180, 159, 121, 119, 166, 190, 144, 10, 91, 76, 230, 221, 80, + 207, 55, 58, 53, 175, 8, 6, 52, 68, 242, 18, 222, 103, 249, 147, 129, + 138, 243, 28, 185, 62, 59, 240, 202, 234, 99, 77, 73, 199, 137, 95, 165, +}; + +//----------------------------------------------------------------------------- +// String +//----------------------------------------------------------------------------- +unsigned FASTCALL HashString(const char* pszKey) +{ + const uint8* k = (const uint8*)pszKey; + unsigned even = 0, + odd = 0, + n; + + while ((n = *k++) != 0) + { + even = g_nRandomValues[odd ^ n]; + if ((n = *k++) != 0) + odd = g_nRandomValues[even ^ n]; + else + break; + } + + return (even << 8) | odd; +} + + +//----------------------------------------------------------------------------- +// Case-insensitive string +//----------------------------------------------------------------------------- +unsigned FASTCALL HashStringCaseless(const char* pszKey) +{ + const uint8* k = (const uint8*)pszKey; + unsigned even = 0, + odd = 0, + n; + + while ((n = toupper(*k++)) != 0) + { + even = g_nRandomValues[odd ^ n]; + if ((n = toupper(*k++)) != 0) + odd = g_nRandomValues[even ^ n]; + else + break; + } + + return (even << 8) | odd; +} + +//----------------------------------------------------------------------------- +// 32 bit conventional case-insensitive string +//----------------------------------------------------------------------------- +uint32 FASTCALL HashStringCaselessConventional(const char* pszKey) +{ + uint32 hash = 0xAAAAAAAA; // Alternating 1's and 0's to maximize the effect of the later multiply and add + hash += (2 * V_strlen(pszKey)); // Add the string length * 2 to the hash to give it more variety + + for (; *pszKey; pszKey++) + { + hash = ((hash << 5) + hash) + (uint8)tolower(*pszKey); + } + + return hash; +} + +//----------------------------------------------------------------------------- +// int hash +//----------------------------------------------------------------------------- +unsigned FASTCALL HashInt(const int n) +{ + register unsigned even, odd; + odd = g_nRandomValues[(((unsigned)n >> 8) & 0xff)]; + even = g_nRandomValues[odd ^ ((unsigned)n >> 24)]; + odd = g_nRandomValues[even ^ ((unsigned)n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ ((unsigned)n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ ((unsigned)n & 0xff)]; + + return (even << 8) | odd; +} + +//----------------------------------------------------------------------------- +// 4-byte hash +//----------------------------------------------------------------------------- +unsigned FASTCALL Hash4(const void* pKey) +{ + register const uint32* p = (const uint32*)pKey; + register unsigned even, + odd, + n; + n = *p; + odd = g_nRandomValues[((n >> 8) & 0xff)]; + even = g_nRandomValues[odd ^ (n >> 24)]; + odd = g_nRandomValues[even ^ (n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ (n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ (n & 0xff)]; + + return (even << 8) | odd; +} + + +//----------------------------------------------------------------------------- +// 8-byte hash +//----------------------------------------------------------------------------- +unsigned FASTCALL Hash8(const void* pKey) +{ + register const uint32* p = (const uint32*)pKey; + register unsigned even, + odd, + n; + n = *p; + odd = g_nRandomValues[((n >> 8) & 0xff)]; + even = g_nRandomValues[odd ^ (n >> 24)]; + odd = g_nRandomValues[even ^ (n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ (n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ (n & 0xff)]; + + n = *(p + 1); + even = g_nRandomValues[odd ^ (n >> 24)]; + odd = g_nRandomValues[even ^ (n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ (n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ (n & 0xff)]; + + return (even << 8) | odd; +} + + +//----------------------------------------------------------------------------- +// 12-byte hash +//----------------------------------------------------------------------------- +unsigned FASTCALL Hash12(const void* pKey) +{ + register const uint32* p = (const uint32*)pKey; + register unsigned even, + odd, + n; + n = *p; + odd = g_nRandomValues[((n >> 8) & 0xff)]; + + even = g_nRandomValues[odd ^ (n >> 24)]; + odd = g_nRandomValues[even ^ (n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ (n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ (n & 0xff)]; + + n = *(p + 1); + even = g_nRandomValues[odd ^ (n >> 24)]; + odd = g_nRandomValues[even ^ (n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ (n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ (n & 0xff)]; + + n = *(p + 2); + even = g_nRandomValues[odd ^ (n >> 24)]; + odd = g_nRandomValues[even ^ (n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ (n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ (n & 0xff)]; + + return (even << 8) | odd; +} + + +//----------------------------------------------------------------------------- +// 16-byte hash +//----------------------------------------------------------------------------- +unsigned FASTCALL Hash16(const void* pKey) +{ + register const uint32* p = (const uint32*)pKey; + register unsigned even, + odd, + n; + n = *p; + odd = g_nRandomValues[((n >> 8) & 0xff)]; + + even = g_nRandomValues[odd ^ (n >> 24)]; + odd = g_nRandomValues[even ^ (n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ (n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ (n & 0xff)]; + + n = *(p + 1); + even = g_nRandomValues[odd ^ (n >> 24)]; + odd = g_nRandomValues[even ^ (n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ (n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ (n & 0xff)]; + + n = *(p + 2); + even = g_nRandomValues[odd ^ (n >> 24)]; + odd = g_nRandomValues[even ^ (n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ (n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ (n & 0xff)]; + + n = *(p + 3); + even = g_nRandomValues[odd ^ (n >> 24)]; + odd = g_nRandomValues[even ^ (n >> 16) & 0xff]; + even = g_nRandomValues[odd ^ (n >> 8) & 0xff]; + odd = g_nRandomValues[even ^ (n & 0xff)]; + + return (even << 8) | odd; +} + + +//----------------------------------------------------------------------------- +// Arbitrary fixed length hash +//----------------------------------------------------------------------------- +unsigned FASTCALL HashBlock(const void* pKey, unsigned size) +{ + const uint8* k = (const uint8*)pKey; + unsigned even = 0, + odd = 0, + n; + + while (size) + { + --size; + n = *k++; + even = g_nRandomValues[odd ^ n]; + if (size) + { + --size; + n = *k++; + odd = g_nRandomValues[even ^ n]; + } + else + break; + } + + return (even << 8) | odd; +} + + +//----------------------------------------------------------------------------- +// Murmur hash +//----------------------------------------------------------------------------- +uint32 MurmurHash2(const void* key, int len, uint32 seed) +{ + // 'm' and 'r' are mixing constants generated offline. + // They're not really 'magic', they just happen to work well. + + const uint32 m = 0x5bd1e995; + const int r = 24; + + // Initialize the hash to a 'random' value + + uint32 h = seed ^ len; + + // Mix 4 bytes at a time into the hash + + const unsigned char* data = (const unsigned char*)key; + + while (len >= 4) + { + uint32 k = LittleDWord(*(uint32*)data); + + k *= m; + k ^= k >> r; + k *= m; + + h *= m; + h ^= k; + + data += 4; + len -= 4; + } + + // Handle the last few bytes of the input array + + switch (len) + { + case 3: h ^= data[2] << 16; + case 2: h ^= data[1] << 8; + case 1: h ^= data[0]; + h *= m; + }; + + // Do a few final mixes of the hash to ensure the last few + // bytes are well-incorporated. + + h ^= h >> 13; + h *= m; + h ^= h >> 15; + + return h; +} + +#define TOLOWERU( c ) ( ( uint32 ) ( ( ( c >= 'A' ) && ( c <= 'Z' ) )? c + 32 : c ) ) +uint32 MurmurHash2LowerCase(char const* pString, uint32 nSeed) +{ + int nLen = V_strlen(pString); + char* p = (char*)stackalloc(nLen + 1); + for (int i = 0; i < nLen; i++) + { + p[i] = TOLOWERU(pString[i]); + } + return MurmurHash2(p, nLen, nSeed); +} + + +//----------------------------------------------------------------------------- +// Murmur hash, 64 bit- endian neutral +//----------------------------------------------------------------------------- +uint64 MurmurHash64(const void* key, int len, uint32 seed) +{ + // 'm' and 'r' are mixing constants generated offline. + // They're not really 'magic', they just happen to work well. + + const uint32 m = 0x5bd1e995; + const int r = 24; + + // Initialize the hash to a 'random' value + + uint32 h1 = seed ^ len; + uint32 h2 = 0; + + // Mix 4 bytes at a time into the hash + + const uint32* data = (const uint32*)key; + while (len >= 8) + { + uint32 k1 = LittleDWord(*data++); + k1 *= m; k1 ^= k1 >> r; k1 *= m; + h1 *= m; h1 ^= k1; + len -= 4; + + uint32 k2 = LittleDWord(*data++); + k2 *= m; k2 ^= k2 >> r; k2 *= m; + h2 *= m; h2 ^= k2; + len -= 4; + } + + if (len >= 4) + { + uint32 k1 = LittleDWord(*data++); + k1 *= m; k1 ^= k1 >> r; k1 *= m; + h1 *= m; h1 ^= k1; + len -= 4; + } + + // Handle the last few bytes of the input array + switch (len) + { + case 3: h2 ^= ((uint8*)data)[2] << 16; + case 2: h2 ^= ((uint8*)data)[1] << 8; + case 1: h2 ^= ((uint8*)data)[0]; + h2 *= m; + }; + + h1 ^= h2 >> 18; h1 *= m; + h2 ^= h1 >> 22; h2 *= m; + h1 ^= h2 >> 17; h1 *= m; + h2 ^= h1 >> 19; h2 *= m; + + uint64 h = h1; + + h = (h << 32) | h2; + + return h; +} diff --git a/r5dev/tier1/utllinkedlist.h b/r5dev/tier1/utllinkedlist.h new file mode 100644 index 00000000..4a7ed1a9 --- /dev/null +++ b/r5dev/tier1/utllinkedlist.h @@ -0,0 +1,1095 @@ +//======= Copyright � 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: Linked list container class +// +// $Revision: $ +// $NoKeywords: $ +//=============================================================================// + +#ifndef UTLLINKEDLIST_H +#define UTLLINKEDLIST_H + +#ifdef _WIN32 +#pragma once +#endif + +#include "tier0/basetypes.h" +#include "utlmemory.h" +#include "utlfixedmemory.h" +#include "utlblockmemory.h" +#include "tier0/dbg.h" + +// define to enable asserts griping about things you shouldn't be doing with multilists +// #define MULTILIST_PEDANTIC_ASSERTS 1 + +// This is a useful macro to iterate from head to tail in a linked list. +#define FOR_EACH_LL( listName, iteratorName ) \ + for( int iteratorName=listName.Head(); iteratorName != listName.InvalidIndex(); iteratorName = listName.Next( iteratorName ) ) + +//----------------------------------------------------------------------------- +// class CUtlLinkedList: +// description: +// A lovely index-based linked list! T is the class type, I is the index +// type, which usually should be an unsigned short or smaller. However, +// you must avoid using 16- or 8-bit arithmetic on PowerPC architectures; +// therefore you should not use UtlLinkedListElem_t::I as the type of +// a local variable... ever. PowerPC integer arithmetic must be 32- or +// 64-bit only; otherwise performance plummets. +//----------------------------------------------------------------------------- + +template +struct UtlLinkedListElem_t +{ + T m_Element; + I m_Previous; + I m_Next; + +private: + // No copy constructor for these... + UtlLinkedListElem_t(const UtlLinkedListElem_t&); +}; + + +// Class S is the storage type; the type you can use to save off indices in +// persistent memory. Class I is the iterator type, which is what should be used +// in local scopes. I defaults to be S, but be aware that on the 360, 16-bit +// arithmetic is catastrophically slow. Therefore you should try to save shorts +// in memory, but always operate on 32's or 64's in local scope. +// The ideal parameter order would be TSMI (you are more likely to override M than I) +// but since M depends on I we can't have the defaults in that order, alas. +template , I > > +class CUtlLinkedList +{ +public: + typedef T ElemType_t; + typedef S IndexType_t; // should really be called IndexStorageType_t, but that would be a huge change + typedef I IndexLocalType_t; + typedef M MemoryAllocator_t; + + // constructor, destructor + CUtlLinkedList(ssize_t growSize = 0, ssize_t initSize = 0); + ~CUtlLinkedList(); + + // gets particular elements + T& Element(I i); + T const& Element(I i) const; + T& operator[](I i); + T const& operator[](I i) const; + + // Make sure we have a particular amount of memory + void EnsureCapacity(int num); + + void SetGrowSize(int growSize); + + // Memory deallocation + void Purge(); + + // Delete all the elements then call Purge. + void PurgeAndDeleteElements(); + + // Insertion methods.... + I InsertBefore(I before); + I InsertAfter(I after); + I AddToHead(); + I AddToTail(); + + I InsertBefore(I before, T const& src); + I InsertAfter(I after, T const& src); + I AddToHead(T const& src); + I AddToTail(T const& src); + + // Find an element and return its index or InvalidIndex() if it couldn't be found. + I Find(const T& src) const; + + // Look for the element. If it exists, remove it and return true. Otherwise, return false. + bool FindAndRemove(const T& src); + + // Removal methods + void Remove(I elem); + void RemoveAll(); + + // Allocation/deallocation methods + // If multilist == true, then list list may contain many + // non-connected lists, and IsInList and Head + Tail are meaningless... + I Alloc(bool multilist = false); + void Free(I elem); + + // Identify the owner of this linked list's memory: + void SetAllocOwner(const char* pszAllocOwner); + + // list modification + void LinkBefore(I before, I elem); + void LinkAfter(I after, I elem); + void Unlink(I elem); + void LinkToHead(I elem); + void LinkToTail(I elem); + + // invalid index (M will never allocate an element at this index) + inline static S InvalidIndex() { return (S)M::InvalidIndex(); } + + // Is a given index valid to use? (representible by S and not the invalid index) + static bool IndexInRange(I index); + + inline static size_t ElementSize() { return sizeof(ListElem_t); } + + // list statistics + int Count() const; + I MaxElementIndex() const; + I NumAllocated(void) const { return m_NumAlloced; } + + // Traversing the list + I Head() const; + I Tail() const; + I Previous(I i) const; + I Next(I i) const; + + // Are nodes in the list or valid? + bool IsValidIndex(I i) const; + bool IsInList(I i) const; + +protected: + + // What the linked list element looks like + typedef UtlLinkedListElem_t ListElem_t; + + // constructs the class + I AllocInternal(bool multilist = false); + void ConstructList(); + + // Gets at the list element.... + ListElem_t& InternalElement(I i) { return m_Memory[i]; } + ListElem_t const& InternalElement(I i) const { return m_Memory[i]; } + + // copy constructors not allowed + CUtlLinkedList(CUtlLinkedList const& list) { Assert(0); } + + M m_Memory; + I m_Head; + I m_Tail; + I m_FirstFree; + I m_ElementCount; // The number actually in the list + I m_NumAlloced; // The number of allocated elements + typename M::Iterator_t m_LastAlloc; // the last index allocated + + // For debugging purposes; + // it's in release builds so this can be used in libraries correctly + ListElem_t* m_pElements; + + FORCEINLINE M const& Memory(void) const + { + return m_Memory; + } + + void ResetDbgInfo() + { + m_pElements = m_Memory.Base(); + } +}; + + +// this is kind of ugly, but until C++ gets templatized typedefs in C++0x, it's our only choice +template < class T > +class CUtlFixedLinkedList : public CUtlLinkedList< T, intptr_t, true, intptr_t, CUtlFixedMemory< UtlLinkedListElem_t< T, intptr_t > > > +{ +public: + CUtlFixedLinkedList(ssize_t growSize = 0, ssize_t initSize = 0) + : CUtlLinkedList< T, intptr_t, true, intptr_t, CUtlFixedMemory< UtlLinkedListElem_t< T, intptr_t > > >(growSize, initSize) {} + + bool IsValidIndex(intptr_t i) const + { + if (!this->Memory().IsIdxValid(i)) + return false; + +#ifdef _DEBUG // it's safe to skip this here, since the only way to get indices after m_LastAlloc is to use MaxElementIndex + if (this->Memory().IsIdxAfter(i, this->m_LastAlloc)) + { + Assert(0); + return false; // don't read values that have been allocated, but not constructed + } +#endif + + return (this->Memory()[i].m_Previous != i) || (this->Memory()[i].m_Next == i); + } + +private: + int MaxElementIndex() const { Assert(0); return this->InvalidIndex(); } // fixedmemory containers don't support iteration from 0..maxelements-1 + void ResetDbgInfo() {} +}; + +// this is kind of ugly, but until C++ gets templatized typedefs in C++0x, it's our only choice +template < class T, class I = unsigned short > +class CUtlBlockLinkedList : public CUtlLinkedList< T, I, true, I, CUtlBlockMemory< UtlLinkedListElem_t< T, I >, I > > +{ +public: + CUtlBlockLinkedList(int growSize = 0, int initSize = 0) + : CUtlLinkedList< T, I, true, I, CUtlBlockMemory< UtlLinkedListElem_t< T, I >, I > >(growSize, initSize) {} +protected: + void ResetDbgInfo() {} +}; + + +//----------------------------------------------------------------------------- +// constructor, destructor +//----------------------------------------------------------------------------- + +template +CUtlLinkedList::CUtlLinkedList(ssize_t growSize, ssize_t initSize) : + m_Memory(growSize, initSize), m_LastAlloc(m_Memory.InvalidIterator()) +{ + // Prevent signed non-int datatypes + COMPILE_TIME_ASSERT(sizeof(S) == 8 || (((S)-1) > 0)); + ConstructList(); + ResetDbgInfo(); +} + +template +CUtlLinkedList::~CUtlLinkedList() +{ + RemoveAll(); +} + +template +void CUtlLinkedList::ConstructList() +{ + m_Head = InvalidIndex(); + m_Tail = InvalidIndex(); + m_FirstFree = InvalidIndex(); + m_ElementCount = 0; + m_NumAlloced = 0; +} + + +//----------------------------------------------------------------------------- +// gets particular elements +//----------------------------------------------------------------------------- + +template +inline T& CUtlLinkedList::Element(I i) +{ + return m_Memory[i].m_Element; +} + +template +inline T const& CUtlLinkedList::Element(I i) const +{ + return m_Memory[i].m_Element; +} + +template +inline T& CUtlLinkedList::operator[](I i) +{ + return m_Memory[i].m_Element; +} + +template +inline T const& CUtlLinkedList::operator[](I i) const +{ + return m_Memory[i].m_Element; +} + +//----------------------------------------------------------------------------- +// list statistics +//----------------------------------------------------------------------------- + +template +inline int CUtlLinkedList::Count() const +{ +#ifdef MULTILIST_PEDANTIC_ASSERTS + AssertMsg(!ML, "CUtlLinkedList::Count() is meaningless for linked lists."); +#endif + return m_ElementCount; +} + +template +inline I CUtlLinkedList::MaxElementIndex() const +{ + return m_Memory.NumAllocated(); +} + + +//----------------------------------------------------------------------------- +// Traversing the list +//----------------------------------------------------------------------------- + +template +inline I CUtlLinkedList::Head() const +{ + return m_Head; +} + +template +inline I CUtlLinkedList::Tail() const +{ + return m_Tail; +} + +template +inline I CUtlLinkedList::Previous(I i) const +{ + Assert(IsValidIndex(i)); + return InternalElement(i).m_Previous; +} + +template +inline I CUtlLinkedList::Next(I i) const +{ + Assert(IsValidIndex(i)); + return InternalElement(i).m_Next; +} + + +//----------------------------------------------------------------------------- +// Are nodes in the list or valid? +//----------------------------------------------------------------------------- + +#pragma warning(push) +#pragma warning( disable: 4310 ) // Allows "(I)(S)M::INVALID_INDEX" below +template +inline bool CUtlLinkedList::IndexInRange(I index) // Static method +{ + // Since S is not necessarily the type returned by M, we need to check that M returns indices + // which are representable by S. A common case is 'S === unsigned short', 'I == int', in which + // case CUtlMemory will have 'InvalidIndex == (int)-1' (which casts to 65535 in S), and will + // happily return elements at index 65535 and above. + + // Do some static checks here: + // 'I' needs to be able to store 'S' + COMPILE_TIME_ASSERT(sizeof(I) >= sizeof(S)); + // 'S' should be unsigned (to avoid signed arithmetic errors for plausibly exhaustible ranges) + COMPILE_TIME_ASSERT((sizeof(S) > 2) || (((S)-1) > 0)); + // M::INVALID_INDEX should be storable in S to avoid ambiguities (e.g. with 65536) + COMPILE_TIME_ASSERT((M::INVALID_INDEX == -1) || (M::INVALID_INDEX == (S)M::INVALID_INDEX)); + + return (((S)index == index) && ((S)index != InvalidIndex())); +} +#pragma warning(pop) + +template +inline bool CUtlLinkedList::IsValidIndex(I i) const +{ + if (!m_Memory.IsIdxValid(i)) + return false; + + if (m_Memory.IsIdxAfter(i, m_LastAlloc)) + return false; // don't read values that have been allocated, but not constructed + + return (m_Memory[i].m_Previous != i) || (m_Memory[i].m_Next == i); +} + +template +inline bool CUtlLinkedList::IsInList(I i) const +{ + if (!m_Memory.IsIdxValid(i) || m_Memory.IsIdxAfter(i, m_LastAlloc)) + return false; // don't read values that have been allocated, but not constructed + + return Previous(i) != i; +} + +/* +template +inline bool CUtlFixedLinkedList::IsInList( int i ) const +{ + return m_Memory.IsIdxValid( i ) && (Previous( i ) != i); +} +*/ + +//----------------------------------------------------------------------------- +// Makes sure we have enough memory allocated to store a requested # of elements +//----------------------------------------------------------------------------- + +template< class T, class S, bool ML, class I, class M > +void CUtlLinkedList::EnsureCapacity(int num) +{ + MEM_ALLOC_CREDIT_CLASS(); + m_Memory.EnsureCapacity(num); + ResetDbgInfo(); +} + +template< class T, class S, bool ML, class I, class M > +void CUtlLinkedList::SetGrowSize(int growSize) +{ + RemoveAll(); + m_Memory.Init(growSize); + ResetDbgInfo(); +} + +template< class T, class S, bool ML, class I, class M > +void CUtlLinkedList::SetAllocOwner(const char* pszAllocOwner) +{ + m_Memory.SetAllocOwner(pszAllocOwner); +} + + +//----------------------------------------------------------------------------- +// Deallocate memory +//----------------------------------------------------------------------------- + +template +void CUtlLinkedList::Purge() +{ + RemoveAll(); + + m_Memory.Purge(); + m_FirstFree = InvalidIndex(); + m_NumAlloced = 0; + + //Routing "m_LastAlloc = m_Memory.InvalidIterator();" through a local const to sidestep an internal compiler error on 360 builds + const typename M::Iterator_t scInvalidIterator = m_Memory.InvalidIterator(); + m_LastAlloc = scInvalidIterator; + ResetDbgInfo(); +} + + +template +void CUtlLinkedList::PurgeAndDeleteElements() +{ + I iNext; + for (I i = Head(); i != InvalidIndex(); i = iNext) + { + iNext = Next(i); + MemAllocSingleton()->Free(Element(i)); + } + + Purge(); +} + +//----------------------------------------------------------------------------- +// Node allocation/deallocation +//----------------------------------------------------------------------------- +template +I CUtlLinkedList::AllocInternal(bool multilist) +{ + Assert(!multilist || ML); +#ifdef MULTILIST_PEDANTIC_ASSERTS + Assert(multilist == ML); +#endif + I elem; + if (m_FirstFree == InvalidIndex()) + { + Assert(m_Memory.IsValidIterator(m_LastAlloc) || m_ElementCount == 0); + + typename M::Iterator_t it = m_Memory.IsValidIterator(m_LastAlloc) ? m_Memory.Next(m_LastAlloc) : m_Memory.First(); + + if (!m_Memory.IsValidIterator(it)) + { + MEM_ALLOC_CREDIT_CLASS(); + m_Memory.Grow(); + ResetDbgInfo(); + + it = m_Memory.IsValidIterator(m_LastAlloc) ? m_Memory.Next(m_LastAlloc) : m_Memory.First(); + + Assert(m_Memory.IsValidIterator(it)); + if (!m_Memory.IsValidIterator(it)) + { + ExecuteNTimes(10, Warning(eDLL_T::ENGINE, "CUtlLinkedList overflow! (exhausted memory allocator)\n")); + return InvalidIndex(); + } + } + + // We can overflow before the utlmemory overflows, since S != I + if (!IndexInRange(m_Memory.GetIndex(it))) + { + ExecuteNTimes(10, Warning(eDLL_T::ENGINE, "CUtlLinkedList overflow! (exhausted index range)\n")); + return InvalidIndex(); + } + + m_LastAlloc = it; + elem = m_Memory.GetIndex(m_LastAlloc); + m_NumAlloced++; + } + else + { + elem = m_FirstFree; + m_FirstFree = InternalElement(m_FirstFree).m_Next; + } + + if (!multilist) + { + InternalElement(elem).m_Next = elem; + InternalElement(elem).m_Previous = elem; + } + else + { + InternalElement(elem).m_Next = InvalidIndex(); + InternalElement(elem).m_Previous = InvalidIndex(); + } + + return elem; +} + +template +I CUtlLinkedList::Alloc(bool multilist) +{ + I elem = AllocInternal(multilist); + if (elem == InvalidIndex()) + return elem; + + Construct(&Element(elem)); + + return elem; +} + +template +void CUtlLinkedList::Free(I elem) +{ + Assert(IsValidIndex(elem) && IndexInRange(elem)); + Unlink(elem); + + ListElem_t& internalElem = InternalElement(elem); + Destruct(&internalElem.m_Element); + internalElem.m_Next = m_FirstFree; + m_FirstFree = elem; +} + +//----------------------------------------------------------------------------- +// Insertion methods; allocates and links (uses default constructor) +//----------------------------------------------------------------------------- + +template +I CUtlLinkedList::InsertBefore(I before) +{ + // Make a new node + I newNode = AllocInternal(); + if (newNode == InvalidIndex()) + return newNode; + + // Link it in + LinkBefore(before, newNode); + + // Construct the data + Construct(&Element(newNode)); + + return newNode; +} + +template +I CUtlLinkedList::InsertAfter(I after) +{ + // Make a new node + I newNode = AllocInternal(); + if (newNode == InvalidIndex()) + return newNode; + + // Link it in + LinkAfter(after, newNode); + + // Construct the data + Construct(&Element(newNode)); + + return newNode; +} + +template +inline I CUtlLinkedList::AddToHead() +{ + return InsertAfter(InvalidIndex()); +} + +template +inline I CUtlLinkedList::AddToTail() +{ + return InsertBefore(InvalidIndex()); +} + + +//----------------------------------------------------------------------------- +// Insertion methods; allocates and links (uses copy constructor) +//----------------------------------------------------------------------------- + +template +I CUtlLinkedList::InsertBefore(I before, T const& src) +{ + // Make a new node + I newNode = AllocInternal(); + if (newNode == InvalidIndex()) + return newNode; + + // Link it in + LinkBefore(before, newNode); + + // Construct the data + CopyConstruct(&Element(newNode), src); + + return newNode; +} + +template +I CUtlLinkedList::InsertAfter(I after, T const& src) +{ + // Make a new node + I newNode = AllocInternal(); + if (newNode == InvalidIndex()) + return newNode; + + // Link it in + LinkAfter(after, newNode); + + // Construct the data + CopyConstruct(&Element(newNode), src); + + return newNode; +} + +template +inline I CUtlLinkedList::AddToHead(T const& src) +{ + return InsertAfter(InvalidIndex(), src); +} + +template +inline I CUtlLinkedList::AddToTail(T const& src) +{ + return InsertBefore(InvalidIndex(), src); +} + + +//----------------------------------------------------------------------------- +// Removal methods +//----------------------------------------------------------------------------- + +template +I CUtlLinkedList::Find(const T& src) const +{ + for (I i = Head(); i != InvalidIndex(); i = Next(i)) + { + if (Element(i) == src) + return i; + } + return InvalidIndex(); +} + + +template +bool CUtlLinkedList::FindAndRemove(const T& src) +{ + I i = Find(src); + if (i == InvalidIndex()) + { + return false; + } + else + { + Remove(i); + return true; + } +} + + +template +void CUtlLinkedList::Remove(I elem) +{ + Free(elem); +} + +template +void CUtlLinkedList::RemoveAll() +{ + // Have to do some convoluted stuff to invoke the destructor on all + // valid elements for the multilist case (since we don't have all elements + // connected to each other in a list). + + if (m_LastAlloc == m_Memory.InvalidIterator()) + { + Assert(m_Head == InvalidIndex()); + Assert(m_Tail == InvalidIndex()); + Assert(m_FirstFree == InvalidIndex()); + Assert(m_ElementCount == 0); + return; + } + + if (ML) + { + for (typename M::Iterator_t it = m_Memory.First(); it != m_Memory.InvalidIterator(); it = m_Memory.Next(it)) + { + I i = m_Memory.GetIndex(it); + if (IsValidIndex(i)) // skip elements already in the free list + { + ListElem_t& internalElem = InternalElement(i); + Destruct(&internalElem.m_Element); + internalElem.m_Previous = i; + internalElem.m_Next = m_FirstFree; + m_FirstFree = i; + } + + if (it == m_LastAlloc) + break; // don't destruct elements that haven't ever been constructed + } + } + else + { + I i = Head(); + I next; + while (i != InvalidIndex()) + { + next = Next(i); + ListElem_t& internalElem = InternalElement(i); + Destruct(&internalElem.m_Element); + internalElem.m_Previous = i; + internalElem.m_Next = next == InvalidIndex() ? m_FirstFree : next; + i = next; + } + if (Head() != InvalidIndex()) + { + m_FirstFree = Head(); + } + } + + // Clear everything else out + m_Head = InvalidIndex(); + m_Tail = InvalidIndex(); + m_ElementCount = 0; +} + + +//----------------------------------------------------------------------------- +// list modification +//----------------------------------------------------------------------------- + +template +void CUtlLinkedList::LinkBefore(I before, I elem) +{ + Assert(IsValidIndex(elem)); + + // Unlink it if it's in the list at the moment + Unlink(elem); + + ListElem_t* pNewElem = &InternalElement(elem); + + // The element *after* our newly linked one is the one we linked before. + pNewElem->m_Next = before; + + S newElem_mPrevious; // we need to hang on to this for the compairson against InvalidIndex() + // below; otherwise we get a a load-hit-store on pNewElem->m_Previous, even + // with + if (before == InvalidIndex()) + { + // In this case, we're linking to the end of the list, so reset the tail + newElem_mPrevious = m_Tail; + pNewElem->m_Previous = m_Tail; + m_Tail = elem; + } + else + { + // Here, we're not linking to the end. Set the prev pointer to point to + // the element we're linking. + Assert(IsInList(before)); + ListElem_t* beforeElem = &InternalElement(before); + pNewElem->m_Previous = newElem_mPrevious = beforeElem->m_Previous; + beforeElem->m_Previous = elem; + } + + // Reset the head if we linked to the head of the list + if (newElem_mPrevious == InvalidIndex()) + m_Head = elem; + else + InternalElement(newElem_mPrevious).m_Next = elem; + + // one more element baby + ++m_ElementCount; +} + +template +void CUtlLinkedList::LinkAfter(I after, I elem) +{ + Assert(IsValidIndex(elem)); + + // Unlink it if it's in the list at the moment + if (IsInList(elem)) + Unlink(elem); + + ListElem_t& newElem = InternalElement(elem); + + // The element *before* our newly linked one is the one we linked after + newElem.m_Previous = after; + if (after == InvalidIndex()) + { + // In this case, we're linking to the head of the list, reset the head + newElem.m_Next = m_Head; + m_Head = elem; + } + else + { + // Here, we're not linking to the end. Set the next pointer to point to + // the element we're linking. + Assert(IsInList(after)); + ListElem_t& afterElem = InternalElement(after); + newElem.m_Next = afterElem.m_Next; + afterElem.m_Next = elem; + } + + // Reset the tail if we linked to the tail of the list + if (newElem.m_Next == InvalidIndex()) + m_Tail = elem; + else + InternalElement(newElem.m_Next).m_Previous = elem; + + // one more element baby + ++m_ElementCount; +} + +template +void CUtlLinkedList::Unlink(I elem) +{ + Assert(IsValidIndex(elem)); + if (IsInList(elem)) + { + ListElem_t* pOldElem = &m_Memory[elem]; + + // If we're the first guy, reset the head + // otherwise, make our previous node's next pointer = our next + if (pOldElem->m_Previous != InvalidIndex()) + { + m_Memory[pOldElem->m_Previous].m_Next = pOldElem->m_Next; + } + else + { + m_Head = pOldElem->m_Next; + } + + // If we're the last guy, reset the tail + // otherwise, make our next node's prev pointer = our prev + if (pOldElem->m_Next != InvalidIndex()) + { + m_Memory[pOldElem->m_Next].m_Previous = pOldElem->m_Previous; + } + else + { + m_Tail = pOldElem->m_Previous; + } + + // This marks this node as not in the list, + // but not in the free list either + pOldElem->m_Previous = pOldElem->m_Next = elem; + + // One less puppy + --m_ElementCount; + } +} + +template +inline void CUtlLinkedList::LinkToHead(I elem) +{ + LinkAfter(InvalidIndex(), elem); +} + +template +inline void CUtlLinkedList::LinkToTail(I elem) +{ + LinkBefore(InvalidIndex(), elem); +} + + +//----------------------------------------------------------------------------- +// Class to drop in to replace a CUtlLinkedList that needs to be more memory agressive +//----------------------------------------------------------------------------- + +DECLARE_POINTER_HANDLE(UtlPtrLinkedListIndex_t); // to enforce correct usage + +template < typename T > +class CUtlPtrLinkedList +{ +public: + CUtlPtrLinkedList() + : m_pFirst(NULL), + m_nElems(0) + { + COMPILE_TIME_ASSERT(sizeof(IndexType_t) == sizeof(Node_t*)); + } + + ~CUtlPtrLinkedList() + { + RemoveAll(); + } + + typedef UtlPtrLinkedListIndex_t IndexType_t; + + T& operator[](IndexType_t i) + { + return ((Node_t*)i)->elem; + } + + const T& operator[](IndexType_t i) const + { + return ((Node_t*)i)->elem; + } + + IndexType_t AddToTail() + { + return DoInsertBefore((IndexType_t)m_pFirst, NULL); + } + + IndexType_t AddToTail(T const& src) + { + return DoInsertBefore((IndexType_t)m_pFirst, &src); + } + + IndexType_t AddToHead() + { + IndexType_t result = DoInsertBefore((IndexType_t)m_pFirst, NULL); + m_pFirst = ((Node_t*)result); + return result; + } + + IndexType_t AddToHead(T const& src) + { + IndexType_t result = DoInsertBefore((IndexType_t)m_pFirst, &src); + m_pFirst = ((Node_t*)result); + return result; + } + + IndexType_t InsertBefore(IndexType_t before) + { + return DoInsertBefore(before, NULL); + } + + IndexType_t InsertAfter(IndexType_t after) + { + Node_t* pBefore = ((Node_t*)after)->next; + return DoInsertBefore(pBefore, NULL); + } + + IndexType_t InsertBefore(IndexType_t before, T const& src) + { + return DoInsertBefore(before, &src); + } + + IndexType_t InsertAfter(IndexType_t after, T const& src) + { + Node_t* pBefore = ((Node_t*)after)->next; + return DoInsertBefore(pBefore, &src); + } + + void Remove(IndexType_t elem) + { + Node_t* p = (Node_t*)elem; + + if (p->pNext == p) + { + m_pFirst = NULL; + } + else + { + if (m_pFirst == p) + { + m_pFirst = p->pNext; + } + p->pNext->pPrev = p->pPrev; + p->pPrev->pNext = p->pNext; + } + + delete p; + m_nElems--; + } + + void RemoveAll() + { + Node_t* p = m_pFirst; + if (p) + { + do + { + Node_t* pNext = p->pNext; + delete p; + p = pNext; + } while (p != m_pFirst); + } + + m_pFirst = NULL; + m_nElems = 0; + } + + int Count() const + { + return m_nElems; + } + + IndexType_t Head() const + { + return (IndexType_t)m_pFirst; + } + + IndexType_t Next(IndexType_t i) const + { + Node_t* p = ((Node_t*)i)->pNext; + if (p != m_pFirst) + { + return (IndexType_t)p; + } + return NULL; + } + + bool IsValidIndex(IndexType_t i) const + { + Node_t* p = ((Node_t*)i); + return (p && p->pNext && p->pPrev); + } + + inline static IndexType_t InvalidIndex() + { + return NULL; + } +private: + + struct Node_t + { + Node_t() {} + Node_t(const T& _elem) : elem(_elem) {} + + // Have to do it like this instead of 'new' because we have to use the internal memalloc singleton! + static Node_t* Alloc(const T* _elem) + { + Node_t* pNode = MemAllocSingleton()->Alloc(sizeof(Node_t)); + pNode->elem(_elem); + + return pNode; + } + + T elem; + Node_t* pPrev, * pNext; + }; + + Node_t* AllocNode(const T* pCopyFrom) + { + MEM_ALLOC_CREDIT_CLASS(); + Node_t* p; + + if (!pCopyFrom) + { + p = MemAllocSingleton()->Alloc(sizeof(Node_t)); + } + else + { + p = Node_t::Alloc(*pCopyFrom); + } + + return p; + } + + IndexType_t DoInsertBefore(IndexType_t before, const T* pCopyFrom) + { + Node_t* p = AllocNode(pCopyFrom); + Node_t* pBefore = (Node_t*)before; + if (pBefore) + { + p->pNext = pBefore; + p->pPrev = pBefore->pPrev; + pBefore->pPrev = p; + p->pPrev->pNext = p; + } + else + { + Assert(!m_pFirst); + m_pFirst = p->pNext = p->pPrev = p; + } + + m_nElems++; + return (IndexType_t)p; + } + + Node_t* m_pFirst; + unsigned m_nElems; +}; + +//----------------------------------------------------------------------------- + +#endif // UTLLINKEDLIST_H diff --git a/r5dev/vproj/clientsdk.vcxproj b/r5dev/vproj/clientsdk.vcxproj index ed27751e..ada35d25 100644 --- a/r5dev/vproj/clientsdk.vcxproj +++ b/r5dev/vproj/clientsdk.vcxproj @@ -111,6 +111,7 @@ + @@ -489,6 +490,7 @@ + @@ -508,6 +510,7 @@ + diff --git a/r5dev/vproj/clientsdk.vcxproj.filters b/r5dev/vproj/clientsdk.vcxproj.filters index 8429a0c7..6e64793f 100644 --- a/r5dev/vproj/clientsdk.vcxproj.filters +++ b/r5dev/vproj/clientsdk.vcxproj.filters @@ -573,6 +573,9 @@ sdk\public\utility + + sdk\tier1 + @@ -1703,6 +1706,12 @@ sdk\tier0 + + sdk\tier1 + + + sdk\vstdlib + diff --git a/r5dev/vproj/dedicated.vcxproj b/r5dev/vproj/dedicated.vcxproj index 36f1ec7b..f36c9d31 100644 --- a/r5dev/vproj/dedicated.vcxproj +++ b/r5dev/vproj/dedicated.vcxproj @@ -449,6 +449,7 @@ + @@ -461,6 +462,7 @@ + @@ -560,6 +562,7 @@ + diff --git a/r5dev/vproj/dedicated.vcxproj.filters b/r5dev/vproj/dedicated.vcxproj.filters index 920e59f5..7ac72b64 100644 --- a/r5dev/vproj/dedicated.vcxproj.filters +++ b/r5dev/vproj/dedicated.vcxproj.filters @@ -1224,6 +1224,12 @@ sdk\tier0 + + sdk\tier1 + + + sdk\vstdlib + @@ -1529,6 +1535,9 @@ sdk\networksystem + + sdk\tier1 + diff --git a/r5dev/vproj/gamesdk.vcxproj b/r5dev/vproj/gamesdk.vcxproj index 5db4516b..19aae70c 100644 --- a/r5dev/vproj/gamesdk.vcxproj +++ b/r5dev/vproj/gamesdk.vcxproj @@ -121,6 +121,7 @@ + @@ -515,6 +516,7 @@ + @@ -534,6 +536,7 @@ + diff --git a/r5dev/vproj/gamesdk.vcxproj.filters b/r5dev/vproj/gamesdk.vcxproj.filters index 37fb95fc..0371742e 100644 --- a/r5dev/vproj/gamesdk.vcxproj.filters +++ b/r5dev/vproj/gamesdk.vcxproj.filters @@ -612,6 +612,9 @@ sdk\networksystem + + sdk\tier1 + @@ -1790,6 +1793,12 @@ sdk\tier0 + + sdk\vstdlib + + + sdk\tier1 + diff --git a/r5dev/vstdlib/concommandhash.h b/r5dev/vstdlib/concommandhash.h new file mode 100644 index 00000000..b39ce2c1 --- /dev/null +++ b/r5dev/vstdlib/concommandhash.h @@ -0,0 +1,214 @@ +//===== Copyright � 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: Special case hash table for console commands +// +// $NoKeywords: $ +// +//===========================================================================// + +#if !defined( CONCOMMANDHASH_H ) +#define CONCOMMANDHASH_H +#ifdef _WIN32 +#pragma once +#endif + +#include "tier1/cmd.h" +#include "tier1/utlvector.h" +#include "tier1/utllinkedlist.h" +#include "tier1/generichash.h" + +// This is a hash table class very similar to the CUtlHashFast, but +// modified specifically so that we can look up ConCommandBases +// by string names without having to actually store those strings in +// the dictionary, and also iterate over all of them. +// It uses separate chaining: each key hashes to a bucket, each +// bucket is a linked list of hashed commands. We store the hash of +// the command's string name as well as its pointer, so we can do +// the linked list march part of the Find() operation more quickly. +class CConCommandHash +{ +public: + typedef int64_t CCommandHashHandle_t; // confirmed 64-bit in r5 see [r5apex_ds.exe+0x597062] + typedef unsigned int HashKey_t; + + // Constructor/Deconstructor. + CConCommandHash(); + ~CConCommandHash(); + + // Memory. + void Purge(bool bReinitialize); + + // Invalid handle. + static CCommandHashHandle_t InvalidHandle(void) { return (CCommandHashHandle_t)~0; } + inline bool IsValidHandle(CCommandHashHandle_t hHash) const; + + /// Initialize. + void Init(void); // bucket count is hardcoded in enum below. + + /// Get hash value for a concommand + static inline HashKey_t Hash(const ConCommandBase* cmd); + + // Size not available; count is meaningless for multilists. + // int Count( void ) const; + + // Insertion. + CCommandHashHandle_t Insert(ConCommandBase* cmd); + CCommandHashHandle_t FastInsert(ConCommandBase* cmd); + + // Removal. + void Remove(CCommandHashHandle_t hHash); + void RemoveAll(void); + + // Retrieval. + inline CCommandHashHandle_t Find(const char* name) const; + CCommandHashHandle_t Find(const ConCommandBase* cmd) const; + // A convenience version of Find that skips the handle part + // and returns a pointer to a concommand, or NULL if none was found. + inline ConCommandBase* FindPtr(const char* name) const; + + inline ConCommandBase*& operator[](CCommandHashHandle_t hHash); + inline ConCommandBase* const& operator[](CCommandHashHandle_t hHash) const; + +//#ifdef _DEBUG + // Dump a report to MSG + void Report(void); +//#endif + + // Iteration + struct CCommandHashIterator_t + { + int bucket; + CCommandHashHandle_t handle; + + CCommandHashIterator_t(int _bucket, const CCommandHashHandle_t& _handle) + : bucket(_bucket), handle(_handle) {}; + // inline operator UtlHashFastHandle_t() const { return handle; }; + }; + inline CCommandHashIterator_t First() const; + inline CCommandHashIterator_t Next(const CCommandHashIterator_t& hHash) const; + inline bool IsValidIterator(const CCommandHashIterator_t& iter) const; + inline ConCommandBase*& operator[](const CCommandHashIterator_t& iter) { return (*this)[iter.handle]; } + inline ConCommandBase* const& operator[](const CCommandHashIterator_t& iter) const { return (*this)[iter.handle]; } +private: + // a find func where we've already computed the hash for the string. + // (hidden private in case we decide to invent a custom string hash func + // for this class) + CCommandHashHandle_t Find(const char* name, HashKey_t hash) const; + +protected: + enum + { + kNUM_BUCKETS = 256, + kBUCKETMASK = kNUM_BUCKETS - 1, + }; + + struct HashEntry_t + { + HashKey_t m_uiKey; + ConCommandBase* m_Data; + + HashEntry_t(unsigned int _hash, ConCommandBase* _cmd) + : m_uiKey(_hash), m_Data(_cmd) {}; + + HashEntry_t() {}; + }; + + typedef CUtlFixedLinkedList datapool_t; + + CUtlVector m_aBuckets; + datapool_t m_aDataPool; +}; + +inline bool CConCommandHash::IsValidHandle(CCommandHashHandle_t hHash) const +{ + return m_aDataPool.IsValidIndex(hHash); +} + + +inline CConCommandHash::CCommandHashHandle_t CConCommandHash::Find(const char* name) const +{ + return Find(name, HashStringCaseless(name)); +} + +inline ConCommandBase*& CConCommandHash::operator[](CCommandHashHandle_t hHash) +{ + return (m_aDataPool[hHash].m_Data); +} + +inline ConCommandBase* const& CConCommandHash::operator[](CCommandHashHandle_t hHash) const +{ + return (m_aDataPool[hHash].m_Data); +} + +//----------------------------------------------------------------------------- +// Purpose: For iterating over the whole hash, return the index of the first element +//----------------------------------------------------------------------------- +CConCommandHash::CCommandHashIterator_t CConCommandHash::First() const +{ + // walk through the buckets to find the first one that has some data + int bucketCount = m_aBuckets.Count(); + const CCommandHashHandle_t invalidIndex = m_aDataPool.InvalidIndex(); + for (int bucket = 0; bucket < bucketCount; ++bucket) + { + CCommandHashHandle_t iElement = m_aBuckets[bucket]; // get the head of the bucket + if (iElement != invalidIndex) + return CCommandHashIterator_t(bucket, iElement); + } + + // if we are down here, the list is empty + return CCommandHashIterator_t(-1, invalidIndex); +} + +//----------------------------------------------------------------------------- +// Purpose: For iterating over the whole hash, return the next element after +// the param one. Or an invalid iterator. +//----------------------------------------------------------------------------- +CConCommandHash::CCommandHashIterator_t +CConCommandHash::Next(const CConCommandHash::CCommandHashIterator_t& iter) const +{ + // look for the next entry in the current bucket + CCommandHashHandle_t next = m_aDataPool.Next(iter.handle); + const CCommandHashHandle_t invalidIndex = m_aDataPool.InvalidIndex(); + if (next != invalidIndex) + { + // this bucket still has more elements in it + return CCommandHashIterator_t(iter.bucket, next); + } + + // otherwise look for the next bucket with data + int bucketCount = m_aBuckets.Count(); + for (int bucket = iter.bucket + 1; bucket < bucketCount; ++bucket) + { + CCommandHashHandle_t next = m_aBuckets[bucket]; // get the head of the bucket + if (next != invalidIndex) + return CCommandHashIterator_t(bucket, next); + } + + // if we're here, there's no more data to be had + return CCommandHashIterator_t(-1, invalidIndex); +} + +bool CConCommandHash::IsValidIterator(const CCommandHashIterator_t& iter) const +{ + return ((iter.bucket >= 0) && (m_aDataPool.IsValidIndex(iter.handle))); +} + +inline CConCommandHash::HashKey_t CConCommandHash::Hash(const ConCommandBase* cmd) +{ + return HashStringCaseless(cmd->GetName()); +} + +inline ConCommandBase* CConCommandHash::FindPtr(const char* name) const +{ + CCommandHashHandle_t handle = Find(name); + if (handle == InvalidHandle()) + { + return NULL; + } + else + { + return (*this)[handle]; + } +} + +#endif