From a68fda98c1590cd6fa0125d6907ddc0c7ab44800 Mon Sep 17 00:00:00 2001 From: Kawe Mazidjatari <48657826+Mauler125@users.noreply.github.com> Date: Wed, 5 Oct 2022 01:59:22 +0200 Subject: [PATCH] Fix crash when loading arena maps caused by recent BSP crash fix CMaterial and CMaterialGlue derive from the same abstract classes. The function that builds the culling map uses IMaterialInternal, which is used by CMaterial, CMaterialInternal and CMaterialSubRect. Added a check for CMaterialGlue as these are valid too. --- r5dev/bsplib/bsplib.cpp | 2 +- r5dev/materialsystem/cmaterialglue.h | 1 - r5dev/materialsystem/cmaterialsystem.cpp | 7 +++++-- r5dev/materialsystem/cmaterialsystem.h | 2 +- 4 files changed, 7 insertions(+), 5 deletions(-) diff --git a/r5dev/bsplib/bsplib.cpp b/r5dev/bsplib/bsplib.cpp index a220d392..2df36bbc 100644 --- a/r5dev/bsplib/bsplib.cpp +++ b/r5dev/bsplib/bsplib.cpp @@ -400,7 +400,7 @@ void* __fastcall BuildPropStaticFrustumCullMap(int64_t a1, int64_t a2, unsigned if (reinterpret_cast(v68) < g_GameDll.m_RunTimeData.m_pSectionBase || // Check bounds (data is mostly within the '.data' segment. reinterpret_cast(v68) > g_GameDll.m_ExceptionTable.m_pSectionBase || error) { - if (!IsMaterialVFTable(reinterpret_cast(v68))) // Last chance. + if (!IsMaterialInternal(reinterpret_cast(v68))) // Last chance. { error = true; continue; diff --git a/r5dev/materialsystem/cmaterialglue.h b/r5dev/materialsystem/cmaterialglue.h index 6ce5ae7d..cdf0e56f 100644 --- a/r5dev/materialsystem/cmaterialglue.h +++ b/r5dev/materialsystem/cmaterialglue.h @@ -1,5 +1,4 @@ #pragma once - #include "materialsystem/cshaderglue.h" #pragma pack(push, 1) // Without this MSVC might get the idea to align our members to completely fuck the offsets up. diff --git a/r5dev/materialsystem/cmaterialsystem.cpp b/r5dev/materialsystem/cmaterialsystem.cpp index 0a102776..7c2beae6 100644 --- a/r5dev/materialsystem/cmaterialsystem.cpp +++ b/r5dev/materialsystem/cmaterialsystem.cpp @@ -7,6 +7,7 @@ #include "tier1/cvar.h" #include "rtech/rtech_utils.h" #include "filesystem/filesystem.h" +#include "materialsystem/cmaterialglue.h" #include "materialsystem/cmaterialsystem.h" //--------------------------------------------------------------------------------- @@ -85,7 +86,7 @@ void* __fastcall DispatchDrawCall(int64_t a1, uint64_t a2, int a3, int a4, int64 // Input : **pCandidate - // Output : true if valid and material, false otherwise //----------------------------------------------------------------------------- -bool IsMaterialVFTable(void** pCandidate) +bool IsMaterialInternal(void** pCandidate) { // NOTE: this is a dirty fix, but for running technically broken BSP's, this is the only fix // besides going bare metal inline assembly (which on its own isn't directly the problem, but @@ -99,13 +100,15 @@ bool IsMaterialVFTable(void** pCandidate) // The first member of the CMaterial data structure should be its VFTable pointer, anything else is invalid. __try { - if (*pCandidate == g_pMaterialVFTable) + if (*pCandidate == g_pMaterialVFTable || + *pCandidate == g_pMaterialGlueVFTable) return true; } __except (GetExceptionCode() == EXCEPTION_ACCESS_VIOLATION) { return false; } + return false; } /////////////////////////////////////////////////////////////////////////////// diff --git a/r5dev/materialsystem/cmaterialsystem.h b/r5dev/materialsystem/cmaterialsystem.h index aeee8eba..ed4a74c2 100644 --- a/r5dev/materialsystem/cmaterialsystem.h +++ b/r5dev/materialsystem/cmaterialsystem.h @@ -26,7 +26,7 @@ inline int* g_nUnfreeStreamingTextureMemory = nullptr; inline int* g_nUnusableStreamingTextureMemory = nullptr; #endif // !DEDICATED -bool IsMaterialVFTable(void** pCandidate); +bool IsMaterialInternal(void** pCandidate); void CMaterialSystem_Attach(); void CMaterialSystem_Detach(); ///////////////////////////////////////////////////////////////////////////////