2020-07-02 16:44:53 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2018 Józef Kucia for CodeWeavers
|
|
|
|
* Copyright 2020 Joshua Ashton for Valve Software
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2020-07-09 15:02:08 +01:00
|
|
|
#define VKD3D_DBG_CHANNEL VKD3D_DBG_CHANNEL_API
|
|
|
|
|
2020-07-02 16:44:53 +01:00
|
|
|
#define VK_NO_PROTOTYPES
|
|
|
|
#include "vkd3d_win32.h"
|
|
|
|
#include "vkd3d_atomic.h"
|
|
|
|
#include "vkd3d_debug.h"
|
2020-07-10 13:55:24 +01:00
|
|
|
#include "vkd3d_threads.h"
|
2020-07-02 16:44:53 +01:00
|
|
|
|
2020-07-07 16:40:05 +01:00
|
|
|
/* We need to specify the __declspec(dllexport) attribute
|
|
|
|
* on MinGW because otherwise the stdcall aliases/fixups
|
|
|
|
* don't get exported.
|
|
|
|
*/
|
|
|
|
#if defined(_MSC_VER)
|
|
|
|
#define DLLEXPORT
|
|
|
|
#else
|
|
|
|
#define DLLEXPORT __declspec(dllexport)
|
|
|
|
#endif
|
|
|
|
|
2020-07-10 13:55:24 +01:00
|
|
|
static pthread_once_t library_once = PTHREAD_ONCE_INIT;
|
2020-07-02 16:44:53 +01:00
|
|
|
static HMODULE vulkan_module = NULL;
|
|
|
|
|
2020-07-10 13:55:24 +01:00
|
|
|
static void load_vulkan_once(void)
|
2020-07-02 16:44:53 +01:00
|
|
|
{
|
|
|
|
if (!vulkan_module)
|
|
|
|
vulkan_module = LoadLibraryA("vulkan-1.dll");
|
2020-07-10 13:55:24 +01:00
|
|
|
}
|
2020-07-02 16:44:53 +01:00
|
|
|
|
2020-07-10 13:55:24 +01:00
|
|
|
static PFN_vkGetInstanceProcAddr load_vulkan(void)
|
|
|
|
{
|
|
|
|
pthread_once(&library_once, load_vulkan_once);
|
2020-07-02 16:44:53 +01:00
|
|
|
if (vulkan_module)
|
|
|
|
return (void *)GetProcAddress(vulkan_module, "vkGetInstanceProcAddr");
|
2020-07-10 13:55:24 +01:00
|
|
|
else
|
|
|
|
return NULL;
|
2020-07-02 16:44:53 +01:00
|
|
|
}
|
|
|
|
|
2020-07-07 16:40:05 +01:00
|
|
|
HRESULT WINAPI DLLEXPORT D3D12GetDebugInterface(REFIID iid, void **debug)
|
2020-07-02 16:44:53 +01:00
|
|
|
{
|
|
|
|
TRACE("iid %s, debug %p.\n", debugstr_guid(iid), debug);
|
|
|
|
|
|
|
|
WARN("Returning DXGI_ERROR_SDK_COMPONENT_MISSING.\n");
|
|
|
|
return DXGI_ERROR_SDK_COMPONENT_MISSING;
|
|
|
|
}
|
|
|
|
|
2020-07-07 16:40:05 +01:00
|
|
|
HRESULT WINAPI DLLEXPORT D3D12EnableExperimentalFeatures(UINT feature_count,
|
2020-07-02 16:44:53 +01:00
|
|
|
const IID *iids, void *configurations, UINT *configurations_sizes)
|
|
|
|
{
|
|
|
|
FIXME("feature_count %u, iids %p, configurations %p, configurations_sizes %p stub!\n",
|
|
|
|
feature_count, iids, configurations, configurations_sizes);
|
|
|
|
|
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT d3d12_signal_event(HANDLE event)
|
|
|
|
{
|
|
|
|
return SetEvent(event) ? S_OK : E_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct d3d12_thread_data
|
|
|
|
{
|
|
|
|
PFN_vkd3d_thread main_pfn;
|
|
|
|
void *data;
|
|
|
|
};
|
|
|
|
|
|
|
|
static DWORD WINAPI d3d12_thread_main(void *data)
|
|
|
|
{
|
|
|
|
struct d3d12_thread_data *thread_data = data;
|
|
|
|
|
|
|
|
thread_data->main_pfn(thread_data->data);
|
|
|
|
free(thread_data);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *d3d12_create_thread(PFN_vkd3d_thread main_pfn, void *data)
|
|
|
|
{
|
|
|
|
struct d3d12_thread_data *thread_data;
|
|
|
|
HANDLE thread;
|
|
|
|
|
|
|
|
if (!(thread_data = malloc(sizeof(*thread_data))))
|
|
|
|
{
|
|
|
|
ERR("Failed to allocate thread data.\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
thread_data->main_pfn = main_pfn;
|
|
|
|
thread_data->data = data;
|
|
|
|
|
|
|
|
if (!(thread = CreateThread(NULL, 0, d3d12_thread_main, thread_data, 0, NULL)))
|
|
|
|
free(thread_data);
|
|
|
|
|
|
|
|
return thread;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT d3d12_join_thread(void *handle)
|
|
|
|
{
|
|
|
|
HANDLE thread = handle;
|
|
|
|
DWORD ret;
|
|
|
|
|
|
|
|
if ((ret = WaitForSingleObject(thread, INFINITE)) != WAIT_OBJECT_0)
|
|
|
|
{
|
|
|
|
ERR("Failed to wait for thread, ret %#x.\n", ret);
|
|
|
|
}
|
|
|
|
CloseHandle(thread);
|
|
|
|
return ret == WAIT_OBJECT_0 ? S_OK : E_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT d3d12_get_adapter(IDXGIAdapter **dxgi_adapter, IUnknown *adapter)
|
|
|
|
{
|
|
|
|
IDXGIFactory4 *factory = NULL;
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
if (!adapter)
|
|
|
|
{
|
2020-07-07 14:03:45 +01:00
|
|
|
if (FAILED(hr = CreateDXGIFactory1(&IID_IDXGIFactory4, (void **)&factory)))
|
2020-07-02 16:44:53 +01:00
|
|
|
{
|
|
|
|
WARN("Failed to create DXGI factory, hr %#x.\n", hr);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FAILED(hr = IDXGIFactory4_EnumAdapters(factory, 0, dxgi_adapter)))
|
|
|
|
{
|
|
|
|
WARN("Failed to enumerate primary adapter, hr %#x.\n", hr);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (FAILED(hr = IUnknown_QueryInterface(adapter, &IID_IDXGIAdapter, (void **)dxgi_adapter)))
|
|
|
|
{
|
|
|
|
WARN("Invalid adapter %p, hr %#x.\n", adapter, hr);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
|
|
|
if (factory)
|
|
|
|
IDXGIFactory4_Release(factory);
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
2021-03-16 16:17:01 +00:00
|
|
|
static VkPhysicalDevice d3d12_find_physical_device(struct vkd3d_instance *instance,
|
2020-07-02 16:44:53 +01:00
|
|
|
PFN_vkGetInstanceProcAddr pfn_vkGetInstanceProcAddr, struct DXGI_ADAPTER_DESC *adapter_desc)
|
|
|
|
{
|
2021-03-16 16:09:43 +00:00
|
|
|
PFN_vkGetPhysicalDeviceProperties2 pfn_vkGetPhysicalDeviceProperties2;
|
2020-07-02 16:44:53 +01:00
|
|
|
PFN_vkGetPhysicalDeviceProperties pfn_vkGetPhysicalDeviceProperties;
|
|
|
|
PFN_vkEnumeratePhysicalDevices pfn_vkEnumeratePhysicalDevices;
|
|
|
|
VkPhysicalDevice vk_physical_device = VK_NULL_HANDLE;
|
|
|
|
VkPhysicalDeviceIDProperties id_properties;
|
|
|
|
VkPhysicalDeviceProperties2 properties2;
|
|
|
|
VkPhysicalDevice *vk_physical_devices;
|
|
|
|
VkInstance vk_instance;
|
|
|
|
unsigned int i;
|
|
|
|
uint32_t count;
|
|
|
|
VkResult vr;
|
|
|
|
|
|
|
|
vk_instance = vkd3d_instance_get_vk_instance(instance);
|
|
|
|
|
|
|
|
pfn_vkEnumeratePhysicalDevices = (void *)pfn_vkGetInstanceProcAddr(vk_instance, "vkEnumeratePhysicalDevices");
|
|
|
|
pfn_vkGetPhysicalDeviceProperties = (void *)pfn_vkGetInstanceProcAddr(vk_instance, "vkGetPhysicalDeviceProperties");
|
2021-03-16 16:09:43 +00:00
|
|
|
pfn_vkGetPhysicalDeviceProperties2 = (void *)pfn_vkGetInstanceProcAddr(vk_instance, "vkGetPhysicalDeviceProperties2");
|
2020-07-02 16:44:53 +01:00
|
|
|
|
|
|
|
if ((vr = pfn_vkEnumeratePhysicalDevices(vk_instance, &count, NULL)) < 0)
|
|
|
|
{
|
|
|
|
WARN("Failed to get device count, vr %d.\n", vr);
|
|
|
|
return VK_NULL_HANDLE;
|
|
|
|
}
|
|
|
|
if (!count)
|
|
|
|
{
|
|
|
|
WARN("No physical device available.\n");
|
|
|
|
return VK_NULL_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(vk_physical_devices = calloc(count, sizeof(*vk_physical_devices))))
|
|
|
|
return VK_NULL_HANDLE;
|
|
|
|
|
|
|
|
if ((vr = pfn_vkEnumeratePhysicalDevices(vk_instance, &count, vk_physical_devices)) < 0)
|
|
|
|
goto done;
|
|
|
|
|
2021-03-16 16:09:43 +00:00
|
|
|
TRACE("Matching adapters by LUIDs.\n");
|
|
|
|
|
|
|
|
for (i = 0; i < count; ++i)
|
2020-07-02 16:44:53 +01:00
|
|
|
{
|
2021-03-16 16:09:43 +00:00
|
|
|
pfn_vkGetPhysicalDeviceProperties(vk_physical_devices[i], &properties2.properties);
|
2020-07-02 16:44:53 +01:00
|
|
|
|
2021-03-16 16:09:43 +00:00
|
|
|
/* Skip over physical devices below our minimum API version */
|
|
|
|
if (properties2.properties.apiVersion < VKD3D_MIN_API_VERSION)
|
2020-07-02 16:44:53 +01:00
|
|
|
{
|
2021-03-16 16:09:43 +00:00
|
|
|
WARN("Skipped adapter %s as it is below our minimum API version.", properties2.properties.deviceName);
|
|
|
|
continue;
|
|
|
|
}
|
2020-07-02 16:44:53 +01:00
|
|
|
|
2021-03-16 16:09:43 +00:00
|
|
|
id_properties.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_ID_PROPERTIES;
|
|
|
|
id_properties.pNext = NULL;
|
2020-07-02 16:44:53 +01:00
|
|
|
|
2021-03-16 16:09:43 +00:00
|
|
|
properties2.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2;
|
|
|
|
properties2.pNext = &id_properties;
|
2020-07-02 16:44:53 +01:00
|
|
|
|
2021-03-16 16:09:43 +00:00
|
|
|
pfn_vkGetPhysicalDeviceProperties2(vk_physical_devices[i], &properties2);
|
|
|
|
|
|
|
|
if (id_properties.deviceLUIDValid && !memcmp(id_properties.deviceLUID, &adapter_desc->AdapterLuid, VK_LUID_SIZE))
|
|
|
|
{
|
|
|
|
vk_physical_device = vk_physical_devices[i];
|
|
|
|
break;
|
2020-07-02 16:44:53 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-16 16:09:43 +00:00
|
|
|
if (!vk_physical_device)
|
2020-07-02 16:44:53 +01:00
|
|
|
{
|
2021-03-16 16:09:43 +00:00
|
|
|
TRACE("Matching adapters by PCI IDs.\n");
|
2020-07-02 16:44:53 +01:00
|
|
|
|
2021-03-16 16:09:43 +00:00
|
|
|
for (i = 0; i < count; ++i)
|
2020-07-02 16:44:53 +01:00
|
|
|
{
|
2021-03-16 16:09:43 +00:00
|
|
|
pfn_vkGetPhysicalDeviceProperties(vk_physical_devices[i], &properties2.properties);
|
|
|
|
|
|
|
|
if (properties2.properties.deviceID == adapter_desc->DeviceId &&
|
|
|
|
properties2.properties.vendorID == adapter_desc->VendorId)
|
|
|
|
{
|
|
|
|
vk_physical_device = vk_physical_devices[i];
|
|
|
|
break;
|
|
|
|
}
|
2020-07-02 16:44:53 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!vk_physical_device)
|
|
|
|
{
|
|
|
|
FIXME("Could not find Vulkan physical device for DXGI adapter.\n");
|
|
|
|
WARN("Using first available physical device...\n");
|
|
|
|
vk_physical_device = vk_physical_devices[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
|
|
|
free(vk_physical_devices);
|
|
|
|
return vk_physical_device;
|
|
|
|
}
|
|
|
|
|
2020-07-07 16:40:05 +01:00
|
|
|
HRESULT WINAPI DLLEXPORT D3D12CreateDevice(IUnknown *adapter, D3D_FEATURE_LEVEL minimum_feature_level,
|
2020-07-02 16:44:53 +01:00
|
|
|
REFIID iid, void **device)
|
|
|
|
{
|
|
|
|
struct vkd3d_instance_create_info instance_create_info;
|
|
|
|
PFN_vkGetInstanceProcAddr pfn_vkGetInstanceProcAddr;
|
|
|
|
struct vkd3d_device_create_info device_create_info;
|
|
|
|
struct DXGI_ADAPTER_DESC adapter_desc;
|
|
|
|
struct vkd3d_instance *instance;
|
|
|
|
IDXGIAdapter *dxgi_adapter;
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
static const char * const instance_extensions[] =
|
|
|
|
{
|
|
|
|
VK_KHR_SURFACE_EXTENSION_NAME,
|
|
|
|
VK_KHR_WIN32_SURFACE_EXTENSION_NAME,
|
|
|
|
};
|
|
|
|
static const char * const device_extensions[] =
|
|
|
|
{
|
|
|
|
VK_KHR_SWAPCHAIN_EXTENSION_NAME,
|
|
|
|
};
|
|
|
|
|
|
|
|
TRACE("adapter %p, minimum_feature_level %#x, iid %s, device %p.\n",
|
|
|
|
adapter, minimum_feature_level, debugstr_guid(iid), device);
|
|
|
|
|
|
|
|
if (!(pfn_vkGetInstanceProcAddr = load_vulkan()))
|
|
|
|
{
|
|
|
|
ERR("Failed to load Vulkan library.\n");
|
|
|
|
return E_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FAILED(hr = d3d12_get_adapter(&dxgi_adapter, adapter)))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
if (FAILED(hr = IDXGIAdapter_GetDesc(dxgi_adapter, &adapter_desc)))
|
|
|
|
{
|
|
|
|
WARN("Failed to get adapter desc, hr %#x.\n", hr);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
instance_create_info.pfn_signal_event = d3d12_signal_event;
|
|
|
|
instance_create_info.pfn_create_thread = d3d12_create_thread;
|
|
|
|
instance_create_info.pfn_join_thread = d3d12_join_thread;
|
|
|
|
instance_create_info.pfn_vkGetInstanceProcAddr = pfn_vkGetInstanceProcAddr;
|
|
|
|
instance_create_info.instance_extensions = instance_extensions;
|
|
|
|
instance_create_info.instance_extension_count = ARRAYSIZE(instance_extensions);
|
2021-04-19 16:54:53 +01:00
|
|
|
instance_create_info.optional_instance_extensions = NULL;
|
|
|
|
instance_create_info.optional_instance_extension_count = 0;
|
2020-07-02 16:44:53 +01:00
|
|
|
|
|
|
|
if (FAILED(hr = vkd3d_create_instance(&instance_create_info, &instance)))
|
|
|
|
{
|
|
|
|
WARN("Failed to create vkd3d instance, hr %#x.\n", hr);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
device_create_info.minimum_feature_level = minimum_feature_level;
|
|
|
|
device_create_info.instance = instance;
|
|
|
|
device_create_info.instance_create_info = NULL;
|
2021-03-16 16:17:01 +00:00
|
|
|
device_create_info.vk_physical_device = d3d12_find_physical_device(instance, pfn_vkGetInstanceProcAddr, &adapter_desc);
|
2020-07-02 16:44:53 +01:00
|
|
|
device_create_info.device_extensions = device_extensions;
|
|
|
|
device_create_info.device_extension_count = ARRAYSIZE(device_extensions);
|
2021-04-20 23:27:54 +01:00
|
|
|
device_create_info.optional_device_extensions = NULL;
|
|
|
|
device_create_info.optional_device_extension_count = 0;
|
2020-07-02 16:44:53 +01:00
|
|
|
device_create_info.parent = (IUnknown *)dxgi_adapter;
|
|
|
|
memcpy(&device_create_info.adapter_luid, &adapter_desc.AdapterLuid, VK_LUID_SIZE);
|
|
|
|
|
|
|
|
hr = vkd3d_create_device(&device_create_info, iid, device);
|
|
|
|
|
|
|
|
vkd3d_instance_decref(instance);
|
|
|
|
|
|
|
|
done:
|
|
|
|
IDXGIAdapter_Release(dxgi_adapter);
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
2020-07-07 16:40:05 +01:00
|
|
|
HRESULT WINAPI DLLEXPORT D3D12CreateRootSignatureDeserializer(const void *data, SIZE_T data_size,
|
2020-07-02 16:44:53 +01:00
|
|
|
REFIID iid, void **deserializer)
|
|
|
|
{
|
|
|
|
TRACE("data %p, data_size %lu, iid %s, deserializer %p.\n",
|
|
|
|
data, data_size, debugstr_guid(iid), deserializer);
|
|
|
|
|
|
|
|
return vkd3d_create_root_signature_deserializer(data, data_size, iid, deserializer);
|
|
|
|
}
|
|
|
|
|
2020-07-07 16:40:05 +01:00
|
|
|
HRESULT WINAPI DLLEXPORT D3D12SerializeRootSignature(const D3D12_ROOT_SIGNATURE_DESC *root_signature_desc,
|
2020-07-02 16:44:53 +01:00
|
|
|
D3D_ROOT_SIGNATURE_VERSION version, ID3DBlob **blob, ID3DBlob **error_blob)
|
|
|
|
{
|
|
|
|
TRACE("root_signature_desc %p, version %#x, blob %p, error_blob %p.\n",
|
|
|
|
root_signature_desc, version, blob, error_blob);
|
|
|
|
|
|
|
|
return vkd3d_serialize_root_signature(root_signature_desc, version, blob, error_blob);
|
|
|
|
}
|
|
|
|
|
2020-07-07 16:40:05 +01:00
|
|
|
HRESULT WINAPI DLLEXPORT D3D12CreateVersionedRootSignatureDeserializer(const void *data, SIZE_T data_size,
|
2020-07-02 16:44:53 +01:00
|
|
|
REFIID iid, void **deserializer)
|
|
|
|
{
|
|
|
|
TRACE("data %p, data_size %lu, iid %s, deserializer %p.\n",
|
|
|
|
data, data_size, debugstr_guid(iid), deserializer);
|
|
|
|
|
|
|
|
return vkd3d_create_versioned_root_signature_deserializer(data, data_size, iid, deserializer);
|
|
|
|
}
|
|
|
|
|
2020-07-07 16:40:05 +01:00
|
|
|
HRESULT WINAPI DLLEXPORT D3D12SerializeVersionedRootSignature(const D3D12_VERSIONED_ROOT_SIGNATURE_DESC *desc,
|
2020-07-02 16:44:53 +01:00
|
|
|
ID3DBlob **blob, ID3DBlob **error_blob)
|
|
|
|
{
|
|
|
|
TRACE("desc %p, blob %p, error_blob %p.\n", desc, blob, error_blob);
|
|
|
|
|
|
|
|
return vkd3d_serialize_versioned_root_signature(desc, blob, error_blob);
|
|
|
|
}
|