c/main: Support compute queue in mirror code

This commit is contained in:
Jakob Bornecrantz 2023-05-15 19:12:14 +01:00
parent 44c81b6ddb
commit 023d5bf232
3 changed files with 450 additions and 63 deletions

View file

@ -18,6 +18,118 @@
*
*/
/*!
* If `COND` is not VK_SUCCESS returns false.
*/
#define C(c) \
do { \
VkResult ret = c; \
if (ret != VK_SUCCESS) { \
comp_mirror_fini(m, vk); \
return ret; \
} \
} 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; \
}
static VkResult
create_blit_descriptor_set_layout(struct vk_bundle *vk, VkDescriptorSetLayout *out_descriptor_set_layout)
{
VkResult ret;
VkDescriptorSetLayoutBinding set_layout_bindings[2] = {
{
.binding = 0,
.descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
.descriptorCount = 1,
.stageFlags = VK_SHADER_STAGE_COMPUTE_BIT,
},
{
.binding = 1,
.descriptorType = VK_DESCRIPTOR_TYPE_STORAGE_IMAGE,
.descriptorCount = 1,
.stageFlags = VK_SHADER_STAGE_COMPUTE_BIT,
},
};
VkDescriptorSetLayoutCreateInfo set_layout_info = {
.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
.bindingCount = ARRAY_SIZE(set_layout_bindings),
.pBindings = set_layout_bindings,
};
VkDescriptorSetLayout descriptor_set_layout = VK_NULL_HANDLE;
ret = vk->vkCreateDescriptorSetLayout( //
vk->device, //
&set_layout_info, //
NULL, //
&descriptor_set_layout); //
if (ret != VK_SUCCESS) {
VK_ERROR(vk, "vkCreateDescriptorSetLayout failed: %s", vk_result_string(ret));
return ret;
}
*out_descriptor_set_layout = descriptor_set_layout;
return VK_SUCCESS;
}
static VkResult
create_blit_pipeline_layout(struct vk_bundle *vk,
VkDescriptorSetLayout descriptor_set_layout,
VkPipelineLayout *out_pipeline_layout)
{
VkResult ret;
VkPipelineLayoutCreateInfo pipeline_layout_info = {
.sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO,
.flags = 0,
.setLayoutCount = 1,
.pSetLayouts = &descriptor_set_layout,
.pushConstantRangeCount = 1,
.pPushConstantRanges =
&(VkPushConstantRange){
.stageFlags = VK_SHADER_STAGE_COMPUTE_BIT,
.offset = 0,
.size = sizeof(struct render_compute_blit_push_data),
},
};
VkPipelineLayout pipeline_layout = VK_NULL_HANDLE;
ret = vk->vkCreatePipelineLayout( //
vk->device, // device
&pipeline_layout_info, // pCreateInfo
NULL, // pAllocator
&pipeline_layout); // pPipelineLayout
if (ret != VK_SUCCESS) {
VK_ERROR(vk, "vkCreatePipelineLayout failed: %s", vk_result_string(ret));
return ret;
}
*out_pipeline_layout = pipeline_layout;
return VK_SUCCESS;
}
static bool
ensure_scratch(struct comp_mirror_to_debug_gui *m, struct vk_bundle *vk)
{
@ -27,25 +139,115 @@ ensure_scratch(struct comp_mirror_to_debug_gui *m, struct vk_bundle *vk)
return true;
}
VkFormat format = VK_FORMAT_R8G8B8A8_SRGB;
VkImageUsageFlags usage = VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT;
VkExtent2D extent = m->image_extent;
const VkFormat unorm_format = VK_FORMAT_R8G8B8A8_UNORM;
const VkImageViewType view_type = VK_IMAGE_VIEW_TYPE_2D;
const VkExtent2D extent = m->image_extent;
ret = vk_create_image_simple( //
// Both usages are common.
const VkImageUsageFlags unorm_usage = VK_IMAGE_USAGE_STORAGE_BIT | VK_IMAGE_USAGE_SAMPLED_BIT;
// Very few cards support SRGB storage.
const VkImageUsageFlags srgb_usage = VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT;
// Combination of both.
const VkImageUsageFlags image_usage = unorm_usage | srgb_usage;
ret = vk_create_image_mutable_rgba( //
vk, // vk_bundle
extent, // extent
format, // format
usage, // usage
&m->bounce.mem, // out_mem
image_usage, // usage
&m->bounce.mem, // out_device_memory
&m->bounce.image); // out_image
if (ret != VK_SUCCESS) {
VK_ERROR(vk, "vk_create_image_simple: %s", vk_result_string(ret));
VK_ERROR(vk, "vk_create_image_mutable_rgba: %s", vk_result_string(ret));
return false;
}
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_usage( //
vk, // vk_bundle
m->bounce.image, // image
view_type, // type
unorm_format, // format
unorm_usage, // image_usage
subresource_range, // subresource_range
&m->bounce.unorm_view); // out_image_view
if (ret != VK_SUCCESS) {
VK_ERROR(vk, "vk_create_view_usage: %s", vk_result_string(ret));
return false;
}
return true;
}
static void
update_blit_descriptor_set(struct vk_bundle *vk,
VkSampler src_sampler,
VkImageView src_view,
VkImageView target_view,
VkDescriptorSet descriptor_set)
{
VkDescriptorImageInfo src_image_info = {
.sampler = src_sampler,
.imageView = src_view,
.imageLayout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL,
};
VkDescriptorImageInfo target_image_info = {
.imageView = target_view,
.imageLayout = VK_IMAGE_LAYOUT_GENERAL,
};
VkWriteDescriptorSet write_descriptor_sets[2] = {
{
.sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
.dstSet = descriptor_set,
.dstBinding = 0,
.descriptorCount = 1,
.descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
.pImageInfo = &src_image_info,
},
{
.sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
.dstSet = descriptor_set,
.dstBinding = 1,
.descriptorCount = 1,
.descriptorType = VK_DESCRIPTOR_TYPE_STORAGE_IMAGE,
.pImageInfo = &target_image_info,
},
};
vk->vkUpdateDescriptorSets( //
vk->device, //
ARRAY_SIZE(write_descriptor_sets), // descriptorWriteCount
write_descriptor_sets, // pDescriptorWrites
0, // descriptorCopyCount
NULL);
}
/*
* For dispatching compute to the blit target, calculate the number of groups.
*/
static void
calc_dispatch_dims(const VkExtent2D extent, uint32_t *out_w, uint32_t *out_h)
{
// Power of two divide and round up.
#define P2_DIVIDE_ROUND_UP(v, div) ((v + (div - 1)) / div)
uint32_t w = P2_DIVIDE_ROUND_UP(extent.width, 8);
uint32_t h = P2_DIVIDE_ROUND_UP(extent.height, 8);
#undef P2_DIVIDE_ROUND_UP
*out_w = w;
*out_h = h;
}
/*
*
@ -54,7 +256,10 @@ ensure_scratch(struct comp_mirror_to_debug_gui *m, struct vk_bundle *vk)
*/
VkResult
comp_mirror_init(struct comp_mirror_to_debug_gui *m, struct vk_bundle *vk, VkExtent2D extent)
comp_mirror_init(struct comp_mirror_to_debug_gui *m,
struct vk_bundle *vk,
struct render_shaders *shaders,
VkExtent2D extent)
{
VkResult ret;
@ -83,14 +288,47 @@ comp_mirror_init(struct comp_mirror_to_debug_gui *m, struct vk_bundle *vk, VkExt
m->image_extent, // extent
&m->pool, // out_pool
XRT_FORMAT_R8G8B8X8, // xrt_format
VK_FORMAT_R8G8B8A8_SRGB); // vk_format
VK_FORMAT_R8G8B8A8_UNORM); // vk_format
ret = vk_cmd_pool_init(vk, &m->cmd_pool, VK_COMMAND_POOL_CREATE_TRANSIENT_BIT);
if (ret != VK_SUCCESS) {
VK_ERROR(vk, "vk_cmd_pool_init: %s", vk_result_string(ret));
comp_mirror_fini(m, vk);
return ret;
}
struct vk_descriptor_pool_info blit_pool_info = {
.uniform_per_descriptor_count = 0,
.sampler_per_descriptor_count = 1,
.storage_image_per_descriptor_count = 1,
.storage_buffer_per_descriptor_count = 0,
.descriptor_count = 1,
.freeable = false,
};
C(vk_create_descriptor_pool( //
vk, // vk_bundle
&blit_pool_info, // info
&m->blit.descriptor_pool)); // out_descriptor_pool
C(vk_create_pipeline_cache(vk, &m->blit.pipeline_cache));
C(create_blit_descriptor_set_layout(vk, &m->blit.descriptor_set_layout));
C(create_blit_pipeline_layout( //
vk, // vk_bundle
m->blit.descriptor_set_layout, // descriptor_set_layout
&m->blit.pipeline_layout)); // out_pipeline_layout
C(vk_create_compute_pipeline( //
vk, // vk_bundle
m->blit.pipeline_cache, // pipeline_cache
shaders->blit_comp, // shader
m->blit.pipeline_layout, // pipeline_layout
NULL, // specialization_info
&m->blit.pipeline)); // out_compute_pipeline
return VK_SUCCESS;
}
@ -163,7 +401,10 @@ comp_mirror_do_blit(struct comp_mirror_to_debug_gui *m,
struct vk_bundle *vk,
uint64_t predicted_display_time_ns,
VkImage from_image,
VkExtent2D from_extent)
VkImageView from_view,
VkSampler from_sampler,
VkExtent2D from_extent,
struct xrt_normalized_rect from_rect)
{
VkResult ret;
@ -177,6 +418,17 @@ comp_mirror_do_blit(struct comp_mirror_to_debug_gui *m,
return;
}
VkDescriptorSet descriptor_set = VK_NULL_HANDLE;
ret = vk_create_descriptor_set( //
vk, //
m->blit.descriptor_pool, // descriptor_pool
m->blit.descriptor_set_layout, // descriptor_set_layout
&descriptor_set); // descriptor_set
if (ret != VK_SUCCESS) {
VK_ERROR(vk, "vk_create_descriptor_set: %s", vk_result_string(ret));
return;
}
struct vk_cmd_pool *pool = &m->cmd_pool;
// For writing and submitting commands.
@ -185,15 +437,18 @@ comp_mirror_do_blit(struct comp_mirror_to_debug_gui *m,
VkCommandBuffer cmd;
ret = vk_cmd_pool_create_and_begin_cmd_buffer_locked(vk, pool, 0, &cmd);
if (ret != VK_SUCCESS) {
vk->vkResetDescriptorPool(vk->device, m->blit.descriptor_pool, 0);
vk_cmd_pool_unlock(pool);
return;
}
// First mip view into the copy from image.
struct vk_cmd_first_mip_image copy_from_fm_image = {
.aspect_mask = VK_IMAGE_ASPECT_COLOR_BIT,
.base_array_layer = 0,
.image = from_image,
// Barrier arguments.
VkImageSubresourceRange first_color_level_subresource_range = {
.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
.baseMipLevel = 0,
.levelCount = 1,
.baseArrayLayer = 0,
.layerCount = 1,
};
// First mip view into the bounce image.
@ -210,6 +465,20 @@ comp_mirror_do_blit(struct comp_mirror_to_debug_gui *m,
.image = wrap->image,
};
VkImageLayout old_layout;
VkPipelineStageFlags src_stage_mask;
VkAccessFlags src_access_mask;
#if 0
// First mip view into the copy from image.
struct vk_cmd_first_mip_image copy_from_fm_image = {
.aspect_mask = VK_IMAGE_ASPECT_COLOR_BIT,
.base_array_layer = 0,
.image = from_image,
};
// Not completely removed just in case we need to go back on some platforms.
if (!use_compute_blit) {
// Blit arguments.
struct vk_cmd_blit_image_info blit_info = {
.src.old_layout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL,
@ -231,11 +500,80 @@ comp_mirror_do_blit(struct comp_mirror_to_debug_gui *m,
vk_cmd_blit_image_locked(vk, cmd, &blit_info);
old_layout = VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL;
src_access_mask = VK_ACCESS_TRANSFER_WRITE_BIT;
src_stage_mask = VK_PIPELINE_STAGE_TRANSFER_BIT;
} else
#endif
{
// Barrier bounce image so it can be safely written to.
vk_cmd_image_barrier_locked( //
vk, // vk_bundle
cmd, // cmdbuffer
bounce_fm_image.image, // image
0, // srcAccessMask
VK_ACCESS_SHADER_WRITE_BIT, // dstAccessMask
VK_IMAGE_LAYOUT_UNDEFINED, // oldImageLayout
VK_IMAGE_LAYOUT_GENERAL, // newImageLayout
VK_PIPELINE_STAGE_TRANSFER_BIT, // srcStageMask
VK_PIPELINE_STAGE_COMPUTE_SHADER_BIT, // dstStageMask
first_color_level_subresource_range); // subresourceRange
update_blit_descriptor_set( //
vk, //
from_sampler, //
from_view, //
m->bounce.unorm_view, //
descriptor_set); //
vk->vkCmdBindPipeline( //
cmd, // commandBuffer
VK_PIPELINE_BIND_POINT_COMPUTE, // pipelineBindPoint
m->blit.pipeline); // pipeline
vk->vkCmdBindDescriptorSets( //
cmd, // commandBuffer
VK_PIPELINE_BIND_POINT_COMPUTE, // pipelineBindPoint
m->blit.pipeline_layout, // layout
0, // firstSet
1, // descriptorSetCount
&descriptor_set, // pDescriptorSets
0, // dynamicOffsetCount
NULL); // pDynamicOffsets
struct render_compute_blit_push_data constants = {
from_rect,
{{0, 0}, {m->image_extent.width, m->image_extent.height}},
};
vk->vkCmdPushConstants( //
cmd, //
m->blit.pipeline_layout, //
VK_SHADER_STAGE_COMPUTE_BIT, //
0, //
sizeof(constants), //
&constants); //
uint32_t w = 0, h = 0;
calc_dispatch_dims(m->image_extent, &w, &h);
assert(w != 0 && h != 0);
vk->vkCmdDispatch( //
cmd, // commandBuffer
w, // groupCountX
h, // groupCountY
1); // groupCountZ
old_layout = VK_IMAGE_LAYOUT_GENERAL;
src_access_mask = VK_ACCESS_SHADER_WRITE_BIT;
src_stage_mask = VK_PIPELINE_STAGE_COMPUTE_SHADER_BIT;
}
// Copy arguments.
struct vk_cmd_copy_image_info copy_info = {
.src.old_layout = VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
.src.src_access_mask = VK_ACCESS_TRANSFER_WRITE_BIT,
.src.src_stage_mask = VK_PIPELINE_STAGE_TRANSFER_BIT,
.src.old_layout = old_layout,
.src.src_access_mask = src_access_mask,
.src.src_stage_mask = src_stage_mask,
.src.fm_image = bounce_fm_image,
.dst.old_layout = wrap->layout,
@ -249,15 +587,6 @@ comp_mirror_do_blit(struct comp_mirror_to_debug_gui *m,
vk_cmd_copy_image_locked(vk, cmd, &copy_info);
// Barrier arguments.
VkImageSubresourceRange first_color_level_subresource_range = {
.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
.baseMipLevel = 0,
.levelCount = 1,
.baseArrayLayer = 0,
.layerCount = 1,
};
// Barrier readback image to host so we can safely read
vk_cmd_image_barrier_locked( //
vk, // vk_bundle
@ -294,6 +623,9 @@ comp_mirror_do_blit(struct comp_mirror_to_debug_gui *m,
u_frame_times_widget_push_sample(&m->push_frame_times, predicted_display_time_ns);
xrt_frame_reference(&frame, NULL);
// Tidies the descriptor we created.
vk->vkResetDescriptorPool(vk->device, m->blit.descriptor_pool, 0);
}
void
@ -305,14 +637,21 @@ comp_mirror_fini(struct comp_mirror_to_debug_gui *m, struct vk_bundle *vk)
// Left eye readback
vk_image_readback_to_xf_pool_destroy(vk, &m->pool);
if (m->bounce.image != VK_NULL_HANDLE) {
vk->vkFreeMemory(vk->device, m->bounce.mem, NULL);
vk->vkDestroyImage(vk->device, m->bounce.image, NULL);
}
// Bounce image resources.
D(ImageView, m->bounce.unorm_view);
D(Image, m->bounce.image);
DF(Memory, m->bounce.mem);
// Command pool for readback code.
vk_cmd_pool_destroy(vk, &m->cmd_pool);
// Destroy blit shader Vulkan resources.
D(Pipeline, m->blit.pipeline);
D(PipelineLayout, m->blit.pipeline_layout);
D(PipelineCache, m->blit.pipeline_cache);
D(DescriptorPool, m->blit.descriptor_pool);
D(DescriptorSetLayout, m->blit.descriptor_set_layout);
// The frame timing widigt.
u_frame_times_widget_teardown(&m->push_frame_times);

View file

@ -48,9 +48,28 @@ struct comp_mirror_to_debug_gui
struct
{
VkImage image;
VkImageView unorm_view;
VkDeviceMemory mem;
} bounce;
struct
{
//! Private here for now.
VkPipelineCache pipeline_cache;
//! Descriptor pool for blit.
VkDescriptorPool descriptor_pool;
//! Descriptor set layout for compute.
VkDescriptorSetLayout descriptor_set_layout;
//! Pipeline layout used for compute distortion.
VkPipelineLayout pipeline_layout;
//! Doesn't depend on target so is static.
VkPipeline pipeline;
} blit;
struct vk_cmd_pool cmd_pool;
};
@ -60,7 +79,10 @@ struct comp_mirror_to_debug_gui
* @public @memberof comp_mirror_to_debug_gui
*/
VkResult
comp_mirror_init(struct comp_mirror_to_debug_gui *m, struct vk_bundle *vk, VkExtent2D extent);
comp_mirror_init(struct comp_mirror_to_debug_gui *m,
struct vk_bundle *vk,
struct render_shaders *shaders,
VkExtent2D extent);
/*!
* One time adding of the debug variables.
@ -99,7 +121,10 @@ comp_mirror_do_blit(struct comp_mirror_to_debug_gui *m,
struct vk_bundle *vk,
uint64_t predicted_display_time_ns,
VkImage from_image,
VkExtent2D from_extent);
VkImageView from_view,
VkSampler from_sampler,
VkExtent2D from_extent,
struct xrt_normalized_rect from_rect);
/*!
* Finalise the struct, frees and resources.

View file

@ -548,7 +548,7 @@ renderer_init(struct comp_renderer *r, struct comp_compositor *c)
struct vk_bundle *vk = &r->c->base.vk;
VkResult ret = comp_mirror_init(&r->mirror_to_debug_gui, vk, r->lr->extent);
VkResult ret = comp_mirror_init(&r->mirror_to_debug_gui, vk, &c->shaders, r->lr->extent);
if (ret != VK_SUCCESS) {
COMP_ERROR(c, "comp_mirror_init: %s", vk_result_string(ret));
assert(false && "Whelp, can't return a error. But should never really fail.");
@ -1441,7 +1441,9 @@ dispatch_compute(struct comp_renderer *r, struct render_compute *crc)
VkImageView target_image_view = r->c->target->images[r->acquired_buffer].view;
uint32_t layer_count = c->base.slot.layer_count;
if (layer_count == 1 && c->base.slot.layers[0].data.type == XRT_LAYER_STEREO_PROJECTION) {
bool fast_path = c->base.slot.one_projection_layer_fast_path;
if (fast_path && c->base.slot.layers[0].data.type == XRT_LAYER_STEREO_PROJECTION) {
int i = 0;
const struct comp_layer *layer = &c->base.slot.layers[i];
const struct xrt_layer_stereo_projection_data *stereo = &layer->data.stereo;
@ -1449,7 +1451,7 @@ dispatch_compute(struct comp_renderer *r, struct render_compute *crc)
const struct xrt_layer_projection_view_data *rvd = &stereo->r;
do_projection_layers(r, crc, layer, lvd, rvd);
} else if (layer_count == 1 && c->base.slot.layers[0].data.type == XRT_LAYER_STEREO_PROJECTION_DEPTH) {
} else if (fast_path && c->base.slot.layers[0].data.type == XRT_LAYER_STEREO_PROJECTION_DEPTH) {
int i = 0;
const struct comp_layer *layer = &c->base.slot.layers[i];
const struct xrt_layer_stereo_projection_depth_data *stereo = &layer->data.stereo_depth;
@ -1772,12 +1774,33 @@ comp_renderer_draw(struct comp_renderer *r)
comp_mirror_fixup_ui_state(&r->mirror_to_debug_gui, c);
if (comp_mirror_is_ready_and_active(&r->mirror_to_debug_gui, c, predicted_display_time_ns)) {
if (use_compute) {
// Covers only the first half of the view.
struct xrt_normalized_rect rect = {0, 0, 0.5f, 1.0f};
comp_mirror_do_blit( //
&r->mirror_to_debug_gui, //
&c->base.vk, //
predicted_display_time_ns, //
c->nr.scratch.color.image, //
c->nr.scratch.color.srgb_view, //
c->nr.compute.default_sampler, //
c->nr.scratch.extent, //
rect); //
} else {
// Covers the whole view.
struct xrt_normalized_rect rect = {0, 0, 1.0f, 1.0f};
comp_mirror_do_blit( //
&r->mirror_to_debug_gui, //
&c->base.vk, //
predicted_display_time_ns, //
r->lr->framebuffers[0].image, //
r->lr->extent); //
r->lr->framebuffers[0].view, //
r->lr->framebuffers[0].sampler, //
r->lr->extent, //
rect); //
}
}
/*