Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature/detect dxvk #41

Merged
merged 7 commits into from
Oct 18, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ if(WIN32) # Setup some variables for Windows build
set(PRIVATE_INGAMEOVERLAY_HEADERS
src/VulkanHelpers.h
src/Windows/DirectXVTables.h
src/Windows/DXVKDetector.h
src/Windows/DX9Hook.h
src/Windows/DX10Hook.h
src/Windows/DX11Hook.h
Expand Down
1 change: 1 addition & 0 deletions include/InGameOverlay/RendererDetector.h
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ namespace InGameOverlay {
/// Starts a detector to automatically find the renderer used by the application.
/// </summary>
/// <param name="timeout">The time before the future will timeout if no renderer has been found.</param>
/// <param name="rendererToDetect">Set this to any combined RendererHookType_t value to filter the renderers you want to detect.</param>
/// <param name="preferSystemLibraries">Prefer hooking the system libraries instead of the first one found.</param>
/// <returns>A future nullptr or the renderer.</returns>
std::future<RendererHook_t*> DetectRenderer(std::chrono::milliseconds timeout = std::chrono::milliseconds{ -1 }, RendererHookType_t rendererToDetect = RendererHookType_t::Any, bool preferSystemLibraries = true);
Expand Down
2 changes: 2 additions & 0 deletions include/InGameOverlay/RendererHook.h
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,8 @@ enum class OverlayHookState : uint8_t
/// <summary>
/// Only one RendererHookType_t will always be returned by RendererHook_t::GetRendererHookType
/// but you can use them as flags to limit the detection in InGameOverlay::DetectRenderer.
/// NOTE: Even if a DirectX hook has been found and DXVK is used, the renderer hook will only report the DirectX hook type.
/// There is no combination like : DirectX11 | Vulkan returned by RendererHook_t::GetRendererHookType.
/// </summary>
enum class RendererHookType_t : uint8_t
{
Expand Down
4 changes: 2 additions & 2 deletions src/Linux/OpenGLXHook.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@

namespace InGameOverlay {

#define TRY_HOOK_FUNCTION(NAME) do { if (!HookFunc(std::make_pair<void**, void*>(&(void*&)_##NAME, (void*)&OpenGLXHook_t::_My##NAME))) { \
#define TRY_HOOK_FUNCTION_OR_FAIL(NAME) do { if (!HookFunc(std::make_pair<void**, void*>(&(void*&)_##NAME, (void*)&OpenGLXHook_t::_My##NAME))) { \
INGAMEOVERLAY_ERROR("Failed to hook {}", #NAME);\
UnhookAll();\
return false;\
Expand All @@ -51,7 +51,7 @@ bool OpenGLXHook_t::StartHook(std::function<void()> keyCombinationCallback, Togg
_X11Hooked = true;

BeginHook();
TRY_HOOK_FUNCTION(GLXSwapBuffers);
TRY_HOOK_FUNCTION_OR_FAIL(GLXSwapBuffers);
EndHook();

INGAMEOVERLAY_INFO("Hooked OpenGLX");
Expand Down
33 changes: 24 additions & 9 deletions src/Linux/VulkanHook.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@

namespace InGameOverlay {

#define TRY_HOOK_FUNCTION(NAME) do { if (!HookFunc(std::make_pair<void**, void*>(&(void*&)_##NAME, (void*)&VulkanHook_t::_My##NAME))) { \
#define TRY_HOOK_FUNCTION_OR_FAIL(NAME) do { if (!HookFunc(std::make_pair<void**, void*>(&(void*&)_##NAME, (void*)&VulkanHook_t::_My##NAME))) { \
INGAMEOVERLAY_ERROR("Failed to hook {}", #NAME);\
return false;\
} } while(0)
Expand Down Expand Up @@ -74,13 +74,13 @@ bool VulkanHook_t::StartHook(std::function<void()> keyCombinationCallback, Toggl
_X11Hooked = true;

BeginHook();
TRY_HOOK_FUNCTION(VkAcquireNextImageKHR);
TRY_HOOK_FUNCTION_OR_FAIL(VkAcquireNextImageKHR);
if (_VkAcquireNextImage2KHR != nullptr)
TRY_HOOK_FUNCTION(VkAcquireNextImage2KHR);
TRY_HOOK_FUNCTION_OR_FAIL(VkAcquireNextImage2KHR);

TRY_HOOK_FUNCTION(VkQueuePresentKHR);
TRY_HOOK_FUNCTION(VkCreateSwapchainKHR);
TRY_HOOK_FUNCTION(VkDestroyDevice);
TRY_HOOK_FUNCTION_OR_FAIL(VkQueuePresentKHR);
TRY_HOOK_FUNCTION_OR_FAIL(VkCreateSwapchainKHR);
TRY_HOOK_FUNCTION_OR_FAIL(VkDestroyDevice);
EndHook();

INGAMEOVERLAY_INFO("Hooked Vulkan");
Expand Down Expand Up @@ -269,7 +269,7 @@ bool VulkanHook_t::_CreateRenderTargets(VkSwapchainKHR swapChain)
VkImageViewCreateInfo info = { };
info.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
info.viewType = VK_IMAGE_VIEW_TYPE_2D;
info.format = VK_FORMAT_B8G8R8A8_UNORM;
info.format = _VulkanTargetFormat;
info.image = frame.BackBuffer;

info.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
Expand Down Expand Up @@ -353,6 +353,8 @@ void VulkanHook_t::_ResetRenderState(OverlayHookState state)
_ImageResources.clear();

_FreeVulkanRessources();

_SentOutOfDate = false;
}
}

Expand Down Expand Up @@ -454,6 +456,7 @@ bool VulkanHook_t::_CreateVulkanInstance()
LOAD_VULKAN_FUNCTION(vkGetPhysicalDeviceMemoryProperties);
LOAD_VULKAN_FUNCTION(vkEnumerateDeviceExtensionProperties);
LOAD_VULKAN_FUNCTION(vkEnumeratePhysicalDevices);
LOAD_VULKAN_FUNCTION(vkGetPhysicalDeviceSurfaceFormatsKHR);
LOAD_VULKAN_FUNCTION(vkGetPhysicalDeviceProperties);
LOAD_VULKAN_FUNCTION(vkGetPhysicalDeviceQueueFamilyProperties);
#undef LOAD_VULKAN_FUNCTION
Expand Down Expand Up @@ -672,7 +675,7 @@ bool VulkanHook_t::_CreateRenderPass()
return true;

VkAttachmentDescription attachment = { };
attachment.format = VK_FORMAT_B8G8R8A8_UNORM;
attachment.format = _VulkanTargetFormat;
attachment.samples = VK_SAMPLE_COUNT_1_BIT;
attachment.loadOp = VK_ATTACHMENT_LOAD_OP_LOAD;
attachment.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
Expand Down Expand Up @@ -925,6 +928,13 @@ VKAPI_ATTR VkResult VKAPI_CALL VulkanHook_t::_MyVkQueuePresentKHR(VkQueue queue,
INGAMEOVERLAY_INFO("vkQueuePresentKHR");
auto inst = VulkanHook_t::Inst();

// Send out of date khr and see if the game recreates its swapchain, so we can get the rendering color space :p
if (!inst->_SentOutOfDate)
{
inst->_SentOutOfDate = true;
return VkResult::VK_ERROR_OUT_OF_DATE_KHR;
}

inst->_PrepareForOverlay(queue, pPresentInfo);
return inst->_VkQueuePresentKHR(queue, pPresentInfo);
}
Expand All @@ -933,14 +943,17 @@ VKAPI_ATTR VkResult VKAPI_CALL VulkanHook_t::_MyVkCreateSwapchainKHR(VkDevice de
{
INGAMEOVERLAY_INFO("vkCreateSwapchainKHR");
auto inst = VulkanHook_t::Inst();
auto createRenderTargets = false;

if (inst->_VulkanDevice == device)
{
createRenderTargets = !inst->_Frames.empty();
inst->_ResetRenderState(OverlayHookState::Reset);
inst->_DestroyRenderTargets();
}
inst->_VulkanTargetFormat = pCreateInfo->imageFormat;
auto res = inst->_VkCreateSwapchainKHR(device, pCreateInfo, pAllocator, pSwapchain);
if (inst->_VulkanDevice == device && res == VkResult::VK_SUCCESS)
if (inst->_VulkanDevice == device && res == VkResult::VK_SUCCESS && createRenderTargets)
{
inst->_CreateRenderTargets(*pSwapchain);
inst->_ResetRenderState(OverlayHookState::Ready);
Expand Down Expand Up @@ -974,6 +987,7 @@ VulkanHook_t::VulkanHook_t():
_VulkanImageSampler(VK_NULL_HANDLE),
_VulkanImageDescriptorSetLayout(VK_NULL_HANDLE),
_VulkanRenderPass(VK_NULL_HANDLE),
_VulkanTargetFormat(VK_FORMAT_R8G8B8A8_UNORM),
_VulkanDevice(VK_NULL_HANDLE),
_VulkanQueue(VK_NULL_HANDLE),
_ImGuiFontAtlas(nullptr),
Expand Down Expand Up @@ -1039,6 +1053,7 @@ VulkanHook_t::VulkanHook_t():
_vkGetBufferMemoryRequirements(nullptr),
_vkGetImageMemoryRequirements(nullptr),
_vkEnumeratePhysicalDevices(nullptr),
_vkGetPhysicalDeviceSurfaceFormatsKHR(nullptr),
_vkGetPhysicalDeviceProperties(nullptr),
_vkGetPhysicalDeviceQueueFamilyProperties(nullptr),
_vkGetPhysicalDeviceMemoryProperties(nullptr),
Expand Down
3 changes: 3 additions & 0 deletions src/Linux/VulkanHook.h
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,7 @@ class VulkanHook_t :
// Variables
bool _Hooked;
bool _X11Hooked;
bool _SentOutOfDate;
void* _Window;
OverlayHookState _HookState;

Expand All @@ -82,6 +83,7 @@ class VulkanHook_t :
std::vector<VulkanFrame_t> _Frames;
VkRenderPass _VulkanRenderPass;
std::vector<VulkanDescriptorPool_t> _DescriptorsPools;
VkFormat _VulkanTargetFormat;

VkDevice _VulkanDevice;
VkQueue _VulkanQueue;
Expand Down Expand Up @@ -203,6 +205,7 @@ class VulkanHook_t :
decltype(::vkGetBufferMemoryRequirements) *_vkGetBufferMemoryRequirements;
decltype(::vkGetImageMemoryRequirements) *_vkGetImageMemoryRequirements;
decltype(::vkEnumeratePhysicalDevices) *_vkEnumeratePhysicalDevices;
decltype(::vkGetPhysicalDeviceSurfaceFormatsKHR) *_vkGetPhysicalDeviceSurfaceFormatsKHR;
decltype(::vkGetPhysicalDeviceProperties) *_vkGetPhysicalDeviceProperties;
decltype(::vkGetPhysicalDeviceQueueFamilyProperties) *_vkGetPhysicalDeviceQueueFamilyProperties;
decltype(::vkGetPhysicalDeviceMemoryProperties) *_vkGetPhysicalDeviceMemoryProperties;
Expand Down
4 changes: 2 additions & 2 deletions src/MacOSX/OpenGLHook.mm
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@

namespace InGameOverlay {

#define TRY_HOOK_FUNCTION(NAME) do { if (!HookFunc(std::make_pair<void**, void*>(&(void*&)_##NAME, (void*)&OpenGLHook_t::_My##NAME))) { \
#define TRY_HOOK_FUNCTION_OR_FAIL(NAME) do { if (!HookFunc(std::make_pair<void**, void*>(&(void*&)_##NAME, (void*)&OpenGLHook_t::_My##NAME))) { \
INGAMEOVERLAY_ERROR("Failed to hook {}", #NAME);\
UnhookAll();\
return false;\
Expand Down Expand Up @@ -63,7 +63,7 @@ static bool ImGuiOpenGL3Init()
else if (_CGLFlushDrawable != nullptr)
{
BeginHook();
TRY_HOOK_FUNCTION(CGLFlushDrawable);
TRY_HOOK_FUNCTION_OR_FAIL(CGLFlushDrawable);
EndHook();
}

Expand Down
22 changes: 18 additions & 4 deletions src/Windows/DX10Hook.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,10 @@ namespace InGameOverlay {

#define TRY_HOOK_FUNCTION(NAME) do { if (!HookFunc(std::make_pair<void**, void*>(&(void*&)_##NAME, (void*)&DX10Hook_t::_My##NAME))) { \
INGAMEOVERLAY_ERROR("Failed to hook {}", #NAME);\
} } while(0)

#define TRY_HOOK_FUNCTION_OR_FAIL(NAME) do { if (!HookFunc(std::make_pair<void**, void*>(&(void*&)_##NAME, (void*)&DX10Hook_t::_My##NAME))) { \
INGAMEOVERLAY_ERROR("Failed to hook {}", #NAME);\
UnhookAll();\
return false;\
} } while(0)
Expand Down Expand Up @@ -60,12 +64,12 @@ bool DX10Hook_t::StartHook(std::function<void()> keyCombinationCallback, ToggleK

BeginHook();
TRY_HOOK_FUNCTION(ID3D10DeviceRelease);
TRY_HOOK_FUNCTION(IDXGISwapChainPresent);
TRY_HOOK_FUNCTION(IDXGISwapChainResizeTarget);
TRY_HOOK_FUNCTION(IDXGISwapChainResizeBuffers);
TRY_HOOK_FUNCTION_OR_FAIL(IDXGISwapChainPresent);
TRY_HOOK_FUNCTION_OR_FAIL(IDXGISwapChainResizeTarget);
TRY_HOOK_FUNCTION_OR_FAIL(IDXGISwapChainResizeBuffers);

if (_IDXGISwapChain1Present1 != nullptr)
TRY_HOOK_FUNCTION(IDXGISwapChain1Present1);
TRY_HOOK_FUNCTION_OR_FAIL(IDXGISwapChain1Present1);

EndHook();

Expand Down Expand Up @@ -275,6 +279,7 @@ HRESULT STDMETHODCALLTYPE DX10Hook_t::_MyIDXGISwapChain1Present1(IDXGISwapChain1
DX10Hook_t::DX10Hook_t():
_Hooked(false),
_WindowsHooked(false),
_UsesDXVK(false),
_DeviceReleasing(0),
_Device(nullptr),
_HookDeviceRefCount(0),
Expand Down Expand Up @@ -320,6 +325,15 @@ RendererHookType_t DX10Hook_t::GetRendererHookType() const
return RendererHookType_t::DirectX10;
}

void DX10Hook_t::SetDXVK()
{
if (!_UsesDXVK)
{
_UsesDXVK = true;
LibraryName += " (DXVK)";
}
}

void DX10Hook_t::LoadFunctions(
decltype(_ID3D10DeviceRelease) releaseFcn,
decltype(_IDXGISwapChainPresent) presentFcn,
Expand Down
2 changes: 2 additions & 0 deletions src/Windows/DX10Hook.h
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ class DX10Hook_t :
// Variables
bool _Hooked;
bool _WindowsHooked;
bool _UsesDXVK;
uint32_t _DeviceReleasing;
ID3D10Device* _Device;
ULONG _HookDeviceRefCount;
Expand Down Expand Up @@ -79,6 +80,7 @@ class DX10Hook_t :
virtual const char* GetLibraryName() const;
virtual RendererHookType_t GetRendererHookType() const;

void SetDXVK();
void LoadFunctions(
decltype(_ID3D10DeviceRelease) releaseFcn,
decltype(_IDXGISwapChainPresent) presentFcn,
Expand Down
27 changes: 21 additions & 6 deletions src/Windows/DX11Hook.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,10 @@ namespace InGameOverlay {

#define TRY_HOOK_FUNCTION(NAME) do { if (!HookFunc(std::make_pair<void**, void*>(&(void*&)_##NAME, (void*)&DX11Hook_t::_My##NAME))) { \
INGAMEOVERLAY_ERROR("Failed to hook {}", #NAME);\
} } while(0)

#define TRY_HOOK_FUNCTION_OR_FAIL(NAME) do { if (!HookFunc(std::make_pair<void**, void*>(&(void*&)_##NAME, (void*)&DX11Hook_t::_My##NAME))) { \
INGAMEOVERLAY_ERROR("Failed to hook {}", #NAME);\
UnhookAll();\
return false;\
} } while(0)
Expand Down Expand Up @@ -70,12 +74,12 @@ bool DX11Hook_t::StartHook(std::function<void()> keyCombinationCallback, ToggleK

BeginHook();
TRY_HOOK_FUNCTION(ID3D11DeviceRelease);
TRY_HOOK_FUNCTION(IDXGISwapChainPresent);
TRY_HOOK_FUNCTION(IDXGISwapChainResizeTarget);
TRY_HOOK_FUNCTION(IDXGISwapChainResizeBuffers);
TRY_HOOK_FUNCTION_OR_FAIL(IDXGISwapChainPresent);
TRY_HOOK_FUNCTION_OR_FAIL(IDXGISwapChainResizeTarget);
TRY_HOOK_FUNCTION_OR_FAIL(IDXGISwapChainResizeBuffers);

if (_IDXGISwapChain1Present1 != nullptr)
TRY_HOOK_FUNCTION(IDXGISwapChain1Present1);
TRY_HOOK_FUNCTION_OR_FAIL(IDXGISwapChain1Present1);

EndHook();

Expand Down Expand Up @@ -105,14 +109,15 @@ bool DX11Hook_t::IsStarted()

void DX11Hook_t::_UpdateHookDeviceRefCount()
{
const int BaseRefCount = 2;
switch (_HookState)
{
// 0 ref from ImGui
case OverlayHookState::Removing: _HookDeviceRefCount = 2; break;
case OverlayHookState::Removing: _HookDeviceRefCount = BaseRefCount; break;
// 1 refs from us, 10 refs from ImGui (device, vertex shader, input layout, vertex constant buffer, pixel shader, blend state, rasterizer state, depth stencil state, texture view, texture sample)
//case OverlayHookState::Reset: _HookDeviceRefCount = 15 + _ImageResources.size(); break;
// 1 refs from us, 12 refs from ImGui (device, vertex shader, input layout, vertex constant buffer, pixel shader, blend state, rasterizer state, depth stencil state, texture view, texture sample, vertex buffer, index buffer)
case OverlayHookState::Ready: _HookDeviceRefCount = 17 + _ImageResources.size();
case OverlayHookState::Ready: _HookDeviceRefCount = BaseRefCount + 15 + _ImageResources.size();
}
}

Expand Down Expand Up @@ -311,6 +316,7 @@ HRESULT STDMETHODCALLTYPE DX11Hook_t::_MyIDXGISwapChain1Present1(IDXGISwapChain1
DX11Hook_t::DX11Hook_t():
_Hooked(false),
_WindowsHooked(false),
_UsesDXVK(false),
_DeviceReleasing(0),
_Device(nullptr),
_HookDeviceRefCount(0),
Expand Down Expand Up @@ -357,6 +363,15 @@ RendererHookType_t DX11Hook_t::GetRendererHookType() const
return RendererHookType_t::DirectX11;
}

void DX11Hook_t::SetDXVK()
{
if (!_UsesDXVK)
{
_UsesDXVK = true;
LibraryName += " (DXVK)";
}
}

void DX11Hook_t::LoadFunctions(
decltype(_ID3D11DeviceRelease) releaseFcn,
decltype(_IDXGISwapChainPresent) presentFcn,
Expand Down
2 changes: 2 additions & 0 deletions src/Windows/DX11Hook.h
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ class DX11Hook_t :
// Variables
bool _Hooked;
bool _WindowsHooked;
bool _UsesDXVK;
uint32_t _DeviceReleasing;
ID3D11Device* _Device;
ULONG _HookDeviceRefCount;
Expand Down Expand Up @@ -80,6 +81,7 @@ class DX11Hook_t :
virtual const char* GetLibraryName() const;
virtual RendererHookType_t GetRendererHookType() const;

void SetDXVK();
void LoadFunctions(
decltype(_ID3D11DeviceRelease) releaseFcn,
decltype(_IDXGISwapChainPresent) presentFcn,
Expand Down
Loading