From 66ef9fb6a0a0dff665033cfea7eeb46268f9454c Mon Sep 17 00:00:00 2001 From: Kawe Mazidjatari <48657826+Mauler125@users.noreply.github.com> Date: Mon, 23 Jan 2023 23:21:29 +0100 Subject: [PATCH] Fix crash in hitbox drawing code and light restructure * Fix null pointer dereference crash in server hitbox drawing code; implemented a proper getter for CStudioHdr. * Moved 'baseentity.h/cpp' to 'game/server/' instead. * Overall light cleanup. --- r5dev/game/server/baseanimating.cpp | 42 +++++++++++--------- r5dev/game/server/baseanimating.h | 17 ++++---- r5dev/{public => game/server}/baseentity.cpp | 20 ++++++++++ r5dev/{public => game/server}/baseentity.h | 19 +++++---- r5dev/game/server/entitylist.cpp | 9 +++++ r5dev/game/server/entitylist.h | 13 ++++++ r5dev/game/server/networkproperty.cpp | 2 +- r5dev/game/server/player.h | 9 +++-- r5dev/game/shared/animation.h | 4 +- r5dev/game/shared/collisionproperty.h | 2 +- r5dev/game/shared/imovehelper.h | 6 +-- r5dev/game/shared/predictioncopy.h | 5 ++- r5dev/public/studio.h | 17 ++++++-- r5dev/vproj/dedicated.vcxproj | 6 ++- r5dev/vproj/dedicated.vcxproj.filters | 18 ++++++--- r5dev/vproj/gamesdk.vcxproj | 6 ++- r5dev/vproj/gamesdk.vcxproj.filters | 18 ++++++--- 17 files changed, 146 insertions(+), 67 deletions(-) rename r5dev/{public => game/server}/baseentity.cpp (71%) rename r5dev/{public => game/server}/baseentity.h (95%) create mode 100644 r5dev/game/server/entitylist.cpp create mode 100644 r5dev/game/server/entitylist.h diff --git a/r5dev/game/server/baseanimating.cpp b/r5dev/game/server/baseanimating.cpp index 18197930..5508a821 100644 --- a/r5dev/game/server/baseanimating.cpp +++ b/r5dev/game/server/baseanimating.cpp @@ -8,7 +8,7 @@ #include "engine/modelinfo.h" #include "public/idebugoverlay.h" -static Vector3D hullcolor[8] = +static const Vector3D s_HullColor[8] = { Vector3D(1.0, 1.0, 1.0), Vector3D(1.0, 0.5, 0.5), @@ -27,17 +27,12 @@ static Vector3D hullcolor[8] = //----------------------------------------------------------------------------- void CBaseAnimating::DrawServerHitboxes(float duration /*= 0.0f*/) { - if (!m_pStudioHdr && g_pModelInfoServer->GetModel(m_nModelIndex)) - { - UpdateModelPtr(); - if (!m_pStudioHdr) - { - return; - } - } + CStudioHdr* pStudioHdr = GetModelPtr(); + if (!GetModelPtr()) + return; - mstudiohitboxset_t* set = m_pStudioHdr->pHitboxSet(m_nHitboxSet); - if (!set) + mstudiohitboxset_t* pSet = pStudioHdr->GetHitboxSet(m_nHitboxSet); + if (!pSet) return; matrix3x4_t transforms; @@ -46,15 +41,15 @@ void CBaseAnimating::DrawServerHitboxes(float duration /*= 0.0f*/) int g = 0; int b = 255; - for (int i = 0; i < set->numhitboxes; i++) + for (int i = 0; i < pSet->numhitboxes; i++) { - mstudiobbox_t* pBox = set->pHitbox(i); + mstudiobbox_t* pBox = pSet->pHitbox(i); int j = (pBox->group % 8); - r = (int)(255.0f * hullcolor[j][0]); - g = (int)(255.0f * hullcolor[j][1]); - b = (int)(255.0f * hullcolor[j][2]); + r = static_cast(255.0f * s_HullColor[j][0]); + g = static_cast(255.0f * s_HullColor[j][1]); + b = static_cast(255.0f * s_HullColor[j][2]); HitboxToWorldTransforms(pBox->bone, &transforms); g_pDebugOverlay->AddBoxOverlay(transforms, pBox->bbmin, pBox->bbmax, r, g, b, 0, true, duration); @@ -63,17 +58,26 @@ void CBaseAnimating::DrawServerHitboxes(float duration /*= 0.0f*/) void CBaseAnimating::HitboxToWorldTransforms(uint32_t iBone, matrix3x4_t* transforms) { - const static int index = 285; + constexpr int index = 285; CallVFunc(index, this, iBone, transforms); } //----------------------------------------------------------------------------- // Purpose: sets studio pointer to an updated studiomdl cache //----------------------------------------------------------------------------- -void CBaseAnimating::UpdateModelPtr() +void CBaseAnimating::LockStudioHdr() { // Populates the 'm_pStudioHdr' field. - v_CBaseAnimating__UpdateModelPtr(this); + v_CBaseAnimating__LockStudioHdr(this); +} + +CStudioHdr* CBaseAnimating::GetModelPtr(void) +{ + if (!m_pStudioHdr && GetModel()) + { + LockStudioHdr(); + } + return (m_pStudioHdr && m_pStudioHdr->IsValid()) ? m_pStudioHdr : nullptr; } void DrawServerHitboxes(CBaseAnimating* thisp, float duration) diff --git a/r5dev/game/server/baseanimating.h b/r5dev/game/server/baseanimating.h index 45e32069..434585ff 100644 --- a/r5dev/game/server/baseanimating.h +++ b/r5dev/game/server/baseanimating.h @@ -9,19 +9,20 @@ #ifdef _WIN32 #pragma once #endif -#include "public/baseentity.h" -#include "game/shared/animation.h" +#include "baseentity.h" #include "public/studio.h" +#include "game/shared/animation.h" class CBaseAnimating : public CBaseEntity { public: void DrawServerHitboxes(float duration = 0.0f); - void UpdateModelPtr(); + void LockStudioHdr(void); void HitboxToWorldTransforms(uint32_t iBone, matrix3x4_t* transforms); + CStudioHdr* GetModelPtr(void); float GetModelScale() const { return m_flModelScale; } protected: @@ -135,8 +136,8 @@ protected: char gap_11E4[12]; // TODO: this might belong to CBaseAnimatingOverlay! }; -inline CMemory p_CBaseAnimating__UpdateModelPtr; -inline auto v_CBaseAnimating__UpdateModelPtr = p_CBaseAnimating__UpdateModelPtr.RCast(); +inline CMemory p_CBaseAnimating__LockStudioHdr; +inline auto v_CBaseAnimating__LockStudioHdr = p_CBaseAnimating__LockStudioHdr.RCast(); inline CMemory p_CBaseAnimating__DrawServerHitboxes; inline auto v_CBaseAnimating__DrawServerHitboxes = p_CBaseAnimating__DrawServerHitboxes.RCast(); @@ -149,14 +150,14 @@ class VBaseAnimating : public IDetour { virtual void GetAdr(void) const { - spdlog::debug("| FUN: CBaseAnimating::UpdateModelPtr : {:#18x} |\n", p_CBaseAnimating__UpdateModelPtr.GetPtr()); + spdlog::debug("| FUN: CBaseAnimating::LockStudioHdr : {:#18x} |\n", p_CBaseAnimating__LockStudioHdr.GetPtr()); spdlog::debug("| FUN: CBaseAnimating::DrawServerHitboxes : {:#18x} |\n", p_CBaseAnimating__DrawServerHitboxes.GetPtr()); spdlog::debug("+----------------------------------------------------------------+\n"); } virtual void GetFun(void) const { - p_CBaseAnimating__UpdateModelPtr = g_GameDll.FindPatternSIMD("48 89 5C 24 ?? 48 89 74 24 ?? 41 56 48 83 EC 20 0F BF 41 58"); - v_CBaseAnimating__UpdateModelPtr = p_CBaseAnimating__UpdateModelPtr.RCast(); + p_CBaseAnimating__LockStudioHdr = g_GameDll.FindPatternSIMD("48 89 5C 24 ?? 48 89 74 24 ?? 41 56 48 83 EC 20 0F BF 41 58"); + v_CBaseAnimating__LockStudioHdr = p_CBaseAnimating__LockStudioHdr.RCast(); p_CBaseAnimating__DrawServerHitboxes = g_GameDll.FindPatternSIMD("41 57 48 81 EC ?? ?? ?? ?? 48 83 B9 ?? ?? ?? ?? ?? 4C 8B F9"); // !FIXME NOT COMPAT WITH S0~S2! v_CBaseAnimating__DrawServerHitboxes = p_CBaseAnimating__DrawServerHitboxes.RCast(); diff --git a/r5dev/public/baseentity.cpp b/r5dev/game/server/baseentity.cpp similarity index 71% rename from r5dev/public/baseentity.cpp rename to r5dev/game/server/baseentity.cpp index 27150199..bfc03f84 100644 --- a/r5dev/public/baseentity.cpp +++ b/r5dev/game/server/baseentity.cpp @@ -5,6 +5,8 @@ //=============================================================================== #include "core/stdafx.h" #include "baseentity.h" +#include "engine/gl_model_private.h" +#include "engine/modelinfo.h" //----------------------------------------------------------------------------- // @@ -37,3 +39,21 @@ const CServerNetworkProperty* CBaseEntity::NetworkProp() const { return &m_Network; } + +//----------------------------------------------------------------------------- +// +//----------------------------------------------------------------------------- +model_t* CBaseEntity::GetModel(void) +{ + return (model_t*)g_pModelInfoServer->GetModel(GetModelIndex()); +} + +inline string_t CBaseEntity::GetModelName(void) const +{ + return m_ModelName; +} + +inline int CBaseEntity::GetModelIndex(void) const +{ + return m_nModelIndex; +} \ No newline at end of file diff --git a/r5dev/public/baseentity.h b/r5dev/game/server/baseentity.h similarity index 95% rename from r5dev/public/baseentity.h rename to r5dev/game/server/baseentity.h index b877361d..71456ad8 100644 --- a/r5dev/public/baseentity.h +++ b/r5dev/game/server/baseentity.h @@ -12,23 +12,28 @@ #endif #include "mathlib/vector.h" -#include "game/server/networkproperty.h" +#include "public/iservernetworkable.h" +#include "public/iserverentity.h" +#include "public/string_t.h" +#include "engine/gl_model_private.h" #include "game/shared/collisionproperty.h" -#include "game/shared/entitylist_base.h" -#include "iservernetworkable.h" -#include "iserverentity.h" +#include "networkproperty.h" +#include "entitylist.h" class CBaseEntity : public IServerEntity { // non-virtual methods. Don't override these! public: - // An inline version the game code can use CCollisionProperty* CollisionProp(); const CCollisionProperty* CollisionProp() const; CServerNetworkProperty* NetworkProp(); const CServerNetworkProperty* NetworkProp() const; + model_t* GetModel(void); + int GetModelIndex(void) const; // Virtual in-engine! + string_t GetModelName(void) const; // Virtual in-engine! + protected: char m_RefEHandle[4]; char gap_c[4]; @@ -40,7 +45,7 @@ protected: void* m_pfnMoveDone; void* m_pfnThink; CServerNetworkProperty m_Network; - const char* m_ModelName; + string_t m_ModelName; int m_entIndex; char gap_74[8]; // Aligns properly in IDA and generated code after setting from 4 to 8. const char* m_iClassname; @@ -251,8 +256,6 @@ protected: inline CMemory p_CBaseEntity__GetBaseEntity; inline auto v_CBaseEntity__GetBaseEntity = p_CBaseEntity__GetBaseEntity.RCast(); -inline CEntInfo* g_pEntityList = nullptr; - /////////////////////////////////////////////////////////////////////////////// class VBaseEntity : public IDetour { diff --git a/r5dev/game/server/entitylist.cpp b/r5dev/game/server/entitylist.cpp new file mode 100644 index 00000000..9965ef87 --- /dev/null +++ b/r5dev/game/server/entitylist.cpp @@ -0,0 +1,9 @@ +//===== Copyright © 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: +// +//===========================================================================// +#include "core/stdafx.h" +#include "entitylist.h" + +CEntInfo* g_pEntityList = nullptr; \ No newline at end of file diff --git a/r5dev/game/server/entitylist.h b/r5dev/game/server/entitylist.h new file mode 100644 index 00000000..cb166122 --- /dev/null +++ b/r5dev/game/server/entitylist.h @@ -0,0 +1,13 @@ +//===== Copyright © 1996-2005, Valve Corporation, All rights reserved. ======// +// +// Purpose: +// +// $NoKeywords: $ +//===========================================================================// +#ifndef ENTITYLIST_H +#define ENTITYLIST_H +#include "game/shared/entitylist_base.h" + +extern CEntInfo* g_pEntityList; + +#endif // ENTITYLIST_H diff --git a/r5dev/game/server/networkproperty.cpp b/r5dev/game/server/networkproperty.cpp index e5284d57..1bf9d6d0 100644 --- a/r5dev/game/server/networkproperty.cpp +++ b/r5dev/game/server/networkproperty.cpp @@ -6,8 +6,8 @@ //===========================================================================// #include "core/stdafx.h" -#include "public/baseentity.h" #include "public/basehandle.h" +#include "baseentity.h" #include "networkproperty.h" edict_t CServerNetworkProperty::GetEdict(void) const diff --git a/r5dev/game/server/player.h b/r5dev/game/server/player.h index baecfb14..d75e006f 100644 --- a/r5dev/game/server/player.h +++ b/r5dev/game/server/player.h @@ -7,18 +7,19 @@ #ifndef PLAYER_H #define PLAYER_H -#include "public/baseentity.h" +#include "baseentity.h" #include "mathlib/mathlib.h" -#include -#include +#include "mathlib/vector4d.h" +#include "public/studio.h" +#include "public/playerstate.h" #include "game/shared/animation.h" #include "game/shared/takedamageinfo.h" #include "game/shared/usercmd.h" #include "game/shared/imovehelper.h" + #include "playerlocaldata.h" #include "basecombatcharacter.h" -#include enum PlayerConnectedState { diff --git a/r5dev/game/shared/animation.h b/r5dev/game/shared/animation.h index 957e1c79..9fb6e81b 100644 --- a/r5dev/game/shared/animation.h +++ b/r5dev/game/shared/animation.h @@ -1,8 +1,8 @@ #ifndef ANIMATION_H #define ANIMATION_H #include "mathlib/vector.h" -#include "public/baseentity.h" #include "public/studio.h" +#include "public/ihandleentity.h" class CAnimationLayer { @@ -17,7 +17,7 @@ class CAnimationLayer int m_nPriority; float m_flLastEventCheck; char gap_24[4]; - CBaseEntity* m_animationLayerOwner; + IHandleEntity* m_animationLayerOwner; // !TODO: CBaseEntity/C_BaseEntity? }; struct __declspec(align(8)) PredictedAnimEventData diff --git a/r5dev/game/shared/collisionproperty.h b/r5dev/game/shared/collisionproperty.h index df0cdd1a..3d5d9688 100644 --- a/r5dev/game/shared/collisionproperty.h +++ b/r5dev/game/shared/collisionproperty.h @@ -9,8 +9,8 @@ #define COLLISIONPROPERTY_H #include "public/engine/ICollideable.h" -#include "public/baseentity.h" #include "mathlib/vector.h" +#include "baseentity.h" class CCollisionProperty : public ICollideable { diff --git a/r5dev/game/shared/imovehelper.h b/r5dev/game/shared/imovehelper.h index c01fe2f1..51bfe4d9 100644 --- a/r5dev/game/shared/imovehelper.h +++ b/r5dev/game/shared/imovehelper.h @@ -13,7 +13,7 @@ #endif #include "tier0/annotations.h" -#include "public/baseentity.h" +#include "public/ihandleentity.h" typedef CBaseHandle EntityHandle_t; class IPhysicsSurfaceProps; // !TODO: reverse vtable. @@ -32,8 +32,8 @@ public: virtual char const* GetName(EntityHandle_t handle) const = 0; // sets the entity being moved - virtual void SetHost(CBaseEntity* host) = 0; - virtual CBaseEntity* GetHost(void) = 0; + virtual void SetHost(IHandleEntity* host) = 0; + virtual IHandleEntity* GetHost(void) = 0; virtual void ResetTouchList(void) = 0; virtual bool AddToTouched(const /*CGameTrace&*/void* tr, const Vector3D& impactvelocity) = 0; diff --git a/r5dev/game/shared/predictioncopy.h b/r5dev/game/shared/predictioncopy.h index 65dc7fe6..980922d5 100644 --- a/r5dev/game/shared/predictioncopy.h +++ b/r5dev/game/shared/predictioncopy.h @@ -12,13 +12,16 @@ #endif #include "ehandle.h" -#include "public/baseentity.h" #if defined( CLIENT_DLL ) +#include "game/client/c_baseentity.h" + class C_BaseEntity; typedef CHandle EHANDLE; #else +#include "game/server/baseentity.h" + class CBaseEntity; typedef CHandle EHANDLE; #endif diff --git a/r5dev/public/studio.h b/r5dev/public/studio.h index 4eed38b0..a5d04e64 100644 --- a/r5dev/public/studio.h +++ b/r5dev/public/studio.h @@ -339,7 +339,7 @@ struct studiohdr_t // Look up hitbox set by index - mstudiohitboxset_t* pHitboxSet(int i) const + mstudiohitboxset_t* GetHitboxSet(int i) const { Assert(i >= 0 && i < numhitboxsets); return (mstudiohitboxset_t*)(((byte*)this) + hitboxsetindex) + i; @@ -1033,13 +1033,22 @@ struct studiohwdata_t class CStudioHdr { -public: +public: // Detour statics: static int LookupSequence(CStudioHdr* pStudio, const char* pszName); - inline mstudiohitboxset_t* pHitboxSet(int i) const { return m_pStudioHdr->pHitboxSet(i); }; +public: + inline bool IsVirtual(void) { return (m_pVModel != NULL); }; + inline bool IsValid(void) { return (m_pStudioHdr != NULL); }; + inline bool IsReadyForAccess(void) const { return (m_pStudioHdr != NULL); }; + +public: + inline const studiohdr_t* GetRenderHdr(void) const { return m_pStudioHdr; }; + inline mstudiohitboxset_t* GetHitboxSet(int i) const { return m_pStudioHdr->GetHitboxSet(i); }; + +private: void* vtbl; studiohdr_t* m_pStudioHdr; - void* m_pVModel; + void* m_pVModel; // !TODO: 'virtualmodel_t'. char gap0[156]; mstudiobbox_t m_pHitBox; char gap_10[1896]; diff --git a/r5dev/vproj/dedicated.vcxproj b/r5dev/vproj/dedicated.vcxproj index cd48fa43..2e58820a 100644 --- a/r5dev/vproj/dedicated.vcxproj +++ b/r5dev/vproj/dedicated.vcxproj @@ -183,7 +183,9 @@ + + @@ -238,7 +240,6 @@ - @@ -564,6 +565,8 @@ + + @@ -608,7 +611,6 @@ NotUsing NotUsing - diff --git a/r5dev/vproj/dedicated.vcxproj.filters b/r5dev/vproj/dedicated.vcxproj.filters index 92e11910..bd9f7fc8 100644 --- a/r5dev/vproj/dedicated.vcxproj.filters +++ b/r5dev/vproj/dedicated.vcxproj.filters @@ -1311,9 +1311,6 @@ sdk\public - - sdk\public - sdk\public\engine @@ -1398,6 +1395,12 @@ sdk\engine + + sdk\game\server + + + sdk\game\server + @@ -1751,9 +1754,6 @@ sdk\game\server - - sdk\public - sdk\game\server @@ -1769,6 +1769,12 @@ sdk\engine + + sdk\game\server + + + sdk\game\server + diff --git a/r5dev/vproj/gamesdk.vcxproj b/r5dev/vproj/gamesdk.vcxproj index 44f20077..efafa247 100644 --- a/r5dev/vproj/gamesdk.vcxproj +++ b/r5dev/vproj/gamesdk.vcxproj @@ -71,6 +71,8 @@ + + @@ -118,7 +120,6 @@ NotUsing NotUsing - @@ -266,7 +267,9 @@ + + @@ -330,7 +333,6 @@ - diff --git a/r5dev/vproj/gamesdk.vcxproj.filters b/r5dev/vproj/gamesdk.vcxproj.filters index eb7a8c8f..9c555431 100644 --- a/r5dev/vproj/gamesdk.vcxproj.filters +++ b/r5dev/vproj/gamesdk.vcxproj.filters @@ -708,9 +708,6 @@ sdk\game\server - - sdk\public - sdk\game\server @@ -729,6 +726,12 @@ sdk\engine + + sdk\game\server + + + sdk\game\server + @@ -2039,9 +2042,6 @@ sdk\public - - sdk\public - sdk\game\server @@ -2135,6 +2135,12 @@ sdk\engine + + sdk\game\server + + + sdk\game\server +