diff --git a/r5dev/common/qlimits.h b/r5dev/common/qlimits.h new file mode 100644 index 00000000..31beb3db --- /dev/null +++ b/r5dev/common/qlimits.h @@ -0,0 +1,36 @@ +//====== Copyright � 1996-2005, Valve Corporation, All rights reserved. =======// +// +// Purpose: +// +// $NoKeywords: $ +//=============================================================================// + +#ifndef QLIMITS_H +#define QLIMITS_H + +#if defined( _WIN32 ) +#pragma once +#endif + +// DATA STRUCTURE INFO + +#define MAX_NUM_ARGVS 50 + +// SYSTEM INFO +#define MAX_QPATH 96 // max length of a game pathname +#define MAX_OSPATH 260 // max length of a filesystem pathname + +#define ON_EPSILON 0.1 // point on plane side epsilon + + +// Resource counts; +// Must have this value in sync(-1) with const.h and effect_dispatch_data.cpp +#define MAX_MODEL_INDEX_BITS 12 // sent as a short +#define MAX_MODELS (1<(&*m_MDLMutex)); + v6 = *(_QWORD*)(m_MDLDict.Deref().GetPtr() + 24 * v4 + 16); + LeaveCriticalSection(reinterpret_cast(&*m_MDLMutex)); + + if (!g_pMDLFallback->m_hErrorMDL || !g_pMDLFallback->m_hEmptyMDL) + { + studiohdr_t* pStudioHDR = **(studiohdr_t***)v6; + string svStudio = ConvertToUnixPath(string(pStudioHDR->name)); + + if (strcmp(svStudio.c_str(), ERROR_MODEL) == 0) + { + g_pMDLFallback->m_pErrorHDR = pStudioHDR; + g_pMDLFallback->m_hErrorMDL = handle; + } + if (strcmp(svStudio.c_str(), EMPTY_MODEL) == 0) + { + g_pMDLFallback->m_pEmptyHDR = pStudioHDR; + g_pMDLFallback->m_hEmptyMDL = handle; + } + } + + if (!v6) + { + if (!g_pMDLFallback->m_hErrorMDL) + Error(eDLL_T::ENGINE, "Model with handle \"%hu\" not found and \"%s\" couldn't be loaded.\n", handle, ERROR_MODEL); + + return g_pMDLFallback->m_pErrorHDR; + } + + if ((*(_WORD*)(v6 + 18) & 0x600) != 0) + { + v7 = *(__int64**)v6; + if (*(_QWORD*)v6) + { + if (a3) + { + CMDLCache::FindCachedMDL(pMDLCache, (void*)v6, a3); + v7 = *(__int64**)v6; + } + LABEL_6: + result = (studiohdr_t*)*v7; + if (result) + return result; + + return CMDLCache::FindUncachedMDL(pMDLCache, v4, (void*)v6, a3); + } + v7 = *(__int64**)(v6 + 8); + if (v7) + goto LABEL_6; + } + return CMDLCache::FindUncachedMDL(pMDLCache, v4, (void*)v6, a3); +} + +//----------------------------------------------------------------------------- +// Purpose: finds an MDL cached +// Input : *this - +// *a2 - +// *a3 - +//----------------------------------------------------------------------------- +void CMDLCache::FindCachedMDL(CMDLCache* pMDLCache, void* a2, void* a3) +{ + v_CMDLCache__FindCachedMDL(pMDLCache, a2, a3); +} + +//----------------------------------------------------------------------------- +// Purpose: finds an MDL uncached +// Input : *this - +// handle - +// *a3 - +// *a4 +// Output : a pointer to the studiohdr_t object +//----------------------------------------------------------------------------- +studiohdr_t* CMDLCache::FindUncachedMDL(CMDLCache* cache, MDLHandle_t handle, void* a3, void* a4) +{ + const char* v8; // rdi + __int64 v9; // rax + int v10; // eax + const char* v11; // r11 + int v12; // eax + const char* v13; // r11 + int v14; // eax + bool v16; // zf + studiohdr_t* v17; // rdi + studiohdr_t** v18; // rax + + //CThreadFastMutexSlow::WaitForLock(a3 + 0x80); + EnterCriticalSection(reinterpret_cast(&*m_MDLMutex)); + void* modelCache = cache->m_pModelCacheSection; + v8 = (const char*)(*(_QWORD*)((int64)modelCache + 24 * static_cast(handle) + 8)); + LeaveCriticalSection(reinterpret_cast(&*m_MDLMutex)); + v9 = -1i64; + do + ++v9; + while (v8[v9]); + + if (v9 < 5 || + (_stricmp(&v8[v9 - 5], ".rmdl") != 0) && + (_stricmp(&v8[v9 - 5], ".rrig") != 0) && + (_stricmp(&v8[v9 - 5], ".rpak") != 0)) + { + Error(eDLL_T::ENGINE, "Attempted to load old model \"%s\"; replace with rmdl.\n", v8); + goto LABEL_ERROR; + } + + g_pRTech->StringToGuid(v8); + v16 = *(_QWORD*)a3 == 0i64; + *(_BYTE*)((int64)a3 + 152) = 0; + if (v16) + { + v18 = *(studiohdr_t***)((int64)a3 + 8); + if (v18) + { + v17 = *v18; + } + else + { + LABEL_ERROR: + if (g_pMDLFallback->m_hErrorMDL) + Error(eDLL_T::ENGINE, "Model \"%s\" not found; replacing with \"%s\".\n", v8, ERROR_MODEL); + else + Error(eDLL_T::ENGINE, "Model \"%s\" not found and \"%s\" couldn't be loaded.\n", v8, ERROR_MODEL); + v17 = g_pMDLFallback->m_pErrorHDR; + } + } + else + { + v_CMDLCache__FindCachedMDL(cache, a3, a4); + v17 = **(studiohdr_t***)a3; + } + //CThreadFastMutexSlow::ReleaseWaiter(a3 + 128); + return v17; +} + +studiohdr_t* CMDLCache::GetStudioHdr(CMDLCache* pMDLCache, MDLHandle_t handle) +{ + __int64 v2; // rbx + __int64 v3; // rbx + __int64 v4; // rdx + studiohdr_t* result; // rax + + if (!handle) + { + if (!g_pMDLFallback->m_hErrorMDL) + Error(eDLL_T::ENGINE, "Model with handle \"%hu\" not found and \"%s\" couldn't be loaded.\n", handle, ERROR_MODEL); + + return g_pMDLFallback->m_pErrorHDR; + } + + v2 = handle; + EnterCriticalSection(reinterpret_cast(&*m_MDLMutex)); + v3 = *(_QWORD*)(m_MDLDict.Deref().GetPtr() + 24 * v2 + 16); + LeaveCriticalSection(reinterpret_cast(&*m_MDLMutex)); + v4 = *(_QWORD*)(*(_QWORD*)(*(_QWORD*)v3 + 8i64) + 24i64); + result = (studiohdr_t*)(v4 + 16); + if (!v4) + result = nullptr;; + return result; +} + +void MDLCache_Attach() +{ + DetourAttach((LPVOID*)&v_CMDLCache__FindMDL, &CMDLCache::FindMDL); + DetourAttach((LPVOID*)&v_CMDLCache__FindUncachedMDL, &CMDLCache::FindUncachedMDL); + //DetourAttach((LPVOID*)&v_CMDLCache__GetStudioHdr, &CMDLCache::GetStudioHdr); +} + +void MDLCache_Detach() +{ + DetourDetach((LPVOID*)&v_CMDLCache__FindMDL, &CMDLCache::FindMDL); + DetourDetach((LPVOID*)&v_CMDLCache__FindUncachedMDL, &CMDLCache::FindUncachedMDL); + //DetourDetach((LPVOID*)&v_CMDLCache__GetStudioHdr, &CMDLCache::GetStudioHdr); +} \ No newline at end of file diff --git a/r5dev/datacache/mdlcache.h b/r5dev/datacache/mdlcache.h new file mode 100644 index 00000000..5069e9fd --- /dev/null +++ b/r5dev/datacache/mdlcache.h @@ -0,0 +1,106 @@ +#ifndef MDLCACHE_H +#define MDLCACHE_H +#include "public/include/studio.h" +#include "datacache/imdlcache.h" + +struct CMDLFallBack +{ + studiohdr_t* m_pErrorHDR{}; + MDLHandle_t m_hErrorMDL{}; + studiohdr_t* m_pEmptyHDR{}; + MDLHandle_t m_hEmptyMDL{}; + + // This has to be cleared if 'common.rpak' is getting unloaded! + void Clear(void) + { + m_pEmptyHDR = nullptr; + m_hErrorMDL = NULL; + m_pEmptyHDR = nullptr; + m_hEmptyMDL = NULL; + } +}; +inline CMDLFallBack* g_pMDLFallback = new CMDLFallBack(); + +class CMDLCache +{ +public: + static studiohdr_t* FindMDL(CMDLCache* pMDLCache, MDLHandle_t handle, void* a3); + static void FindCachedMDL(CMDLCache* pMDLCache, void* a2, void* a3); + static studiohdr_t* FindUncachedMDL(CMDLCache* pMDLCache, MDLHandle_t handle, void* a3, void* a4); + static studiohdr_t* GetStudioHdr(CMDLCache* pMDLCache, MDLHandle_t handle); + + CMDLCache* m_pVTable; + void* m_pStrCmp; // string compare func; + void* m_pModelCacheSection; // IDataCacheSection* + int m_nModelCacheFrameLocks; // + // TODO.. +}; + +extern studiohdr_t* pErrorStudioHDR; +extern MDLHandle_t hErrorMDL; + +inline CMemory p_CMDLCache__FindMDL; +inline auto v_CMDLCache__FindMDL = p_CMDLCache__FindMDL.RCast(); + +inline CMemory p_CMDLCache__FindCachedMDL; +inline auto v_CMDLCache__FindCachedMDL = p_CMDLCache__FindCachedMDL.RCast(); + +inline CMemory p_CMDLCache__FindUncachedMDL; +inline auto v_CMDLCache__FindUncachedMDL = p_CMDLCache__FindUncachedMDL.RCast(); + +inline CMemory p_CMDLCache__GetStudioHdr; +inline auto v_CMDLCache__GetStudioHdr = p_CMDLCache__GetStudioHdr.RCast(); + +inline CMemory m_MDLDict; +inline LPCRITICAL_SECTION* m_MDLMutex = nullptr; +inline CMDLCache* g_MDLCache = nullptr; + + +void MDLCache_Attach(); +void MDLCache_Detach(); +/////////////////////////////////////////////////////////////////////////////// +class HMDLCache : public IDetour +{ + virtual void GetAdr(void) const + { + std::cout << "| FUN: CMDLCache::FindMDL : 0x" << std::hex << std::uppercase << p_CMDLCache__FindMDL.GetPtr() << std::setw(nPad) << " |" << std::endl; + std::cout << "| FUN: CMDLCache::FindCachedMDL : 0x" << std::hex << std::uppercase << p_CMDLCache__FindCachedMDL.GetPtr() << std::setw(nPad) << " |" << std::endl; + std::cout << "| FUN: CMDLCache::FindUncachedMDL : 0x" << std::hex << std::uppercase << p_CMDLCache__FindUncachedMDL.GetPtr() << std::setw(nPad) << " |" << std::endl; + std::cout << "| FUN: CMDLCache::GetStudioHdr : 0x" << std::hex << std::uppercase << p_CMDLCache__GetStudioHdr.GetPtr() << std::setw(nPad) << " |" << std::endl; + std::cout << "| VAR: m_MDLMutex : 0x" << std::hex << std::uppercase << m_MDLMutex << std::setw(nPad) << " |" << std::endl; + std::cout << "| VAR: m_MDLDict : 0x" << std::hex << std::uppercase << m_MDLDict.GetPtr() << std::setw(nPad) << " |" << std::endl; + std::cout << "| VAR: g_MDLCache : 0x" << std::hex << std::uppercase << g_MDLCache << std::setw(0) << " |" << std::endl; + std::cout << "+----------------------------------------------------------------+" << std::endl; + } + virtual void GetFun(void) const + { + p_CMDLCache__FindMDL = g_mGameDll.FindPatternSIMD(reinterpret_cast("\x48\x89\x5C\x24\x00\x48\x89\x6C\x24\x00\x48\x89\x74\x24\x00\x57\x48\x83\xEC\x20\x48\x8B\xF1\x0F\xB7\xEA"), "xxxx?xxxx?xxxx?xxxxxxxxxxx"); + v_CMDLCache__FindMDL = p_CMDLCache__FindMDL.RCast(); /*48 89 5C 24 ? 48 89 6C 24 ? 48 89 74 24 ? 57 48 83 EC 20 48 8B F1 0F B7 EA*/ + + p_CMDLCache__FindCachedMDL = g_mGameDll.FindPatternSIMD(reinterpret_cast("\x4D\x85\xC0\x74\x7A\x48\x89\x6C\x24\x00"), "xxxxxxxxx?"); + v_CMDLCache__FindCachedMDL = p_CMDLCache__FindCachedMDL.RCast(); /*4D 85 C0 74 7A 48 89 6C 24 ?*/ + + p_CMDLCache__FindUncachedMDL = g_mGameDll.FindPatternSIMD(reinterpret_cast("\x48\x89\x5C\x24\x00\x48\x89\x6C\x24\x00\x48\x89\x74\x24\x00\x48\x89\x7C\x24\x00\x41\x56\x48\x83\xEC\x20\x48\x8B\xE9\x0F\xB7\xFA"), "xxxx?xxxx?xxxx?xxxx?xxxxxxxxxxxx"); + v_CMDLCache__FindUncachedMDL = p_CMDLCache__FindUncachedMDL.RCast(); + + p_CMDLCache__GetStudioHdr = g_mGameDll.FindPatternSIMD(reinterpret_cast("\x40\x53\x48\x83\xEC\x20\x48\x8D\x0D\x00\x00\x00\x00\x0F\xB7\xDA\xFF\x15\x00\x00\x00\x00\x48\x8B\x05\x00\x00\x00\x00\x48\x8D\x14\x5B\x48\x8D\x0D\x00\x00\x00\x00\x48\x8B\x5C\xD0\x00\xFF\x15\x00\x00\x00\x00\x48\x8B\x03\x48\x8B\x48\x08"), "xxxxxxxxx????xxxxx????xxx????xxxxxxx????xxxx?xx????xxxxxxx");; + v_CMDLCache__GetStudioHdr = p_CMDLCache__GetStudioHdr.RCast(); + } + virtual void GetVar(void) const + { + m_MDLMutex = g_mGameDll.FindPatternSIMD(reinterpret_cast("\x48\x83\xEC\x28\xBA\x00\x00\x00\x00\x48\x8D\x0D\x00\x00\x00\x00\xFF\x15\x00\x00\x00\x00\x0F\xB6\x05\x00\x00\x00\x00"), "xxxxx????xxx????xx????xxx????") + .FindPatternSelf("48 8D 0D", CMemory::Direction::DOWN).ResolveRelativeAddressSelf(0x3, 0x7).RCast(); + + m_MDLDict = p_CMDLCache__FindMDL.FindPattern("48 8B 05", CMemory::Direction::DOWN).ResolveRelativeAddressSelf(0x3, 0x7); + + g_MDLCache = g_mGameDll.FindPatternSIMD(reinterpret_cast("\x48\x8B\x0D\x00\x00\x00\x00\x44\x0F\xB7\x82\x00\x00\x00\x00\x48\x8B\x01\x48\xFF\xA0\x30\x01\x00\x00"), "xxx????xxxx????xxxxxxxxxx") + .ResolveRelativeAddressSelf(0x3, 0x7).RCast(); + } + virtual void GetCon(void) const { } + virtual void Attach(void) const { } + virtual void Detach(void) const { } +}; +/////////////////////////////////////////////////////////////////////////////// + +REGISTER(HMDLCache); +#endif // MDLCACHE_H diff --git a/r5dev/engine/gl_model_private.h b/r5dev/engine/gl_model_private.h new file mode 100644 index 00000000..d9e5b5fc --- /dev/null +++ b/r5dev/engine/gl_model_private.h @@ -0,0 +1,69 @@ +//=============================================================================// +// +// Purpose: +// +// $Workfile: $ +// $Date: $ +// $NoKeywords: $ +//=============================================================================// +#include "vpc/keyvalues.h" +#include "mathlib/vector.h" +#include "common/qlimits.h" +#include "datacache/imdlcache.h" +#include "public/include/model_types.h" + +#ifndef DEDICATED +#include "game/client/enginesprite.h" +#endif // !DEDICATED +typedef int FileNameHandle_t; // 4 bytes in r5, void* originally. + +struct brushdata_t // !! UNCONFIRMED !! +{ + void* pShared; // worldbrushdata_t + int firstmodelsurface; + int nummodelsurfaces; + + // track the union of all lightstyles on this brush. That way we can avoid + // searching all faces if the lightstyle hasn't changed since the last update + int nLightstyleLastComputedFrame; + unsigned short nLightstyleIndex; // g_ModelLoader actually holds the allocated data here + unsigned short nLightstyleCount; + + unsigned short renderHandle; + unsigned short firstnode; +}; + +#ifndef DEDICATED +struct spritedata_t // !! UNCONFIRMED !! +{ + int numframes; + int width; + int height; + CEngineSprite* sprite; +}; +#endif // !DEDICATED + +struct model_t // !! CONFIRMED !! +{ + FileNameHandle_t fnHandle; + char szPathName[MAX_OSPATH]; + + int nLoadFlags; // mark loaded/not loaded + int nServerCount; // marked at load + + modtype_t type; + int flags; // MODELFLAG_??? + + // volume occupied by the model graphics + Vector3 mins, maxs; + float radius; + KeyValues* m_pKeyValues; + union + { + brushdata_t brush; + MDLHandle_t studio; +#ifndef DEDICATED + spritedata_t sprite; +#endif // !DEDICATED + }; +}; \ No newline at end of file diff --git a/r5dev/engine/host_state.cpp b/r5dev/engine/host_state.cpp index a2db235b..684fbc48 100644 --- a/r5dev/engine/host_state.cpp +++ b/r5dev/engine/host_state.cpp @@ -12,6 +12,7 @@ #include "tier1/NetAdr2.h" #include "tier2/socketcreator.h" #include "vpc/keyvalues.h" +#include "datacache/mdlcache.h" #ifdef DEDICATED #include "engine/sv_rcon.h" #else // diff --git a/r5dev/engine/modelloader.cpp b/r5dev/engine/modelloader.cpp index 6eded16a..9f451ee6 100644 --- a/r5dev/engine/modelloader.cpp +++ b/r5dev/engine/modelloader.cpp @@ -8,19 +8,63 @@ #include "core/stdafx.h" #include "engine/cmodel_bsp.h" #include "engine/modelloader.h" +#include "datacache/mdlcache.h" +#include "engine/sys_utils.h" -uint64_t HCModelLoader__Map_LoadModelGuts(void* thisptr, void* mod) +model_t* pErrorMDL = nullptr; + +//----------------------------------------------------------------------------- +// Purpose: +// Input : *loader - +// *model - +//----------------------------------------------------------------------------- +void CModelLoader::LoadModel(CModelLoader* loader, model_t* model) { - return CModelLoader__Map_LoadModelGuts(thisptr, mod); + //if (!pErrorMDL) + //{ + // if (strcmp(model->szPathName, ERROR_MODEL) == 0) + // { + // pErrorMDL = model; + // } + //} + + //string svExtension = model->szPathName; + //size_t npos = svExtension.find("."); + //if (npos != string::npos) + //{ + // svExtension = svExtension.substr(npos + 1); + //} + + //if (strcmp(svExtension.c_str(), "rmdl") == 0 && strcmp(model->szPathName, ERROR_MODEL) != 0) + //{ + // studiohdr_t* pStudioHDR = g_MDLCache->FindMDL(g_MDLCache->m_pVTable, model->studio, 0); + // if (pStudioHDR == pErrorStudioHDR) + // { + // model = pErrorMDL; + // } + //} + return CModelLoader__LoadModel(loader, model); +} + +//----------------------------------------------------------------------------- +// Purpose: +// Input : *loader - +// *model - +//----------------------------------------------------------------------------- +uint64_t CModelLoader::Map_LoadModelGuts(CModelLoader* loader, model_t* model) +{ + return CModelLoader__Map_LoadModelGuts(loader, model); } /////////////////////////////////////////////////////////////////////////////// void CModelLoader_Attach() { - DetourAttach((LPVOID*)&CModelLoader__Map_LoadModelGuts, &HCModelLoader__Map_LoadModelGuts); + DetourAttach((LPVOID*)&CModelLoader__LoadModel, &CModelLoader::LoadModel); + DetourAttach((LPVOID*)&CModelLoader__Map_LoadModelGuts, &CModelLoader::Map_LoadModelGuts); } void CModelLoader_Detach() { - DetourDetach((LPVOID*)&CModelLoader__Map_LoadModelGuts, &HCModelLoader__Map_LoadModelGuts); + DetourDetach((LPVOID*)&CModelLoader__LoadModel, &CModelLoader::LoadModel); + DetourDetach((LPVOID*)&CModelLoader__Map_LoadModelGuts, &CModelLoader::Map_LoadModelGuts); } \ No newline at end of file diff --git a/r5dev/engine/modelloader.h b/r5dev/engine/modelloader.h index 3477d623..0b413df7 100644 --- a/r5dev/engine/modelloader.h +++ b/r5dev/engine/modelloader.h @@ -1,22 +1,72 @@ #pragma once +#include "engine/gl_model_private.h" + +//----------------------------------------------------------------------------- +// Purpose: +//----------------------------------------------------------------------------- +class IModelLoader +{ +public: + enum REFERENCETYPE + { + // The name is allocated, but nothing else is in memory or being referenced + FMODELLOADER_NOTLOADEDORREFERENCED = 0, + // The model has been loaded into memory + FMODELLOADER_LOADED = (1 << 0), + + // The model is being referenced by the server code + FMODELLOADER_SERVER = (1 << 1), + // The model is being referenced by the client code + FMODELLOADER_CLIENT = (1 << 2), + // The model is being referenced in the client .dll + FMODELLOADER_CLIENTDLL = (1 << 3), + // The model is being referenced by static props + FMODELLOADER_STATICPROP = (1 << 4), + // The model is a detail prop + FMODELLOADER_DETAILPROP = (1 << 5), + // The model is the simple version of the world geometry + FMODELLOADER_SIMPLEWORLD = (1 << 6), + // The model is dynamically loaded + FMODELLOADER_DYNSERVER = (1 << 7), + FMODELLOADER_DYNCLIENT = (1 << 8), + FMODELLOADER_COMBINED = (1 << 9), + FMODELLOADER_DYNAMIC = FMODELLOADER_DYNSERVER | FMODELLOADER_DYNCLIENT | FMODELLOADER_COMBINED, + + FMODELLOADER_REFERENCEMASK = (FMODELLOADER_SERVER | FMODELLOADER_CLIENT | FMODELLOADER_CLIENTDLL | FMODELLOADER_STATICPROP | FMODELLOADER_DETAILPROP | FMODELLOADER_DYNAMIC | FMODELLOADER_SIMPLEWORLD), + + // The model was touched by the preload method + FMODELLOADER_TOUCHED_BY_PRELOAD = (1 << 15), + // The model was loaded by the preload method, a postload fixup is required + FMODELLOADER_LOADED_BY_PRELOAD = (1 << 16), + // The model touched its materials as part of its load + FMODELLOADER_TOUCHED_MATERIALS = (1 << 17), + }; +}; + +class CModelLoader +{ +public: + static void LoadModel(CModelLoader* loader, model_t* model); + static uint64_t Map_LoadModelGuts(CModelLoader* loader, model_t* model); +}; inline CMemory p_CModelLoader__FindModel; -inline auto CModelLoader__FindModel = p_CModelLoader__FindModel.RCast(); +inline auto CModelLoader__FindModel = p_CModelLoader__FindModel.RCast(); inline CMemory p_CModelLoader__LoadModel; -inline auto CModelLoader__LoadModel = p_CModelLoader__LoadModel.RCast(); +inline auto CModelLoader__LoadModel = p_CModelLoader__LoadModel.RCast(); inline CMemory p_CModelLoader__UnloadModel; -inline auto CModelLoader__UnloadModel = p_CModelLoader__UnloadModel.RCast(); +inline auto CModelLoader__UnloadModel = p_CModelLoader__UnloadModel.RCast(); inline CMemory p_CModelLoader__Studio_LoadModel; -inline auto CModelLoader__Studio_LoadModel = p_CModelLoader__Studio_LoadModel.RCast(); +inline auto CModelLoader__Studio_LoadModel = p_CModelLoader__Studio_LoadModel.RCast(); inline CMemory p_CModelLoader__Map_LoadModelGuts; -inline auto CModelLoader__Map_LoadModelGuts = p_CModelLoader__Map_LoadModelGuts.RCast(); +inline auto CModelLoader__Map_LoadModelGuts = p_CModelLoader__Map_LoadModelGuts.RCast(); inline CMemory p_CModelLoader__Map_IsValid; -inline auto CModelLoader__Map_IsValid = p_CModelLoader__Map_IsValid.RCast(); +inline auto CModelLoader__Map_IsValid = p_CModelLoader__Map_IsValid.RCast(); inline CMemory p_GetSpriteInfo; inline auto GetSpriteInfo = p_GetSpriteInfo.RCast(); @@ -24,7 +74,7 @@ inline auto GetSpriteInfo = p_GetSpriteInfo.RCast(); -inline void* g_pModelLoader; +inline CModelLoader* g_pModelLoader; void CModelLoader_Attach(); void CModelLoader_Detach(); @@ -66,12 +116,12 @@ class HModelLoader : public IDetour #endif p_BuildSpriteLoadName = g_mGameDll.FindPatternSIMD(reinterpret_cast("\x48\x89\x5C\x24\x00\x48\x89\x6C\x24\x00\x48\x89\x74\x24\x00\x48\x89\x7C\x24\x00\x41\x56\x48\x81\xEC\x00\x00\x00\x00\x4D\x8B\xF1\x48\x8B\xF2"), "xxxx?xxxx?xxxx?xxxx?xxxxx????xxxxxx"); - CModelLoader__FindModel = p_CModelLoader__FindModel.RCast(); - CModelLoader__LoadModel = p_CModelLoader__LoadModel.RCast(); - CModelLoader__UnloadModel = p_CModelLoader__UnloadModel.RCast(); - CModelLoader__Studio_LoadModel = p_CModelLoader__Studio_LoadModel.RCast(); - CModelLoader__Map_LoadModelGuts = p_CModelLoader__Map_LoadModelGuts.RCast(); - CModelLoader__Map_IsValid = p_CModelLoader__Map_IsValid.RCast(); + CModelLoader__FindModel = p_CModelLoader__FindModel.RCast(); + CModelLoader__LoadModel = p_CModelLoader__LoadModel.RCast(); + CModelLoader__UnloadModel = p_CModelLoader__UnloadModel.RCast(); + CModelLoader__Studio_LoadModel = p_CModelLoader__Studio_LoadModel.RCast(); + CModelLoader__Map_LoadModelGuts = p_CModelLoader__Map_LoadModelGuts.RCast(); + CModelLoader__Map_IsValid = p_CModelLoader__Map_IsValid.RCast(); GetSpriteInfo = p_GetSpriteInfo.RCast(); BuildSpriteLoadName = p_BuildSpriteLoadName.RCast(); } @@ -79,7 +129,7 @@ class HModelLoader : public IDetour { g_pModelLoader = g_mGameDll.FindPatternSIMD( reinterpret_cast("\x48\x89\x4C\x24\x00\x53\x55\x56\x41\x54\x41\x55\x41\x56\x41\x57\x48\x81\xEC\x00\x00\x00\x00"), - "xxxx?xxxxxxxxxxxxxx????").FindPatternSelf("48 ?? 0D", CMemory::Direction::DOWN).ResolveRelativeAddressSelf(3, 7); + "xxxx?xxxxxxxxxxxxxx????").FindPatternSelf("48 ?? 0D", CMemory::Direction::DOWN).ResolveRelativeAddressSelf(3, 7).RCast(); } virtual void GetCon(void) const { } virtual void Attach(void) const { } diff --git a/r5dev/game/client/enginesprite.h b/r5dev/game/client/enginesprite.h new file mode 100644 index 00000000..2909e5f8 --- /dev/null +++ b/r5dev/game/client/enginesprite.h @@ -0,0 +1,58 @@ +//===== Copyright � 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: +// +// $NoKeywords: $ +//===========================================================================// +#ifndef ENGINESPRITE_H +#define ENGINESPRITE_H +#include "public/avi/iavi.h" +#include "public/avi/ibik.h" +#include "public/include/const.h" +#include "game/client/hud.h" + +typedef void* IMaterial; // HACK + +//----------------------------------------------------------------------------- +// Purpose: Sprite Models +//----------------------------------------------------------------------------- +class CEngineSprite // !! UNCONFIRMED !! +{ + // NOTE: don't define a constructor or destructor so that this can be allocated + // as before. +public: + int GetWidth(void) const { return m_width; } + int GetHeight(void) const { return m_height; } + int GetNumFrames(void) const { return m_numFrames; } + //IMaterial* GetMaterial(RenderMode_t nRenderMode) { return m_material[nRenderMode]; } + //IMaterial* GetMaterial(RenderMode_t nRenderMode, int nFrame); + //void SetFrame(RenderMode_t nRenderMode, int nFrame); + //bool Init(const char* name); + //void Shutdown(void); + //void UnloadMaterial(); + //void SetColor(float r, float g, float b); + //int GetOrientation(void); + //void GetHUDSpriteColor(float* color); + float GetUp(void) const { return up; } + float GetDown(void) const { return down; } + float GetLeft(void) const { return left; } + float GetRight(void) const { return right; } + //void DrawFrame(RenderMode_t nRenderMode, int frame, int x, int y, const wrect_t* prcSubRect); + //void DrawFrameOfSize(RenderMode_t nRenderMode, int frame, int x, int y, int iWidth, int iHeight, const wrect_t* prcSubRect); + bool IsAVI(void) const; + bool IsBIK(void) const; + //void GetTexCoordRange(float* pMinU, float* pMinV, float* pMaxU, float* pMaxV); + +private: + AVIMaterial_t m_hAVIMaterial; + BIKMaterial_t m_hBIKMaterial; + int m_width; + int m_height; + int m_numFrames; + IMaterial* m_material[kRenderModeCount]; + int m_orientation; + float m_hudSpriteColor[3]; + float up, down, left, right; +}; + +#endif // ENGINESPRITE_H \ No newline at end of file diff --git a/r5dev/game/client/hud.h b/r5dev/game/client/hud.h new file mode 100644 index 00000000..61db9100 --- /dev/null +++ b/r5dev/game/client/hud.h @@ -0,0 +1,19 @@ +//====== Copyright � 1996-2005, Valve Corporation, All rights reserved. =======// +// +// Purpose: CHud handles the message, calculation, and drawing the HUD +// +// $NoKeywords: $ +//=============================================================================// +#ifndef HUD_H +#define HUD_H + +// basic rectangle struct used for drawing +typedef struct wrect_s +{ + int left; + int right; + int top; + int bottom; +} wrect_t; + +#endif // HUD_H \ No newline at end of file diff --git a/r5dev/game/client/spritemodel.cpp b/r5dev/game/client/spritemodel.cpp new file mode 100644 index 00000000..4306cf67 --- /dev/null +++ b/r5dev/game/client/spritemodel.cpp @@ -0,0 +1,24 @@ +//===== Copyright � 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: +// +// $NoKeywords: $ +//===========================================================================// +#include "core/stdafx.h" +#include "game/client/enginesprite.h" + +//----------------------------------------------------------------------------- +// Is the sprite an AVI? +//----------------------------------------------------------------------------- +bool CEngineSprite::IsAVI(void) const +{ + return (m_hAVIMaterial != AVIMATERIAL_INVALID); +} + +//----------------------------------------------------------------------------- +// Is the sprite an BIK? +//----------------------------------------------------------------------------- +bool CEngineSprite::IsBIK(void) const +{ + return (m_hBIKMaterial != BIKMATERIAL_INVALID); +} diff --git a/r5dev/public/avi/iavi.h b/r5dev/public/avi/iavi.h new file mode 100644 index 00000000..0e01f49d --- /dev/null +++ b/r5dev/public/avi/iavi.h @@ -0,0 +1,29 @@ +//===== Copyright � 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: +// +// $NoKeywords: $ +//===========================================================================// +#ifndef IAVI_H +#define IAVI_H + +//----------------------------------------------------------------------------- +// Handle to an AVI +//----------------------------------------------------------------------------- +typedef unsigned short AVIHandle_t; +enum +{ + AVIHANDLE_INVALID = (AVIHandle_t)~0 +}; + + +//----------------------------------------------------------------------------- +// Handle to an AVI material +//----------------------------------------------------------------------------- +typedef unsigned short AVIMaterial_t; +enum +{ + AVIMATERIAL_INVALID = (AVIMaterial_t)~0 +}; + +#endif // IAVI_H \ No newline at end of file diff --git a/r5dev/public/avi/ibik.h b/r5dev/public/avi/ibik.h new file mode 100644 index 00000000..88952982 --- /dev/null +++ b/r5dev/public/avi/ibik.h @@ -0,0 +1,29 @@ +//===== Copyright � 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: +// +// $NoKeywords: $ +//===========================================================================// +#ifndef IBIK_H +#define IBIK_H + +//----------------------------------------------------------------------------- +// Handle to an BINK +//----------------------------------------------------------------------------- +typedef unsigned short BIKHandle_t; +enum +{ + BIKHANDLE_INVALID = (BIKHandle_t)~0 +}; + + +//----------------------------------------------------------------------------- +// Handle to an BINK material +//----------------------------------------------------------------------------- +typedef unsigned short BIKMaterial_t; +enum +{ + BIKMATERIAL_INVALID = (BIKMaterial_t)~0 +}; + +#endif // IBIK_H \ No newline at end of file diff --git a/r5dev/public/include/const.h b/r5dev/public/include/const.h new file mode 100644 index 00000000..806799a3 --- /dev/null +++ b/r5dev/public/include/const.h @@ -0,0 +1,28 @@ +//===== Copyright (c) 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: +// +// $NoKeywords: $ +// +//=============================================================================// +#ifndef CONST_H +#define CONST_H + +enum RenderMode_t +{ + kRenderNormal = 0, // src + kRenderTransColor, // c*a+dest*(1-a) + kRenderTransTexture, // src*a+dest*(1-a) + kRenderGlow, // src*a+dest -- No Z buffer checks -- Fixed size in screen space + kRenderTransAlpha, // src*srca+dest*(1-srca) + kRenderTransAdd, // src*a+dest + kRenderEnvironmental, // not drawn, used for environmental effects + kRenderTransAddFrameBlend, // use a fractional frame value to blend between animation frames + kRenderTransAlphaAdd, // src + dest*(1-a) + kRenderWorldGlow, // Same as kRenderGlow but not fixed size in screen space + kRenderNone, // Don't render. + + kRenderModeCount, // must be last +}; + +#endif \ No newline at end of file diff --git a/r5dev/public/include/model_types.h b/r5dev/public/include/model_types.h new file mode 100644 index 00000000..84c82109 --- /dev/null +++ b/r5dev/public/include/model_types.h @@ -0,0 +1,60 @@ +//===== Copyright � 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: +// +// $Workfile: $ +// $Date: $ +// $NoKeywords: $ +//===========================================================================// +#if !defined( MODEL_TYPES_H ) +#define MODEL_TYPES_H +#ifdef _WIN32 +#pragma once +#endif + +#define STUDIO_NONE 0x00000000 +#define STUDIO_RENDER 0x00000001 +#define STUDIO_VIEWXFORMATTACHMENTS 0x00000002 +#define STUDIO_DRAWTRANSLUCENTSUBMODELS 0x00000004 +#define STUDIO_TWOPASS 0x00000008 +#define STUDIO_STATIC_LIGHTING 0x00000010 +#define STUDIO_WIREFRAME 0x00000020 +#define STUDIO_ITEM_BLINK 0x00000040 +#define STUDIO_NOSHADOWS 0x00000080 +#define STUDIO_WIREFRAME_VCOLLIDE 0x00000100 +#define STUDIO_NOLIGHTING_OR_CUBEMAP 0x00000200 +#define STUDIO_SKIP_FLEXES 0x00000400 +#define STUDIO_DONOTMODIFYSTENCILSTATE 0x00000800 // TERROR + +// not a studio flag, but used to signify that the draw call should cause a depth imposter to be drawn as well. +#define STUDIO_DEPTH_ONLY 0x01000000 + +// Not a studio flag, but used to flag model to render using special settings for AO pre-pass +#define STUDIO_AOPREPASSTEXURE 0x02000000 + +// Not a studio flag, but used to flag model to keep the shadow state it starts with +#define STUDIO_KEEP_SHADOWS 0x04000000 + +// Not a studio flag, but used to flag model as using shadow depth material override +#define STUDIO_SSAODEPTHTEXTURE 0x08000000 + +// Not a studio flag, but used to flag model as a non-sorting brush model +#define STUDIO_TRANSPARENCY 0x80000000 + +// Not a studio flag, but used to flag model as using shadow depth material override +#define STUDIO_SHADOWDEPTHTEXTURE 0x40000000 + +// Not a studio flag, but used to flag model as doing custom rendering into shadow texture +#define STUDIO_SHADOWTEXTURE 0x20000000 + +#define STUDIO_SKIP_DECALS 0x10000000 + +enum modtype_t +{ + mod_bad = 0, + mod_brush, + mod_sprite, + mod_studio +}; + +#endif // MODEL_TYPES_H diff --git a/r5dev/public/include/studio.h b/r5dev/public/include/studio.h new file mode 100644 index 00000000..709419e0 --- /dev/null +++ b/r5dev/public/include/studio.h @@ -0,0 +1,70 @@ +#ifndef STUDIO_H +#define STUDIO_H +#include "mathlib/vector.h" + +#pragma pack(push, 1) +struct studiohdr_t +{ + int id; // 'IDST' + int version; // R5 = '6' + int checksum; + int tableIndex; // Offset + + char name[0x40]; + + int length; // size of data + + Vector3 eyeposition; // ideal eye position + Vector3 illumposition; // illumination center + Vector3 hull_min; // ideal movement hull size + Vector3 hull_max; + Vector3 view_bbmin; // clipping bounding box + Vector3 view_bbmax; + + int flags; + + int numbones; // bones + int boneindex; + + int numbonecontrollers; + int bonecontrollerindex; + + int numhitboxsets; + int hitboxsetindex; + + int numlocalanim; // animations/poses + int localanimindex; // animation descriptions + + int numlocalseq; // sequences + int localseqindex; + + int activitylistversion; // initialization flag - have the sequences been indexed ? + int eventsindexed; + + int numtextures; + int textureindex; + + int numcdtextures; + int cdtextureindex; + + int numskinref; // Total number of references (submeshes) + int numskinfamilies; // Total skins per reference + int skinindex; // Offset to data + + int numbodyparts; + int bodypartindex; + + int numlocalattachments; + int localattachmentindex; + + uint8_t Unknown2[0x14]; + + int submeshLodsIndex; + + uint8_t Unknown3[0x64]; + int boneRemapInfoIndex; + int boneRemapCount; +}; +#pragma pack(pop) + +#endif // STUDIO_H diff --git a/r5dev/public/include/utility.h b/r5dev/public/include/utility.h index 7e1f2264..2687b244 100644 --- a/r5dev/public/include/utility.h +++ b/r5dev/public/include/utility.h @@ -16,6 +16,7 @@ void HexDump(const char* szHeader, const char* szLogger, const void* pData, int string CreateDirectories(string svFilePath); string ConvertToWinPath(const string& svInput); +string ConvertToUnixPath(const string& svInput); string Base64Encode(const string& svInput); string Base64Decode(const string& svInput); diff --git a/r5dev/public/utility.cpp b/r5dev/public/utility.cpp index 1e650032..2d660391 100644 --- a/r5dev/public/utility.cpp +++ b/r5dev/public/utility.cpp @@ -267,6 +267,25 @@ string ConvertToWinPath(const string& svInput) return results = szFilePath; } +/////////////////////////////////////////////////////////////////////////////// +// For converting filepaths to unix filepaths. +string ConvertToUnixPath(const string& svInput) +{ + char szFilePath[MAX_PATH] = { 0 }; + string results; + sprintf_s(szFilePath, MAX_PATH, "%s", svInput.c_str()); + + // Flip forward slashes in filepath to windows-style backslash + for (int i = 0; i < strlen(szFilePath); i++) + { + if (szFilePath[i] == '\\') + { + szFilePath[i] = '/'; + } + } + return results = szFilePath; +} + /////////////////////////////////////////////////////////////////////////////// // For encoding data in base64. string Base64Encode(const string& svInput) diff --git a/r5dev/vproj/clientsdk.vcxproj b/r5dev/vproj/clientsdk.vcxproj index 7ac72918..96b186ca 100644 --- a/r5dev/vproj/clientsdk.vcxproj +++ b/r5dev/vproj/clientsdk.vcxproj @@ -24,6 +24,7 @@ Create + @@ -45,6 +46,7 @@ + @@ -119,6 +121,7 @@ + @@ -127,6 +130,8 @@ + + @@ -136,6 +141,7 @@ + @@ -153,6 +159,8 @@ + + @@ -175,6 +183,9 @@ + + + @@ -183,8 +194,10 @@ + + diff --git a/r5dev/vproj/clientsdk.vcxproj.filters b/r5dev/vproj/clientsdk.vcxproj.filters index 3d22e6f0..27fc0049 100644 --- a/r5dev/vproj/clientsdk.vcxproj.filters +++ b/r5dev/vproj/clientsdk.vcxproj.filters @@ -169,6 +169,12 @@ {fd290792-d36d-400c-9f4d-366a9ce57427} + + {99b7722c-1c10-431d-b86d-bc4d03655aee} + + + {05663bd0-2f29-41e4-acd3-4e3580f16549} + @@ -429,6 +435,12 @@ sdk\common + + sdk\datacache + + + sdk\game\client + @@ -1211,6 +1223,39 @@ windows + + sdk\public\include + + + sdk\engine + + + sdk\common + + + sdk\public\include + + + sdk\game\client + + + sdk\datacache + + + sdk\datacache + + + sdk\game\client + + + sdk\public\avi + + + sdk\public\avi + + + sdk\public\include + diff --git a/r5dev/vproj/dedicated.vcxproj b/r5dev/vproj/dedicated.vcxproj index b380c39f..e45dfa50 100644 --- a/r5dev/vproj/dedicated.vcxproj +++ b/r5dev/vproj/dedicated.vcxproj @@ -131,6 +131,7 @@ + @@ -139,6 +140,8 @@ + + @@ -146,6 +149,7 @@ + @@ -187,6 +191,7 @@ + @@ -195,7 +200,9 @@ + + @@ -375,6 +382,7 @@ Create + diff --git a/r5dev/vproj/dedicated.vcxproj.filters b/r5dev/vproj/dedicated.vcxproj.filters index 61b903c3..6529d52b 100644 --- a/r5dev/vproj/dedicated.vcxproj.filters +++ b/r5dev/vproj/dedicated.vcxproj.filters @@ -145,6 +145,9 @@ {cc54d9ba-f73a-48af-af6a-3b2064710e61} + + {4573ce75-0337-41b1-a43e-e9c17773b127} + @@ -873,6 +876,27 @@ sdk\rtech\rui + + sdk\public\include + + + sdk\engine + + + sdk\common + + + sdk\public\include + + + sdk\datacache + + + sdk\datacache + + + sdk\public\include + @@ -1103,6 +1127,9 @@ sdk\tier0 + + sdk\datacache + diff --git a/r5dev/vproj/gamesdk.vcxproj b/r5dev/vproj/gamesdk.vcxproj index f336e4f0..442c5b49 100644 --- a/r5dev/vproj/gamesdk.vcxproj +++ b/r5dev/vproj/gamesdk.vcxproj @@ -24,6 +24,7 @@ Create + @@ -47,6 +48,7 @@ + @@ -127,6 +129,7 @@ + @@ -135,6 +138,8 @@ + + @@ -145,6 +150,7 @@ + @@ -163,6 +169,8 @@ + + @@ -192,6 +200,9 @@ + + + @@ -200,8 +211,10 @@ + + diff --git a/r5dev/vproj/gamesdk.vcxproj.filters b/r5dev/vproj/gamesdk.vcxproj.filters index 15e43c09..9103ec3c 100644 --- a/r5dev/vproj/gamesdk.vcxproj.filters +++ b/r5dev/vproj/gamesdk.vcxproj.filters @@ -175,6 +175,12 @@ {48abe326-8ad7-43fa-875d-2e73c7c64106} + + {42f3eba8-1c16-4a48-b9c7-209eb4e0eb36} + + + {67444ecb-b115-4231-a33b-aab424f785fc} + @@ -459,6 +465,12 @@ sdk\common + + sdk\datacache + + + sdk\game\client + @@ -1271,6 +1283,39 @@ windows + + sdk\datacache + + + sdk\public\include + + + sdk\engine + + + sdk\common + + + sdk\public\include + + + sdk\datacache + + + sdk\game\client + + + sdk\public\include + + + sdk\public\avi + + + sdk\public\avi + + + sdk\game\client +