// Copyright 2019-2023, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Code for handling distortion resources (not shaders). * @author Jakob Bornecrantz * @ingroup comp_render */ #include "xrt/xrt_device.h" #include "math/m_api.h" #include "math/m_matrix_2x2.h" #include "math/m_vec2.h" #include "render/render_interface.h" /* * * Helper defines. * */ /*! * This define will error if `RET` is not `VK_SUCCESS`, printing out that the * `FUNC_STR` string has failed, then goto `GOTO`, `VK` will be used for the * `VK_ERROR` call. */ #define CG(VK, RET, FUNC_STR, GOTO) \ do { \ VkResult CG_ret = RET; \ if (CG_ret != VK_SUCCESS) { \ VK_ERROR(VK, FUNC_STR ": %s", vk_result_string(CG_ret)); \ goto GOTO; \ } \ } while (false) /*! * Calls `vkDestroy##TYPE` on `THING` if it is not `VK_NULL_HANDLE`, sets it to * `VK_NULL_HANDLE` afterwards. */ #define D(TYPE, THING) \ if (THING != VK_NULL_HANDLE) { \ vk->vkDestroy##TYPE(vk->device, THING, NULL); \ THING = VK_NULL_HANDLE; \ } /*! * Calls `vkFree##TYPE` on `THING` if it is not `VK_NULL_HANDLE`, sets it to * `VK_NULL_HANDLE` afterwards. */ #define DF(TYPE, THING) \ if (THING != VK_NULL_HANDLE) { \ vk->vkFree##TYPE(vk->device, THING, NULL); \ THING = VK_NULL_HANDLE; \ } /* * * Helper functions. * */ static VkResult create_distortion_image_and_view(struct vk_bundle *vk, VkExtent2D extent, VkDeviceMemory *out_device_memory, VkImage *out_image, VkImageView *out_image_view) { VkFormat format = VK_FORMAT_R32G32_SFLOAT; VkImage image = VK_NULL_HANDLE; VkDeviceMemory device_memory = VK_NULL_HANDLE; VkImageView image_view = VK_NULL_HANDLE; VkImageViewType view_type = VK_IMAGE_VIEW_TYPE_2D; VkResult ret; ret = vk_create_image_simple( // vk, // vk_bundle extent, // extent format, // format VK_IMAGE_USAGE_SAMPLED_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT, // usage &device_memory, // out_device_memory &image); // out_image if (ret != VK_SUCCESS) { VK_ERROR(vk, "vk_create_image_simple: %s", vk_result_string(ret)); return ret; } VkImageSubresourceRange subresource_range = { .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT, .baseMipLevel = 0, .levelCount = VK_REMAINING_MIP_LEVELS, .baseArrayLayer = 0, .layerCount = VK_REMAINING_ARRAY_LAYERS, }; ret = vk_create_view( // vk, // vk_bundle image, // image view_type, // type format, // format subresource_range, // subresource_range &image_view); // out_image_view if (ret != VK_SUCCESS) { VK_ERROR(vk, "vk_create_view: %s", vk_result_string(ret)); D(Image, image); DF(Memory, device_memory); return ret; } *out_device_memory = device_memory; *out_image = image; *out_image_view = image_view; return VK_SUCCESS; } static void queue_upload_for_first_level_and_layer_locked( struct vk_bundle *vk, VkCommandBuffer cmd, VkBuffer src, VkImage dst, VkExtent2D extent) { VkImageSubresourceRange subresource_range = { .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT, .baseMipLevel = 0, .levelCount = VK_REMAINING_MIP_LEVELS, .baseArrayLayer = 0, .layerCount = VK_REMAINING_ARRAY_LAYERS, }; vk_cmd_image_barrier_gpu_locked( // vk, // cmd, // dst, // 0, // VK_ACCESS_TRANSFER_WRITE_BIT, // VK_IMAGE_LAYOUT_UNDEFINED, // VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, // subresource_range); // VkImageSubresourceLayers subresource_layers = { .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT, .mipLevel = 0, .baseArrayLayer = 0, .layerCount = 1, }; VkBufferImageCopy region = { .bufferOffset = 0, .bufferRowLength = 0, .bufferImageHeight = 0, .imageSubresource = subresource_layers, .imageOffset = {0, 0, 0}, .imageExtent = {extent.width, extent.height, 1}, }; vk->vkCmdCopyBufferToImage( // cmd, // commandBuffer src, // srcBuffer dst, // dstImage VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, // dstImageLayout 1, // regionCount ®ion); // pRegions vk_cmd_image_barrier_gpu_locked( // vk, // cmd, // dst, // VK_ACCESS_TRANSFER_WRITE_BIT, // VK_ACCESS_SHADER_READ_BIT, // VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, // VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL, // subresource_range); // } static VkResult create_and_queue_upload_locked(struct vk_bundle *vk, struct vk_cmd_pool *pool, VkCommandBuffer cmd, VkBuffer src_buffer, VkDeviceMemory *out_image_device_memory, VkImage *out_image, VkImageView *out_image_view) { VkExtent2D extent = {COMP_DISTORTION_IMAGE_DIMENSIONS, COMP_DISTORTION_IMAGE_DIMENSIONS}; VkDeviceMemory device_memory = VK_NULL_HANDLE; VkImage image = VK_NULL_HANDLE; VkImageView image_view = VK_NULL_HANDLE; VkResult ret; ret = create_distortion_image_and_view( // vk, // vk_bundle extent, // extent &device_memory, // out_device_memory &image, // out_image &image_view); // out_image_view if (ret != VK_SUCCESS) { return ret; } queue_upload_for_first_level_and_layer_locked( // vk, // vk_bundle cmd, // cmd src_buffer, // src image, // dst extent); // extent *out_image_device_memory = device_memory; *out_image = image; *out_image_view = image_view; return VK_SUCCESS; } /*! * Helper struct to make code easier to read. */ struct texture { struct xrt_vec2 pixels[COMP_DISTORTION_IMAGE_DIMENSIONS][COMP_DISTORTION_IMAGE_DIMENSIONS]; }; struct tan_angles_transforms { struct xrt_vec2 offset; struct xrt_vec2 scale; }; static void calc_uv_to_tanangle(struct xrt_device *xdev, uint32_t view, struct xrt_normalized_rect *out_rect) { const struct xrt_fov fov = xdev->hmd->distortion.fov[view]; const double tan_left = tan(fov.angle_left); const double tan_right = tan(fov.angle_right); const double tan_down = tan(fov.angle_down); const double tan_up = tan(fov.angle_up); const double tan_width = tan_right - tan_left; const double tan_height = tan_up - tan_down; /* * I do not know why we have to calculate the offsets like this, but * this one is the one that seems to work with what is currently in the * calc timewarp matrix function and the distortion shader. It works * with Index (unbalanced left and right angles) and WMR (unbalanced up * and down angles) so here it is. In so far it matches what the gfx * and non-timewarp compute pipeline produces. */ const double tan_offset_x = ((tan_right + tan_left) - tan_width) / 2; const double tan_offset_y = (-(tan_up + tan_down) - tan_height) / 2; struct xrt_normalized_rect transform = { .x = (float)tan_offset_x, .y = (float)tan_offset_y, .w = (float)tan_width, .h = (float)tan_height, }; *out_rect = transform; } static XRT_CHECK_RESULT VkResult create_and_fill_in_distortion_buffer_for_view(struct vk_bundle *vk, struct xrt_device *xdev, struct render_buffer *r_buffer, struct render_buffer *g_buffer, struct render_buffer *b_buffer, uint32_t view, bool pre_rotate) { VkBufferUsageFlags usage_flags = VK_BUFFER_USAGE_TRANSFER_SRC_BIT; VkMemoryPropertyFlags properties = VK_MEMORY_PROPERTY_HOST_COHERENT_BIT | VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT; VkResult ret; struct xrt_matrix_2x2 rot = xdev->hmd->views[view].rot; const struct xrt_matrix_2x2 rotation_90_cw = {{ .vecs = { {0, 1}, {-1, 0}, }, }}; if (pre_rotate) { m_mat2x2_multiply(&rot, &rotation_90_cw, &rot); } VkDeviceSize size = sizeof(struct texture); ret = render_buffer_init(vk, r_buffer, usage_flags, properties, size); CG(vk, ret, "render_buffer_init", err_buffers); ret = render_buffer_init(vk, g_buffer, usage_flags, properties, size); CG(vk, ret, "render_buffer_init", err_buffers); ret = render_buffer_init(vk, b_buffer, usage_flags, properties, size); CG(vk, ret, "render_buffer_init", err_buffers); ret = render_buffer_map(vk, r_buffer); CG(vk, ret, "render_buffer_map", err_buffers); ret = render_buffer_map(vk, g_buffer); CG(vk, ret, "render_buffer_map", err_buffers); ret = render_buffer_map(vk, b_buffer); CG(vk, ret, "render_buffer_map", err_buffers); struct texture *r = r_buffer->mapped; struct texture *g = g_buffer->mapped; struct texture *b = b_buffer->mapped; const double dim_minus_one_f64 = COMP_DISTORTION_IMAGE_DIMENSIONS - 1; for (int row = 0; row < COMP_DISTORTION_IMAGE_DIMENSIONS; row++) { // This goes from 0 to 1.0 inclusive. float v = (float)(row / dim_minus_one_f64); for (int col = 0; col < COMP_DISTORTION_IMAGE_DIMENSIONS; col++) { // This goes from 0 to 1.0 inclusive. float u = (float)(col / dim_minus_one_f64); // These need to go from -0.5 to 0.5 for the rotation struct xrt_vec2 uv = {u - 0.5f, v - 0.5f}; m_mat2x2_transform_vec2(&rot, &uv, &uv); uv.x += 0.5f; uv.y += 0.5f; struct xrt_uv_triplet result; xrt_device_compute_distortion(xdev, view, uv.x, uv.y, &result); r->pixels[row][col] = result.r; g->pixels[row][col] = result.g; b->pixels[row][col] = result.b; } } render_buffer_unmap(vk, r_buffer); render_buffer_unmap(vk, g_buffer); render_buffer_unmap(vk, b_buffer); return VK_SUCCESS; err_buffers: render_buffer_close(vk, r_buffer); render_buffer_close(vk, g_buffer); render_buffer_close(vk, b_buffer); return ret; } static bool render_distortion_buffer_init(struct render_resources *r, struct vk_bundle *vk, struct xrt_device *xdev, bool pre_rotate) { struct render_buffer bufs[COMP_DISTORTION_NUM_IMAGES]; VkDeviceMemory device_memories[COMP_DISTORTION_NUM_IMAGES]; VkImage images[COMP_DISTORTION_NUM_IMAGES]; VkImageView image_views[COMP_DISTORTION_NUM_IMAGES]; VkCommandBuffer upload_buffer = VK_NULL_HANDLE; VkResult ret; /* * Basics */ static_assert(COMP_DISTORTION_NUM_IMAGES == 6, "Wrong number of distortion images!"); calc_uv_to_tanangle(xdev, 0, &r->distortion.uv_to_tanangle[0]); calc_uv_to_tanangle(xdev, 1, &r->distortion.uv_to_tanangle[1]); /* * Buffers with data to upload. */ ret = create_and_fill_in_distortion_buffer_for_view(vk, xdev, &bufs[0], &bufs[2], &bufs[4], 0, pre_rotate); CG(vk, ret, "create_and_fill_in_distortion_buffer_for_view", err_resources); ret = create_and_fill_in_distortion_buffer_for_view(vk, xdev, &bufs[1], &bufs[3], &bufs[5], 1, pre_rotate); CG(vk, ret, "create_and_fill_in_distortion_buffer_for_view", err_resources); /* * Command submission. */ struct vk_cmd_pool *pool = &r->distortion_pool; vk_cmd_pool_lock(pool); ret = vk_cmd_pool_create_and_begin_cmd_buffer_locked(vk, pool, 0, &upload_buffer); CG(vk, ret, "vk_cmd_pool_create_and_begin_cmd_buffer_locked", err_unlock); for (uint32_t i = 0; i < COMP_DISTORTION_NUM_IMAGES; i++) { ret = create_and_queue_upload_locked( // vk, // vk_bundle pool, // pool upload_buffer, // cmd bufs[i].buffer, // src_buffer &device_memories[i], // out_image_device_memory &images[i], // out_image &image_views[i]); // out_image_view CG(vk, ret, "create_and_queue_upload_locked", err_cmd); } ret = vk_cmd_pool_end_submit_wait_and_free_cmd_buffer_locked(vk, pool, upload_buffer); CG(vk, ret, "vk_cmd_pool_end_submit_wait_and_free_cmd_buffer_locked", err_cmd); vk_cmd_pool_unlock(pool); /* * Write results. */ r->distortion.pre_rotated = pre_rotate; for (uint32_t i = 0; i < COMP_DISTORTION_NUM_IMAGES; i++) { r->distortion.device_memories[i] = device_memories[i]; r->distortion.images[i] = images[i]; r->distortion.image_views[i] = image_views[i]; } /* * Tidy */ for (uint32_t i = 0; i < COMP_DISTORTION_NUM_IMAGES; i++) { render_buffer_close(vk, &bufs[i]); } return true; err_cmd: vk->vkFreeCommandBuffers(vk->device, pool->pool, 1, &upload_buffer); err_unlock: vk_cmd_pool_unlock(pool); err_resources: for (uint32_t i = 0; i < COMP_DISTORTION_NUM_IMAGES; i++) { D(ImageView, image_views[i]); D(Image, images[i]); DF(Memory, device_memories[i]); render_buffer_close(vk, &bufs[i]); } return false; } /* * * 'Exported' functions. * */ void render_distortion_images_close(struct render_resources *r) { struct vk_bundle *vk = r->vk; static_assert(COMP_DISTORTION_NUM_IMAGES == ARRAY_SIZE(r->distortion.image_views), "Array size is wrong!"); static_assert(COMP_DISTORTION_NUM_IMAGES == ARRAY_SIZE(r->distortion.images), "Array size is wrong!"); static_assert(COMP_DISTORTION_NUM_IMAGES == ARRAY_SIZE(r->distortion.device_memories), "Array size is wrong!"); for (uint32_t i = 0; i < COMP_DISTORTION_NUM_IMAGES; i++) { D(ImageView, r->distortion.image_views[i]); D(Image, r->distortion.images[i]); DF(Memory, r->distortion.device_memories[i]); } } bool render_distortion_images_ensure(struct render_resources *r, struct vk_bundle *vk, struct xrt_device *xdev, bool pre_rotate) { if (r->distortion.image_views[0] == VK_NULL_HANDLE || pre_rotate != r->distortion.pre_rotated) { render_distortion_images_close(r); return render_distortion_buffer_init(r, vk, xdev, pre_rotate); } return true; }