mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2025-01-22 17:01:12 +00:00
eb8fc0fb9a
This get pretty confusing when you refer to both the sets and the set layouts in close vicinity.
1417 lines
42 KiB
C
1417 lines
42 KiB
C
/*
|
|
render.c
|
|
|
|
Vulkan render manager
|
|
|
|
Copyright (C) 2023 Bill Currie <bill@taniwha.org>
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to:
|
|
|
|
Free Software Foundation, Inc.
|
|
59 Temple Place - Suite 330
|
|
Boston, MA 02111-1307, USA
|
|
|
|
*/
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_MATH_H
|
|
# include <math.h>
|
|
#endif
|
|
#ifdef HAVE_STRING_H
|
|
# include <string.h>
|
|
#endif
|
|
#ifdef HAVE_STRINGS_H
|
|
# include <strings.h>
|
|
#endif
|
|
|
|
#include "QF/cmem.h"
|
|
#include "QF/hash.h"
|
|
#include "QF/va.h"
|
|
#include "QF/Vulkan/command.h"
|
|
#include "QF/Vulkan/debug.h"
|
|
#include "QF/Vulkan/device.h"
|
|
#include "QF/Vulkan/render.h"
|
|
#include "QF/Vulkan/resource.h"
|
|
#include "QF/Vulkan/pipeline.h"
|
|
#include "vid_vulkan.h"
|
|
|
|
#include "vkparse.h"
|
|
|
|
static VkCommandBuffer
|
|
get_cmd_buffer (vulkan_ctx_t *ctx, int secondary)
|
|
{
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
__auto_type rctx = ctx->render_context;
|
|
__auto_type job = rctx->job;
|
|
|
|
VkCommandBufferAllocateInfo cinfo = {
|
|
.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO,
|
|
.commandPool = job->command_pool,
|
|
.level = secondary ? VK_COMMAND_BUFFER_LEVEL_SECONDARY
|
|
: VK_COMMAND_BUFFER_LEVEL_PRIMARY,
|
|
.commandBufferCount = 1,
|
|
};
|
|
VkCommandBuffer cmd;
|
|
dfunc->vkAllocateCommandBuffers (device->dev, &cinfo, &cmd);
|
|
return cmd;
|
|
}
|
|
|
|
static void
|
|
run_tasks (uint32_t task_count, qfv_taskinfo_t *tasks, void *ctx)
|
|
{
|
|
for (uint32_t i = 0; i < task_count; i++) {
|
|
tasks[i].func->func (tasks[i].params, 0, ctx);
|
|
}
|
|
}
|
|
|
|
static void
|
|
run_pipeline (qfv_pipeline_t *pipeline, VkCommandBuffer cmd, vulkan_ctx_t *ctx)
|
|
{
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
dfunc->vkCmdBindPipeline (cmd, pipeline->bindPoint, pipeline->pipeline);
|
|
dfunc->vkCmdSetViewport (cmd, 0, 1, &pipeline->viewport);
|
|
dfunc->vkCmdSetScissor (cmd, 0, 1, &pipeline->scissor);
|
|
|
|
qfv_taskctx_t taskctx = {
|
|
.ctx = ctx,
|
|
.pipeline = pipeline,
|
|
};
|
|
run_tasks (pipeline->task_count, pipeline->tasks, &taskctx);
|
|
|
|
if (pipeline->num_descriptorsets) {
|
|
dfunc->vkCmdBindDescriptorSets (cmd, pipeline->bindPoint,
|
|
pipeline->layout,
|
|
pipeline->first_descriptorset,
|
|
pipeline->num_descriptorsets,
|
|
pipeline->descriptorsets,
|
|
0, 0);
|
|
}
|
|
if (pipeline->num_push_constants) {
|
|
QFV_PushConstants (device, cmd, pipeline->layout,
|
|
pipeline->num_push_constants,
|
|
pipeline->push_constants);
|
|
}
|
|
}
|
|
|
|
// https://themaister.net/blog/2019/08/14/yet-another-blog-explaining-vulkan-synchronization/
|
|
static void
|
|
run_subpass (qfv_subpass_t_ *sp, VkCommandBuffer cmd, vulkan_ctx_t *ctx)
|
|
{
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
dfunc->vkResetCommandBuffer (cmd, 0);
|
|
dfunc->vkBeginCommandBuffer (cmd, &sp->beginInfo);
|
|
QFV_duCmdBeginLabel (device, cmd, sp->label.name,
|
|
{VEC4_EXP (sp->label.color)});
|
|
|
|
for (uint32_t i = 0; i < sp->pipeline_count; i++) {
|
|
__auto_type pipeline = &sp->pipelines[i];
|
|
run_pipeline (pipeline, cmd, ctx);
|
|
}
|
|
|
|
QFV_duCmdEndLabel (device, cmd);
|
|
dfunc->vkEndCommandBuffer (cmd);
|
|
}
|
|
|
|
static void
|
|
run_renderpass (qfv_renderpass_t_ *rp, vulkan_ctx_t *ctx)
|
|
{
|
|
printf ("run_renderpass: %s\n", rp->label.name);
|
|
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
|
|
VkCommandBuffer cmd = get_cmd_buffer (ctx, 1);
|
|
VkCommandBufferBeginInfo beginInfo = {
|
|
VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO,
|
|
};
|
|
dfunc->vkResetCommandBuffer (cmd, 0);
|
|
dfunc->vkBeginCommandBuffer (cmd, &beginInfo);
|
|
QFV_duCmdBeginLabel (device, cmd, rp->label.name,
|
|
{VEC4_EXP (rp->label.color)});
|
|
dfunc->vkCmdBeginRenderPass (cmd, &rp->beginInfo, rp->subpassContents);
|
|
for (uint32_t i = 0; i < rp->subpass_count; i++) {
|
|
__auto_type sp = &rp->subpasses[i];
|
|
VkCommandBuffer subcmd = get_cmd_buffer (ctx, 1);
|
|
run_subpass (sp, subcmd, ctx);
|
|
dfunc->vkCmdExecuteCommands (cmd, 1, &subcmd);
|
|
//FIXME comment is a bit off as exactly one buffer is always submitted
|
|
//
|
|
//Regardless of whether any commands were submitted for this
|
|
//subpass, must step through each and every subpass, otherwise
|
|
//the attachments won't be transitioned correctly.
|
|
//However, only if not the last (or only) subpass.
|
|
if (i < rp->subpass_count - 1) {
|
|
dfunc->vkCmdNextSubpass (cmd, rp->subpassContents);
|
|
}
|
|
}
|
|
QFV_CmdEndLabel (device, cmd);
|
|
}
|
|
|
|
static void
|
|
run_compute_pipeline (qfv_pipeline_t *pipeline, VkCommandBuffer cmd,
|
|
vulkan_ctx_t *ctx)
|
|
{
|
|
printf ("run_compute_pipeline: %s\n", pipeline->label.name);
|
|
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
dfunc->vkCmdBindPipeline (cmd, pipeline->bindPoint, pipeline->pipeline);
|
|
|
|
qfv_taskctx_t taskctx = {
|
|
.ctx = ctx,
|
|
.pipeline = pipeline,
|
|
};
|
|
run_tasks (pipeline->task_count, pipeline->tasks, &taskctx);
|
|
|
|
if (pipeline->num_descriptorsets) {
|
|
dfunc->vkCmdBindDescriptorSets (cmd, pipeline->bindPoint,
|
|
pipeline->layout,
|
|
pipeline->first_descriptorset,
|
|
pipeline->num_descriptorsets,
|
|
pipeline->descriptorsets,
|
|
0, 0);
|
|
}
|
|
if (pipeline->num_push_constants) {
|
|
QFV_PushConstants (device, cmd, pipeline->layout,
|
|
pipeline->num_push_constants,
|
|
pipeline->push_constants);
|
|
}
|
|
uint32_t *d = pipeline->dispatch;
|
|
dfunc->vkCmdDispatch (cmd, d[0], d[1], d[2]);
|
|
}
|
|
|
|
static void
|
|
run_compute (qfv_compute_t *comp, vulkan_ctx_t *ctx)
|
|
{
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
__auto_type rctx = ctx->render_context;
|
|
__auto_type job = rctx->job;
|
|
|
|
VkCommandBuffer cmd = get_cmd_buffer (ctx, 0);
|
|
|
|
VkCommandBufferBeginInfo beginInfo = {
|
|
.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO,
|
|
.flags = VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT,
|
|
};
|
|
dfunc->vkBeginCommandBuffer (cmd, &beginInfo);
|
|
for (uint32_t i = 0; i < comp->pipeline_count; i++) {
|
|
__auto_type pipeline = &comp->pipelines[i];
|
|
run_compute_pipeline (pipeline, cmd, ctx);
|
|
}
|
|
dfunc->vkEndCommandBuffer (cmd);
|
|
DARRAY_APPEND (&job->commands, cmd);
|
|
}
|
|
|
|
static void
|
|
run_process (qfv_process_t *proc, vulkan_ctx_t *ctx)
|
|
{
|
|
qfv_taskctx_t taskctx = {
|
|
.ctx = ctx,
|
|
};
|
|
run_tasks (proc->task_count, proc->tasks, &taskctx);
|
|
}
|
|
|
|
void
|
|
QFV_RunRenderJob (vulkan_ctx_t *ctx)
|
|
{
|
|
__auto_type rctx = ctx->render_context;
|
|
__auto_type job = rctx->job;
|
|
|
|
for (uint32_t i = 0; i < job->num_steps; i++) {
|
|
__auto_type step = &job->steps[i];
|
|
printf ("run_step: %s\n", step->label.name);
|
|
if (step->render) {
|
|
run_renderpass (step->render->active, ctx);
|
|
}
|
|
if (step->compute) {
|
|
run_compute (step->compute, ctx);
|
|
}
|
|
if (step->process) {
|
|
run_process (step->process, ctx);
|
|
}
|
|
}
|
|
}
|
|
|
|
static qfv_output_t
|
|
get_output (vulkan_ctx_t *ctx, plitem_t *item)
|
|
{
|
|
qfv_output_t output = {};
|
|
Vulkan_ConfigOutput (ctx, &output);
|
|
|
|
plitem_t *output_def = PL_ObjectForKey (item, "output");
|
|
if (output_def) {
|
|
// QFV_ParseOutput clears the structure, but extent and frames need to
|
|
// be preserved
|
|
qfv_output_t o = output;
|
|
QFV_ParseOutput (ctx, &o, output_def, item);
|
|
output.format = o.format;
|
|
output.finalLayout = o.finalLayout;
|
|
}
|
|
return output;
|
|
}
|
|
|
|
void
|
|
QFV_LoadRenderInfo (vulkan_ctx_t *ctx)
|
|
{
|
|
__auto_type rctx = ctx->render_context;
|
|
|
|
plitem_t *item = Vulkan_GetConfig (ctx, "main_def");
|
|
__auto_type output = get_output (ctx, item);
|
|
Vulkan_Script_SetOutput (ctx, &output);
|
|
rctx->jobinfo = QFV_ParseJobInfo (ctx, item, rctx);
|
|
if (rctx->jobinfo) {
|
|
rctx->jobinfo->plitem = item;
|
|
}
|
|
}
|
|
|
|
typedef struct {
|
|
uint32_t num_images;
|
|
uint32_t num_imageviews;
|
|
uint32_t num_buffers;
|
|
uint32_t num_bufferviews;
|
|
uint32_t num_layouts;
|
|
|
|
uint32_t num_steps;
|
|
uint32_t num_render;
|
|
uint32_t num_compute;
|
|
uint32_t num_process;
|
|
uint32_t num_tasks;
|
|
|
|
uint32_t num_renderpasses;
|
|
uint32_t num_attachments;
|
|
uint32_t num_subpasses;
|
|
uint32_t num_dependencies;
|
|
uint32_t num_attachmentrefs;
|
|
uint32_t num_colorblend;
|
|
uint32_t num_preserve;
|
|
uint32_t num_graph_pipelines;
|
|
uint32_t num_comp_pipelines;
|
|
|
|
uint32_t num_descriptorsets;
|
|
} objcount_t;
|
|
|
|
static void
|
|
count_as_stuff (qfv_attachmentsetinfo_t *as, objcount_t *counts)
|
|
{
|
|
counts->num_attachmentrefs += as->num_input;
|
|
counts->num_attachmentrefs += as->num_color;
|
|
counts->num_colorblend += as->num_color;
|
|
if (as->resolve) {
|
|
counts->num_attachmentrefs += as->num_color;
|
|
}
|
|
if (as->depth) {
|
|
counts->num_attachmentrefs += 1;
|
|
}
|
|
counts->num_preserve += as->num_preserve;
|
|
}
|
|
|
|
static void
|
|
count_sp_stuff (qfv_subpassinfo_t *spi, objcount_t *counts)
|
|
{
|
|
counts->num_dependencies += spi->num_dependencies;
|
|
if (spi->attachments) {
|
|
count_as_stuff (spi->attachments, counts);
|
|
}
|
|
for (uint32_t i = 0; i < spi->num_pipelines; i++) {
|
|
__auto_type pli = &spi->pipelines[i];
|
|
if (pli->num_graph_stages && !pli->compute_stage) {
|
|
counts->num_graph_pipelines++;
|
|
counts->num_tasks += pli->num_tasks;
|
|
} else {
|
|
Sys_Error ("%s:%s: invalid graphics pipeline",
|
|
spi->name, pli->name);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
count_rp_stuff (qfv_renderpassinfo_t *rpi, objcount_t *counts)
|
|
{
|
|
counts->num_attachments += rpi->num_attachments;
|
|
counts->num_subpasses += rpi->num_subpasses;
|
|
for (uint32_t i = 0; i < rpi->num_subpasses; i++) {
|
|
count_sp_stuff (&rpi->subpasses[i], counts);
|
|
}
|
|
}
|
|
|
|
static void
|
|
count_comp_stuff (qfv_computeinfo_t *ci, objcount_t *counts)
|
|
{
|
|
for (uint32_t i = 0; i < ci->num_pipelines; i++) {
|
|
__auto_type pli = &ci->pipelines[i];
|
|
if (!pli->num_graph_stages && pli->compute_stage) {
|
|
counts->num_comp_pipelines++;
|
|
counts->num_tasks += pli->num_tasks;
|
|
} else {
|
|
Sys_Error ("%s:%s: invalid compute pipeline",
|
|
ci->name, pli->name);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
count_step_stuff (qfv_stepinfo_t *step, objcount_t *counts)
|
|
{
|
|
if ((step->render && step->compute)
|
|
|| (step->render && step->process)
|
|
|| (step->compute && step->process)) {
|
|
Sys_Error ("%s: invalid step: must be one of render/compute/process",
|
|
step->name);
|
|
}
|
|
if (step->render) {
|
|
__auto_type rinfo = step->render;
|
|
counts->num_renderpasses += rinfo->num_renderpasses;
|
|
for (uint32_t i = 0; i < rinfo->num_renderpasses; i++) {
|
|
count_rp_stuff (&rinfo->renderpasses[i], counts);
|
|
}
|
|
counts->num_render++;
|
|
}
|
|
if (step->compute) {
|
|
count_comp_stuff (step->compute, counts);
|
|
counts->num_compute++;
|
|
}
|
|
if (step->process) {
|
|
counts->num_process++;
|
|
counts->num_tasks += step->process->num_tasks;
|
|
}
|
|
counts->num_steps++;
|
|
}
|
|
|
|
static void
|
|
count_stuff (qfv_jobinfo_t *jobinfo, objcount_t *counts)
|
|
{
|
|
counts->num_images += jobinfo->num_images;
|
|
counts->num_imageviews += jobinfo->num_imageviews;
|
|
counts->num_buffers += jobinfo->num_buffers;
|
|
counts->num_bufferviews += jobinfo->num_bufferviews;
|
|
for (uint32_t i = 0; i < jobinfo->num_steps; i++) {
|
|
count_step_stuff (&jobinfo->steps[i], counts);
|
|
}
|
|
}
|
|
|
|
static void
|
|
create_resources (vulkan_ctx_t *ctx, objcount_t *counts)
|
|
{
|
|
__auto_type rctx = ctx->render_context;
|
|
__auto_type jinfo = rctx->jobinfo;
|
|
__auto_type job = rctx->job;
|
|
|
|
job->resources = malloc (sizeof(qfv_resource_t)
|
|
+ counts->num_images * sizeof (qfv_resobj_t)
|
|
+ counts->num_imageviews * sizeof (qfv_resobj_t));
|
|
job->images = (qfv_resobj_t *) &job->resources[1];
|
|
job->image_views = &job->images[counts->num_images];
|
|
|
|
job->resources[0] = (qfv_resource_t) {
|
|
.name = "render",
|
|
.va_ctx = ctx->va_ctx,
|
|
.memory_properties = VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT,
|
|
.num_objects = counts->num_images + counts->num_imageviews,
|
|
.objects = job->images,
|
|
};
|
|
for (uint32_t i = 0; i < counts->num_images; i++) {
|
|
__auto_type img = &jinfo->images[i];
|
|
job->images[i] = (qfv_resobj_t) {
|
|
.name = img->name,
|
|
.type = qfv_res_image,
|
|
.image = {
|
|
.flags = img->flags,
|
|
.type = img->imageType,
|
|
.format = img->format,
|
|
.extent = img->extent,
|
|
.num_mipmaps = img->mipLevels,
|
|
.num_layers = img->arrayLayers,
|
|
.samples = img->samples,
|
|
.tiling = img->tiling,
|
|
.usage = img->usage,
|
|
.initialLayout = img->initialLayout,
|
|
},
|
|
};
|
|
}
|
|
int error = 0;
|
|
for (uint32_t i = 0; i < counts->num_imageviews; i++) {
|
|
__auto_type view = &jinfo->imageviews[i];
|
|
job->image_views[i] = (qfv_resobj_t) {
|
|
.name = view->name,
|
|
.type = qfv_res_image_view,
|
|
.image_view = {
|
|
.flags = view->flags,
|
|
.type = view->viewType,
|
|
.format = view->format,
|
|
.components = view->components,
|
|
.subresourceRange = view->subresourceRange,
|
|
},
|
|
};
|
|
if (strcmp (view->image.name, "$output.image") == 0) {
|
|
//__auto_type image = jinfo->output.image;
|
|
//job->image_views[i].image_view.external_image = image;
|
|
//job->image_views[i].image_view.image = -1;
|
|
} else {
|
|
qfv_resobj_t *img = 0;
|
|
for (uint32_t j = 0; j < jinfo->num_images; j++) {
|
|
if (strcmp (view->image.name, jinfo->images[j].name) == 0) {
|
|
img = &job->images[j];
|
|
}
|
|
}
|
|
if (img) {
|
|
uint32_t ind = img - job->resources->objects;
|
|
job->image_views[i].image_view.image = ind;
|
|
} else {
|
|
Sys_Printf ("%d: unknown image reference: %s\n",
|
|
view->image.line, view->image.name);
|
|
error = 1;
|
|
}
|
|
}
|
|
}
|
|
if (error) {
|
|
free (job->resources);
|
|
job->resources = 0;
|
|
return;
|
|
}
|
|
QFV_CreateResource (ctx->device, job->resources);
|
|
}
|
|
|
|
typedef struct {
|
|
VkRenderPassCreateInfo *rpCreate;
|
|
VkAttachmentDescription *attach;
|
|
VkClearValue *clear;
|
|
VkSubpassDescription *subpass;
|
|
VkSubpassDependency *depend;
|
|
VkAttachmentReference *attachref;
|
|
VkPipelineColorBlendAttachmentState *cbAttach;
|
|
uint32_t *preserve;
|
|
const char **rpName;
|
|
const char **plName;
|
|
VkComputePipelineCreateInfo *cplCreate;
|
|
VkGraphicsPipelineCreateInfo *gplCreate;
|
|
VkPipelineColorBlendStateCreateInfo *cbState;
|
|
qfv_layoutinfo_t *layouts;
|
|
|
|
uint32_t *pl_counts;
|
|
|
|
VkPipeline *gpl;
|
|
VkPipeline *cpl;
|
|
VkRenderPass *rp;
|
|
} objptr_t;
|
|
|
|
typedef struct {
|
|
objcount_t inds;
|
|
objptr_t ptr;
|
|
|
|
vulkan_ctx_t *ctx;
|
|
qfv_jobinfo_t *jinfo;
|
|
exprtab_t *symtab;
|
|
qfv_renderpassinfo_t *rpi;
|
|
VkRenderPassCreateInfo *rpc;
|
|
qfv_subpassinfo_t *spi;
|
|
VkSubpassDescription *spc;
|
|
qfv_pipelineinfo_t *pli;
|
|
VkGraphicsPipelineCreateInfo *plc;
|
|
} objstate_t;
|
|
|
|
static uint32_t __attribute__((pure))
|
|
find_subpass (qfv_dependencyinfo_t *d, uint32_t spind,
|
|
qfv_subpassinfo_t *subpasses)
|
|
{
|
|
if (strcmp (d->name, "$external") == 0) {
|
|
return VK_SUBPASS_EXTERNAL;
|
|
}
|
|
for (uint32_t i = 0; i <= spind; i++) {
|
|
__auto_type s = &subpasses[i];
|
|
if (strcmp (d->name, s->name) == 0) {
|
|
return i;
|
|
}
|
|
}
|
|
Sys_Error ("invalid dependency: [%d] %s", spind, d->name);
|
|
}
|
|
|
|
static VkDescriptorSetLayout
|
|
find_descriptorSet (const qfv_reference_t *ref, objstate_t *s)
|
|
{
|
|
for (uint32_t i = 0; i < s->jinfo->num_descriptorsetlayouts; i++) {
|
|
__auto_type ds = &s->jinfo->descriptorsetlayouts[i];
|
|
if (strcmp (ds->name, ref->name) == 0) {
|
|
if (!ds->setLayout) {
|
|
VkDescriptorSetLayoutCreateInfo cInfo = {
|
|
.sType=VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
|
|
.flags = ds->flags,
|
|
.bindingCount = ds->num_bindings,
|
|
.pBindings = ds->bindings,
|
|
};
|
|
qfv_device_t *device = s->ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
dfunc->vkCreateDescriptorSetLayout (device->dev, &cInfo, 0,
|
|
&ds->setLayout);
|
|
QFV_duSetObjectName (device,
|
|
VK_OBJECT_TYPE_DESCRIPTOR_SET_LAYOUT,
|
|
ds->setLayout,
|
|
va (s->ctx->va_ctx, "descriptorSet:%s",
|
|
ds->name));
|
|
}
|
|
return ds->setLayout;
|
|
}
|
|
}
|
|
Sys_Error ("%s.%s:%d: invalid descriptor set layout: %s",
|
|
s->rpi->name, s->spi->name, ref->line, ref->name);
|
|
}
|
|
|
|
static qfv_layoutinfo_t *
|
|
find_layout (const qfv_reference_t *ref, objstate_t *s)
|
|
{
|
|
for (uint32_t i = 0; i < s->inds.num_layouts; i++) {
|
|
if (strcmp (s->ptr.layouts[i].name, ref->name) == 0) {
|
|
return &s->ptr.layouts[i];
|
|
}
|
|
}
|
|
if (!QFV_ParseLayoutInfo (s->ctx, s->jinfo->memsuper, s->symtab, ref->name,
|
|
&s->ptr.layouts[s->inds.num_layouts])) {
|
|
Sys_Error ("%s.%s:%d: invalid layout: %s",
|
|
s->rpi->name, s->spi->name, ref->line, ref->name);
|
|
}
|
|
__auto_type li = &s->ptr.layouts[s->inds.num_layouts++];
|
|
li->name = ref->name;
|
|
VkDescriptorSetLayout sets[li->num_sets];
|
|
for (uint32_t i = 0; i < li->num_sets; i++) {
|
|
sets[i] = find_descriptorSet (&li->sets[i], s);
|
|
}
|
|
VkPipelineLayoutCreateInfo cInfo = {
|
|
.sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO,
|
|
.setLayoutCount = li->num_sets,
|
|
.pSetLayouts = sets,
|
|
.pushConstantRangeCount = li->num_ranges,
|
|
.pPushConstantRanges = li->ranges,
|
|
};
|
|
qfv_device_t *device = s->ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
dfunc->vkCreatePipelineLayout (device->dev, &cInfo, 0, &li->layout);
|
|
QFV_duSetObjectName (device, VK_OBJECT_TYPE_PIPELINE_LAYOUT, li->layout,
|
|
va (s->ctx->va_ctx, "layout:%s", li->name));
|
|
return li;
|
|
}
|
|
|
|
static void
|
|
init_plCreate (VkGraphicsPipelineCreateInfo *plc, const qfv_pipelineinfo_t *pli,
|
|
objstate_t *s)
|
|
{
|
|
if (pli->num_graph_stages) {
|
|
plc->stageCount = pli->num_graph_stages;
|
|
}
|
|
if (pli->graph_stages) {
|
|
plc->pStages = pli->graph_stages;
|
|
}
|
|
if (pli->vertexInput) {
|
|
plc->pVertexInputState = pli->vertexInput;
|
|
}
|
|
if (pli->inputAssembly) {
|
|
plc->pInputAssemblyState = pli->inputAssembly;
|
|
}
|
|
if (pli->tessellation) {
|
|
plc->pTessellationState = pli->tessellation;
|
|
}
|
|
if (pli->viewport) {
|
|
plc->pViewportState = pli->viewport;
|
|
}
|
|
if (pli->rasterization) {
|
|
plc->pRasterizationState = pli->rasterization;
|
|
}
|
|
if (pli->multisample) {
|
|
plc->pMultisampleState = pli->multisample;
|
|
}
|
|
if (pli->depthStencil) {
|
|
plc->pDepthStencilState = pli->depthStencil;
|
|
}
|
|
if (pli->colorBlend) {
|
|
VkPipelineColorBlendStateCreateInfo *cb;
|
|
cb = (VkPipelineColorBlendStateCreateInfo *) plc->pColorBlendState;
|
|
*cb = *pli->colorBlend;
|
|
}
|
|
if (pli->dynamic) {
|
|
plc->pDynamicState = pli->dynamic;
|
|
}
|
|
if (pli->layout.name) {
|
|
__auto_type li = find_layout (&pli->layout, s);
|
|
plc->layout = li->layout;
|
|
s->inds.num_descriptorsets += li->num_sets;
|
|
}
|
|
}
|
|
|
|
static uint32_t __attribute__((pure))
|
|
find_attachment (qfv_reference_t *ref, objstate_t *s)
|
|
{
|
|
for (uint32_t i = 0; i < s->rpi->num_attachments; i++) {
|
|
__auto_type a = &s->rpi->attachments[i];
|
|
if (strcmp (ref->name, a->name) == 0) {
|
|
return i;
|
|
}
|
|
}
|
|
Sys_Error ("%s.%s:%d: invalid attachment: %s",
|
|
s->rpi->name, s->spi->name, ref->line, ref->name);
|
|
}
|
|
|
|
static void
|
|
init_arCreate (const qfv_attachmentrefinfo_t *ari, objstate_t *s)
|
|
{
|
|
__auto_type arc = &s->ptr.attachref[s->inds.num_attachmentrefs];
|
|
qfv_reference_t ref = {
|
|
.name = ari->name,
|
|
.line = ari->line,
|
|
};
|
|
|
|
*arc = (VkAttachmentReference) {
|
|
.attachment = find_attachment (&ref, s),
|
|
.layout = ari->layout,
|
|
};
|
|
}
|
|
|
|
static void
|
|
init_cbCreate (const qfv_attachmentrefinfo_t *ari, objstate_t *s)
|
|
{
|
|
__auto_type cbc = &s->ptr.cbAttach[s->inds.num_colorblend];
|
|
|
|
*cbc = ari->blend;
|
|
}
|
|
|
|
static void
|
|
init_spCreate (uint32_t index, qfv_subpassinfo_t *sub, objstate_t *s)
|
|
{
|
|
s->spi = &sub[index];
|
|
s->plc = &s->ptr.gplCreate[s->inds.num_graph_pipelines];
|
|
s->spc = &s->ptr.subpass[s->inds.num_subpasses];
|
|
__auto_type pln = &s->ptr.plName[s->inds.num_graph_pipelines];
|
|
__auto_type cbs = &s->ptr.cbState[s->inds.num_graph_pipelines];
|
|
|
|
s->ptr.pl_counts[s->inds.num_renderpasses] += s->spi->num_pipelines;
|
|
|
|
*s->spc = (VkSubpassDescription) {
|
|
.pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS,
|
|
};
|
|
for (uint32_t i = 0; i < s->spi->num_dependencies; i++) {
|
|
__auto_type d = &s->spi->dependencies[i];
|
|
__auto_type dep = &s->ptr.depend[s->inds.num_dependencies++];
|
|
*dep = (VkSubpassDependency) {
|
|
.srcSubpass = find_subpass (d, index, s->rpi->subpasses),
|
|
.dstSubpass = index,
|
|
.srcStageMask = d->src.stage,
|
|
.dstStageMask = d->dst.stage,
|
|
.srcAccessMask = d->src.access,
|
|
.dstAccessMask = d->dst.access,
|
|
.dependencyFlags = d->flags,
|
|
};
|
|
}
|
|
|
|
for (uint32_t i = 0; i < s->spi->num_pipelines; i++) {
|
|
s->plc[i] = (VkGraphicsPipelineCreateInfo) {
|
|
.sType = VK_STRUCTURE_TYPE_GRAPHICS_PIPELINE_CREATE_INFO,
|
|
.pColorBlendState = &cbs[i],
|
|
.subpass = index,
|
|
};
|
|
if (s->spi->base_pipeline) {
|
|
init_plCreate (&s->plc[i], s->spi->base_pipeline, s);
|
|
}
|
|
init_plCreate (&s->plc[i], &s->spi->pipelines[i], s);
|
|
pln[i] = s->spi->name;
|
|
s->inds.num_graph_pipelines++;
|
|
}
|
|
|
|
__auto_type att = s->spi->attachments;
|
|
if (!att) {
|
|
return;
|
|
}
|
|
for (uint32_t i = 0; i < s->spi->num_pipelines; i++) {
|
|
cbs[i].attachmentCount = att->num_color;
|
|
cbs[i].pAttachments = &s->ptr.cbAttach[s->inds.num_colorblend];
|
|
}
|
|
s->spc->inputAttachmentCount = att->num_input;
|
|
s->spc->pInputAttachments = &s->ptr.attachref[s->inds.num_attachmentrefs];
|
|
for (uint32_t i = 0; i < att->num_input; i++) {
|
|
init_arCreate (&att->input[i], s);
|
|
s->inds.num_attachmentrefs++;
|
|
}
|
|
s->spc->colorAttachmentCount = att->num_color;
|
|
s->spc->pColorAttachments = &s->ptr.attachref[s->inds.num_attachmentrefs];
|
|
for (uint32_t i = 0; i < att->num_color; i++) {
|
|
init_arCreate (&att->color[i], s);
|
|
s->inds.num_attachmentrefs++;
|
|
init_cbCreate (&att->color[i], s);
|
|
s->inds.num_colorblend++;
|
|
}
|
|
if (att->resolve) {
|
|
s->spc->pResolveAttachments
|
|
= &s->ptr.attachref[s->inds.num_attachmentrefs];
|
|
for (uint32_t i = 0; i < att->num_color; i++) {
|
|
init_arCreate (&att->resolve[i], s);
|
|
s->inds.num_attachmentrefs++;
|
|
}
|
|
}
|
|
if (att->depth) {
|
|
s->spc->pDepthStencilAttachment
|
|
= &s->ptr.attachref[s->inds.num_attachmentrefs];
|
|
init_arCreate (att->depth, s);
|
|
s->inds.num_attachmentrefs++;
|
|
}
|
|
s->spc->preserveAttachmentCount = att->num_preserve;
|
|
s->spc->pPreserveAttachments = &s->ptr.preserve[s->inds.num_preserve];
|
|
for (uint32_t i = 0; i < att->num_preserve; i++) {
|
|
s->ptr.preserve[s->inds.num_preserve]
|
|
= find_attachment (&att->preserve[i], s);
|
|
s->inds.num_preserve++;
|
|
}
|
|
}
|
|
|
|
static void
|
|
init_atCreate (uint32_t index, qfv_attachmentinfo_t *attachments, objstate_t *s)
|
|
{
|
|
__auto_type ati = &attachments[index];
|
|
__auto_type atc = &s->ptr.attach[s->inds.num_attachments];
|
|
__auto_type cvc = &s->ptr.clear[s->inds.num_attachments];
|
|
|
|
*atc = (VkAttachmentDescription) {
|
|
.flags = ati->flags,
|
|
.format = ati->format,
|
|
.samples = ati->samples,
|
|
.loadOp = ati->loadOp,
|
|
.storeOp = ati->storeOp,
|
|
.stencilLoadOp = ati->stencilLoadOp,
|
|
.stencilStoreOp = ati->stencilStoreOp,
|
|
.initialLayout = ati->initialLayout,
|
|
.finalLayout = ati->finalLayout,
|
|
};
|
|
*cvc = ati->clearValue;
|
|
}
|
|
|
|
static void
|
|
init_rpCreate (uint32_t index, const qfv_renderinfo_t *rinfo, objstate_t *s)
|
|
{
|
|
s->rpi = &rinfo->renderpasses[index];
|
|
s->rpc = &s->ptr.rpCreate[s->inds.num_renderpasses];
|
|
s->ptr.rpName[s->inds.num_renderpasses] = s->rpi->name;
|
|
|
|
__auto_type attachments = &s->ptr.attach[s->inds.num_attachments];
|
|
__auto_type subpasses = &s->ptr.subpass[s->inds.num_subpasses];
|
|
__auto_type dependencies = &s->ptr.depend[s->inds.num_dependencies];
|
|
|
|
for (uint32_t i = 0; i < s->rpi->num_attachments; i++) {
|
|
init_atCreate (i, s->rpi->attachments, s);
|
|
s->inds.num_attachments++;
|
|
}
|
|
|
|
uint32_t num_dependencies = s->inds.num_dependencies;
|
|
for (uint32_t i = 0; i < s->rpi->num_subpasses; i++) {
|
|
init_spCreate (i, s->rpi->subpasses, s);
|
|
s->inds.num_subpasses++;
|
|
}
|
|
num_dependencies = s->inds.num_dependencies - num_dependencies;
|
|
|
|
*s->rpc = (VkRenderPassCreateInfo) {
|
|
.sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO,
|
|
.attachmentCount = s->rpi->num_attachments,
|
|
.pAttachments = attachments,
|
|
.subpassCount = s->rpi->num_subpasses,
|
|
.pSubpasses = subpasses,
|
|
.dependencyCount = num_dependencies,
|
|
.pDependencies = dependencies,
|
|
};
|
|
}
|
|
|
|
typedef struct {
|
|
qfv_step_t *steps;
|
|
qfv_render_t *renders;
|
|
qfv_compute_t *computes;
|
|
qfv_process_t *processes;
|
|
qfv_renderpass_t_ *renderpasses;
|
|
VkClearValue *clearvalues;
|
|
qfv_subpass_t_ *subpasses;
|
|
qfv_pipeline_t *pipelines;
|
|
qfv_taskinfo_t *tasks;
|
|
VkDescriptorSet *descriptorsets;
|
|
} jobptr_t;
|
|
|
|
static void
|
|
init_pipeline (qfv_pipeline_t *pl, qfv_pipelineinfo_t *plinfo,
|
|
jobptr_t *jp, objstate_t *s, int is_compute)
|
|
{
|
|
__auto_type li = find_layout (&plinfo->layout, s);
|
|
*pl = (qfv_pipeline_t) {
|
|
.label = {
|
|
.name = plinfo->name,
|
|
.color = plinfo->color,
|
|
},
|
|
.bindPoint = is_compute ? VK_PIPELINE_BIND_POINT_COMPUTE
|
|
: VK_PIPELINE_BIND_POINT_GRAPHICS,
|
|
.pipeline = is_compute ? s->ptr.cpl[s->inds.num_comp_pipelines]
|
|
: s->ptr.gpl[s->inds.num_graph_pipelines],
|
|
.layout = li->layout,
|
|
.task_count = plinfo->num_tasks,
|
|
.tasks = &jp->tasks[s->inds.num_tasks],
|
|
.descriptorsets = &jp->descriptorsets[s->inds.num_descriptorsets],
|
|
};
|
|
s->inds.num_tasks += plinfo->num_tasks;
|
|
s->inds.num_descriptorsets += li->num_sets;
|
|
for (uint32_t i = 0; i < li->num_sets; i++) {
|
|
pl->descriptorsets[i] = 0;
|
|
}
|
|
for (uint32_t i = 0; i < pl->task_count; i++) {
|
|
pl->tasks[i] = plinfo->tasks[i];
|
|
}
|
|
}
|
|
|
|
static void
|
|
init_subpass (qfv_subpass_t_ *sp, qfv_subpassinfo_t *isp,
|
|
jobptr_t *jp, objstate_t *s)
|
|
{
|
|
uint32_t np = s->inds.num_graph_pipelines + s->inds.num_comp_pipelines;
|
|
*sp = (qfv_subpass_t_) {
|
|
.label = {
|
|
.name = isp->name,
|
|
.color = isp->color,
|
|
},
|
|
.inherit = {
|
|
.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_INHERITANCE_INFO,
|
|
},
|
|
.beginInfo = {
|
|
.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO,
|
|
.flags = VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT
|
|
| VK_COMMAND_BUFFER_USAGE_RENDER_PASS_CONTINUE_BIT,
|
|
.pInheritanceInfo = &sp->inherit,
|
|
},
|
|
.pipeline_count = isp->num_pipelines,
|
|
.pipelines = &jp->pipelines[np],
|
|
};
|
|
for (uint32_t i = 0; i < isp->num_pipelines; i++) {
|
|
init_pipeline (&sp->pipelines[i], &isp->pipelines[i], jp, s, 0);
|
|
s->inds.num_graph_pipelines++;
|
|
}
|
|
}
|
|
|
|
static void
|
|
init_renderpass (qfv_renderpass_t_ *rp, qfv_renderpassinfo_t *rpinfo,
|
|
jobptr_t *jp, objstate_t *s)
|
|
{
|
|
*rp = (qfv_renderpass_t_) {
|
|
.vulkan_ctx = s->ctx,
|
|
.label.name = rpinfo->name,
|
|
.label.color = rpinfo->color,
|
|
.subpass_count = rpinfo->num_subpasses,
|
|
.subpasses = &jp->subpasses[s->inds.num_subpasses],
|
|
.beginInfo = (VkRenderPassBeginInfo) {
|
|
.sType = VK_STRUCTURE_TYPE_RENDER_PASS_BEGIN_INFO,
|
|
.renderPass = s->ptr.rp[s->inds.num_renderpasses],
|
|
.clearValueCount = rpinfo->num_attachments,
|
|
.pClearValues = &jp->clearvalues[s->inds.num_attachments],
|
|
},
|
|
.subpassContents = VK_SUBPASS_CONTENTS_SECONDARY_COMMAND_BUFFERS,
|
|
};
|
|
s->inds.num_attachments += rpinfo->num_attachments;
|
|
for (uint32_t i = 0; i < rpinfo->num_subpasses; i++) {
|
|
init_subpass (&rp->subpasses[i], &rpinfo->subpasses[i], jp, s);
|
|
rp->subpasses[i].inherit.renderPass = rp->beginInfo.renderPass;
|
|
rp->subpasses[i].inherit.subpass = i;
|
|
s->inds.num_subpasses++;
|
|
}
|
|
}
|
|
|
|
static void
|
|
init_render (qfv_render_t *rend, qfv_renderinfo_t *rinfo,
|
|
jobptr_t *jp, objstate_t *s)
|
|
{
|
|
*rend = (qfv_render_t) {
|
|
.label.color = rinfo->color,
|
|
.label.name = rinfo->name,
|
|
.num_renderpasses = rinfo->num_renderpasses,
|
|
.renderpasses = &jp->renderpasses[s->inds.num_renderpasses],
|
|
};
|
|
s->inds.num_renderpasses += rinfo->num_renderpasses;
|
|
for (uint32_t i = 0; i < rend->num_renderpasses; i++) {
|
|
init_renderpass (&rend->renderpasses[i], &rinfo->renderpasses[i],
|
|
jp, s);
|
|
}
|
|
rend->active = &rend->renderpasses[0];
|
|
}
|
|
|
|
static void
|
|
init_compute (qfv_compute_t *comp, qfv_computeinfo_t *cinfo,
|
|
jobptr_t *jp, objstate_t *s)
|
|
{
|
|
uint32_t np = s->inds.num_graph_pipelines + s->inds.num_comp_pipelines;
|
|
*comp = (qfv_compute_t) {
|
|
.label.color = cinfo->color,
|
|
.label.name = cinfo->name,
|
|
.pipelines = &jp->pipelines[np],
|
|
.pipeline_count = cinfo->num_pipelines,
|
|
};
|
|
for (uint32_t i = 0; i < cinfo->num_pipelines; i++) {
|
|
init_pipeline (&comp->pipelines[i], &cinfo->pipelines[i], jp, s, 1);
|
|
s->inds.num_comp_pipelines++;
|
|
}
|
|
}
|
|
|
|
static void
|
|
init_process (qfv_process_t *proc, qfv_processinfo_t *pinfo,
|
|
jobptr_t *jp, objstate_t *s)
|
|
{
|
|
*proc = (qfv_process_t) {
|
|
.label.color = pinfo->color,
|
|
.label.name = pinfo->name,
|
|
.tasks = &jp->tasks[s->inds.num_tasks],
|
|
.task_count = pinfo->num_tasks,
|
|
};
|
|
s->inds.num_tasks += pinfo->num_tasks;
|
|
for (uint32_t i = 0; i < proc->task_count; i++) {
|
|
proc->tasks[i] = pinfo->tasks[i];
|
|
}
|
|
}
|
|
|
|
static void
|
|
init_step (uint32_t ind, jobptr_t *jp, objstate_t *s)
|
|
{
|
|
__auto_type step = &jp->steps[s->inds.num_steps++];
|
|
__auto_type sinfo = &s->jinfo->steps[ind];
|
|
|
|
*step = (qfv_step_t) {
|
|
.label.name = sinfo->name,
|
|
.label.color = sinfo->color,
|
|
};
|
|
if (sinfo->render) {
|
|
step->render = &jp->renders[s->inds.num_render++];
|
|
init_render (step->render, sinfo->render, jp, s);
|
|
}
|
|
if (sinfo->compute) {
|
|
step->compute = &jp->computes[s->inds.num_compute++];
|
|
init_compute (step->compute, sinfo->compute, jp, s);
|
|
}
|
|
if (sinfo->process) {
|
|
step->process = &jp->processes[s->inds.num_process++];
|
|
init_process (step->process, sinfo->process, jp, s);
|
|
}
|
|
}
|
|
|
|
static void
|
|
init_job (vulkan_ctx_t *ctx, objcount_t *counts, objstate_t s)
|
|
{
|
|
__auto_type rctx = ctx->render_context;
|
|
|
|
size_t size = sizeof (qfv_job_t);
|
|
|
|
size += counts->num_renderpasses * sizeof (VkRenderPass);
|
|
size += counts->num_graph_pipelines * sizeof (VkPipeline);
|
|
size += counts->num_comp_pipelines * sizeof (VkPipeline);
|
|
size += s.inds.num_layouts * sizeof (VkPipelineLayout);
|
|
|
|
size += counts->num_steps * sizeof (qfv_step_t);
|
|
size += counts->num_render * sizeof (qfv_render_t);
|
|
size += counts->num_compute * sizeof (qfv_compute_t);
|
|
size += counts->num_process * sizeof (qfv_process_t);
|
|
size += counts->num_renderpasses * sizeof (qfv_renderpass_t_);
|
|
size += counts->num_attachments * sizeof (VkClearValue);
|
|
size += counts->num_subpasses * sizeof (qfv_subpass_t_);
|
|
size += counts->num_graph_pipelines * sizeof (qfv_pipeline_t);
|
|
size += counts->num_comp_pipelines * sizeof (qfv_pipeline_t);
|
|
size += counts->num_tasks * sizeof (qfv_taskinfo_t);
|
|
size += counts->num_descriptorsets * sizeof (VkDescriptorSet);
|
|
|
|
rctx->job = malloc (size);
|
|
__auto_type job = rctx->job;
|
|
*job = (qfv_job_t) {
|
|
.num_renderpasses = counts->num_renderpasses,
|
|
.num_pipelines = counts->num_graph_pipelines
|
|
+ counts->num_comp_pipelines,
|
|
.num_layouts = s.inds.num_layouts,
|
|
.num_steps = counts->num_steps,
|
|
.commands = DARRAY_STATIC_INIT (16),
|
|
};
|
|
job->renderpasses = (VkRenderPass *) &job[1];
|
|
job->pipelines = (VkPipeline *) &job->renderpasses[job->num_renderpasses];
|
|
job->layouts = (VkPipelineLayout *) &job->pipelines[job->num_pipelines];
|
|
job->steps = (qfv_step_t *) &job->layouts[job->num_layouts];
|
|
__auto_type rn = (qfv_render_t *) &job->steps[job->num_steps];
|
|
__auto_type cp = (qfv_compute_t *) &rn[counts->num_render];
|
|
__auto_type pr = (qfv_process_t *) &cp[counts->num_compute];
|
|
__auto_type rp = (qfv_renderpass_t_ *) &pr[counts->num_process];
|
|
__auto_type cv = (VkClearValue *) &rp[counts->num_renderpasses];
|
|
__auto_type sp = (qfv_subpass_t_ *) &cv[counts->num_attachments];
|
|
__auto_type pl = (qfv_pipeline_t *) &sp[counts->num_subpasses];
|
|
__auto_type ti = (qfv_taskinfo_t *) &pl[job->num_pipelines];
|
|
__auto_type ds = (VkDescriptorSet *) &ti[counts->num_tasks];
|
|
jobptr_t jp = {
|
|
.steps = job->steps,
|
|
.renders = rn,
|
|
.computes = cp,
|
|
.processes = pr,
|
|
.renderpasses = rp,
|
|
.clearvalues = cv,
|
|
.subpasses = sp,
|
|
.pipelines = pl,
|
|
.tasks = ti,
|
|
.descriptorsets = ds,
|
|
};
|
|
|
|
for (uint32_t i = 0; i < job->num_renderpasses; i++) {
|
|
job->renderpasses[i] = s.ptr.rp[i];
|
|
}
|
|
for (uint32_t i = 0; i < job->num_pipelines; i++) {
|
|
// compute pipelines come immediately after the graphics pipelines
|
|
job->pipelines[i] = s.ptr.gpl[i];
|
|
}
|
|
for (uint32_t i = 0; i < s.inds.num_layouts; i++) {
|
|
job->layouts[i] = s.ptr.layouts[i].layout;
|
|
}
|
|
memcpy (cv, s.ptr.clear, counts->num_attachments * sizeof (VkClearValue));
|
|
|
|
uint32_t num_layouts = s.inds.num_layouts;
|
|
s.inds = (objcount_t) {};
|
|
s.inds.num_layouts = num_layouts;
|
|
for (uint32_t i = 0; i < job->num_steps; i++) {
|
|
init_step (i, &jp, &s);
|
|
}
|
|
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
VkCommandPoolCreateInfo poolCInfo = {
|
|
.sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO,
|
|
.flags = VK_COMMAND_POOL_CREATE_TRANSIENT_BIT,
|
|
.queueFamilyIndex = device->queue.queueFamily,
|
|
};
|
|
dfunc->vkCreateCommandPool (device->dev, &poolCInfo, 0, &job->command_pool);
|
|
}
|
|
|
|
static void
|
|
create_step_render_objects (uint32_t index, const qfv_stepinfo_t *step,
|
|
objstate_t *s)
|
|
{
|
|
__auto_type rinfo = step->render;
|
|
if (!rinfo) {
|
|
return;
|
|
}
|
|
for (uint32_t i = 0; i < rinfo->num_renderpasses; i++) {
|
|
s->ptr.pl_counts[s->inds.num_renderpasses] = 0;
|
|
init_rpCreate (i, rinfo, s);
|
|
s->inds.num_renderpasses++;
|
|
}
|
|
}
|
|
|
|
static void
|
|
create_step_compute_objects (uint32_t index, const qfv_stepinfo_t *step,
|
|
objstate_t *s)
|
|
{
|
|
__auto_type cinfo = step->compute;
|
|
if (!cinfo) {
|
|
return;
|
|
}
|
|
|
|
uint32_t base = s->inds.num_graph_pipelines;
|
|
for (uint32_t i = 0; i < cinfo->num_pipelines; i++) {
|
|
__auto_type pli = &cinfo->pipelines[i];
|
|
__auto_type plc = &s->ptr.cplCreate[s->inds.num_comp_pipelines];
|
|
__auto_type li = find_layout (&pli->layout, s);
|
|
s->ptr.plName[base + s->inds.num_comp_pipelines] = pli->name;
|
|
*plc = (VkComputePipelineCreateInfo) {
|
|
.sType = VK_STRUCTURE_TYPE_COMPUTE_PIPELINE_CREATE_INFO,
|
|
.flags = pli->flags,
|
|
.stage = *pli->compute_stage,
|
|
.layout = li->layout,
|
|
};
|
|
plc->stage.stage = VK_SHADER_STAGE_COMPUTE_BIT;
|
|
s->inds.num_comp_pipelines++;
|
|
s->inds.num_descriptorsets += li->num_sets;
|
|
}
|
|
}
|
|
|
|
static void
|
|
create_step_process_objects (uint32_t index, const qfv_stepinfo_t *step,
|
|
objstate_t *s)
|
|
{
|
|
__auto_type pinfo = step->process;
|
|
if (!pinfo) {
|
|
return;
|
|
}
|
|
// nothing to create at this stage
|
|
}
|
|
|
|
static void
|
|
create_objects (vulkan_ctx_t *ctx, objcount_t *counts)
|
|
{
|
|
__auto_type rctx = ctx->render_context;
|
|
__auto_type jinfo = rctx->jobinfo;
|
|
|
|
VkRenderPass renderpasses[counts->num_renderpasses];
|
|
VkPipeline pipelines[counts->num_graph_pipelines
|
|
+ counts->num_comp_pipelines];
|
|
VkRenderPassCreateInfo rpCreate[counts->num_renderpasses];
|
|
VkAttachmentDescription attach[counts->num_attachments];
|
|
VkClearValue clear[counts->num_attachments];
|
|
VkSubpassDescription subpass[counts->num_subpasses];
|
|
VkSubpassDependency depend[counts->num_dependencies];
|
|
VkAttachmentReference attachref[counts->num_attachmentrefs];
|
|
VkPipelineColorBlendAttachmentState cbAttach[counts->num_colorblend];
|
|
uint32_t preserve[counts->num_preserve];
|
|
const char *rpName[counts->num_renderpasses];
|
|
const char *plName[counts->num_graph_pipelines
|
|
+ counts->num_comp_pipelines];
|
|
VkComputePipelineCreateInfo cplCreate[counts->num_comp_pipelines];
|
|
VkGraphicsPipelineCreateInfo gplCreate[counts->num_graph_pipelines];
|
|
VkPipelineColorBlendStateCreateInfo cbState[counts->num_graph_pipelines];
|
|
qfv_layoutinfo_t layouts[counts->num_graph_pipelines
|
|
+ counts->num_comp_pipelines];
|
|
uint32_t pl_counts[counts->num_renderpasses];
|
|
|
|
exprctx_t ectx = { .hashctx = &ctx->script_context->hashctx };
|
|
objstate_t s = {
|
|
.ptr = {
|
|
.rpCreate = rpCreate,
|
|
.attach = attach,
|
|
.clear = clear,
|
|
.subpass = subpass,
|
|
.depend = depend,
|
|
.attachref = attachref,
|
|
.cbAttach = cbAttach,
|
|
.preserve = preserve,
|
|
.rpName = rpName,
|
|
.plName = plName,
|
|
.cplCreate = cplCreate,
|
|
.gplCreate = gplCreate,
|
|
.cbState = cbState,
|
|
.layouts = layouts,
|
|
.pl_counts = pl_counts,
|
|
.rp = renderpasses,
|
|
.gpl = pipelines,
|
|
.cpl = pipelines + counts->num_graph_pipelines,
|
|
},
|
|
.ctx = ctx,
|
|
.jinfo = jinfo,
|
|
.symtab = QFV_CreateSymtab (jinfo->plitem, "properties", 0, 0, &ectx),
|
|
};
|
|
for (uint32_t i = 0; i < jinfo->num_steps; i++) {
|
|
create_step_render_objects (i, &jinfo->steps[i], &s);
|
|
}
|
|
for (uint32_t i = 0; i < jinfo->num_steps; i++) {
|
|
create_step_compute_objects (i, &jinfo->steps[i], &s);
|
|
}
|
|
for (uint32_t i = 0; i < jinfo->num_steps; i++) {
|
|
create_step_process_objects (i, &jinfo->steps[i], &s);
|
|
}
|
|
if (s.inds.num_renderpasses != counts->num_renderpasses
|
|
|| s.inds.num_attachments != counts->num_attachments
|
|
|| s.inds.num_subpasses != counts->num_subpasses
|
|
|| s.inds.num_dependencies != counts->num_dependencies
|
|
|| s.inds.num_attachmentrefs != counts->num_attachmentrefs
|
|
|| s.inds.num_colorblend != counts->num_colorblend
|
|
|| s.inds.num_preserve != counts->num_preserve
|
|
|| s.inds.num_graph_pipelines != counts->num_graph_pipelines
|
|
|| s.inds.num_comp_pipelines != counts->num_comp_pipelines) {
|
|
Sys_Error ("create_objects: something was missed");
|
|
}
|
|
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
uint32_t plInd = 0;
|
|
for (uint32_t i = 0; i < counts->num_renderpasses; i++) {
|
|
dfunc->vkCreateRenderPass (device->dev, &s.ptr.rpCreate[i], 0,
|
|
&renderpasses[i]);
|
|
QFV_duSetObjectName (device, VK_OBJECT_TYPE_RENDER_PASS,
|
|
renderpasses[i],
|
|
va (ctx->va_ctx, "renderpass:%s", rpName[i]));
|
|
for (uint32_t j = 0; j < pl_counts[i]; j++) {
|
|
s.ptr.gplCreate[plInd++].renderPass = renderpasses[i];
|
|
}
|
|
}
|
|
if (s.inds.num_graph_pipelines) {
|
|
dfunc->vkCreateGraphicsPipelines (device->dev, 0,
|
|
s.inds.num_graph_pipelines,
|
|
s.ptr.gplCreate, 0, pipelines);
|
|
}
|
|
if (s.inds.num_comp_pipelines) {
|
|
__auto_type p = &pipelines[s.inds.num_graph_pipelines];
|
|
dfunc->vkCreateComputePipelines (device->dev, 0,
|
|
s.inds.num_comp_pipelines,
|
|
s.ptr.cplCreate, 0, p);
|
|
}
|
|
for (uint32_t i = 0;
|
|
i < s.inds.num_graph_pipelines + s.inds.num_comp_pipelines; i++) {
|
|
QFV_duSetObjectName (device, VK_OBJECT_TYPE_PIPELINE, pipelines[i],
|
|
va (ctx->va_ctx, "pipeline:%s", plName[i]));
|
|
}
|
|
|
|
counts->num_descriptorsets = s.inds.num_descriptorsets;
|
|
init_job (ctx, counts, s);
|
|
}
|
|
|
|
void
|
|
QFV_BuildRender (vulkan_ctx_t *ctx)
|
|
{
|
|
__auto_type rctx = ctx->render_context;
|
|
|
|
objcount_t counts = {};
|
|
count_stuff (rctx->jobinfo, &counts);
|
|
|
|
create_objects (ctx, &counts);
|
|
create_resources (ctx, &counts);
|
|
}
|
|
|
|
static VkImageView __attribute__((pure, used))
|
|
find_imageview (qfv_reference_t *ref, qfv_renderctx_t *rctx)
|
|
{
|
|
__auto_type jinfo = rctx->jobinfo;
|
|
__auto_type job = rctx->job;
|
|
const char *name = ref->name;
|
|
|
|
if (strncmp (name, "$imageviews.", 7) == 0) {
|
|
name += 7;
|
|
}
|
|
|
|
for (uint32_t i = 0; i < jinfo->num_imageviews; i++) {
|
|
__auto_type vi = &jinfo->imageviews[i];
|
|
__auto_type vo = &job->image_views[i];
|
|
if (strcmp (name, vi->name) == 0) {
|
|
return vo->image_view.view;
|
|
}
|
|
}
|
|
Sys_Error ("%d:invalid imageview: %s", ref->line, ref->name);
|
|
}
|
|
|
|
void
|
|
QFV_DestroyFramebuffer (vulkan_ctx_t *ctx)
|
|
{
|
|
#if 0
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
__auto_type rctx = ctx->render_context;
|
|
__auto_type job = rctx->job;
|
|
__auto_type rp = &job->renderpasses[0];
|
|
|
|
if (rp->beginInfo.framebuffer) {
|
|
VkFramebuffer framebuffer = rp->beginInfo.framebuffer;
|
|
rp->beginInfo.framebuffer = 0;
|
|
dfunc->vkDestroyFramebuffer (device->dev, framebuffer, 0);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void
|
|
QFV_CreateFramebuffer (vulkan_ctx_t *ctx)
|
|
{
|
|
#if 0
|
|
__auto_type rctx = ctx->render_context;
|
|
__auto_type rinfo = rctx->renderinfo;
|
|
__auto_type job = rctx->job;
|
|
__auto_type rpInfo = &rinfo->renderpasses[0];
|
|
__auto_type rp = &job->renderpasses[0];
|
|
|
|
VkImageView attachments[rpInfo->num_attachments];
|
|
VkFramebufferCreateInfo cInfo = {
|
|
.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO,
|
|
.attachmentCount = rpInfo->num_attachments,
|
|
.pAttachments = attachments,
|
|
.renderPass = rp->beginInfo.renderPass,
|
|
.width = rpInfo->framebuffer.width,
|
|
.height = rpInfo->framebuffer.height,
|
|
.layers = rpInfo->framebuffer.layers,
|
|
};
|
|
for (uint32_t i = 0; i < rpInfo->num_attachments; i++) {
|
|
attachments[i] = find_imageview (&rpInfo->attachments[i].view, rctx);
|
|
}
|
|
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
VkFramebuffer framebuffer;
|
|
dfunc->vkCreateFramebuffer (device->dev, &cInfo, 0, &framebuffer);
|
|
QFV_duSetObjectName (device, VK_OBJECT_TYPE_FRAMEBUFFER, framebuffer,
|
|
va (ctx->va_ctx, "framebuffer:%s", rpInfo->name));
|
|
|
|
rp->beginInfo.framebuffer = framebuffer;
|
|
for (uint32_t i = 0; i < rp->subpass_count; i++) {
|
|
__auto_type sp = &rp->subpasses[i];
|
|
sp->inherit.framebuffer = framebuffer;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void
|
|
QFV_Render_Init (vulkan_ctx_t *ctx)
|
|
{
|
|
qfv_renderctx_t *rctx = calloc (1, sizeof (*rctx));
|
|
ctx->render_context = rctx;
|
|
|
|
exprctx_t ectx = { .hashctx = &rctx->hashctx };
|
|
exprsym_t syms[] = { {} };
|
|
rctx->task_functions.symbols = syms;
|
|
cexpr_init_symtab (&rctx->task_functions, &ectx);
|
|
rctx->task_functions.symbols = 0;
|
|
}
|
|
|
|
void
|
|
QFV_Render_Shutdown (vulkan_ctx_t *ctx)
|
|
{
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
__auto_type rctx = ctx->render_context;
|
|
if (rctx->job) {
|
|
__auto_type job = rctx->job;
|
|
QFV_DestroyFramebuffer (ctx); //FIXME do properly
|
|
for (uint32_t i = 0; i < job->num_renderpasses; i++) {
|
|
dfunc->vkDestroyRenderPass (device->dev, job->renderpasses[i], 0);
|
|
}
|
|
for (uint32_t i = 0; i < job->num_pipelines; i++) {
|
|
dfunc->vkDestroyPipeline (device->dev, job->pipelines[i], 0);
|
|
}
|
|
for (uint32_t i = 0; i < job->num_layouts; i++) {
|
|
dfunc->vkDestroyPipelineLayout (device->dev, job->layouts[i], 0);
|
|
}
|
|
if (job->resources) {
|
|
QFV_DestroyResource (ctx->device, job->resources);
|
|
free (job->resources);
|
|
}
|
|
if (job->command_pool) {
|
|
dfunc->vkDestroyCommandPool (device->dev, job->command_pool, 0);
|
|
}
|
|
DARRAY_CLEAR (&job->commands);
|
|
free (rctx->job);
|
|
}
|
|
if (rctx->jobinfo) {
|
|
__auto_type jinfo = rctx->jobinfo;
|
|
for (uint32_t i = 0; i < jinfo->num_descriptorsetlayouts; i++) {
|
|
__auto_type setLayout = jinfo->descriptorsetlayouts[i].setLayout;
|
|
dfunc->vkDestroyDescriptorSetLayout (device->dev, setLayout, 0);
|
|
}
|
|
delete_memsuper (jinfo->memsuper);
|
|
}
|
|
if (rctx->task_functions.tab) {
|
|
Hash_DelTable (rctx->task_functions.tab);
|
|
}
|
|
Hash_DelContext (rctx->hashctx);
|
|
}
|
|
|
|
void
|
|
QFV_Render_AddTasks (vulkan_ctx_t *ctx, exprsym_t *task_syms)
|
|
{
|
|
__auto_type rctx = ctx->render_context;
|
|
exprctx_t ectx = { .hashctx = &rctx->hashctx };
|
|
for (exprsym_t *sym = task_syms; sym->name; sym++) {
|
|
Hash_Add (rctx->task_functions.tab, sym);
|
|
for (exprfunc_t *f = sym->value; f->func; f++) {
|
|
for (int i = 0; i < f->num_params; i++) {
|
|
exprenum_t *e = f->param_types[i]->data;
|
|
if (e && !e->symtab->tab) {
|
|
cexpr_init_symtab (e->symtab, &ectx);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|