|
|
|
#if defined(__FreeBSD__)
|
|
|
|
#undef _POSIX_C_SOURCE
|
|
|
|
#endif
|
|
|
|
#include <assert.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <xf86drm.h>
|
|
|
|
#include <vulkan/vulkan.h>
|
|
|
|
#include <wlr/util/log.h>
|
|
|
|
#include <wlr/version.h>
|
|
|
|
#include <wlr/config.h>
|
|
|
|
#include "render/dmabuf.h"
|
|
|
|
#include "render/vulkan.h"
|
|
|
|
|
|
|
|
#if defined(__linux__)
|
|
|
|
#include <sys/sysmacros.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static bool check_extension(const VkExtensionProperties *avail,
|
|
|
|
uint32_t avail_len, const char *name) {
|
|
|
|
for (size_t i = 0; i < avail_len; i++) {
|
|
|
|
if (strcmp(avail[i].extensionName, name) == 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
render/vulkan: Fix clang compiler error in 32 bit ARM Android
This adds proper calling convention attribute for callbacks used in vulkan
and fixes the following compiler error.
../src/render/vulkan/vulkan.c:175:22: error: incompatible function pointer types
initializing 'PFN_vkDebugUtilsMessengerCallbackEXT' (aka 'unsigned int (*)
(enum VkDebugUtilsMessageSeverityFlagBitsEXT, unsigned int, const struct
VkDebugUtilsMessengerCallbackDataEXT *, void *) __attribute__((pcs("aapcs-vfp")))')
with an expression of type 'VkBool32 (*)(VkDebugUtilsMessageSeverityFlagBitsEXT,
VkDebugUtilsMessageTypeFlagsEXT, const VkDebugUtilsMessengerCallbackDataEXT *, void *)'
(aka 'unsigned int (*)(enum VkDebugUtilsMessageSeverityFlagBitsEXT, unsigned int,
const struct VkDebugUtilsMessengerCallbackDataEXT *, void *)')
[-Wincompatible-function-pointer-types]
.pfnUserCallback = &debug_callback,
^~~~~~~~~~~~~~~
9 months ago
|
|
|
static VKAPI_ATTR VkBool32 debug_callback(VkDebugUtilsMessageSeverityFlagBitsEXT severity,
|
|
|
|
VkDebugUtilsMessageTypeFlagsEXT type,
|
|
|
|
const VkDebugUtilsMessengerCallbackDataEXT *debug_data,
|
|
|
|
void *data) {
|
|
|
|
// we ignore some of the non-helpful warnings
|
|
|
|
static const char *const ignored[] = {
|
|
|
|
// notifies us that shader output is not consumed since
|
|
|
|
// we use the shared vertex buffer with uv output
|
|
|
|
"UNASSIGNED-CoreValidation-Shader-OutputNotConsumed",
|
|
|
|
};
|
|
|
|
|
|
|
|
if (debug_data->pMessageIdName) {
|
|
|
|
for (unsigned i = 0; i < sizeof(ignored) / sizeof(ignored[0]); ++i) {
|
|
|
|
if (strcmp(debug_data->pMessageIdName, ignored[i]) == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum wlr_log_importance importance;
|
|
|
|
switch (severity) {
|
|
|
|
case VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT:
|
|
|
|
case VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT:
|
|
|
|
importance = WLR_ERROR;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
case VK_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT:
|
|
|
|
importance = WLR_INFO;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
wlr_log(importance, "%s (%s)", debug_data->pMessage,
|
|
|
|
debug_data->pMessageIdName);
|
|
|
|
if (debug_data->queueLabelCount > 0) {
|
|
|
|
const char *name = debug_data->pQueueLabels[0].pLabelName;
|
|
|
|
if (name) {
|
|
|
|
wlr_log(importance, " last label '%s'", name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < debug_data->objectCount; ++i) {
|
|
|
|
if (debug_data->pObjects[i].pObjectName) {
|
|
|
|
wlr_log(importance, " involving '%s'", debug_data->pMessage);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct wlr_vk_instance *vulkan_instance_create(bool debug) {
|
|
|
|
// we require vulkan 1.1
|
|
|
|
PFN_vkEnumerateInstanceVersion pfEnumInstanceVersion =
|
|
|
|
(PFN_vkEnumerateInstanceVersion)
|
|
|
|
vkGetInstanceProcAddr(VK_NULL_HANDLE, "vkEnumerateInstanceVersion");
|
|
|
|
if (!pfEnumInstanceVersion) {
|
|
|
|
wlr_log(WLR_ERROR, "wlroots requires vulkan 1.1 which is not available");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t ini_version;
|
|
|
|
if (pfEnumInstanceVersion(&ini_version) != VK_SUCCESS ||
|
|
|
|
ini_version < VK_API_VERSION_1_1) {
|
|
|
|
wlr_log(WLR_ERROR, "wlroots requires vulkan 1.1 which is not available");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t avail_extc = 0;
|
|
|
|
VkResult res;
|
|
|
|
res = vkEnumerateInstanceExtensionProperties(NULL, &avail_extc, NULL);
|
|
|
|
if ((res != VK_SUCCESS) || (avail_extc == 0)) {
|
|
|
|
wlr_vk_error("Could not enumerate instance extensions (1)", res);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
VkExtensionProperties avail_ext_props[avail_extc + 1];
|
|
|
|
res = vkEnumerateInstanceExtensionProperties(NULL, &avail_extc,
|
|
|
|
avail_ext_props);
|
|
|
|
if (res != VK_SUCCESS) {
|
|
|
|
wlr_vk_error("Could not enumerate instance extensions (2)", res);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t j = 0; j < avail_extc; ++j) {
|
|
|
|
wlr_log(WLR_DEBUG, "Vulkan instance extension %s v%"PRIu32,
|
|
|
|
avail_ext_props[j].extensionName, avail_ext_props[j].specVersion);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct wlr_vk_instance *ini = calloc(1, sizeof(*ini));
|
|
|
|
if (!ini) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "allocation failed");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t extensions_len = 0;
|
|
|
|
const char *extensions[1] = {0};
|
|
|
|
|
|
|
|
bool debug_utils_found = false;
|
|
|
|
if (debug && check_extension(avail_ext_props, avail_extc,
|
|
|
|
VK_EXT_DEBUG_UTILS_EXTENSION_NAME)) {
|
|
|
|
debug_utils_found = true;
|
|
|
|
extensions[extensions_len++] = VK_EXT_DEBUG_UTILS_EXTENSION_NAME;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(extensions_len <= sizeof(extensions) / sizeof(extensions[0]));
|
|
|
|
|
|
|
|
VkApplicationInfo application_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_APPLICATION_INFO,
|
|
|
|
.pEngineName = "wlroots",
|
|
|
|
.engineVersion = WLR_VERSION_NUM,
|
|
|
|
.apiVersion = VK_API_VERSION_1_1,
|
|
|
|
};
|
|
|
|
|
|
|
|
VkInstanceCreateInfo instance_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO,
|
|
|
|
.pApplicationInfo = &application_info,
|
|
|
|
.enabledExtensionCount = extensions_len,
|
|
|
|
.ppEnabledExtensionNames = extensions,
|
|
|
|
.enabledLayerCount = 0,
|
|
|
|
.ppEnabledLayerNames = NULL,
|
|
|
|
};
|
|
|
|
|
|
|
|
VkDebugUtilsMessageSeverityFlagsEXT severity =
|
|
|
|
// VK_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT |
|
|
|
|
VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT |
|
|
|
|
VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT;
|
|
|
|
VkDebugUtilsMessageTypeFlagsEXT types =
|
|
|
|
// VK_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT |
|
|
|
|
VK_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT |
|
|
|
|
VK_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT;
|
|
|
|
|
|
|
|
VkDebugUtilsMessengerCreateInfoEXT debug_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT,
|
|
|
|
.messageSeverity = severity,
|
|
|
|
.messageType = types,
|
|
|
|
.pfnUserCallback = &debug_callback,
|
|
|
|
.pUserData = ini,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (debug_utils_found) {
|
|
|
|
// already adding the debug utils messenger extension to
|
|
|
|
// instance creation gives us additional information during
|
|
|
|
// instance creation and destruction, can be useful for debugging
|
|
|
|
// layers/extensions not being found.
|
|
|
|
instance_info.pNext = &debug_info;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = vkCreateInstance(&instance_info, NULL, &ini->instance);
|
|
|
|
if (res != VK_SUCCESS) {
|
|
|
|
wlr_vk_error("Could not create instance", res);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (debug_utils_found) {
|
|
|
|
ini->api.createDebugUtilsMessengerEXT =
|
|
|
|
(PFN_vkCreateDebugUtilsMessengerEXT) vkGetInstanceProcAddr(
|
|
|
|
ini->instance, "vkCreateDebugUtilsMessengerEXT");
|
|
|
|
ini->api.destroyDebugUtilsMessengerEXT =
|
|
|
|
(PFN_vkDestroyDebugUtilsMessengerEXT) vkGetInstanceProcAddr(
|
|
|
|
ini->instance, "vkDestroyDebugUtilsMessengerEXT");
|
|
|
|
|
|
|
|
if (ini->api.createDebugUtilsMessengerEXT) {
|
|
|
|
ini->api.createDebugUtilsMessengerEXT(ini->instance,
|
|
|
|
&debug_info, NULL, &ini->messenger);
|
|
|
|
} else {
|
|
|
|
wlr_log(WLR_ERROR, "vkCreateDebugUtilsMessengerEXT not found");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ini;
|
|
|
|
|
|
|
|
error:
|
|
|
|
vulkan_instance_destroy(ini);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void vulkan_instance_destroy(struct wlr_vk_instance *ini) {
|
|
|
|
if (!ini) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ini->messenger && ini->api.destroyDebugUtilsMessengerEXT) {
|
|
|
|
ini->api.destroyDebugUtilsMessengerEXT(ini->instance,
|
|
|
|
ini->messenger, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ini->instance) {
|
|
|
|
vkDestroyInstance(ini->instance, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(ini);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void log_phdev(const VkPhysicalDeviceProperties *props) {
|
|
|
|
uint32_t vv_major = VK_VERSION_MAJOR(props->apiVersion);
|
|
|
|
uint32_t vv_minor = VK_VERSION_MINOR(props->apiVersion);
|
|
|
|
uint32_t vv_patch = VK_VERSION_PATCH(props->apiVersion);
|
|
|
|
|
|
|
|
uint32_t dv_major = VK_VERSION_MAJOR(props->driverVersion);
|
|
|
|
uint32_t dv_minor = VK_VERSION_MINOR(props->driverVersion);
|
|
|
|
uint32_t dv_patch = VK_VERSION_PATCH(props->driverVersion);
|
|
|
|
|
|
|
|
const char *dev_type = "unknown";
|
|
|
|
switch (props->deviceType) {
|
|
|
|
case VK_PHYSICAL_DEVICE_TYPE_INTEGRATED_GPU:
|
|
|
|
dev_type = "integrated";
|
|
|
|
break;
|
|
|
|
case VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU:
|
|
|
|
dev_type = "discrete";
|
|
|
|
break;
|
|
|
|
case VK_PHYSICAL_DEVICE_TYPE_CPU:
|
|
|
|
dev_type = "cpu";
|
|
|
|
break;
|
|
|
|
case VK_PHYSICAL_DEVICE_TYPE_VIRTUAL_GPU:
|
|
|
|
dev_type = "vgpu";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
wlr_log(WLR_INFO, "Vulkan device: '%s'", props->deviceName);
|
|
|
|
wlr_log(WLR_INFO, " Device type: '%s'", dev_type);
|
|
|
|
wlr_log(WLR_INFO, " Supported API version: %u.%u.%u", vv_major, vv_minor, vv_patch);
|
|
|
|
wlr_log(WLR_INFO, " Driver version: %u.%u.%u", dv_major, dv_minor, dv_patch);
|
|
|
|
}
|
|
|
|
|
|
|
|
VkPhysicalDevice vulkan_find_drm_phdev(struct wlr_vk_instance *ini, int drm_fd) {
|
|
|
|
VkResult res;
|
|
|
|
uint32_t num_phdevs;
|
|
|
|
|
|
|
|
res = vkEnumeratePhysicalDevices(ini->instance, &num_phdevs, NULL);
|
|
|
|
if (res != VK_SUCCESS) {
|
|
|
|
wlr_vk_error("Could not retrieve physical devices", res);
|
|
|
|
return VK_NULL_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
VkPhysicalDevice phdevs[1 + num_phdevs];
|
|
|
|
res = vkEnumeratePhysicalDevices(ini->instance, &num_phdevs, phdevs);
|
|
|
|
if (res != VK_SUCCESS) {
|
|
|
|
wlr_vk_error("Could not retrieve physical devices", res);
|
|
|
|
return VK_NULL_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct stat drm_stat = {0};
|
|
|
|
if (fstat(drm_fd, &drm_stat) != 0) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "fstat failed");
|
|
|
|
return VK_NULL_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < num_phdevs; ++i) {
|
|
|
|
VkPhysicalDevice phdev = phdevs[i];
|
|
|
|
|
|
|
|
// check whether device supports vulkan 1.1, needed for
|
|
|
|
// vkGetPhysicalDeviceProperties2
|
|
|
|
VkPhysicalDeviceProperties phdev_props;
|
|
|
|
vkGetPhysicalDeviceProperties(phdev, &phdev_props);
|
|
|
|
|
|
|
|
log_phdev(&phdev_props);
|
|
|
|
|
|
|
|
if (phdev_props.apiVersion < VK_API_VERSION_1_1) {
|
|
|
|
// NOTE: we could additionaly check whether the
|
|
|
|
// VkPhysicalDeviceProperties2KHR extension is supported but
|
|
|
|
// implementations not supporting 1.1 are unlikely in future
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for extensions
|
|
|
|
uint32_t avail_extc = 0;
|
|
|
|
res = vkEnumerateDeviceExtensionProperties(phdev, NULL,
|
|
|
|
&avail_extc, NULL);
|
|
|
|
if ((res != VK_SUCCESS) || (avail_extc == 0)) {
|
|
|
|
wlr_vk_error(" Could not enumerate device extensions", res);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
VkExtensionProperties avail_ext_props[avail_extc + 1];
|
|
|
|
res = vkEnumerateDeviceExtensionProperties(phdev, NULL,
|
|
|
|
&avail_extc, avail_ext_props);
|
|
|
|
if (res != VK_SUCCESS) {
|
|
|
|
wlr_vk_error(" Could not enumerate device extensions", res);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool has_drm_props = check_extension(avail_ext_props, avail_extc,
|
|
|
|
VK_EXT_PHYSICAL_DEVICE_DRM_EXTENSION_NAME);
|
|
|
|
bool has_driver_props = check_extension(avail_ext_props, avail_extc,
|
|
|
|
VK_KHR_DRIVER_PROPERTIES_EXTENSION_NAME);
|
|
|
|
|
|
|
|
VkPhysicalDeviceProperties2 props = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2,
|
|
|
|
};
|
|
|
|
|
|
|
|
VkPhysicalDeviceDrmPropertiesEXT drm_props = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_DRM_PROPERTIES_EXT,
|
|
|
|
};
|
|
|
|
if (has_drm_props) {
|
|
|
|
drm_props.pNext = props.pNext;
|
|
|
|
props.pNext = &drm_props;
|
|
|
|
}
|
|
|
|
|
|
|
|
VkPhysicalDeviceDriverPropertiesKHR driver_props = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_DRIVER_PROPERTIES,
|
|
|
|
};
|
|
|
|
if (has_driver_props) {
|
|
|
|
driver_props.pNext = props.pNext;
|
|
|
|
props.pNext = &driver_props;
|
|
|
|
}
|
|
|
|
|
|
|
|
vkGetPhysicalDeviceProperties2(phdev, &props);
|
|
|
|
|
|
|
|
if (has_driver_props) {
|
|
|
|
wlr_log(WLR_INFO, " Driver name: %s (%s)", driver_props.driverName, driver_props.driverInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_drm_props) {
|
|
|
|
wlr_log(WLR_DEBUG, " Ignoring physical device \"%s\": "
|
|
|
|
"VK_EXT_physical_device_drm not supported",
|
|
|
|
phdev_props.deviceName);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev_t primary_devid = makedev(drm_props.primaryMajor, drm_props.primaryMinor);
|
|
|
|
dev_t render_devid = makedev(drm_props.renderMajor, drm_props.renderMinor);
|
|
|
|
if (primary_devid == drm_stat.st_rdev ||
|
|
|
|
render_devid == drm_stat.st_rdev) {
|
|
|
|
wlr_log(WLR_INFO, "Found matching Vulkan physical device: %s",
|
|
|
|
phdev_props.deviceName);
|
|
|
|
return phdev;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return VK_NULL_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
int vulkan_open_phdev_drm_fd(VkPhysicalDevice phdev) {
|
|
|
|
// vulkan_find_drm_phdev() already checks that VK_EXT_physical_device_drm
|
|
|
|
// is supported
|
|
|
|
VkPhysicalDeviceDrmPropertiesEXT drm_props = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_DRM_PROPERTIES_EXT,
|
|
|
|
};
|
|
|
|
VkPhysicalDeviceProperties2 props = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2,
|
|
|
|
.pNext = &drm_props,
|
|
|
|
};
|
|
|
|
vkGetPhysicalDeviceProperties2(phdev, &props);
|
|
|
|
|
|
|
|
dev_t devid;
|
|
|
|
if (drm_props.hasRender) {
|
|
|
|
devid = makedev(drm_props.renderMajor, drm_props.renderMinor);
|
|
|
|
} else if (drm_props.hasPrimary) {
|
|
|
|
devid = makedev(drm_props.primaryMajor, drm_props.primaryMinor);
|
|
|
|
} else {
|
|
|
|
wlr_log(WLR_ERROR, "Physical device is missing both render and primary nodes");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
drmDevice *device = NULL;
|
|
|
|
if (drmGetDeviceFromDevId(devid, 0, &device) != 0) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "drmGetDeviceFromDevId failed");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *name = NULL;
|
|
|
|
if (device->available_nodes & (1 << DRM_NODE_RENDER)) {
|
|
|
|
name = device->nodes[DRM_NODE_RENDER];
|
|
|
|
} else {
|
|
|
|
assert(device->available_nodes & (1 << DRM_NODE_PRIMARY));
|
|
|
|
name = device->nodes[DRM_NODE_PRIMARY];
|
|
|
|
wlr_log(WLR_DEBUG, "DRM device %s has no render node, "
|
|
|
|
"falling back to primary node", name);
|
|
|
|
}
|
|
|
|
|
|
|
|
int drm_fd = open(name, O_RDWR | O_NONBLOCK | O_CLOEXEC);
|
|
|
|
if (drm_fd < 0) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "Failed to open DRM node %s", name);
|
|
|
|
}
|
|
|
|
drmFreeDevice(&device);
|
|
|
|
return drm_fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void load_device_proc(struct wlr_vk_device *dev, const char *name,
|
|
|
|
void *proc_ptr) {
|
|
|
|
void *proc = (void *)vkGetDeviceProcAddr(dev->dev, name);
|
|
|
|
if (proc == NULL) {
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
*(void **)proc_ptr = proc;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct wlr_vk_device *vulkan_device_create(struct wlr_vk_instance *ini,
|
|
|
|
VkPhysicalDevice phdev) {
|
|
|
|
VkResult res;
|
|
|
|
|
|
|
|
uint32_t avail_extc = 0;
|
|
|
|
res = vkEnumerateDeviceExtensionProperties(phdev, NULL,
|
|
|
|
&avail_extc, NULL);
|
|
|
|
if (res != VK_SUCCESS || avail_extc == 0) {
|
|
|
|
wlr_vk_error("Could not enumerate device extensions (1)", res);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
VkExtensionProperties avail_ext_props[avail_extc + 1];
|
|
|
|
res = vkEnumerateDeviceExtensionProperties(phdev, NULL,
|
|
|
|
&avail_extc, avail_ext_props);
|
|
|
|
if (res != VK_SUCCESS) {
|
|
|
|
wlr_vk_error("Could not enumerate device extensions (2)", res);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t j = 0; j < avail_extc; ++j) {
|
|
|
|
wlr_log(WLR_DEBUG, "Vulkan device extension %s v%"PRIu32,
|
|
|
|
avail_ext_props[j].extensionName, avail_ext_props[j].specVersion);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct wlr_vk_device *dev = calloc(1, sizeof(*dev));
|
|
|
|
if (!dev) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "allocation failed");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev->phdev = phdev;
|
|
|
|
dev->instance = ini;
|
|
|
|
dev->drm_fd = -1;
|
|
|
|
|
|
|
|
// For dmabuf import we require at least the external_memory_fd,
|
|
|
|
// external_memory_dma_buf, queue_family_foreign,
|
|
|
|
// image_drm_format_modifier, and image_format_list extensions.
|
|
|
|
// The size is set to a large number to allow for other conditional
|
|
|
|
// extensions before the device is created
|
|
|
|
const char *extensions[32] = {0};
|
|
|
|
size_t extensions_len = 0;
|
|
|
|
extensions[extensions_len++] = VK_KHR_EXTERNAL_MEMORY_FD_EXTENSION_NAME;
|
|
|
|
extensions[extensions_len++] = VK_KHR_EXTERNAL_SEMAPHORE_FD_EXTENSION_NAME;
|
|
|
|
extensions[extensions_len++] = VK_KHR_IMAGE_FORMAT_LIST_EXTENSION_NAME; // or vulkan 1.2
|
|
|
|
extensions[extensions_len++] = VK_EXT_EXTERNAL_MEMORY_DMA_BUF_EXTENSION_NAME;
|
|
|
|
extensions[extensions_len++] = VK_EXT_QUEUE_FAMILY_FOREIGN_EXTENSION_NAME;
|
|
|
|
extensions[extensions_len++] = VK_EXT_IMAGE_DRM_FORMAT_MODIFIER_EXTENSION_NAME;
|
|
|
|
extensions[extensions_len++] = VK_KHR_TIMELINE_SEMAPHORE_EXTENSION_NAME; // or vulkan 1.2
|
|
|
|
extensions[extensions_len++] = VK_KHR_SYNCHRONIZATION_2_EXTENSION_NAME; // or vulkan 1.3
|
|
|
|
|
|
|
|
for (size_t i = 0; i < extensions_len; i++) {
|
|
|
|
if (!check_extension(avail_ext_props, avail_extc, extensions[i])) {
|
|
|
|
wlr_log(WLR_ERROR, "vulkan: required device extension %s not found",
|
|
|
|
extensions[i]);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
uint32_t qfam_count;
|
|
|
|
vkGetPhysicalDeviceQueueFamilyProperties(phdev, &qfam_count, NULL);
|
|
|
|
assert(qfam_count > 0);
|
|
|
|
VkQueueFamilyProperties queue_props[qfam_count];
|
|
|
|
vkGetPhysicalDeviceQueueFamilyProperties(phdev, &qfam_count,
|
|
|
|
queue_props);
|
|
|
|
|
|
|
|
bool graphics_found = false;
|
|
|
|
for (unsigned i = 0u; i < qfam_count; ++i) {
|
|
|
|
graphics_found = queue_props[i].queueFlags & VK_QUEUE_GRAPHICS_BIT;
|
|
|
|
if (graphics_found) {
|
|
|
|
dev->queue_family = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(graphics_found);
|
|
|
|
}
|
|
|
|
|
|
|
|
const VkPhysicalDeviceExternalSemaphoreInfo ext_semaphore_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_EXTERNAL_SEMAPHORE_INFO,
|
|
|
|
.handleType = VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT,
|
|
|
|
};
|
|
|
|
VkExternalSemaphoreProperties ext_semaphore_props = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_EXTERNAL_SEMAPHORE_PROPERTIES,
|
|
|
|
};
|
|
|
|
vkGetPhysicalDeviceExternalSemaphoreProperties(phdev,
|
|
|
|
&ext_semaphore_info, &ext_semaphore_props);
|
|
|
|
bool exportable_semaphore = ext_semaphore_props.externalSemaphoreFeatures &
|
|
|
|
VK_EXTERNAL_SEMAPHORE_FEATURE_EXPORTABLE_BIT;
|
|
|
|
bool importable_semaphore = ext_semaphore_props.externalSemaphoreFeatures &
|
|
|
|
VK_EXTERNAL_SEMAPHORE_FEATURE_IMPORTABLE_BIT;
|
|
|
|
if (!exportable_semaphore) {
|
|
|
|
wlr_log(WLR_DEBUG, "VkSemaphore is not exportable to a sync_file");
|
|
|
|
}
|
|
|
|
if (!importable_semaphore) {
|
|
|
|
wlr_log(WLR_DEBUG, "VkSemaphore is not importable from a sync_file");
|
|
|
|
}
|
|
|
|
|
|
|
|
bool dmabuf_sync_file_import_export = dmabuf_check_sync_file_import_export();
|
|
|
|
if (!dmabuf_sync_file_import_export) {
|
|
|
|
wlr_log(WLR_DEBUG, "DMA-BUF sync_file import/export not supported");
|
|
|
|
}
|
|
|
|
|
|
|
|
dev->implicit_sync_interop =
|
|
|
|
exportable_semaphore && importable_semaphore && dmabuf_sync_file_import_export;
|
|
|
|
if (dev->implicit_sync_interop) {
|
|
|
|
wlr_log(WLR_DEBUG, "Implicit sync interop supported");
|
|
|
|
} else {
|
|
|
|
wlr_log(WLR_INFO, "Implicit sync interop not supported, "
|
|
|
|
"falling back to blocking");
|
|
|
|
}
|
|
|
|
|
|
|
|
VkPhysicalDeviceSamplerYcbcrConversionFeatures phdev_sampler_ycbcr_features = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SAMPLER_YCBCR_CONVERSION_FEATURES,
|
|
|
|
};
|
|
|
|
VkPhysicalDeviceFeatures2 phdev_features = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_FEATURES_2,
|
|
|
|
.pNext = &phdev_sampler_ycbcr_features,
|
|
|
|
};
|
|
|
|
vkGetPhysicalDeviceFeatures2(phdev, &phdev_features);
|
|
|
|
|
|
|
|
dev->sampler_ycbcr_conversion = phdev_sampler_ycbcr_features.samplerYcbcrConversion;
|
|
|
|
wlr_log(WLR_DEBUG, "Sampler YCbCr conversion %s",
|
|
|
|
dev->sampler_ycbcr_conversion ? "supported" : "not supported");
|
|
|
|
|
|
|
|
const float prio = 1.f;
|
|
|
|
VkDeviceQueueCreateInfo qinfo = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO,
|
|
|
|
.queueFamilyIndex = dev->queue_family,
|
|
|
|
.queueCount = 1,
|
|
|
|
.pQueuePriorities = &prio,
|
|
|
|
};
|
|
|
|
|
|
|
|
VkDeviceQueueGlobalPriorityCreateInfoKHR global_priority;
|
|
|
|
bool has_global_priority = check_extension(avail_ext_props, avail_extc,
|
|
|
|
VK_KHR_GLOBAL_PRIORITY_EXTENSION_NAME);
|
|
|
|
if (has_global_priority) {
|
|
|
|
// If global priorities are supported, request a high-priority context
|
|
|
|
global_priority = (VkDeviceQueueGlobalPriorityCreateInfoKHR){
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_GLOBAL_PRIORITY_CREATE_INFO_KHR,
|
|
|
|
.globalPriority = VK_QUEUE_GLOBAL_PRIORITY_HIGH_KHR,
|
|
|
|
};
|
|
|
|
qinfo.pNext = &global_priority;
|
|
|
|
extensions[extensions_len++] = VK_KHR_GLOBAL_PRIORITY_EXTENSION_NAME;
|
|
|
|
wlr_log(WLR_DEBUG, "Requesting a high-priority device queue");
|
|
|
|
} else {
|
|
|
|
wlr_log(WLR_DEBUG, "Global priorities are not supported, "
|
|
|
|
"falling back to regular queue priority");
|
|
|
|
}
|
|
|
|
|
|
|
|
VkPhysicalDeviceSamplerYcbcrConversionFeatures sampler_ycbcr_features = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SAMPLER_YCBCR_CONVERSION_FEATURES,
|
|
|
|
.samplerYcbcrConversion = dev->sampler_ycbcr_conversion,
|
|
|
|
};
|
|
|
|
VkPhysicalDeviceSynchronization2FeaturesKHR sync2_features = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SYNCHRONIZATION_2_FEATURES_KHR,
|
|
|
|
.pNext = &sampler_ycbcr_features,
|
|
|
|
.synchronization2 = VK_TRUE,
|
|
|
|
};
|
|
|
|
VkPhysicalDeviceTimelineSemaphoreFeaturesKHR timeline_features = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_TIMELINE_SEMAPHORE_FEATURES_KHR,
|
|
|
|
.pNext = &sync2_features,
|
|
|
|
.timelineSemaphore = VK_TRUE,
|
|
|
|
};
|
|
|
|
VkDeviceCreateInfo dev_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO,
|
|
|
|
.pNext = &timeline_features,
|
|
|
|
.queueCreateInfoCount = 1u,
|
|
|
|
.pQueueCreateInfos = &qinfo,
|
|
|
|
.enabledExtensionCount = extensions_len,
|
|
|
|
.ppEnabledExtensionNames = extensions,
|
|
|
|
};
|
|
|
|
|
|
|
|
assert(extensions_len < sizeof(extensions) / sizeof(extensions[0]));
|
|
|
|
|
|
|
|
res = vkCreateDevice(phdev, &dev_info, NULL, &dev->dev);
|
|
|
|
|
|
|
|
if (has_global_priority && (res == VK_ERROR_NOT_PERMITTED_EXT ||
|
|
|
|
res == VK_ERROR_INITIALIZATION_FAILED)) {
|
|
|
|
// Try to recover from the driver denying a global priority queue
|
|
|
|
wlr_log(WLR_DEBUG, "Failed to obtain a high-priority device queue, "
|
|
|
|
"falling back to regular queue priority");
|
|
|
|
qinfo.pNext = NULL;
|
|
|
|
res = vkCreateDevice(phdev, &dev_info, NULL, &dev->dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res != VK_SUCCESS) {
|
|
|
|
wlr_vk_error("Failed to create vulkan device", res);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
vkGetDeviceQueue(dev->dev, dev->queue_family, 0, &dev->queue);
|
|
|
|
|
|
|
|
load_device_proc(dev, "vkGetMemoryFdPropertiesKHR",
|
|
|
|
&dev->api.vkGetMemoryFdPropertiesKHR);
|
|
|
|
load_device_proc(dev, "vkWaitSemaphoresKHR", &dev->api.vkWaitSemaphoresKHR);
|
|
|
|
load_device_proc(dev, "vkGetSemaphoreCounterValueKHR",
|
|
|
|
&dev->api.vkGetSemaphoreCounterValueKHR);
|
|
|
|
load_device_proc(dev, "vkGetSemaphoreFdKHR", &dev->api.vkGetSemaphoreFdKHR);
|
|
|
|
load_device_proc(dev, "vkImportSemaphoreFdKHR", &dev->api.vkImportSemaphoreFdKHR);
|
|
|
|
load_device_proc(dev, "vkQueueSubmit2KHR", &dev->api.vkQueueSubmit2KHR);
|
|
|
|
|
|
|
|
size_t max_fmts;
|
|
|
|
const struct wlr_vk_format *fmts = vulkan_get_format_list(&max_fmts);
|
|
|
|
dev->shm_formats = calloc(max_fmts, sizeof(*dev->shm_formats));
|
|
|
|
dev->format_props = calloc(max_fmts, sizeof(*dev->format_props));
|
|
|
|
if (!dev->shm_formats || !dev->format_props) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "allocation failed");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
wlr_log(WLR_DEBUG, "Supported Vulkan formats:");
|
|
|
|
for (unsigned i = 0u; i < max_fmts; ++i) {
|
|
|
|
vulkan_format_props_query(dev, &fmts[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return dev;
|
|
|
|
|
|
|
|
error:
|
|
|
|
vulkan_device_destroy(dev);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void vulkan_device_destroy(struct wlr_vk_device *dev) {
|
|
|
|
if (!dev) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dev->dev) {
|
|
|
|
vkDestroyDevice(dev->dev, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dev->drm_fd > 0) {
|
|
|
|
close(dev->drm_fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
wlr_drm_format_set_finish(&dev->dmabuf_render_formats);
|
|
|
|
wlr_drm_format_set_finish(&dev->dmabuf_texture_formats);
|
|
|
|
|
|
|
|
for (unsigned i = 0u; i < dev->format_prop_count; ++i) {
|
|
|
|
vulkan_format_props_finish(&dev->format_props[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(dev->shm_formats);
|
|
|
|
free(dev->format_props);
|
|
|
|
free(dev);
|
|
|
|
}
|