monado/src/xrt/compositor/main/comp_distortion.c

1000 lines
31 KiB
C
Raw Normal View History

2019-03-18 05:52:32 +00:00
// Copyright 2019, Collabora, Ltd.
// SPDX-License-Identifier: BSL-1.0
/*!
* @file
* @brief Distortion shader code.
* @author Lubosz Sarnecki <lubosz.sarnecki@collabora.com>
* @author Jakob Bornecrantz <jakob@collabora.com>
* @ingroup comp
*/
#include <stdio.h>
#include <stdlib.h>
2019-03-18 05:52:32 +00:00
#include <string.h>
2019-10-07 23:12:06 +00:00
#include <assert.h>
2019-03-18 05:52:32 +00:00
#include "main/comp_settings.h"
#include "main/comp_compositor.h"
#include "comp_distortion.h"
#pragma GCC diagnostic push
#pragma GCC diagnostic ignored "-Wpragmas"
#pragma GCC diagnostic ignored "-Wnewline-eof"
#include "shaders/distortion.vert.h"
2019-10-07 23:12:06 +00:00
#include "shaders/mesh.frag.h"
#include "shaders/mesh.vert.h"
2019-03-18 05:52:32 +00:00
#include "shaders/none.frag.h"
#include "shaders/panotools.frag.h"
#include "shaders/vive.frag.h"
2019-10-07 23:12:06 +00:00
2019-03-18 05:52:32 +00:00
#pragma GCC diagnostic pop
/*
*
* Pre declare functions.
*
*/
static void
comp_distortion_update_uniform_buffer_warp(struct comp_distortion *d,
struct comp_compositor *c);
static void
2019-10-07 23:12:06 +00:00
comp_distortion_init_buffers(struct comp_distortion *d,
struct comp_compositor *c);
2019-03-18 05:52:32 +00:00
XRT_MAYBE_UNUSED static void
comp_distortion_update_descriptor_sets(struct comp_distortion *d,
VkSampler samplers[2],
VkImageView views[2]);
static void
comp_distortion_init_descriptor_set_layout(struct comp_distortion *d);
static void
comp_distortion_init_pipeline_layout(struct comp_distortion *d);
static void
comp_distortion_init_pipeline(struct comp_distortion *d,
VkRenderPass render_pass,
VkPipelineCache pipeline_cache);
2019-03-18 05:52:32 +00:00
static VkWriteDescriptorSet
comp_distortion_get_uniform_write_descriptor_set(struct comp_distortion *d,
uint32_t binding,
uint32_t eye);
static VkWriteDescriptorSet
comp_distortion_get_uniform_write_descriptor_set_vp(struct comp_distortion *d,
uint32_t binding,
uint32_t eye);
static VkWriteDescriptorSet
comp_distortion_get_image_write_descriptor_set(
2019-08-16 21:59:06 +00:00
VkDescriptorSet descriptor_set,
VkDescriptorImageInfo *descriptor_position,
2019-03-18 05:52:32 +00:00
uint32_t binding);
static void
comp_distortion_init_descriptor_sets(struct comp_distortion *d,
2019-08-16 21:59:06 +00:00
VkDescriptorPool descriptor_pool);
2019-03-18 05:52:32 +00:00
/*
*
* Buffer functions.
*
*/
static void
_buffer_destroy(struct vk_bundle *vk, struct comp_uniform_buffer *buffer)
{
if (buffer->buffer != VK_NULL_HANDLE) {
vk->vkDestroyBuffer(buffer->device, buffer->buffer, NULL);
}
if (buffer->memory != VK_NULL_HANDLE) {
vk->vkFreeMemory(buffer->device, buffer->memory, NULL);
}
2019-03-18 05:52:32 +00:00
}
static VkResult
_buffer_map(struct vk_bundle *vk,
struct comp_uniform_buffer *buffer,
VkDeviceSize size,
VkDeviceSize offset)
{
return vk->vkMapMemory(vk->device, buffer->memory, offset, size, 0,
&buffer->mapped);
}
static void
_buffer_unmap(struct vk_bundle *vk, struct comp_uniform_buffer *buffer)
{
if (buffer->mapped) {
vk->vkUnmapMemory(vk->device, buffer->memory);
buffer->mapped = NULL;
}
}
static void
_buffer_setup_descriptor(struct vk_bundle *vk,
struct comp_uniform_buffer *buffer,
VkDeviceSize size,
VkDeviceSize offset)
{
buffer->descriptor.offset = offset;
buffer->descriptor.buffer = buffer->buffer;
buffer->descriptor.range = size;
}
/*
*
* Shader functions.
*
*/
static VkPipelineShaderStageCreateInfo
_shader_load(struct vk_bundle *vk,
const uint32_t *code,
size_t size,
VkShaderStageFlagBits flags)
{
VkResult ret;
VkShaderModuleCreateInfo info = {
.sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO,
.pNext = NULL,
.flags = 0,
2019-03-18 05:52:32 +00:00
.codeSize = size,
.pCode = code,
};
VkShaderModule module;
ret = vk->vkCreateShaderModule(vk->device, &info, NULL, &module);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "vkCreateShaderModule failed %u", ret);
}
return (VkPipelineShaderStageCreateInfo){
.sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO,
.pNext = NULL,
.flags = 0,
2019-03-18 05:52:32 +00:00
.stage = flags,
.module = module,
.pName = "main",
.pSpecializationInfo = NULL,
2019-03-18 05:52:32 +00:00
};
}
/*
*
* Functions.
*
*/
void
comp_distortion_init(struct comp_distortion *d,
struct comp_compositor *c,
VkRenderPass render_pass,
VkPipelineCache pipeline_cache,
enum xrt_distortion_model distortion_model,
2019-10-07 23:12:06 +00:00
struct xrt_hmd_parts *parts,
2019-03-18 05:52:32 +00:00
VkDescriptorPool descriptor_pool,
bool flip_y)
{
d->vk = &c->vk;
d->distortion_model = distortion_model;
//! Add support for 1 channels as well.
2019-10-23 21:00:22 +00:00
assert(parts->distortion.mesh.vertices == NULL ||
parts->distortion.mesh.num_uv_channels == 3);
2019-10-23 21:00:22 +00:00
assert(parts->distortion.mesh.indices == NULL ||
parts->distortion.mesh.total_num_indices != 0);
assert(parts->distortion.mesh.indices == NULL ||
parts->distortion.mesh.num_indices[0] != 0);
assert(parts->distortion.mesh.indices == NULL ||
parts->distortion.mesh.num_indices[1] != 0);
d->mesh.vertices = parts->distortion.mesh.vertices;
d->mesh.stride = parts->distortion.mesh.stride;
d->mesh.num_vertices = parts->distortion.mesh.num_vertices;
d->mesh.indices = parts->distortion.mesh.indices;
d->mesh.total_num_indices = parts->distortion.mesh.total_num_indices;
d->mesh.num_indices[0] = parts->distortion.mesh.num_indices[0];
d->mesh.num_indices[1] = parts->distortion.mesh.num_indices[1];
d->mesh.offset_indices[0] = parts->distortion.mesh.offset_indices[0];
d->mesh.offset_indices[1] = parts->distortion.mesh.offset_indices[1];
2019-10-07 23:12:06 +00:00
2019-03-18 05:52:32 +00:00
d->ubo_vp_data[0].flip_y = flip_y;
d->ubo_vp_data[1].flip_y = flip_y;
2019-10-07 23:12:06 +00:00
comp_distortion_init_buffers(d, c);
2019-03-18 05:52:32 +00:00
comp_distortion_update_uniform_buffer_warp(d, c);
comp_distortion_init_descriptor_set_layout(d);
comp_distortion_init_pipeline_layout(d);
comp_distortion_init_pipeline(d, render_pass, pipeline_cache);
2019-03-18 05:52:32 +00:00
comp_distortion_init_descriptor_sets(d, descriptor_pool);
}
void
comp_distortion_destroy(struct comp_distortion *d)
{
struct vk_bundle *vk = d->vk;
vk->vkDestroyDescriptorSetLayout(vk->device, d->descriptor_set_layout,
NULL);
_buffer_destroy(vk, &d->ubo_handle);
2019-10-07 23:12:06 +00:00
_buffer_destroy(vk, &d->vbo_handle);
2019-10-23 21:00:22 +00:00
_buffer_destroy(vk, &d->index_handle);
2019-03-18 05:52:32 +00:00
_buffer_destroy(vk, &d->ubo_viewport_handles[0]);
_buffer_destroy(vk, &d->ubo_viewport_handles[1]);
vk->vkDestroyPipeline(vk->device, d->pipeline, NULL);
vk->vkDestroyPipelineLayout(vk->device, d->pipeline_layout, NULL);
free(d);
2019-03-18 05:52:32 +00:00
}
static void
comp_distortion_init_pipeline(struct comp_distortion *d,
VkRenderPass render_pass,
VkPipelineCache pipeline_cache)
2019-03-18 05:52:32 +00:00
{
struct vk_bundle *vk = d->vk;
VkResult ret;
2019-10-23 21:00:22 +00:00
VkPolygonMode polygonMode = VK_POLYGON_MODE_FILL;
if (d->quirk_draw_lines) {
polygonMode = VK_POLYGON_MODE_LINE;
}
VkPrimitiveTopology topology = VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST;
if (d->mesh.total_num_indices > 0) {
topology = VK_PRIMITIVE_TOPOLOGY_TRIANGLE_STRIP;
}
2019-03-18 05:52:32 +00:00
VkPipelineInputAssemblyStateCreateInfo input_assembly_state = {
.sType =
VK_STRUCTURE_TYPE_PIPELINE_INPUT_ASSEMBLY_STATE_CREATE_INFO,
.pNext = NULL,
.flags = 0,
2019-10-23 21:00:22 +00:00
.topology = topology,
2019-03-18 05:52:32 +00:00
.primitiveRestartEnable = VK_FALSE,
};
VkPipelineRasterizationStateCreateInfo rasterization_state = {
.sType = VK_STRUCTURE_TYPE_PIPELINE_RASTERIZATION_STATE_CREATE_INFO,
.pNext = NULL,
.flags = 0,
2019-03-18 05:52:32 +00:00
.depthClampEnable = VK_FALSE,
.rasterizerDiscardEnable = VK_FALSE,
2019-10-23 21:00:22 +00:00
.polygonMode = polygonMode,
.cullMode = VK_CULL_MODE_BACK_BIT,
2019-03-18 05:52:32 +00:00
.frontFace = VK_FRONT_FACE_CLOCKWISE,
.depthBiasEnable = VK_FALSE,
.depthBiasConstantFactor = 0.f,
.depthBiasClamp = 0.f,
.depthBiasSlopeFactor = 0.f,
2019-03-18 05:52:32 +00:00
.lineWidth = 1.0f,
};
VkPipelineColorBlendAttachmentState blend_attachment_state = {0};
blend_attachment_state.blendEnable = VK_FALSE;
blend_attachment_state.colorWriteMask = 0xf;
2019-03-18 05:52:32 +00:00
VkPipelineColorBlendStateCreateInfo color_blend_state = {
.sType = VK_STRUCTURE_TYPE_PIPELINE_COLOR_BLEND_STATE_CREATE_INFO,
.pNext = NULL,
.flags = 0,
.logicOpEnable = VK_FALSE,
.logicOp = VK_LOGIC_OP_CLEAR,
2019-03-18 05:52:32 +00:00
.attachmentCount = 1,
.pAttachments = &blend_attachment_state,
//! @todo what's the right value for blendConstants?
.blendConstants = {1.f, 1.f, 1.f, 1.f},
2019-03-18 05:52:32 +00:00
};
VkPipelineDepthStencilStateCreateInfo depth_stencil_state = {
.sType = VK_STRUCTURE_TYPE_PIPELINE_DEPTH_STENCIL_STATE_CREATE_INFO,
.pNext = NULL,
.flags = 0,
2019-03-18 05:52:32 +00:00
.depthTestEnable = VK_TRUE,
.depthWriteEnable = VK_TRUE,
.depthCompareOp = VK_COMPARE_OP_LESS_OR_EQUAL,
.depthBoundsTestEnable = VK_FALSE,
.stencilTestEnable = VK_FALSE,
2019-03-18 05:52:32 +00:00
.front =
{
.failOp = VK_STENCIL_OP_KEEP,
.passOp = VK_STENCIL_OP_KEEP,
.depthFailOp = VK_STENCIL_OP_KEEP,
.compareOp = VK_COMPARE_OP_ALWAYS, // this is the only
// meaningful value here.
.compareMask = 0,
.writeMask = 0,
.reference = 0,
2019-03-18 05:52:32 +00:00
},
.back =
{
.failOp = VK_STENCIL_OP_KEEP,
.passOp = VK_STENCIL_OP_KEEP,
.depthFailOp = VK_STENCIL_OP_KEEP,
.compareOp = VK_COMPARE_OP_ALWAYS, // this is the only
// meaningful value here.
.compareMask = 0,
.writeMask = 0,
.reference = 0,
2019-03-18 05:52:32 +00:00
},
.minDepthBounds = 0.f,
.maxDepthBounds = 0.f,
2019-03-18 05:52:32 +00:00
};
VkPipelineViewportStateCreateInfo viewport_state = {
.sType = VK_STRUCTURE_TYPE_PIPELINE_VIEWPORT_STATE_CREATE_INFO,
.pNext = NULL,
.flags = 0,
2019-03-18 05:52:32 +00:00
.viewportCount = 1,
.pViewports = NULL, // assigned valid value later
2019-03-18 05:52:32 +00:00
.scissorCount = 1,
.pScissors = NULL, // assigned valid value later
2019-03-18 05:52:32 +00:00
};
VkPipelineMultisampleStateCreateInfo multisample_state = {
.sType = VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO,
.pNext = NULL,
.flags = 0,
.rasterizationSamples = VK_SAMPLE_COUNT_1_BIT,
.sampleShadingEnable = VK_FALSE,
.minSampleShading = 0,
.pSampleMask = NULL,
.alphaToCoverageEnable = VK_FALSE,
.alphaToOneEnable = VK_FALSE,
};
2019-03-18 05:52:32 +00:00
VkDynamicState dynamic_states[] = {
VK_DYNAMIC_STATE_VIEWPORT,
VK_DYNAMIC_STATE_SCISSOR,
};
VkPipelineDynamicStateCreateInfo dynamic_state = {
.sType = VK_STRUCTURE_TYPE_PIPELINE_DYNAMIC_STATE_CREATE_INFO,
.pNext = NULL,
.flags = 0,
2019-03-18 05:52:32 +00:00
.dynamicStateCount = 2,
.pDynamicStates = dynamic_states,
};
2019-10-07 23:12:06 +00:00
VkVertexInputBindingDescription vertex_input_binding_description;
VkVertexInputAttributeDescription
vertex_input_attribute_descriptions[2];
2019-03-18 05:52:32 +00:00
const uint32_t *fragment_shader_code;
size_t fragment_shader_size;
2019-10-07 23:12:06 +00:00
/*
* By default, we will generate positions and UVs for the full screen
* quad from the gl_VertexIndex
*/
VkPipelineVertexInputStateCreateInfo vertex_input_state = {
.sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO,
.pNext = NULL,
.flags = 0,
.vertexBindingDescriptionCount = 0,
.pVertexBindingDescriptions = NULL,
.vertexAttributeDescriptionCount = 0,
.pVertexAttributeDescriptions = NULL,
};
const uint32_t *vertex_shader_code = shaders_distortion_vert;
size_t vertex_shader_size = sizeof(shaders_distortion_vert);
switch (d->distortion_model) {
2019-03-18 05:52:32 +00:00
case XRT_DISTORTION_MODEL_NONE:
fragment_shader_code = shaders_none_frag;
fragment_shader_size = sizeof(shaders_none_frag);
break;
case XRT_DISTORTION_MODEL_PANOTOOLS:
fragment_shader_code = shaders_panotools_frag;
fragment_shader_size = sizeof(shaders_panotools_frag);
break;
case XRT_DISTORTION_MODEL_VIVE:
fragment_shader_code = shaders_vive_frag;
fragment_shader_size = sizeof(shaders_vive_frag);
break;
2019-10-07 23:12:06 +00:00
case XRT_DISTORTION_MODEL_MESHUV:
// clang-format off
2019-10-07 23:12:06 +00:00
vertex_input_attribute_descriptions[0].binding = 0;
vertex_input_attribute_descriptions[0].location = 0;
vertex_input_attribute_descriptions[0].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2019-10-07 23:12:06 +00:00
vertex_input_attribute_descriptions[0].offset = 0;
vertex_input_attribute_descriptions[1].binding = 0;
vertex_input_attribute_descriptions[1].location = 1;
vertex_input_attribute_descriptions[1].format = VK_FORMAT_R32G32B32A32_SFLOAT;
vertex_input_attribute_descriptions[1].offset = 16;
2019-10-07 23:12:06 +00:00
vertex_input_binding_description.binding = 0;
vertex_input_binding_description.inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2019-10-23 21:00:22 +00:00
vertex_input_binding_description.stride = d->mesh.stride;
2019-10-07 23:12:06 +00:00
vertex_input_state.vertexAttributeDescriptionCount = 2;
vertex_input_state.pVertexAttributeDescriptions = vertex_input_attribute_descriptions;
2019-10-07 23:12:06 +00:00
vertex_input_state.vertexBindingDescriptionCount = 1;
vertex_input_state.pVertexBindingDescriptions = &vertex_input_binding_description;
// clang-format on
2019-10-07 23:12:06 +00:00
vertex_shader_code = shaders_mesh_vert;
vertex_shader_size = sizeof(shaders_mesh_vert);
fragment_shader_code = shaders_mesh_frag;
fragment_shader_size = sizeof(shaders_mesh_frag);
break;
2019-03-18 05:52:32 +00:00
default:
fragment_shader_code = shaders_panotools_frag;
fragment_shader_size = sizeof(shaders_panotools_frag);
break;
}
VkPipelineShaderStageCreateInfo shader_stages[2] = {
2019-10-07 23:12:06 +00:00
_shader_load(d->vk, vertex_shader_code, vertex_shader_size,
2019-03-18 05:52:32 +00:00
VK_SHADER_STAGE_VERTEX_BIT),
_shader_load(d->vk, fragment_shader_code, fragment_shader_size,
VK_SHADER_STAGE_FRAGMENT_BIT),
};
2019-10-07 23:12:06 +00:00
2019-03-18 05:52:32 +00:00
VkGraphicsPipelineCreateInfo pipeline_info = {
.sType = VK_STRUCTURE_TYPE_GRAPHICS_PIPELINE_CREATE_INFO,
.pNext = NULL,
2019-03-18 05:52:32 +00:00
.flags = 0,
.stageCount = ARRAY_SIZE(shader_stages),
.pStages = shader_stages,
2019-10-07 23:12:06 +00:00
.pVertexInputState = &vertex_input_state,
2019-03-18 05:52:32 +00:00
.pInputAssemblyState = &input_assembly_state,
.pTessellationState = NULL,
2019-03-18 05:52:32 +00:00
.pViewportState = &viewport_state,
.pRasterizationState = &rasterization_state,
.pMultisampleState = &multisample_state,
.pDepthStencilState = &depth_stencil_state,
.pColorBlendState = &color_blend_state,
.pDynamicState = &dynamic_state,
.layout = d->pipeline_layout,
.renderPass = render_pass,
.subpass = 0, //! @todo
2019-03-18 05:52:32 +00:00
.basePipelineHandle = VK_NULL_HANDLE,
.basePipelineIndex = -1,
};
ret = vk->vkCreateGraphicsPipelines(vk->device, pipeline_cache, 1,
&pipeline_info, NULL, &d->pipeline);
if (ret != VK_SUCCESS) {
VK_DEBUG(d->vk, "vkCreateGraphicsPipelines failed %u!", ret);
}
vk->vkDestroyShaderModule(vk->device, shader_stages[0].module, NULL);
vk->vkDestroyShaderModule(vk->device, shader_stages[1].module, NULL);
}
static VkWriteDescriptorSet
comp_distortion_get_uniform_write_descriptor_set(struct comp_distortion *d,
uint32_t binding,
uint32_t eye)
{
return (VkWriteDescriptorSet){
.sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
.pNext = NULL,
2019-03-18 05:52:32 +00:00
.dstSet = d->descriptor_sets[eye],
.dstBinding = binding,
.dstArrayElement = 0, //! @todo
2019-03-18 05:52:32 +00:00
.descriptorCount = 1,
.descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
.pImageInfo = NULL,
2019-03-18 05:52:32 +00:00
.pBufferInfo = &d->ubo_handle.descriptor,
.pTexelBufferView = NULL,
2019-03-18 05:52:32 +00:00
};
}
static VkWriteDescriptorSet
comp_distortion_get_uniform_write_descriptor_set_vp(struct comp_distortion *d,
uint32_t binding,
uint32_t eye)
{
return (VkWriteDescriptorSet){
.sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
.pNext = NULL,
2019-03-18 05:52:32 +00:00
.dstSet = d->descriptor_sets[eye],
.dstBinding = binding,
.dstArrayElement = 0, //! @todo
2019-03-18 05:52:32 +00:00
.descriptorCount = 1,
.descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
.pImageInfo = NULL,
2019-03-18 05:52:32 +00:00
.pBufferInfo = &d->ubo_viewport_handles[eye].descriptor,
.pTexelBufferView = NULL,
2019-03-18 05:52:32 +00:00
};
}
static VkWriteDescriptorSet
comp_distortion_get_image_write_descriptor_set(
VkDescriptorSet descriptor_set,
VkDescriptorImageInfo *descriptor_position,
uint32_t binding)
{
return (VkWriteDescriptorSet){
.sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
.pNext = NULL,
2019-03-18 05:52:32 +00:00
.dstSet = descriptor_set,
.dstBinding = binding,
.dstArrayElement = 0, //! @todo
2019-03-18 05:52:32 +00:00
.descriptorCount = 1,
.descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
.pImageInfo = descriptor_position,
.pBufferInfo = NULL,
.pTexelBufferView = NULL,
2019-03-18 05:52:32 +00:00
};
}
static void
comp_distortion_init_descriptor_sets(struct comp_distortion *d,
VkDescriptorPool descriptor_pool)
{
struct vk_bundle *vk = d->vk;
VkResult ret;
VkDescriptorSetAllocateInfo alloc_info = {
.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_ALLOCATE_INFO,
.pNext = NULL,
2019-03-18 05:52:32 +00:00
.descriptorPool = descriptor_pool,
.descriptorSetCount = 1,
.pSetLayouts = &d->descriptor_set_layout,
};
for (uint32_t i = 0; i < 2; i++) {
ret = vk->vkAllocateDescriptorSets(d->vk->device, &alloc_info,
&d->descriptor_sets[i]);
if (ret != VK_SUCCESS) {
VK_DEBUG(d->vk, "vkAllocateDescriptorSets failed %u",
ret);
}
}
}
void
comp_distortion_update_descriptor_set(struct comp_distortion *d,
VkSampler sampler,
VkImageView view,
uint32_t eye)
{
struct vk_bundle *vk = d->vk;
VkDescriptorImageInfo image_info = {
.sampler = sampler,
.imageView = view,
.imageLayout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL,
};
VkWriteDescriptorSet write_descriptor_sets[3] = {
// Binding 0 : Render texture target
comp_distortion_get_image_write_descriptor_set(
d->descriptor_sets[eye], &image_info, 0),
// Binding 1 : Fragment shader uniform buffer
comp_distortion_get_uniform_write_descriptor_set(d, 1, eye),
// Binding 2 : view uniform buffer
comp_distortion_get_uniform_write_descriptor_set_vp(d, 2, eye),
};
vk->vkUpdateDescriptorSets(vk->device,
ARRAY_SIZE(write_descriptor_sets),
write_descriptor_sets, 0, NULL);
}
static void
comp_distortion_update_descriptor_sets(struct comp_distortion *d,
VkSampler samplers[2],
VkImageView views[2])
{
for (uint32_t i = 0; i < 2; i++) {
comp_distortion_update_descriptor_set(d, samplers[i], views[i],
i);
}
}
static void
comp_distortion_init_descriptor_set_layout(struct comp_distortion *d)
{
struct vk_bundle *vk = d->vk;
VkResult ret;
VkDescriptorSetLayoutBinding set_layout_bindings[3] = {
// Binding 0 : Render texture target left
{
.binding = 0,
.descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
.descriptorCount = 1,
.stageFlags = VK_SHADER_STAGE_FRAGMENT_BIT,
.pImmutableSamplers = NULL,
2019-03-18 05:52:32 +00:00
},
// Binding 1 : Fragment shader uniform buffer
{
.binding = 1,
.descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
.descriptorCount = 1,
.stageFlags = VK_SHADER_STAGE_FRAGMENT_BIT,
.pImmutableSamplers = NULL,
2019-03-18 05:52:32 +00:00
},
// binding 2: viewport index
{
.binding = 2,
.descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
.descriptorCount = 1,
.stageFlags = VK_SHADER_STAGE_VERTEX_BIT,
.pImmutableSamplers = NULL,
2019-03-18 05:52:32 +00:00
},
};
VkDescriptorSetLayoutCreateInfo set_layout_info = {
.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
.pNext = NULL,
.flags = 0,
2019-03-18 05:52:32 +00:00
.bindingCount = ARRAY_SIZE(set_layout_bindings),
.pBindings = set_layout_bindings,
};
ret = vk->vkCreateDescriptorSetLayout(d->vk->device, &set_layout_info,
NULL, &d->descriptor_set_layout);
if (ret != VK_SUCCESS) {
VK_DEBUG(d->vk, "vkCreateDescriptorSetLayout failed %u", ret);
}
}
static void
comp_distortion_init_pipeline_layout(struct comp_distortion *d)
{
struct vk_bundle *vk = d->vk;
VkResult ret;
VkPipelineLayoutCreateInfo pipeline_layout_info = {
.sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO,
.pNext = NULL,
.flags = 0,
2019-03-18 05:52:32 +00:00
.setLayoutCount = 1,
.pSetLayouts = &d->descriptor_set_layout,
.pushConstantRangeCount = 0,
.pPushConstantRanges = NULL};
2019-03-18 05:52:32 +00:00
ret = vk->vkCreatePipelineLayout(d->vk->device, &pipeline_layout_info,
NULL, &d->pipeline_layout);
if (ret != VK_SUCCESS) {
VK_DEBUG(d->vk, "Failed to create pipeline layout!");
}
}
void
comp_distortion_draw_quad(struct comp_distortion *d,
VkCommandBuffer command_buffer,
int eye)
{
struct vk_bundle *vk = d->vk;
vk->vkCmdBindDescriptorSets(
command_buffer, VK_PIPELINE_BIND_POINT_GRAPHICS, d->pipeline_layout,
0, 1, &d->descriptor_sets[eye], 0, NULL);
vk->vkCmdBindPipeline(command_buffer, VK_PIPELINE_BIND_POINT_GRAPHICS,
d->pipeline);
/* Draw 3 verts from which we construct the fullscreen quad in
* the shader*/
vk->vkCmdDraw(command_buffer, 3, 1, 0, 0);
}
2019-10-07 23:12:06 +00:00
void
comp_distortion_draw_mesh(struct comp_distortion *d,
VkCommandBuffer command_buffer,
int eye)
{
struct vk_bundle *vk = d->vk;
vk->vkCmdBindDescriptorSets(
command_buffer, VK_PIPELINE_BIND_POINT_GRAPHICS, d->pipeline_layout,
0, 1, &d->descriptor_sets[eye], 0, NULL);
vk->vkCmdBindPipeline(command_buffer, VK_PIPELINE_BIND_POINT_GRAPHICS,
d->pipeline);
VkDeviceSize offsets[] = {0};
vk->vkCmdBindVertexBuffers(command_buffer, 0, 1,
&(d->vbo_handle.buffer), offsets);
2019-10-23 21:00:22 +00:00
if (d->mesh.total_num_indices > 0) {
vk->vkCmdBindIndexBuffer(command_buffer, d->index_handle.buffer,
0, VK_INDEX_TYPE_UINT32);
vk->vkCmdDrawIndexed(command_buffer, d->mesh.num_indices[eye],
1, d->mesh.offset_indices[eye], 0, 0);
} else {
vk->vkCmdDraw(command_buffer, d->mesh.num_vertices, 1, 0, 0);
}
2019-10-07 23:12:06 +00:00
}
2019-03-18 05:52:32 +00:00
// Update fragment shader hmd warp uniform block
static void
comp_distortion_update_uniform_buffer_warp(struct comp_distortion *d,
struct comp_compositor *c)
{
// clang-format off
switch (d->distortion_model) {
case XRT_DISTORTION_MODEL_VIVE:
/*
* VIVE fragment shader
*/
d->ubo_vive.aspect_x_over_y = c->xdev->hmd->distortion.vive.aspect_x_over_y;
d->ubo_vive.grow_for_undistort = c->xdev->hmd->distortion.vive.grow_for_undistort;
for (uint32_t i = 0; i < 2; i++)
d->ubo_vive.undistort_r2_cutoff[i] = c->xdev->hmd->distortion.vive.undistort_r2_cutoff[i];
2019-03-18 05:52:32 +00:00
for (uint32_t i = 0; i < 2; i++)
for (uint32_t j = 0; j < 2; j++)
d->ubo_vive.center[i][j] = c->xdev->hmd->distortion.vive.center[i][j];
for (uint32_t i = 0; i < 2; i++)
for (uint32_t j = 0; j < 3; j++)
for (uint32_t k = 0; k < 3; k++)
d->ubo_vive.coefficients[i][j][k] = c->xdev->hmd->distortion.vive.coefficients[i][j][k];
memcpy(d->ubo_handle.mapped, &d->ubo_vive, sizeof(d->ubo_vive));
break;
case XRT_DISTORTION_MODEL_PANOTOOLS:
2019-10-07 23:12:06 +00:00
case XRT_DISTORTION_MODEL_MESHUV:
default:
/*
* Pano vision fragment shader
*/
d->ubo_pano.hmd_warp_param[0] = c->xdev->hmd->distortion.pano.distortion_k[0];
d->ubo_pano.hmd_warp_param[1] = c->xdev->hmd->distortion.pano.distortion_k[1];
d->ubo_pano.hmd_warp_param[2] = c->xdev->hmd->distortion.pano.distortion_k[2];
d->ubo_pano.hmd_warp_param[3] = c->xdev->hmd->distortion.pano.distortion_k[3];
d->ubo_pano.aberr[0] = c->xdev->hmd->distortion.pano.aberration_k[0];
d->ubo_pano.aberr[1] = c->xdev->hmd->distortion.pano.aberration_k[1];
d->ubo_pano.aberr[2] = c->xdev->hmd->distortion.pano.aberration_k[2];
d->ubo_pano.aberr[3] = c->xdev->hmd->distortion.pano.aberration_k[3];
2019-10-23 21:00:22 +00:00
d->ubo_pano.lens_center[0][0] = 0.5;//c->xdev->hmd->views[0].lens_center.x_meters;
d->ubo_pano.lens_center[0][1] = 0.5;//c->xdev->hmd->views[0].lens_center.y_meters;
d->ubo_pano.lens_center[1][0] = 0.5;//c->xdev->hmd->views[1].lens_center.x_meters;
d->ubo_pano.lens_center[1][1] = 0.5;//c->xdev->hmd->views[1].lens_center.y_meters;
d->ubo_pano.viewport_scale[0] = 1.0;//c->xdev->hmd->views[0].display.w_meters;
d->ubo_pano.viewport_scale[1] = 1.0;//c->xdev->hmd->views[0].display.h_meters;
d->ubo_pano.warp_scale = c->xdev->hmd->distortion.pano.warp_scale;
memcpy(d->ubo_handle.mapped, &d->ubo_pano, sizeof(d->ubo_pano));
2019-10-07 23:12:06 +00:00
break;
}
// clang-format on
2019-03-18 05:52:32 +00:00
/*
* Common vertex shader stuff.
*/
// clang-format off
d->ubo_vp_data[0].viewport_id = 0;
d->ubo_vp_data[0].rot = c->xdev->hmd->views[0].rot;
2019-03-18 05:52:32 +00:00
d->ubo_vp_data[1].viewport_id = 1;
d->ubo_vp_data[1].rot = c->xdev->hmd->views[1].rot;
2019-03-18 05:52:32 +00:00
memcpy(d->ubo_viewport_handles[0].mapped, &d->ubo_vp_data[0], sizeof(d->ubo_vp_data[0]));
memcpy(d->ubo_viewport_handles[1].mapped, &d->ubo_vp_data[1], sizeof(d->ubo_vp_data[1]));
// clang-format on
}
static VkResult
_create_buffer(struct vk_bundle *vk,
VkBufferUsageFlags usage_flags,
VkMemoryPropertyFlags memory_property_flags,
struct comp_uniform_buffer *buffer,
VkDeviceSize size,
void *data)
{
buffer->device = vk->device;
VkResult ret;
// Create the buffer handle.
VkBufferCreateInfo buffer_info = {
.sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO,
.pNext = NULL,
.flags = 0,
2019-03-18 05:52:32 +00:00
.size = size,
.usage = usage_flags,
.sharingMode = VK_SHARING_MODE_EXCLUSIVE,
.queueFamilyIndexCount = 0,
.pQueueFamilyIndices = NULL,
2019-03-18 05:52:32 +00:00
};
ret =
vk->vkCreateBuffer(vk->device, &buffer_info, NULL, &buffer->buffer);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to create buffer!");
return ret;
}
// Create the memory backing up the buffer handle.
VkMemoryRequirements mem_reqs;
vk->vkGetBufferMemoryRequirements(vk->device, buffer->buffer,
&mem_reqs);
// Find a memory type index that fits the properties of the buffer.
uint32_t memory_type_index;
vk_get_memory_type(vk, mem_reqs.memoryTypeBits, memory_property_flags,
&memory_type_index);
VkMemoryAllocateInfo mem_alloc = {
.sType = VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO,
.pNext = NULL,
2019-03-18 05:52:32 +00:00
.allocationSize = mem_reqs.size,
.memoryTypeIndex = memory_type_index,
};
ret =
vk->vkAllocateMemory(vk->device, &mem_alloc, NULL, &buffer->memory);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to allocate memory!");
goto err_buffer;
}
buffer->alignment = mem_reqs.alignment;
buffer->size = mem_alloc.allocationSize;
buffer->usageFlags = usage_flags;
buffer->memoryPropertyFlags = memory_property_flags;
// If a pointer to the buffer data has been passed, map the
// buffer and copy over the data
if (data != NULL) {
ret = _buffer_map(vk, buffer, VK_WHOLE_SIZE, 0);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to map buffer!");
goto err_memory;
}
memcpy(buffer->mapped, data, size);
_buffer_unmap(vk, buffer);
}
// Initialize a default descriptor that covers the whole buffer size
_buffer_setup_descriptor(vk, buffer, VK_WHOLE_SIZE, 0);
// Attach the memory to the buffer object
ret = vk->vkBindBufferMemory(vk->device, buffer->buffer, buffer->memory,
0);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to bind buffer to memory!");
goto err_memory;
}
return VK_SUCCESS;
err_memory:
vk->vkFreeMemory(vk->device, buffer->memory, NULL);
err_buffer:
vk->vkDestroyBuffer(vk->device, buffer->buffer, NULL);
return ret;
}
static void
2019-10-07 23:12:06 +00:00
comp_distortion_init_buffers(struct comp_distortion *d,
struct comp_compositor *c)
2019-03-18 05:52:32 +00:00
{
struct vk_bundle *vk = &c->vk;
VkMemoryPropertyFlags memory_property_flags = 0;
2019-10-07 23:12:06 +00:00
VkBufferUsageFlags ubo_usage_flags = 0;
VkBufferUsageFlags vbo_usage_flags = 0;
2019-10-23 21:00:22 +00:00
VkBufferUsageFlags index_usage_flags = 0;
2019-10-07 23:12:06 +00:00
2019-03-18 05:52:32 +00:00
VkResult ret;
2019-10-07 23:12:06 +00:00
// Using the same flags for all ubos and vbos uniform buffers.
ubo_usage_flags |= VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT;
vbo_usage_flags |= VK_BUFFER_USAGE_VERTEX_BUFFER_BIT;
2019-10-23 21:00:22 +00:00
index_usage_flags |= VK_BUFFER_USAGE_INDEX_BUFFER_BIT;
2019-03-18 05:52:32 +00:00
memory_property_flags |= VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT;
memory_property_flags |= VK_MEMORY_PROPERTY_HOST_COHERENT_BIT;
2019-10-07 23:12:06 +00:00
// Distortion ubo and vbo sizes.
VkDeviceSize ubo_size = 0;
VkDeviceSize vbo_size = 0;
2019-10-23 21:00:22 +00:00
VkDeviceSize index_size = 0;
switch (d->distortion_model) {
case XRT_DISTORTION_MODEL_PANOTOOLS:
ubo_size = sizeof(d->ubo_pano);
break;
2019-10-07 23:12:06 +00:00
case XRT_DISTORTION_MODEL_MESHUV:
ubo_size = sizeof(d->ubo_pano);
2019-10-23 21:00:22 +00:00
vbo_size = d->mesh.stride * d->mesh.num_vertices;
index_size = sizeof(int) * d->mesh.total_num_indices;
2019-10-07 23:12:06 +00:00
break;
case XRT_DISTORTION_MODEL_VIVE:
// Vive data
ubo_size = sizeof(d->ubo_vive);
break;
default:
// Should this be a error?
ubo_size = sizeof(d->ubo_pano);
break;
}
2019-10-07 23:12:06 +00:00
// fp ubo
ret = _create_buffer(vk, ubo_usage_flags, memory_property_flags,
&d->ubo_handle, ubo_size, NULL);
2019-03-18 05:52:32 +00:00
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to create warp ubo buffer!");
}
ret = _buffer_map(vk, &d->ubo_handle, VK_WHOLE_SIZE, 0);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to map warp ubo buffer!");
}
2019-10-07 23:12:06 +00:00
// vp ubo[0]
ret = _create_buffer(vk, ubo_usage_flags, memory_property_flags,
2019-03-18 05:52:32 +00:00
&d->ubo_viewport_handles[0],
sizeof(d->ubo_vp_data[0]), NULL);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to create vp ubo buffer[0]!");
}
ret = _buffer_map(vk, &d->ubo_viewport_handles[0], VK_WHOLE_SIZE, 0);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to map vp ubo buffer[0]!");
}
2019-10-07 23:12:06 +00:00
// vp ubo[1]
ret = _create_buffer(vk, ubo_usage_flags, memory_property_flags,
2019-03-18 05:52:32 +00:00
&d->ubo_viewport_handles[1],
sizeof(d->ubo_vp_data[1]), NULL);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to create vp ubo buffer[1]!");
}
ret = _buffer_map(vk, &d->ubo_viewport_handles[1], VK_WHOLE_SIZE, 0);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to map vp ubo buffer[1]!");
}
2019-10-07 23:12:06 +00:00
// Don't create vbo if size is zero.
if (vbo_size == 0) {
return;
}
ret = _create_buffer(vk, vbo_usage_flags, memory_property_flags,
2019-10-23 21:00:22 +00:00
&d->vbo_handle, vbo_size, d->mesh.vertices);
2019-10-07 23:12:06 +00:00
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to create mesh vbo buffer!");
}
ret = _buffer_map(vk, &d->vbo_handle, vbo_size, 0);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to map mesh vbo buffer!");
}
2019-10-23 21:00:22 +00:00
if (index_size == 0) {
return;
}
ret = _create_buffer(vk, index_usage_flags, memory_property_flags,
&d->index_handle, index_size, d->mesh.indices);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to create mesh index buffer!");
}
ret = _buffer_map(vk, &d->index_handle, index_size, 0);
if (ret != VK_SUCCESS) {
VK_DEBUG(vk, "Failed to map mesh vbo buffer!");
}
2019-03-18 05:52:32 +00:00
}