mirror of
https://gitlab.freedesktop.org/mesa/mesa.git
synced 2026-05-17 09:38:05 +02:00
Signed-off-by: Samuel Pitoiset <samuel.pitoiset@gmail.com> Part-of: <https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/40657>
197 lines
8.6 KiB
C
197 lines
8.6 KiB
C
/*
|
|
* Copyright © 2024 Valve Corporation
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
* to deal in the Software without restriction, including without limitation
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice (including the next
|
|
* paragraph) shall be included in all copies or substantial portions of the
|
|
* Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
|
* IN THE SOFTWARE.
|
|
*/
|
|
|
|
#include "vk_alloc.h"
|
|
#include "vk_device.h"
|
|
#include "vk_device_generated_commands.h"
|
|
#include "vk_log.h"
|
|
#include "vk_util.h"
|
|
|
|
#include "util/compiler.h"
|
|
|
|
static int
|
|
compare_pc_layout(const void *_pc1, const void *_pc2)
|
|
{
|
|
const struct vk_indirect_command_push_constant_layout *pc1 = _pc1;
|
|
const struct vk_indirect_command_push_constant_layout *pc2 = _pc2;
|
|
|
|
assert(pc1->dst_offset_B <= INT_MAX);
|
|
assert(pc2->dst_offset_B <= INT_MAX);
|
|
|
|
return (int)pc1->dst_offset_B - (int)pc2->dst_offset_B;
|
|
}
|
|
|
|
void *
|
|
vk_indirect_command_layout_create(struct vk_device *device,
|
|
const VkIndirectCommandsLayoutCreateInfoEXT* pCreateInfo,
|
|
const VkAllocationCallbacks* pAllocator,
|
|
size_t struct_size)
|
|
{
|
|
struct vk_indirect_command_layout *elayout;
|
|
uint32_t n_pc_layouts = 0, n_vb_layouts = 0;
|
|
|
|
for (unsigned i = 0; i < pCreateInfo->tokenCount; i++) {
|
|
const VkIndirectCommandsLayoutTokenEXT *token = &pCreateInfo->pTokens[i];
|
|
switch (token->type) {
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_PUSH_CONSTANT_EXT:
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_PUSH_DATA_EXT:
|
|
n_pc_layouts++;
|
|
break;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_VERTEX_BUFFER_EXT:
|
|
n_vb_layouts++;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
VK_MULTIALLOC(ma);
|
|
vk_multialloc_add_size_align(&ma, (void **)&elayout, struct_size, 8);
|
|
VK_MULTIALLOC_DECL(&ma, struct vk_indirect_command_push_constant_layout, pc_layouts, n_pc_layouts);
|
|
VK_MULTIALLOC_DECL(&ma, struct vk_indirect_command_vertex_layout, vb_layouts, n_vb_layouts);
|
|
elayout = vk_multialloc_zalloc2(&ma, &device->alloc, pAllocator,
|
|
VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
|
|
if (!elayout)
|
|
return NULL;
|
|
|
|
vk_object_base_init(device, &elayout->base, VK_OBJECT_TYPE_INDIRECT_COMMANDS_LAYOUT_EXT);
|
|
|
|
elayout->pc_layouts = pc_layouts;
|
|
elayout->vb_layouts = vb_layouts;
|
|
|
|
for (unsigned i = 0; i < pCreateInfo->tokenCount; i++) {
|
|
const VkIndirectCommandsLayoutTokenEXT *token = &pCreateInfo->pTokens[i];
|
|
switch (token->type) {
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_EXECUTION_SET_EXT:
|
|
elayout->is_shaders = token->data.pExecutionSet->type == VK_INDIRECT_EXECUTION_SET_INFO_TYPE_SHADER_OBJECTS_EXT;
|
|
elayout->ies_src_offset_B = token->offset;
|
|
elayout->dgc_info |= BITFIELD_BIT(MESA_VK_DGC_IES);
|
|
break;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_VERTEX_BUFFER_EXT:
|
|
assert(token->data.pVertexBuffer->vertexBindingUnit < 32);
|
|
elayout->vertex_bindings |= BITFIELD_BIT(token->data.pVertexBuffer->vertexBindingUnit);
|
|
elayout->dgc_info |= BITFIELD_BIT(MESA_VK_DGC_VB);
|
|
vb_layouts[elayout->n_vb_layouts++] = (struct vk_indirect_command_vertex_layout) {
|
|
.binding = token->data.pVertexBuffer->vertexBindingUnit,
|
|
.src_offset_B = token->offset,
|
|
};
|
|
break;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_INDEX_BUFFER_EXT:
|
|
elayout->index_mode_is_dx = token->data.pIndexBuffer->mode == VK_INDIRECT_COMMANDS_INPUT_MODE_DXGI_INDEX_BUFFER_EXT;
|
|
elayout->index_src_offset_B = token->offset;
|
|
elayout->dgc_info |= BITFIELD_BIT(MESA_VK_DGC_IB);
|
|
break;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_PUSH_CONSTANT_EXT:
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_PUSH_DATA_EXT:
|
|
elayout->dgc_info |= BITFIELD_BIT(MESA_VK_DGC_PC);
|
|
pc_layouts[elayout->n_pc_layouts++] = (struct vk_indirect_command_push_constant_layout) {
|
|
.stages = token->data.pPushConstant->updateRange.stageFlags,
|
|
.dst_offset_B = token->data.pPushConstant->updateRange.offset,
|
|
.src_offset_B = token->offset,
|
|
.size_B = token->data.pPushConstant->updateRange.size,
|
|
};
|
|
break;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_SEQUENCE_INDEX_EXT:
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_PUSH_DATA_SEQUENCE_INDEX_EXT:
|
|
assert(token->data.pPushConstant->updateRange.size == 4);
|
|
elayout->dgc_info |= BITFIELD_BIT(MESA_VK_DGC_SI);
|
|
elayout->si_layout = (struct vk_indirect_command_push_constant_layout) {
|
|
.stages = token->data.pPushConstant->updateRange.stageFlags,
|
|
.dst_offset_B = token->data.pPushConstant->updateRange.offset,
|
|
.src_offset_B = token->offset,
|
|
.size_B = token->data.pPushConstant->updateRange.size,
|
|
};
|
|
break;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_DRAW_COUNT_EXT:
|
|
elayout->draw_count = true;
|
|
FALLTHROUGH;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_DRAW_EXT:
|
|
elayout->dgc_info |= BITFIELD_BIT(MESA_VK_DGC_DRAW);
|
|
elayout->draw_src_offset_B = token->offset;
|
|
break;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_DRAW_INDEXED_COUNT_EXT:
|
|
elayout->draw_count = true;
|
|
FALLTHROUGH;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_DRAW_INDEXED_EXT:
|
|
elayout->dgc_info |= BITFIELD_BIT(MESA_VK_DGC_DRAW_INDEXED);
|
|
elayout->draw_src_offset_B = token->offset;
|
|
break;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_DRAW_MESH_TASKS_COUNT_NV_EXT:
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_DRAW_MESH_TASKS_COUNT_EXT:
|
|
elayout->draw_count = true;
|
|
FALLTHROUGH;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_DRAW_MESH_TASKS_NV_EXT:
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_DRAW_MESH_TASKS_EXT:
|
|
elayout->dgc_info |= BITFIELD_BIT(MESA_VK_DGC_DRAW_MESH);
|
|
elayout->draw_src_offset_B = token->offset;
|
|
break;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_DISPATCH_EXT:
|
|
elayout->dgc_info |= BITFIELD_BIT(MESA_VK_DGC_DISPATCH);
|
|
elayout->dispatch_src_offset_B = token->offset;
|
|
break;
|
|
case VK_INDIRECT_COMMANDS_TOKEN_TYPE_TRACE_RAYS2_EXT:
|
|
elayout->dgc_info |= BITFIELD_BIT(MESA_VK_DGC_RT);
|
|
elayout->dispatch_src_offset_B = token->offset;
|
|
break;
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
if (elayout->dgc_info & (BITFIELD_BIT(MESA_VK_DGC_PC) | BITFIELD_BIT(MESA_VK_DGC_SI))) {
|
|
const VkPipelineLayoutCreateInfo *plci = vk_find_struct_const(pCreateInfo->pNext, PIPELINE_LAYOUT_CREATE_INFO);
|
|
|
|
if (pCreateInfo->pipelineLayout) {
|
|
elayout->layout = pCreateInfo->pipelineLayout;
|
|
} else if (plci) {
|
|
const struct vk_device_dispatch_table *disp = &device->dispatch_table;
|
|
|
|
assert(device->enabled_features.dynamicGeneratedPipelineLayout);
|
|
disp->CreatePipelineLayout(vk_device_to_handle(device), plci, NULL, &elayout->layout);
|
|
elayout->delete_layout = true;
|
|
}
|
|
}
|
|
|
|
qsort(elayout->pc_layouts, elayout->n_pc_layouts,
|
|
sizeof(elayout->pc_layouts[0]), compare_pc_layout);
|
|
|
|
elayout->stages = pCreateInfo->shaderStages;
|
|
elayout->usage = pCreateInfo->flags;
|
|
elayout->stride = pCreateInfo->indirectStride;
|
|
elayout->token_count = pCreateInfo->tokenCount;
|
|
|
|
return elayout;
|
|
}
|
|
|
|
void
|
|
vk_indirect_command_layout_destroy(struct vk_device *device,
|
|
const VkAllocationCallbacks *pAllocator,
|
|
struct vk_indirect_command_layout *elayout)
|
|
{
|
|
if (elayout->delete_layout) {
|
|
const struct vk_device_dispatch_table *disp = &device->dispatch_table;
|
|
assert(device->enabled_features.dynamicGeneratedPipelineLayout);
|
|
disp->DestroyPipelineLayout(vk_device_to_handle(device), elayout->layout, NULL);
|
|
}
|
|
vk_object_base_finish(&elayout->base);
|
|
vk_free2(&device->alloc, pAllocator, elayout);
|
|
}
|