2020-06-01 20:32:43 +00:00
|
|
|
// Copyright 2018-2020, Collabora, Ltd.
|
2019-03-18 05:52:32 +00:00
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief Holds Vulkan related functions.
|
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
|
|
|
* @ingroup oxr_main
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
2022-04-28 09:27:11 +00:00
|
|
|
#include <inttypes.h>
|
2019-03-18 05:52:32 +00:00
|
|
|
|
2019-03-21 20:19:52 +00:00
|
|
|
#include "util/u_misc.h"
|
2020-07-16 12:55:37 +00:00
|
|
|
#include "util/u_debug.h"
|
2022-03-11 00:57:54 +00:00
|
|
|
#include "util/u_string_list.h"
|
2019-03-21 20:19:52 +00:00
|
|
|
|
2022-04-28 09:27:11 +00:00
|
|
|
#include "vk/vk_helpers.h"
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
#include "xrt/xrt_gfx_vk.h"
|
|
|
|
|
|
|
|
#include "oxr_objects.h"
|
|
|
|
#include "oxr_logger.h"
|
|
|
|
#include "oxr_two_call.h"
|
|
|
|
|
|
|
|
|
2022-04-30 12:09:30 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Helpers
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
#define GET_PROC(name) PFN_##name name = (PFN_##name)getProc(vkInstance, #name)
|
|
|
|
|
2022-05-01 08:33:37 +00:00
|
|
|
#define UUID_STR_SIZE (XRT_UUID_SIZE * 3 + 1)
|
2022-04-30 12:09:30 +00:00
|
|
|
|
|
|
|
static void
|
2022-05-01 08:33:37 +00:00
|
|
|
snprint_uuid(char *str, size_t size, const xrt_uuid_t *uuid)
|
2022-04-30 12:09:30 +00:00
|
|
|
{
|
2022-05-01 08:33:37 +00:00
|
|
|
for (size_t i = 0, offset = 0; i < ARRAY_SIZE(uuid->data) && offset < size; i++, offset += 3) {
|
|
|
|
snprintf(str + offset, size - offset, "%02x ", uuid->data[i]);
|
2022-04-30 12:09:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-04 16:17:16 +00:00
|
|
|
static void
|
|
|
|
snprint_luid(char *str, size_t size, xrt_luid_t *luid)
|
|
|
|
{
|
|
|
|
for (size_t i = 0, offset = 0; i < ARRAY_SIZE(luid->data) && offset < size; i++, offset += 3) {
|
|
|
|
snprintf(str + offset, size - offset, "%02x ", luid->data[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-30 12:09:30 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Misc functions (to be organized).
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
XrResult
|
|
|
|
oxr_vk_get_instance_exts(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
uint32_t namesCapacityInput,
|
|
|
|
uint32_t *namesCountOutput,
|
|
|
|
char *namesString)
|
|
|
|
{
|
|
|
|
size_t length = strlen(xrt_gfx_vk_instance_extensions) + 1;
|
|
|
|
|
2021-01-14 14:13:48 +00:00
|
|
|
OXR_TWO_CALL_HELPER(log, namesCapacityInput, namesCountOutput, namesString, length,
|
|
|
|
xrt_gfx_vk_instance_extensions, XR_SUCCESS);
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_vk_get_device_exts(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
uint32_t namesCapacityInput,
|
|
|
|
uint32_t *namesCountOutput,
|
|
|
|
char *namesString)
|
|
|
|
{
|
|
|
|
size_t length = strlen(xrt_gfx_vk_device_extensions) + 1;
|
|
|
|
|
2021-01-14 14:13:48 +00:00
|
|
|
OXR_TWO_CALL_HELPER(log, namesCapacityInput, namesCountOutput, namesString, length,
|
|
|
|
xrt_gfx_vk_device_extensions, XR_SUCCESS);
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_vk_get_requirements(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
XrGraphicsRequirementsVulkanKHR *graphicsRequirements)
|
|
|
|
{
|
|
|
|
struct xrt_api_requirements ver;
|
|
|
|
|
|
|
|
xrt_gfx_vk_get_versions(&ver);
|
2021-01-14 14:13:48 +00:00
|
|
|
graphicsRequirements->minApiVersionSupported = XR_MAKE_VERSION(ver.min_major, ver.min_minor, ver.min_patch);
|
|
|
|
graphicsRequirements->maxApiVersionSupported = XR_MAKE_VERSION(ver.max_major, ver.max_minor, ver.max_patch);
|
2019-03-18 05:52:32 +00:00
|
|
|
|
2020-05-30 11:42:41 +00:00
|
|
|
sys->gotten_requirements = true;
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2020-12-16 17:20:51 +00:00
|
|
|
DEBUG_GET_ONCE_LOG_OPTION(compositor_log, "XRT_COMPOSITOR_LOG", U_LOGGING_WARN)
|
2020-07-16 12:55:37 +00:00
|
|
|
|
2021-01-04 15:44:50 +00:00
|
|
|
//! @todo extension lists are duplicated as long strings in comp_vk_glue.c
|
|
|
|
static const char *required_vk_instance_extensions[] = {
|
|
|
|
VK_KHR_EXTERNAL_FENCE_CAPABILITIES_EXTENSION_NAME,
|
|
|
|
VK_KHR_EXTERNAL_MEMORY_CAPABILITIES_EXTENSION_NAME,
|
|
|
|
VK_KHR_EXTERNAL_SEMAPHORE_CAPABILITIES_EXTENSION_NAME,
|
|
|
|
VK_KHR_GET_PHYSICAL_DEVICE_PROPERTIES_2_EXTENSION_NAME,
|
|
|
|
};
|
|
|
|
|
|
|
|
// The device extensions do vary by platform, but in a very regular way.
|
|
|
|
// This should match the list in comp_compositor, except it shouldn't include
|
|
|
|
// VK_KHR_SWAPCHAIN_EXTENSION_NAME
|
|
|
|
static const char *required_vk_device_extensions[] = {
|
2021-01-14 14:13:48 +00:00
|
|
|
VK_KHR_DEDICATED_ALLOCATION_EXTENSION_NAME, VK_KHR_EXTERNAL_FENCE_EXTENSION_NAME,
|
|
|
|
VK_KHR_EXTERNAL_MEMORY_EXTENSION_NAME, VK_KHR_EXTERNAL_SEMAPHORE_EXTENSION_NAME,
|
2021-01-04 15:44:50 +00:00
|
|
|
VK_KHR_GET_MEMORY_REQUIREMENTS_2_EXTENSION_NAME,
|
|
|
|
|
|
|
|
// Platform version of "external_memory"
|
|
|
|
#if defined(XRT_GRAPHICS_BUFFER_HANDLE_IS_FD)
|
|
|
|
VK_KHR_EXTERNAL_MEMORY_FD_EXTENSION_NAME,
|
|
|
|
|
|
|
|
#elif defined(XRT_GRAPHICS_BUFFER_HANDLE_IS_AHARDWAREBUFFER)
|
|
|
|
VK_ANDROID_EXTERNAL_MEMORY_ANDROID_HARDWARE_BUFFER_EXTENSION_NAME,
|
|
|
|
|
|
|
|
#elif defined(XRT_GRAPHICS_BUFFER_HANDLE_IS_WIN32_HANDLE)
|
|
|
|
VK_KHR_EXTERNAL_MEMORY_WIN32_EXTENSION_NAME,
|
|
|
|
#else
|
|
|
|
#error "Need port!"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Platform version of "external_fence" and "external_semaphore"
|
|
|
|
#if defined(XRT_GRAPHICS_SYNC_HANDLE_IS_FD)
|
2021-01-14 14:13:48 +00:00
|
|
|
VK_KHR_EXTERNAL_SEMAPHORE_FD_EXTENSION_NAME, VK_KHR_EXTERNAL_FENCE_FD_EXTENSION_NAME,
|
2021-01-04 15:44:50 +00:00
|
|
|
|
|
|
|
#elif defined(XRT_GRAPHICS_SYNC_HANDLE_IS_WIN32_HANDLE)
|
|
|
|
VK_KHR_EXTERNAL_SEMAPHORE_WIN32_EXTENSION_NAME,
|
|
|
|
VK_KHR_EXTERNAL_FENCE_WIN32_EXTENSION_NAME,
|
|
|
|
|
|
|
|
#else
|
|
|
|
#error "Need port!"
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2022-03-10 13:11:52 +00:00
|
|
|
static const char *optional_device_extensions[] = {
|
|
|
|
#ifdef VK_KHR_timeline_semaphore
|
|
|
|
VK_KHR_TIMELINE_SEMAPHORE_EXTENSION_NAME,
|
|
|
|
#else
|
|
|
|
NULL, // avoid zero sized array with UB
|
|
|
|
#endif
|
|
|
|
};
|
2021-01-04 15:44:50 +00:00
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_vk_create_vulkan_instance(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
const XrVulkanInstanceCreateInfoKHR *createInfo,
|
|
|
|
VkInstance *vulkanInstance,
|
|
|
|
VkResult *vulkanResult)
|
|
|
|
{
|
|
|
|
|
2021-01-14 14:13:48 +00:00
|
|
|
PFN_vkGetInstanceProcAddr GetInstanceProcAddr = createInfo->pfnGetInstanceProcAddr;
|
2021-01-04 15:44:50 +00:00
|
|
|
|
2021-01-14 14:13:48 +00:00
|
|
|
PFN_vkCreateInstance CreateInstance = (PFN_vkCreateInstance)GetInstanceProcAddr(NULL, "vkCreateInstance");
|
2021-01-04 15:44:50 +00:00
|
|
|
if (!CreateInstance) {
|
|
|
|
//! @todo: clarify in spec
|
|
|
|
*vulkanResult = VK_ERROR_INITIALIZATION_FAILED;
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2022-03-11 00:57:54 +00:00
|
|
|
struct u_string_list *instance_ext_list = u_string_list_create_from_array(
|
|
|
|
required_vk_instance_extensions, ARRAY_SIZE(required_vk_instance_extensions));
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < createInfo->vulkanCreateInfo->enabledExtensionCount; i++) {
|
|
|
|
u_string_list_append_unique(instance_ext_list,
|
|
|
|
createInfo->vulkanCreateInfo->ppEnabledExtensionNames[i]);
|
|
|
|
}
|
2021-01-04 15:44:50 +00:00
|
|
|
|
|
|
|
VkInstanceCreateInfo modified_info = *createInfo->vulkanCreateInfo;
|
2022-03-11 00:57:54 +00:00
|
|
|
modified_info.ppEnabledExtensionNames = u_string_list_get_data(instance_ext_list);
|
|
|
|
modified_info.enabledExtensionCount = u_string_list_get_size(instance_ext_list);
|
2021-01-04 15:44:50 +00:00
|
|
|
|
2022-03-11 00:57:54 +00:00
|
|
|
*vulkanResult = CreateInstance(&modified_info, createInfo->vulkanAllocator, vulkanInstance);
|
2021-01-04 15:44:50 +00:00
|
|
|
|
2022-04-28 09:27:11 +00:00
|
|
|
|
|
|
|
// Logging
|
|
|
|
{
|
|
|
|
struct oxr_sink_logger slog = {0};
|
|
|
|
|
|
|
|
oxr_slog(&slog, "Creation of VkInstance:");
|
|
|
|
oxr_slog(&slog, "\n\tresult: %s", vk_result_string(*vulkanResult));
|
|
|
|
oxr_slog(&slog, "\n\tvulkanInstance: 0x%" PRIx64, (uint64_t)(intptr_t)*vulkanInstance);
|
|
|
|
oxr_slog(&slog, "\n\textensions:");
|
|
|
|
for (uint32_t i = 0; i < modified_info.enabledExtensionCount; i++) {
|
|
|
|
oxr_slog(&slog, "\n\t\t%s", modified_info.ppEnabledExtensionNames[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
oxr_log_slog(log, &slog);
|
|
|
|
}
|
|
|
|
|
2022-03-11 00:57:54 +00:00
|
|
|
u_string_list_destroy(&instance_ext_list);
|
2021-01-04 15:44:50 +00:00
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2022-03-10 13:11:52 +00:00
|
|
|
static XrResult
|
|
|
|
vk_get_device_ext_props(struct oxr_logger *log,
|
|
|
|
VkInstance instance,
|
|
|
|
PFN_vkGetInstanceProcAddr GetInstanceProcAddr,
|
|
|
|
VkPhysicalDevice physical_device,
|
|
|
|
VkExtensionProperties **out_props,
|
|
|
|
uint32_t *out_prop_count)
|
|
|
|
{
|
|
|
|
PFN_vkEnumerateDeviceExtensionProperties EnumerateDeviceExtensionProperties =
|
|
|
|
(PFN_vkEnumerateDeviceExtensionProperties)GetInstanceProcAddr(instance,
|
|
|
|
"vkEnumerateDeviceExtensionProperties");
|
|
|
|
|
|
|
|
if (!EnumerateDeviceExtensionProperties) {
|
|
|
|
oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Failed to get vkEnumerateDeviceExtensionProperties fp");
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t prop_count = 0;
|
|
|
|
VkResult res = EnumerateDeviceExtensionProperties(physical_device, NULL, &prop_count, NULL);
|
|
|
|
if (res != VK_SUCCESS) {
|
|
|
|
oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Failed to enumerate device extension properties count (%d)",
|
|
|
|
res);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
VkExtensionProperties *props = U_TYPED_ARRAY_CALLOC(VkExtensionProperties, prop_count);
|
|
|
|
|
|
|
|
res = EnumerateDeviceExtensionProperties(physical_device, NULL, &prop_count, props);
|
|
|
|
if (res != VK_SUCCESS) {
|
|
|
|
free(props);
|
|
|
|
oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Failed to enumerate device extension properties (%d)", res);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check above returns on failure.
|
|
|
|
*out_props = props;
|
|
|
|
*out_prop_count = prop_count;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
vk_check_extension(VkExtensionProperties *props, uint32_t prop_count, const char *ext)
|
|
|
|
{
|
|
|
|
for (uint32_t i = 0; i < prop_count; i++) {
|
|
|
|
if (strcmp(props[i].extensionName, ext) == 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
vk_get_device_features(struct oxr_logger *log,
|
|
|
|
VkInstance instance,
|
|
|
|
PFN_vkGetInstanceProcAddr GetInstanceProcAddr,
|
|
|
|
VkPhysicalDevice physical_device,
|
|
|
|
VkPhysicalDeviceFeatures2 *physical_device_features)
|
|
|
|
{
|
|
|
|
PFN_vkGetPhysicalDeviceFeatures2 GetPhysicalDeviceFeatures2 =
|
|
|
|
(PFN_vkGetPhysicalDeviceFeatures2)GetInstanceProcAddr(instance, "vkGetPhysicalDeviceFeatures2");
|
|
|
|
|
|
|
|
if (!GetPhysicalDeviceFeatures2) {
|
|
|
|
oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Failed to get vkGetPhysicalDeviceFeatures2 fp");
|
|
|
|
}
|
|
|
|
|
|
|
|
GetPhysicalDeviceFeatures2( //
|
|
|
|
physical_device, // physicalDevice
|
|
|
|
physical_device_features); // pFeatures
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2021-01-04 15:44:50 +00:00
|
|
|
XrResult
|
|
|
|
oxr_vk_create_vulkan_device(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
const XrVulkanDeviceCreateInfoKHR *createInfo,
|
|
|
|
VkDevice *vulkanDevice,
|
|
|
|
VkResult *vulkanResult)
|
|
|
|
{
|
2022-03-10 13:11:52 +00:00
|
|
|
XrResult res;
|
|
|
|
|
2021-01-14 14:13:48 +00:00
|
|
|
PFN_vkGetInstanceProcAddr GetInstanceProcAddr = createInfo->pfnGetInstanceProcAddr;
|
2021-01-04 15:44:50 +00:00
|
|
|
|
|
|
|
PFN_vkCreateDevice CreateDevice =
|
2021-01-14 14:13:48 +00:00
|
|
|
(PFN_vkCreateDevice)GetInstanceProcAddr(sys->vulkan_enable2_instance, "vkCreateDevice");
|
2021-01-04 15:44:50 +00:00
|
|
|
if (!CreateDevice) {
|
|
|
|
//! @todo: clarify in spec
|
|
|
|
*vulkanResult = VK_ERROR_INITIALIZATION_FAILED;
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
VkPhysicalDevice physical_device = createInfo->vulkanPhysicalDevice;
|
|
|
|
|
2022-03-11 00:57:54 +00:00
|
|
|
struct u_string_list *device_extension_list =
|
|
|
|
u_string_list_create_from_array(required_vk_device_extensions, ARRAY_SIZE(required_vk_device_extensions));
|
2021-01-04 15:44:50 +00:00
|
|
|
|
2022-03-11 00:57:54 +00:00
|
|
|
for (uint32_t i = 0; i < createInfo->vulkanCreateInfo->enabledExtensionCount; i++) {
|
|
|
|
u_string_list_append_unique(device_extension_list,
|
|
|
|
createInfo->vulkanCreateInfo->ppEnabledExtensionNames[i]);
|
2021-01-04 15:44:50 +00:00
|
|
|
}
|
|
|
|
|
2022-03-10 13:11:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
VkExtensionProperties *props = NULL;
|
|
|
|
uint32_t prop_count = 0;
|
|
|
|
res = vk_get_device_ext_props(log, sys->vulkan_enable2_instance, createInfo->pfnGetInstanceProcAddr,
|
|
|
|
physical_device, &props, &prop_count);
|
|
|
|
if (res != XR_SUCCESS) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < ARRAY_SIZE(optional_device_extensions); i++) {
|
|
|
|
if (optional_device_extensions[i] &&
|
|
|
|
vk_check_extension(props, prop_count, optional_device_extensions[i])) {
|
|
|
|
u_string_list_append_unique(device_extension_list, optional_device_extensions[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
free(props);
|
|
|
|
|
|
|
|
|
|
|
|
VkPhysicalDeviceFeatures2 physical_device_features = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_FEATURES_2,
|
|
|
|
.pNext = NULL,
|
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef VK_KHR_timeline_semaphore
|
|
|
|
VkPhysicalDeviceTimelineSemaphoreFeaturesKHR timeline_semaphore_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_TIMELINE_SEMAPHORE_FEATURES_KHR,
|
|
|
|
.pNext = NULL,
|
|
|
|
.timelineSemaphore = VK_FALSE,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (u_string_list_contains(device_extension_list, VK_KHR_TIMELINE_SEMAPHORE_EXTENSION_NAME)) {
|
|
|
|
physical_device_features.pNext = &timeline_semaphore_info;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
res = vk_get_device_features(log, sys->vulkan_enable2_instance, createInfo->pfnGetInstanceProcAddr,
|
|
|
|
physical_device, &physical_device_features);
|
|
|
|
if (res != XR_SUCCESS) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-03-11 00:57:54 +00:00
|
|
|
VkDeviceCreateInfo modified_info = *createInfo->vulkanCreateInfo;
|
|
|
|
modified_info.ppEnabledExtensionNames = u_string_list_get_data(device_extension_list);
|
|
|
|
modified_info.enabledExtensionCount = u_string_list_get_size(device_extension_list);
|
2022-03-10 13:11:52 +00:00
|
|
|
|
|
|
|
#ifdef VK_KHR_timeline_semaphore
|
|
|
|
VkPhysicalDeviceTimelineSemaphoreFeatures timeline_semaphore = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_TIMELINE_SEMAPHORE_FEATURES,
|
|
|
|
.pNext = NULL,
|
|
|
|
.timelineSemaphore = timeline_semaphore_info.timelineSemaphore,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (timeline_semaphore_info.timelineSemaphore) {
|
|
|
|
/*
|
|
|
|
* Insert timeline semaphore request first to override
|
|
|
|
* any the app may have put on the next chain.
|
|
|
|
*/
|
|
|
|
// Have to cast away const.
|
|
|
|
timeline_semaphore.pNext = (void *)modified_info.pNext;
|
|
|
|
modified_info.pNext = &timeline_semaphore;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2022-03-11 00:57:54 +00:00
|
|
|
*vulkanResult = CreateDevice(physical_device, &modified_info, createInfo->vulkanAllocator, vulkanDevice);
|
|
|
|
|
2022-04-28 09:27:11 +00:00
|
|
|
|
|
|
|
// Logging
|
|
|
|
{
|
|
|
|
struct oxr_sink_logger slog = {0};
|
|
|
|
|
|
|
|
oxr_slog(&slog, "Creation of VkDevice:");
|
|
|
|
oxr_slog(&slog, "\n\tresult: %s", vk_result_string(*vulkanResult));
|
|
|
|
oxr_slog(&slog, "\n\tvulkanDevice: 0x%" PRIx64, (uint64_t)(intptr_t)*vulkanDevice);
|
|
|
|
oxr_slog(&slog, "\n\tvulkanInstance: 0x%" PRIx64, (uint64_t)(intptr_t)sys->vulkan_enable2_instance);
|
|
|
|
#ifdef VK_KHR_timeline_semaphore
|
|
|
|
oxr_slog(&slog, "\n\ttimelineSemaphore: %s",
|
|
|
|
timeline_semaphore_info.timelineSemaphore ? "true" : "false");
|
|
|
|
#endif
|
|
|
|
oxr_slog(&slog, "\n\textensions:");
|
|
|
|
for (uint32_t i = 0; i < modified_info.enabledExtensionCount; i++) {
|
|
|
|
oxr_slog(&slog, "\n\t\t%s", modified_info.ppEnabledExtensionNames[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
oxr_log_slog(log, &slog);
|
|
|
|
}
|
|
|
|
|
2022-03-10 13:11:52 +00:00
|
|
|
#ifdef VK_KHR_timeline_semaphore
|
|
|
|
// Have timeline semaphores added and as such enabled.
|
|
|
|
if (*vulkanResult == VK_SUCCESS) {
|
|
|
|
sys->vk.timeline_semaphore_enabled = timeline_semaphore_info.timelineSemaphore;
|
|
|
|
U_LOG_D("timeline semaphores enabled: %d", timeline_semaphore_info.timelineSemaphore);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2022-03-11 00:57:54 +00:00
|
|
|
u_string_list_destroy(&device_extension_list);
|
|
|
|
|
2021-01-04 15:44:50 +00:00
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
XrResult
|
|
|
|
oxr_vk_get_physical_device(struct oxr_logger *log,
|
|
|
|
struct oxr_instance *inst,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
VkInstance vkInstance,
|
|
|
|
PFN_vkGetInstanceProcAddr getProc,
|
|
|
|
VkPhysicalDevice *vkPhysicalDevice)
|
|
|
|
{
|
|
|
|
GET_PROC(vkEnumeratePhysicalDevices);
|
2020-07-16 12:55:37 +00:00
|
|
|
GET_PROC(vkGetPhysicalDeviceProperties2);
|
2019-03-18 05:52:32 +00:00
|
|
|
VkResult vk_ret;
|
|
|
|
uint32_t count;
|
|
|
|
|
|
|
|
vk_ret = vkEnumeratePhysicalDevices(vkInstance, &count, NULL);
|
|
|
|
if (vk_ret != VK_SUCCESS) {
|
2021-01-14 14:13:48 +00:00
|
|
|
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Call to vkEnumeratePhysicalDevices returned %u",
|
|
|
|
vk_ret);
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
if (count == 0) {
|
2021-01-14 14:13:48 +00:00
|
|
|
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE,
|
2022-04-30 12:15:25 +00:00
|
|
|
"Call to vkEnumeratePhysicalDevices returned zero VkPhysicalDevices");
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
2019-03-21 20:19:52 +00:00
|
|
|
VkPhysicalDevice *phys = U_TYPED_ARRAY_CALLOC(VkPhysicalDevice, count);
|
2019-03-18 05:52:32 +00:00
|
|
|
vk_ret = vkEnumeratePhysicalDevices(vkInstance, &count, phys);
|
|
|
|
if (vk_ret != VK_SUCCESS) {
|
|
|
|
free(phys);
|
2021-01-14 14:13:48 +00:00
|
|
|
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Call to vkEnumeratePhysicalDevices returned %u",
|
|
|
|
vk_ret);
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
if (count == 0) {
|
|
|
|
free(phys);
|
2021-01-14 14:13:48 +00:00
|
|
|
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE,
|
2022-04-30 12:15:25 +00:00
|
|
|
"Call to vkEnumeratePhysicalDevices returned zero VkPhysicalDevices");
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
2022-04-30 12:09:30 +00:00
|
|
|
char suggested_uuid_str[UUID_STR_SIZE] = {0};
|
2022-05-01 08:33:37 +00:00
|
|
|
snprint_uuid(suggested_uuid_str, ARRAY_SIZE(suggested_uuid_str), &sys->xsysc->info.client_vk_deviceUUID);
|
2019-03-18 05:52:32 +00:00
|
|
|
|
2021-11-20 14:50:47 +00:00
|
|
|
enum u_logging_level log_level = debug_get_log_option_compositor_log();
|
2020-07-16 12:55:37 +00:00
|
|
|
int gpu_index = -1;
|
2019-03-18 05:52:32 +00:00
|
|
|
for (uint32_t i = 0; i < count; i++) {
|
2022-04-30 12:15:25 +00:00
|
|
|
VkPhysicalDeviceIDProperties pdidp = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_ID_PROPERTIES,
|
|
|
|
};
|
|
|
|
|
|
|
|
VkPhysicalDeviceProperties2 pdp2 = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2,
|
|
|
|
.pNext = &pdidp,
|
|
|
|
};
|
2020-07-16 12:55:37 +00:00
|
|
|
|
|
|
|
vkGetPhysicalDeviceProperties2(phys[i], &pdp2);
|
|
|
|
|
2022-05-01 08:33:37 +00:00
|
|
|
// These should always be true
|
|
|
|
static_assert(VK_UUID_SIZE == XRT_UUID_SIZE, "uuid sizes mismatch");
|
|
|
|
static_assert(ARRAY_SIZE(pdidp.deviceUUID) == XRT_UUID_SIZE, "array size mismatch");
|
|
|
|
|
2022-05-04 16:17:16 +00:00
|
|
|
char buffer[UUID_STR_SIZE] = {0};
|
2021-11-20 14:50:47 +00:00
|
|
|
if (log_level <= U_LOGGING_DEBUG) {
|
2022-05-04 16:17:16 +00:00
|
|
|
snprint_uuid(buffer, ARRAY_SIZE(buffer), (xrt_uuid_t *)pdidp.deviceUUID);
|
|
|
|
oxr_log(log, "GPU: #%d, uuid: %s", i, buffer);
|
2022-05-02 22:07:07 +00:00
|
|
|
if (pdidp.deviceLUIDValid == VK_TRUE) {
|
2022-05-04 16:17:16 +00:00
|
|
|
snprint_luid(buffer, ARRAY_SIZE(buffer), (xrt_luid_t *)pdidp.deviceLUID);
|
|
|
|
oxr_log(log, " LUID: %s", buffer);
|
2022-05-02 22:07:07 +00:00
|
|
|
}
|
2020-07-16 12:55:37 +00:00
|
|
|
}
|
|
|
|
|
2022-05-01 08:33:37 +00:00
|
|
|
if (memcmp(pdidp.deviceUUID, sys->xsysc->info.client_vk_deviceUUID.data, XRT_UUID_SIZE) == 0) {
|
2019-03-18 05:52:32 +00:00
|
|
|
gpu_index = i;
|
2021-11-20 14:50:47 +00:00
|
|
|
if (log_level <= U_LOGGING_DEBUG) {
|
2022-05-04 16:17:16 +00:00
|
|
|
oxr_log(log, "Using GPU #%d with uuid %s suggested by runtime", gpu_index, buffer);
|
2020-07-16 12:55:37 +00:00
|
|
|
}
|
|
|
|
break;
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-16 12:55:37 +00:00
|
|
|
if (gpu_index == -1) {
|
2022-04-28 09:29:56 +00:00
|
|
|
oxr_warn(log, "Did not find runtime suggested GPU, fall back to GPU 0\n\tuuid: %s", suggested_uuid_str);
|
2020-07-16 12:55:37 +00:00
|
|
|
gpu_index = 0;
|
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
*vkPhysicalDevice = phys[gpu_index];
|
|
|
|
|
2021-01-04 15:44:50 +00:00
|
|
|
// vulkan_enable2 needs the physical device in xrCreateVulkanDeviceKHR
|
|
|
|
if (inst->extensions.KHR_vulkan_enable2) {
|
2021-01-26 02:45:43 +00:00
|
|
|
sys->vulkan_enable2_instance = vkInstance;
|
2021-05-06 22:34:47 +00:00
|
|
|
}
|
|
|
|
sys->suggested_vulkan_physical_device = *vkPhysicalDevice;
|
2021-11-20 14:50:47 +00:00
|
|
|
if (log_level <= U_LOGGING_DEBUG) {
|
2021-05-06 22:34:47 +00:00
|
|
|
oxr_log(log, "Suggesting vulkan physical device %p", (void *)*vkPhysicalDevice);
|
2021-01-04 15:44:50 +00:00
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
free(phys);
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|