// Copyright 2020-2022, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief D3D11 backed image buffer allocator. * @author Ryan Pavlik * @author Fernando Velazquez Innella * @ingroup aux_d3d */ #include "d3d_d3d11_allocator.h" #include "d3d_d3d11_allocator.hpp" #include "d3d_d3d11_bits.h" #include "d3d_dxgi_formats.h" #include "util/u_misc.h" #include "util/u_logging.h" #include "util/u_debug.h" #include "util/u_handles.h" #include "xrt/xrt_windows.h" #include #include #include #include #include #include #define DEFAULT_CATCH(...) \ catch (wil::ResultException const &e) \ { \ U_LOG_E("Caught exception: %s", e.what()); \ return __VA_ARGS__; \ } \ catch (std::exception const &e) \ { \ U_LOG_E("Caught exception: %s", e.what()); \ return __VA_ARGS__; \ } \ catch (...) \ { \ U_LOG_E("Caught exception"); \ return __VA_ARGS__; \ } DEBUG_GET_ONCE_LOG_OPTION(d3d11_log, "DXGI_LOG", U_LOGGING_WARN) #define D3DA_TRACE(...) U_LOG_IFL_T(debug_get_log_option_d3d11_log(), __VA_ARGS__) #define D3DA_DEBUG(...) U_LOG_IFL_D(debug_get_log_option_d3d11_log(), __VA_ARGS__) #define D3DA_INFO(...) U_LOG_IFL_I(debug_get_log_option_d3d11_log(), __VA_ARGS__) #define D3DA_WARN(...) U_LOG_IFL_W(debug_get_log_option_d3d11_log(), __VA_ARGS__) #define D3DA_ERROR(...) U_LOG_IFL_E(debug_get_log_option_d3d11_log(), __VA_ARGS__) namespace xrt::auxiliary::d3d::d3d11 { HANDLE getSharedHandle(const wil::com_ptr &image) { wil::com_ptr dxgiRes; HANDLE h; image.query_to(dxgiRes.put()); THROW_IF_FAILED(dxgiRes->GetSharedHandle(&h)); return h; } xrt_result_t allocateSharedImages(ID3D11Device5 &device, const xrt_swapchain_create_info &xsci, size_t image_count, bool keyed_mutex, std::vector> &out_images, std::vector &out_handles) try { if (0 != (xsci.create & XRT_SWAPCHAIN_CREATE_PROTECTED_CONTENT)) { return XRT_ERROR_SWAPCHAIN_FLAG_VALID_BUT_UNSUPPORTED; } if (0 != (xsci.create & XRT_SWAPCHAIN_CREATE_STATIC_IMAGE) && image_count > 1) { D3DA_ERROR("Got XRT_SWAPCHAIN_CREATE_STATIC_IMAGE but an image count greater than 1!"); return XRT_ERROR_ALLOCATION; } if (xsci.array_size == 0) { D3DA_ERROR("Array size must not be 0"); return XRT_ERROR_ALLOCATION; } CD3D11_TEXTURE2D_DESC1 desc{d3d_dxgi_format_to_typeless_dxgi((DXGI_FORMAT)xsci.format), xsci.width, xsci.height, xsci.array_size, xsci.mip_count, d3d_convert_usage_bits_to_d3d11_bind_flags(xsci.bits)}; desc.SampleDesc.Count = xsci.sample_count; // Using NT handles for sharing textures has a lot of limitations and issues. // Depth formats are not supported and Vulkan may fail at vkAllocateMemory desc.MiscFlags = keyed_mutex ? D3D11_RESOURCE_MISC_SHARED_KEYEDMUTEX : D3D11_RESOURCE_MISC_SHARED; if (desc.Format == 0) { D3DA_ERROR("Invalid format %04" PRIx64 "!", (uint64_t)xsci.format); return XRT_ERROR_SWAPCHAIN_FORMAT_UNSUPPORTED; } if (xsci.face_count == 6) { desc.ArraySize *= 6; desc.MiscFlags |= D3D11_RESOURCE_MISC_TEXTURECUBE; } // Create textures std::vector> images; for (size_t i = 0; i < image_count; ++i) { wil::com_ptr tex; if (FAILED(LOG_IF_FAILED(device.CreateTexture2D1(&desc, nullptr, tex.put())))) { return XRT_ERROR_ALLOCATION; } images.emplace_back(std::move(tex)); } std::vector handles; handles.reserve(image_count); for (const auto &tex : images) { handles.emplace_back(getSharedHandle(tex)); } out_images = std::move(images); out_handles = std::move(handles); return XRT_SUCCESS; } DEFAULT_CATCH(XRT_ERROR_ALLOCATION) } // namespace xrt::auxiliary::d3d::d3d11 struct d3d11_allocator { struct xrt_image_native_allocator base; wil::com_ptr device; }; static xrt_result_t d3d11_images_allocate(struct xrt_image_native_allocator *xina, const struct xrt_swapchain_create_info *xsci, size_t image_count, struct xrt_image_native *out_images) { try { d3d11_allocator *d3da = reinterpret_cast(xina); std::vector> images; std::vector handles; auto result = xrt::auxiliary::d3d::d3d11::allocateSharedImages(*(d3da->device), *xsci, image_count, false, images, handles); if (result != XRT_SUCCESS) { return result; } for (size_t i = 0; i < image_count; ++i) { out_images[i].handle = handles[i]; out_images[i].is_dxgi_handle = true; } return XRT_SUCCESS; } DEFAULT_CATCH(XRT_ERROR_ALLOCATION) } static xrt_result_t d3d11_images_free(struct xrt_image_native_allocator *xina, size_t image_count, struct xrt_image_native *images) { for (size_t i = 0; i < image_count; ++i) { if (!images[i].is_dxgi_handle) { u_graphics_buffer_unref(&(images[i].handle)); } } return XRT_SUCCESS; } static void d3d11_destroy(struct xrt_image_native_allocator *xina) { d3d11_allocator *d3da = reinterpret_cast(xina); delete d3da; } struct xrt_image_native_allocator * d3d11_allocator_create(ID3D11Device *device) try { auto ret = std::make_unique(); U_ZERO(&(ret->base)); ret->base.images_allocate = d3d11_images_allocate; ret->base.images_free = d3d11_images_free; ret->base.destroy = d3d11_destroy; return &(ret.release()->base); } DEFAULT_CATCH(nullptr)