2023-04-02 11:05:52 +00:00
|
|
|
// Copyright 2019-2023, Collabora, Ltd.
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief Code for handling distortion resources (not shaders).
|
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
|
|
|
* @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;
|
|
|
|
|
2023-04-02 11:14:59 +00:00
|
|
|
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++) {
|
2023-04-02 11:05:52 +00:00
|
|
|
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;
|
|
|
|
}
|