From 7a3a853845adda1734f54d0afdda55ef5c9de888 Mon Sep 17 00:00:00 2001 From: Krzysztof Bogacki Date: Tue, 10 Dec 2024 12:06:10 +0100 Subject: [PATCH 1/3] util: Refactor DLSS version detection --- src/util/util_env.cpp | 46 ++++++++++++++++++++----------------------- 1 file changed, 21 insertions(+), 25 deletions(-) diff --git a/src/util/util_env.cpp b/src/util/util_env.cpp index ede74fbb..c8cb4c57 100644 --- a/src/util/util_env.cpp +++ b/src/util/util_env.cpp @@ -45,16 +45,13 @@ namespace dxvk::env { return stream.str(); } - // Function to check for WAR to DLSS Bug 3634851 if an affected DLSS DLL is - // detected this function will return true; false will be returned for all - // non-affected DLLs. - static bool isDLSSVersion20To24(void* pReturnAddress) { + static std::optional> getDLSSVersion(void* pReturnAddress) { // Get file path of caller DLL char modulePath[MAX_PATH]; HMODULE hModule = nullptr; if (!GetModuleHandleEx(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS | GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT, (const char*)pReturnAddress, &hModule)) { // Failed to get the path, won't try to WAR. - return false; + return {}; } uint32_t pathLen = GetModuleFileName(hModule, modulePath, MAX_PATH); @@ -66,57 +63,56 @@ namespace dxvk::env { // app_???????.bin matches OTA pattern } else { // Filename does not match known patterns - return false; + return {}; } // Get the version metadata from the DLL uint32_t infoSize = GetFileVersionInfoSizeA(modulePath, nullptr); if (infoSize == 0) { - return false; + return {}; } - auto* verInfo = new uint8_t[infoSize]; + auto verInfo = std::vector(infoSize); - if (!GetFileVersionInfoA(modulePath, 0, infoSize, verInfo)) { - delete[] verInfo; - return false; + if (!GetFileVersionInfoA(modulePath, 0, infoSize, verInfo.data())) { + return {}; } VS_FIXEDFILEINFO* fixedInfo = nullptr; uint32_t fixedSize = 0; - VerQueryValueA(verInfo, "\\", (void**)&fixedInfo, &fixedSize); + VerQueryValueA(verInfo.data(), "\\", (void**)&fixedInfo, &fixedSize); // Double-check that we are reading a VS_FIXEDFILEINFO structure if (fixedInfo->dwSignature != 0xFEEF04BD) { - delete[] verInfo; - return false; + return {}; } - // Only DLSS 2.x versions prior to 2.4 are affected by this bug uint32_t majorVersion = (fixedInfo->dwFileVersionMS & 0xFFFF0000) >> 16; uint32_t minorVersion = fixedInfo->dwFileVersionMS & 0x0000FFFF; - if (majorVersion != 2 || minorVersion >= 4) { - delete[] verInfo; - return false; - } // Lastly compare the product name to be absolutely certain we were // called by DLSS char* productName; uint32_t productNameSize; - VerQueryValueA(verInfo, R"(\StringFileInfo\040904E4\ProductName)", (void**)&productName, &productNameSize); + VerQueryValueA(verInfo.data(), R"(\StringFileInfo\040904E4\ProductName)", (void**)&productName, &productNameSize); if (strcmp(productName, "NVIDIA Deep Learning SuperSampling") == 0) { - // A DLSS version between 2.0 and 2.4 was detected, applying WAR - delete[] verInfo; - return true; + return std::make_pair(majorVersion, minorVersion); } else { - delete[] verInfo; - return false; + return {}; } } + // Function to check for WAR to DLSS Bug 3634851 if an affected DLSS DLL is + // detected this function will return true; false will be returned for all + // non-affected DLLs. + static bool isDLSSVersion20To24(void* returnAddress) { + auto [majorVersion, minorVersion] = getDLSSVersion(returnAddress).value_or(std::make_pair(0, 0)); + // Only DLSS 2.x versions prior to 2.4 are affected by this bug + return majorVersion == 2 && minorVersion < 4; + } + bool isTheGreatCircle() { static constexpr auto name = std::string_view("TheGreatCircle.exe"); return getExecutableName() == name; From 369f770f6ac922ccb10cb1853ede9303c2653c1c Mon Sep 17 00:00:00 2001 From: Krzysztof Bogacki Date: Tue, 10 Dec 2024 12:06:58 +0100 Subject: [PATCH 2/3] util: Add a function for detecting DLSS 2.0 --- src/util/util_env.cpp | 5 +++++ src/util/util_env.h | 2 ++ 2 files changed, 7 insertions(+) diff --git a/src/util/util_env.cpp b/src/util/util_env.cpp index c8cb4c57..7d8fd994 100644 --- a/src/util/util_env.cpp +++ b/src/util/util_env.cpp @@ -113,6 +113,11 @@ namespace dxvk::env { return majorVersion == 2 && minorVersion < 4; } + bool isDLSSVersion20(void* returnAddress) { + auto [majorVersion, minorVersion] = getDLSSVersion(returnAddress).value_or(std::make_pair(0, 0)); + return majorVersion == 2 && minorVersion == 0; + } + bool isTheGreatCircle() { static constexpr auto name = std::string_view("TheGreatCircle.exe"); return getExecutableName() == name; diff --git a/src/util/util_env.h b/src/util/util_env.h index 3d644c41..97d6ec68 100644 --- a/src/util/util_env.h +++ b/src/util/util_env.h @@ -9,6 +9,8 @@ namespace dxvk::env { std::string getCurrentDateTime(); + bool isDLSSVersion20(void* returnAddress); + bool needsSucceededGpuQuery(); std::optional needsGpuArchitectureSpoofing(NV_GPU_ARCHITECTURE_ID architectureId, void* returnAddress); From b5332ff6e2da70f9e5a70408ea0ca426a39a646c Mon Sep 17 00:00:00 2001 From: Krzysztof Bogacki Date: Tue, 10 Dec 2024 12:10:03 +0100 Subject: [PATCH 3/3] nvapi: Return no-implementation from NvAPI_GetLogicalGPUFromPhysicalGPU to DLSS 2.0 callers --- src/nvapi.cpp | 3 +++ src/nvapi_private.h | 2 +- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/src/nvapi.cpp b/src/nvapi.cpp index 02463122..aa8a10f6 100644 --- a/src/nvapi.cpp +++ b/src/nvapi.cpp @@ -152,6 +152,9 @@ extern "C" { if (log::tracing()) log::trace(n, log::fmt::hnd(hPhysicalGPU), log::fmt::ptr(pLogicalGPU)); + if (env::isDLSSVersion20(_ReturnAddress())) + return NoImplementation(n); + if (nvapiAdapterRegistry == nullptr) return ApiNotInitialized(n); diff --git a/src/nvapi_private.h b/src/nvapi_private.h index 16298634..1798309f 100644 --- a/src/nvapi_private.h +++ b/src/nvapi_private.h @@ -57,7 +57,7 @@ #endif // __GNUC__ #if defined(__GNUC__) -#define _ReturnAddress() __builtin_return_address(0); +#define _ReturnAddress() __builtin_return_address(0) #elif defined(_MSC_VER) #include #endif