2020-07-08 14:53:52 +00:00
|
|
|
// Copyright 2020, Collabora, Ltd.
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief Vulkan image allocator helper.
|
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
|
|
|
* @ingroup aux_vk
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "util/u_misc.h"
|
|
|
|
#include "util/u_logging.h"
|
|
|
|
|
|
|
|
#include "vk/vk_image_allocator.h"
|
|
|
|
|
|
|
|
#ifdef XRT_OS_LINUX
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Helper functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef XRT_OS_LINUX
|
|
|
|
static VkResult
|
|
|
|
get_device_memory_fd(struct vk_bundle *vk,
|
|
|
|
VkDeviceMemory device_memory,
|
|
|
|
int *out_fd)
|
|
|
|
{
|
|
|
|
// vkGetMemoryFdKHR parameter
|
|
|
|
VkMemoryGetFdInfoKHR fd_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_MEMORY_GET_FD_INFO_KHR,
|
|
|
|
.memory = device_memory,
|
|
|
|
.handleType = VK_EXTERNAL_MEMORY_HANDLE_TYPE_OPAQUE_FD_BIT_KHR,
|
|
|
|
};
|
|
|
|
int fd;
|
|
|
|
VkResult ret = vk->vkGetMemoryFdKHR(vk->device, &fd_info, &fd);
|
|
|
|
if (ret != VK_SUCCESS) {
|
|
|
|
// COMP_ERROR(c, "->image - vkGetMemoryFdKHR: %s",
|
|
|
|
// vk_result_string(ret));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
*out_fd = fd;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static VkResult
|
|
|
|
create_image(struct vk_bundle *vk,
|
2020-08-07 15:58:05 +00:00
|
|
|
const struct xrt_swapchain_create_info *info,
|
2020-07-08 14:53:52 +00:00
|
|
|
struct vk_image *out_image)
|
|
|
|
{
|
2020-07-27 15:22:00 +00:00
|
|
|
VkImageUsageFlags image_usage = vk_swapchain_usage_flags(info->bits);
|
2020-07-08 14:53:52 +00:00
|
|
|
VkDeviceMemory device_memory = VK_NULL_HANDLE;
|
|
|
|
VkImage image = VK_NULL_HANDLE;
|
|
|
|
VkResult ret = VK_SUCCESS;
|
|
|
|
VkDeviceSize size;
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Create the image.
|
|
|
|
*/
|
|
|
|
|
|
|
|
VkExternalMemoryImageCreateInfoKHR external_memory_image_create_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_EXTERNAL_MEMORY_IMAGE_CREATE_INFO_KHR,
|
|
|
|
.handleTypes = VK_EXTERNAL_MEMORY_HANDLE_TYPE_OPAQUE_FD_BIT_KHR,
|
|
|
|
};
|
|
|
|
|
|
|
|
VkImageCreateInfo create_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
|
|
|
|
.pNext = &external_memory_image_create_info,
|
|
|
|
.imageType = VK_IMAGE_TYPE_2D,
|
|
|
|
.format = (VkFormat)info->format,
|
|
|
|
.extent = {.width = info->width,
|
|
|
|
.height = info->height,
|
|
|
|
.depth = 1},
|
|
|
|
.mipLevels = info->mip_count,
|
|
|
|
.arrayLayers = info->array_size,
|
|
|
|
.samples = VK_SAMPLE_COUNT_1_BIT,
|
|
|
|
.tiling = VK_IMAGE_TILING_OPTIMAL,
|
|
|
|
.usage = image_usage,
|
|
|
|
.sharingMode = VK_SHARING_MODE_EXCLUSIVE,
|
|
|
|
.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED,
|
|
|
|
};
|
|
|
|
|
|
|
|
ret = vk->vkCreateImage(vk->device, &create_info, NULL, &image);
|
|
|
|
if (ret != VK_SUCCESS) {
|
|
|
|
U_LOG_E("vkCreateImage: %s", vk_result_string(ret));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Create and bind the memory.
|
|
|
|
*/
|
|
|
|
#ifdef XRT_OS_LINUX
|
|
|
|
// vkAllocateMemory parameters
|
|
|
|
VkMemoryDedicatedAllocateInfoKHR dedicated_memory_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_MEMORY_DEDICATED_ALLOCATE_INFO_KHR,
|
|
|
|
.image = image,
|
|
|
|
.buffer = VK_NULL_HANDLE,
|
|
|
|
};
|
|
|
|
|
|
|
|
VkExportMemoryAllocateInfo export_alloc_info = {
|
|
|
|
.sType = VK_STRUCTURE_TYPE_EXPORT_MEMORY_ALLOCATE_INFO_KHR,
|
|
|
|
.pNext = &dedicated_memory_info,
|
|
|
|
.handleTypes = VK_EXTERNAL_MEMORY_HANDLE_TYPE_OPAQUE_FD_BIT_KHR,
|
|
|
|
};
|
|
|
|
|
|
|
|
ret = vk_alloc_and_bind_image_memory(
|
|
|
|
vk, image, SIZE_MAX, &export_alloc_info, &device_memory, &size);
|
|
|
|
#else
|
|
|
|
ret = VK_ERROR_INITIALIZATION_FAILED;
|
|
|
|
#endif
|
|
|
|
if (ret != VK_SUCCESS) {
|
|
|
|
U_LOG_E("vkAllocateMemory: %s", vk_result_string(ret));
|
|
|
|
vk->vkDestroyImage(vk->device, image, NULL);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
out_image->handle = image;
|
|
|
|
out_image->memory = device_memory;
|
|
|
|
out_image->size = size;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
destroy_image(struct vk_bundle *vk, struct vk_image *image)
|
|
|
|
{
|
|
|
|
if (image->handle != VK_NULL_HANDLE) {
|
|
|
|
vk->vkDestroyImage(vk->device, image->handle, NULL);
|
|
|
|
image->handle = VK_NULL_HANDLE;
|
|
|
|
}
|
|
|
|
if (image->memory != VK_NULL_HANDLE) {
|
|
|
|
vk->vkFreeMemory(vk->device, image->memory, NULL);
|
|
|
|
image->memory = VK_NULL_HANDLE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* 'Exported' functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
VkResult
|
|
|
|
vk_ic_allocate(struct vk_bundle *vk,
|
2020-08-07 15:58:05 +00:00
|
|
|
const struct xrt_swapchain_create_info *xscci,
|
2020-07-08 14:53:52 +00:00
|
|
|
uint32_t num_images,
|
|
|
|
struct vk_image_collection *out_vkic)
|
|
|
|
{
|
|
|
|
VkResult ret = VK_SUCCESS;
|
|
|
|
|
|
|
|
if (num_images > ARRAY_SIZE(out_vkic->images)) {
|
|
|
|
return VK_ERROR_OUT_OF_HOST_MEMORY;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
size_t i = 0;
|
|
|
|
for (; i < num_images; i++) {
|
|
|
|
ret = create_image(vk, xscci, &out_vkic->images[i]);
|
|
|
|
if (ret != VK_SUCCESS) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the fields.
|
|
|
|
out_vkic->num_images = num_images;
|
|
|
|
out_vkic->info = *xscci;
|
|
|
|
|
|
|
|
if (ret == VK_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// i is the index of the failed image, everything before that index
|
|
|
|
// succeeded and needs to be destroyed. If i is zero no call succeeded.
|
|
|
|
while (i > 0) {
|
|
|
|
i--;
|
|
|
|
destroy_image(vk, &out_vkic->images[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
U_ZERO(out_vkic);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* Imports and set images from the given FDs.
|
|
|
|
*/
|
|
|
|
VkResult
|
|
|
|
vk_ic_from_natives(struct vk_bundle *vk,
|
2020-08-07 15:58:05 +00:00
|
|
|
const struct xrt_swapchain_create_info *xscci,
|
2020-07-08 14:53:52 +00:00
|
|
|
struct xrt_image_native *native_images,
|
|
|
|
uint32_t num_images,
|
|
|
|
struct vk_image_collection *out_vkic)
|
|
|
|
{
|
|
|
|
VkResult ret = VK_ERROR_INITIALIZATION_FAILED;
|
|
|
|
|
|
|
|
if (num_images > ARRAY_SIZE(out_vkic->images)) {
|
|
|
|
return VK_ERROR_OUT_OF_HOST_MEMORY;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
size_t i = 0;
|
|
|
|
#ifdef XRT_OS_LINUX
|
|
|
|
for (; i < num_images; i++) {
|
|
|
|
// Ensure that all fds are consumed or none are.
|
|
|
|
int fd = dup(native_images[i].fd);
|
|
|
|
|
|
|
|
ret = vk_create_image_from_native(vk, xscci, &native_images[i],
|
|
|
|
&out_vkic->images[i].handle,
|
|
|
|
&out_vkic->images[i].memory);
|
|
|
|
if (ret != VK_SUCCESS) {
|
|
|
|
close(fd);
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
native_images[i].fd = fd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Set the fields.
|
|
|
|
out_vkic->num_images = num_images;
|
|
|
|
out_vkic->info = *xscci;
|
|
|
|
|
|
|
|
if (ret == VK_SUCCESS) {
|
|
|
|
#ifdef XRT_OS_LINUX
|
|
|
|
// We have consumed all fds now, close all of the copies we
|
|
|
|
// made, all this to make sure we do all or nothing.
|
|
|
|
for (size_t k = 0; k < num_images; k++) {
|
|
|
|
close(native_images[k].fd);
|
|
|
|
native_images[k].fd = -1;
|
|
|
|
native_images[k].size = 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// i is the index of the failed image, everything before that index
|
|
|
|
// succeeded and needs to be destroyed. If i is zero no call succeeded.
|
|
|
|
while (i > 0) {
|
|
|
|
i--;
|
|
|
|
destroy_image(vk, &out_vkic->images[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
U_ZERO(out_vkic);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
vk_ic_destroy(struct vk_bundle *vk, struct vk_image_collection *vkic)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < vkic->num_images; i++) {
|
|
|
|
destroy_image(vk, &vkic->images[i]);
|
|
|
|
}
|
|
|
|
vkic->num_images = 0;
|
|
|
|
U_ZERO(&vkic->info);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef XRT_OS_LINUX
|
|
|
|
VkResult
|
|
|
|
vk_ic_get_fds(struct vk_bundle *vk,
|
|
|
|
struct vk_image_collection *vkic,
|
|
|
|
uint32_t max_fds,
|
|
|
|
int *out_fds)
|
|
|
|
{
|
|
|
|
VkResult ret = VK_SUCCESS;
|
|
|
|
|
|
|
|
size_t i = 0;
|
|
|
|
for (; i < vkic->num_images && i < max_fds; i++) {
|
|
|
|
ret = get_device_memory_fd(vk, vkic->images[i].memory,
|
|
|
|
&out_fds[i]);
|
|
|
|
if (ret != VK_SUCCESS) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret == VK_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// i is the index of the failed fd, everything before that index
|
|
|
|
// succeeded and needs to be closed. If i is zero no call succeeded.
|
|
|
|
while (i > 0) {
|
|
|
|
i--;
|
|
|
|
close(out_fds[i]);
|
|
|
|
out_fds[i] = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|