// Copyright 2020, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Vulkan image allocator helper. * @author Jakob Bornecrantz * @ingroup aux_vk */ #include "util/u_misc.h" #include "util/u_logging.h" #include "vk/vk_image_allocator.h" #ifdef XRT_OS_LINUX #include #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, const struct xrt_swapchain_create_info *info, struct vk_image *out_image) { VkImageUsageFlags image_usage = vk_swapchain_usage_flags(info->bits); 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, const struct xrt_swapchain_create_info *xscci, 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, const struct xrt_swapchain_create_info *xscci, 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].handle); 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; } native_images[i].handle = 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].handle); native_images[k].handle = XRT_GRAPHICS_BUFFER_HANDLE_INVALID; 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