vrclient: Add DXVK support for D3D11 submissions

This commit is contained in:
Philip Rebohle 2018-05-10 17:18:06 -07:00 committed by Pierre-Loup A. Griffais
parent bc88c17a36
commit e01a5a2855
3 changed files with 161 additions and 62 deletions

View File

@ -15,6 +15,7 @@
#include "vrclient_private.h" #include "vrclient_private.h"
#include "initguid.h" #include "initguid.h"
#include "dxvk-interop.h"
#include "wined3d-interop.h" #include "wined3d-interop.h"
#include "cppIVRCompositor_IVRCompositor_021.h" #include "cppIVRCompositor_IVRCompositor_021.h"
@ -478,21 +479,21 @@ EVRCompositorError ivrcompositor_submit(
ID3D11Device *device; ID3D11Device *device;
HRESULT hr; HRESULT hr;
IDXGIVkInteropSurface *dxvk_surface;
IDXGIVkInteropDevice *dxvk_device;
TRACE("%p, %#x, %p, %p, %#x\n", linux_side, eye, texture, bounds, flags); TRACE("%p, %#x, %p, %p, %#x\n", linux_side, eye, texture, bounds, flags);
switch (texture->eType) switch (texture->eType)
{ {
case TextureType_DirectX: case TextureType_DirectX:
{
TRACE("D3D11\n"); TRACE("D3D11\n");
texture_iface = texture->handle; texture_iface = texture->handle;
hr = texture_iface->lpVtbl->QueryInterface(texture_iface, hr = texture_iface->lpVtbl->QueryInterface(texture_iface,
&IID_IWineD3D11Texture2D, (void **)&wine_texture); &IID_IWineD3D11Texture2D, (void **)&wine_texture);
if (FAILED(hr)) if (SUCCEEDED(hr)) {
{
ERR("Invalid D3D11 texture %p.\n", texture);
return cpp_func(linux_side, eye, texture, bounds, flags);
}
wine_texture->lpVtbl->GetDevice(wine_texture, &device); wine_texture->lpVtbl->GetDevice(wine_texture, &device);
if (user_data->d3d11_device != device) if (user_data->d3d11_device != device)
@ -560,6 +561,90 @@ EVRCompositorError ivrcompositor_submit(
wine_texture->lpVtbl->Release(wine_texture); wine_texture->lpVtbl->Release(wine_texture);
return 0; return 0;
}
hr = texture_iface->lpVtbl->QueryInterface(texture_iface,
&IID_IDXGIVkInteropSurface, (void **)&dxvk_surface);
if (SUCCEEDED(hr)) {
struct VRVulkanTextureData_t vkdata;
struct Texture_t vktexture;
VkImage image_handle;
VkImageLayout image_layout;
VkImageCreateInfo image_info;
VkImageSubresourceRange subresources;
EVRCompositorError err;
dxvk_surface->lpVtbl->GetDevice(
dxvk_surface, &dxvk_device);
user_data->dxvk_device = dxvk_device;
dxvk_device->lpVtbl->GetVulkanHandles(
dxvk_device,
&vkdata.m_pInstance,
&vkdata.m_pPhysicalDevice,
&vkdata.m_pDevice);
dxvk_device->lpVtbl->GetSubmissionQueue(
dxvk_device,
&vkdata.m_pQueue,
&vkdata.m_nQueueFamilyIndex);
// DXVK needs this to be initialized correctly
image_info.sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO;
image_info.pNext = NULL;
dxvk_surface->lpVtbl->GetVulkanImageInfo(
dxvk_surface, &image_handle,
&image_layout, &image_info);
load_vk_unwrappers();
vkdata.m_nImage = (uint64_t)image_handle;
vkdata.m_pDevice = get_native_VkDevice(vkdata.m_pDevice);
vkdata.m_pPhysicalDevice = get_native_VkPhysicalDevice(vkdata.m_pPhysicalDevice);
vkdata.m_pInstance = get_native_VkInstance(vkdata.m_pInstance);
vkdata.m_pQueue = get_native_VkQueue(vkdata.m_pQueue);
vkdata.m_nWidth = image_info.extent.width;
vkdata.m_nHeight = image_info.extent.height;
vkdata.m_nFormat = image_info.format;
vkdata.m_nSampleCount = image_info.samples;
vktexture = *texture;
vktexture.handle = &vkdata;
vktexture.eType = TextureType_Vulkan;
subresources.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
subresources.baseMipLevel = 0;
subresources.levelCount = image_info.mipLevels;
subresources.baseArrayLayer = 0;
subresources.layerCount = image_info.arrayLayers;
dxvk_device->lpVtbl->TransitionSurfaceLayout(
dxvk_device, dxvk_surface, &subresources,
image_layout, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL);
dxvk_device->lpVtbl->FlushRenderingCommands(dxvk_device);
dxvk_device->lpVtbl->LockSubmissionQueue(dxvk_device);
err = cpp_func(linux_side, eye, &vktexture, bounds, flags);
dxvk_device->lpVtbl->ReleaseSubmissionQueue(dxvk_device);
dxvk_device->lpVtbl->TransitionSurfaceLayout(
dxvk_device, dxvk_surface, &subresources,
VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL, image_layout);
dxvk_device->lpVtbl->Release(dxvk_device);
dxvk_surface->lpVtbl->Release(dxvk_surface);
return err;
}
ERR("Invalid D3D11 texture %p.\n", texture);
return cpp_func(linux_side, eye, texture, bounds, flags);
}
case TextureType_Vulkan: case TextureType_Vulkan:
{ {
@ -670,7 +755,13 @@ void ivrcompositor_post_present_handoff(void (*cpp_func)(void *),
return; return;
} }
if (user_data->dxvk_device)
user_data->dxvk_device->lpVtbl->LockSubmissionQueue(user_data->dxvk_device);
cpp_func(linux_side); cpp_func(linux_side);
if (user_data->dxvk_device)
user_data->dxvk_device->lpVtbl->ReleaseSubmissionQueue(user_data->dxvk_device);
} }
struct explicit_timing_data struct explicit_timing_data
@ -714,8 +805,14 @@ EVRCompositorError ivrcompositor_wait_get_poses(
TRACE("%p, %p, %u, %p, %u\n", linux_side, render_poses, render_pose_count, game_poses, game_pose_count); TRACE("%p, %p, %u, %p, %u\n", linux_side, render_poses, render_pose_count, game_poses, game_pose_count);
if (user_data->dxvk_device)
user_data->dxvk_device->lpVtbl->LockSubmissionQueue(user_data->dxvk_device);
r = cpp_func(linux_side, render_poses, render_pose_count, game_poses, game_pose_count); r = cpp_func(linux_side, render_poses, render_pose_count, game_poses, game_pose_count);
if (user_data->dxvk_device)
user_data->dxvk_device->lpVtbl->ReleaseSubmissionQueue(user_data->dxvk_device);
if ((wined3d_device = user_data->wined3d_device)) if ((wined3d_device = user_data->wined3d_device))
{ {
TRACE("wined3d device %p\n", wined3d_device); TRACE("wined3d device %p\n", wined3d_device);

View File

@ -37,6 +37,7 @@ void *create_LinuxMatchmakingServerListResponse(void *win);
#ifndef __cplusplus #ifndef __cplusplus
typedef struct ID3D11Device ID3D11Device; typedef struct ID3D11Device ID3D11Device;
typedef struct IWineD3D11Device IWineD3D11Device; typedef struct IWineD3D11Device IWineD3D11Device;
typedef struct IDXGIVkInteropDevice IDXGIVkInteropDevice;
struct generic_interface struct generic_interface
{ {
@ -56,6 +57,7 @@ struct compositor_data
{ {
ID3D11Device *d3d11_device; ID3D11Device *d3d11_device;
IWineD3D11Device *wined3d_device; IWineD3D11Device *wined3d_device;
IDXGIVkInteropDevice *dxvk_device;
}; };
void destroy_compositor_data(struct compositor_data *data); void destroy_compositor_data(struct compositor_data *data);

2
wine

@ -1 +1 @@
Subproject commit 2855342f8188e687a6a93c4a0d9fb4ec5cbaa177 Subproject commit 348865780b2a950aeb6028f16cd6d3c6d8974155