From 5e3a8b313c3043f325b321bb1fe9d8ca36b29aee Mon Sep 17 00:00:00 2001 From: momo5502 Date: Tue, 6 Jun 2017 14:02:56 +0200 Subject: [PATCH] [Materials] Recover images to prevent vid_restart crashes --- src/Components/Modules/Materials.cpp | 19 ++++++++++++++-- src/Components/Modules/Materials.hpp | 2 ++ src/Components/Modules/QuickPatch.hpp | 1 - src/Components/Modules/Renderer.cpp | 31 +++++++++++++++++++++++++++ src/Components/Modules/Renderer.hpp | 4 ++++ src/Components/Modules/Toast.cpp | 13 ++++++----- 6 files changed, 60 insertions(+), 10 deletions(-) diff --git a/src/Components/Modules/Materials.cpp b/src/Components/Modules/Materials.cpp index a1e2528e..7bbda630 100644 --- a/src/Components/Modules/Materials.cpp +++ b/src/Components/Modules/Materials.cpp @@ -120,6 +120,21 @@ namespace Components } } + bool Materials::IsValid(Game::Material* material) + { + if (!material || !material->textureCount || !material->textureTable) return false; + + for(char i = 0; i < material->textureCount; ++i) + { + if (!material->textureTable[i].info.image || !material->textureTable[i].info.image->map) + { + return false; + } + } + + return true; + } + __declspec(naked) void Materials::ImageVersionCheck() { __asm @@ -311,13 +326,13 @@ namespace Components } }); - Renderer::OnDeviceRecoveryEnd([]() + /*Renderer::OnDeviceRecoveryEnd([]() { for (auto& image : Materials::ImageTable) { Utils::Hook::Call(0x51F7B0)(image); } - }); + });*/ } Materials::~Materials() diff --git a/src/Components/Modules/Materials.hpp b/src/Components/Modules/Materials.hpp index c08755af..fe1c9f03 100644 --- a/src/Components/Modules/Materials.hpp +++ b/src/Components/Modules/Materials.hpp @@ -16,6 +16,8 @@ namespace Components static Game::GfxImage* CreateImage(std::string name, unsigned int width, unsigned int height, unsigned int depth, unsigned int flags, _D3DFORMAT format); static void DeleteImage(Game::GfxImage* image); + static bool IsValid(Game::Material* material); + private: static std::vector ImageTable; static std::vector MaterialTable; diff --git a/src/Components/Modules/QuickPatch.hpp b/src/Components/Modules/QuickPatch.hpp index 097fb7ce..d7e43d1e 100644 --- a/src/Components/Modules/QuickPatch.hpp +++ b/src/Components/Modules/QuickPatch.hpp @@ -17,7 +17,6 @@ namespace Components static int FrameTime; static int64_t* GetStatsID(); - static void ShutdownStub(int num); static void SelectStringTableEntryInDvarStub(); diff --git a/src/Components/Modules/Renderer.cpp b/src/Components/Modules/Renderer.cpp index 37bfd0c3..424d7978 100644 --- a/src/Components/Modules/Renderer.cpp +++ b/src/Components/Modules/Renderer.cpp @@ -71,6 +71,29 @@ namespace Components return Utils::Hook::Get(0x66E1C6C); } + void Renderer::PreVidRestart() + { + Renderer::BeginRecoverDeviceSignal(); + } + + void Renderer::PostVidRestart() + { + Renderer::EndRecoverDeviceSignal(); + } + + __declspec(naked) void Renderer::PostVidRestartStub() + { + __asm + { + pushad + call Renderer::PostVidRestart + popad + + push 4F84C0h + retn + } + } + Renderer::Renderer() { if (Dedicated::IsEnabled()) return; @@ -108,17 +131,25 @@ namespace Components Utils::Hook(0x536A80, Renderer::BackendFrameStub, HOOK_JUMP).install()->quick(); + // Begin device recovery (not D3D9Ex) Utils::Hook(0x508298, [] () { Game::DB_BeginRecoverLostDevice(); Renderer::BeginRecoverDeviceSignal(); }, HOOK_CALL).install()->quick(); + // End device recovery (not D3D9Ex) Utils::Hook(0x508355, [] () { Renderer::EndRecoverDeviceSignal(); Game::DB_EndRecoverLostDevice(); }, HOOK_CALL).install()->quick(); + + // Begin vid_restart + Utils::Hook(0x4CA2FD, Renderer::PreVidRestart, HOOK_CALL).install()->quick(); + + // End vid_restart + Utils::Hook(0x4CA3A7, Renderer::PostVidRestartStub, HOOK_CALL).install()->quick(); } Renderer::~Renderer() diff --git a/src/Components/Modules/Renderer.hpp b/src/Components/Modules/Renderer.hpp index 25c428a4..50b8c338 100644 --- a/src/Components/Modules/Renderer.hpp +++ b/src/Components/Modules/Renderer.hpp @@ -24,6 +24,10 @@ namespace Components static void BackendFrameStub(); static void BackendFrameHandler(); + static void PreVidRestart(); + static void PostVidRestart(); + static void PostVidRestartStub(); + static Utils::Signal EndRecoverDeviceSignal; static Utils::Signal BeginRecoverDeviceSignal; diff --git a/src/Components/Modules/Toast.cpp b/src/Components/Modules/Toast.cpp index 00a7af9f..c9281965 100644 --- a/src/Components/Modules/Toast.cpp +++ b/src/Components/Modules/Toast.cpp @@ -59,9 +59,9 @@ namespace Components float fontSize = 0.9f; float descSize = 0.9f; - Game::Material* white = Game::DB_FindXAssetHeader(Game::XAssetType::ASSET_TYPE_MATERIAL, "white").material; - Game::Font* font = Game::DB_FindXAssetHeader(Game::XAssetType::ASSET_TYPE_FONT, "fonts/objectiveFont").font; - Game::Font* descfont = Game::DB_FindXAssetHeader(Game::XAssetType::ASSET_TYPE_FONT, "fonts/normalFont").font; + Game::Material* white = Game::DB_FindXAssetHeader(Game::XAssetType::ASSET_TYPE_MATERIAL, "white").material; if (!white) return; + Game::Font* font = Game::DB_FindXAssetHeader(Game::XAssetType::ASSET_TYPE_FONT, "fonts/objectiveFont").font; if (!font) return; + Game::Font* descfont = Game::DB_FindXAssetHeader(Game::XAssetType::ASSET_TYPE_FONT, "fonts/normalFont").font; if (!descfont) return; Game::vec4_t wColor = { 1.0f, 1.0f, 1.0f, 1.0f }; Game::vec4_t bgColor = { 0.0f, 0.0f, 0.0f, 0.5f }; Game::vec4_t borderColor = { 1.0f, 1.0f, 1.0f, 0.2f }; @@ -116,10 +116,9 @@ namespace Components Game::CL_DrawStretchPicPhysical(static_cast(width / 2 - bWidth / 2), static_cast(height + bHeight / 2), bWidth * 1.0f, border * 1.0f, 0, 0, 1.0f, 1.0f, borderColor, white); // Bottom // Image - if (toast->image && toast->image->textureTable && toast->image->textureCount && toast->image->textureTable->info.image && toast->image->textureTable->info.image->map) - { - Game::CL_DrawStretchPicPhysical(static_cast(width / 2 - bWidth / 2 + iOffsetLeft), static_cast(height - bHeight / 2 + iOffset), imgDim * 1.0f, imgDim * 1.0f, 0, 0, 1.0f, 1.0f, wColor, toast->image); - } + Game::Material* image = toast->image; + if (!Materials::IsValid(image)) image = Game::DB_FindXAssetDefaultHeaderInternal(Game::XAssetType::ASSET_TYPE_MATERIAL).material; + Game::CL_DrawStretchPicPhysical(static_cast(width / 2 - bWidth / 2 + iOffsetLeft), static_cast(height - bHeight / 2 + iOffset), imgDim * 1.0f, imgDim * 1.0f, 0, 0, 1.0f, 1.0f, wColor, image); // Text float leftText = width / 2 - bWidth / 2 - cornerSize + iOffsetLeft * 2 + imgDim;