/* vulkan_main.c Vulkan output Copyright (C) 2022 Bill Currie Author: Bill Currie Date: 2022/11/21 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_STRING_H # include "string.h" #endif #ifdef HAVE_STRINGS_H # include "strings.h" #endif #include "QF/cvar.h" #include "QF/render.h" #include "QF/sys.h" #include "QF/Vulkan/qf_matrices.h" #include "QF/Vulkan/qf_output.h" #include "QF/Vulkan/qf_renderpass.h" #include "QF/Vulkan/qf_vid.h" #include "QF/Vulkan/debug.h" #include "QF/Vulkan/descriptor.h" #include "QF/Vulkan/device.h" #include "QF/Vulkan/image.h" #include "QF/Vulkan/instance.h" #include "QF/Vulkan/swapchain.h" #include "vid_vulkan.h" #include "vkparse.h"//FIXME static void update_input (qfv_renderframe_t *rFrame) { vulkan_ctx_t *ctx = rFrame->vulkan_ctx; qfv_device_t *device = ctx->device; qfv_devfuncs_t *dfunc = device->funcs; outputctx_t *octx = ctx->output_context; uint32_t curFrame = ctx->curFrame; outputframe_t *oframe = &octx->frames.a[curFrame]; qfv_renderpass_t *rp = ctx->output_renderpass; if (rp->attachment_views->a[0] == oframe->view) { return; } oframe->view = rp->attachment_views->a[0]; VkDescriptorImageInfo imageInfo = { octx->sampler, oframe->view, VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL, }; VkWriteDescriptorSet write[] = { { VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET, 0, oframe->set, 0, 0, 1, VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER, &imageInfo, 0, 0 } }; dfunc->vkUpdateDescriptorSets (device->dev, 1, write, 0, 0); } static void output_draw (qfv_renderframe_t *rFrame) { vulkan_ctx_t *ctx = rFrame->vulkan_ctx; qfv_device_t *device = ctx->device; qfv_devfuncs_t *dfunc = device->funcs; outputctx_t *octx = ctx->output_context; outputframe_t *oframe = &octx->frames.a[ctx->curFrame]; VkCommandBuffer cmd = oframe->cmd; DARRAY_APPEND (&rFrame->subpassCmdSets[0], cmd); dfunc->vkResetCommandBuffer (cmd, 0); VkCommandBufferInheritanceInfo inherit = { VK_STRUCTURE_TYPE_COMMAND_BUFFER_INHERITANCE_INFO, 0, rFrame->renderpass->renderpass, 0, rFrame->framebuffer, 0, 0, 0, }; VkCommandBufferBeginInfo beginInfo = { VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO, 0, VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT | VK_COMMAND_BUFFER_USAGE_RENDER_PASS_CONTINUE_BIT, &inherit, }; dfunc->vkBeginCommandBuffer (cmd, &beginInfo); QFV_duCmdBeginLabel (device, cmd, "output:output"); dfunc->vkCmdBindPipeline (cmd, VK_PIPELINE_BIND_POINT_GRAPHICS, octx->pipeline); dfunc->vkCmdSetViewport (cmd, 0, 1, &rFrame->renderpass->viewport); dfunc->vkCmdSetScissor (cmd, 0, 1, &rFrame->renderpass->scissor); VkDescriptorSet set[] = { Vulkan_Matrix_Descriptors (ctx, ctx->curFrame), oframe->set, }; dfunc->vkCmdBindDescriptorSets (cmd, VK_PIPELINE_BIND_POINT_GRAPHICS, octx->layout, 0, 2, set, 0, 0); dfunc->vkCmdDraw (cmd, 3, 1, 0, 0); QFV_duCmdEndLabel (device, cmd); dfunc->vkEndCommandBuffer (cmd); } void Vulkan_Output_CreateRenderPasses (vulkan_ctx_t *ctx) { outputctx_t *octx = calloc (1, sizeof (outputctx_t)); ctx->output_context = octx; qfv_output_t output = { .extent = ctx->swapchain->extent, .view = ctx->swapchain->imageViews->a[0], .format = ctx->swapchain->format, .frames = ctx->swapchain->numImages, .view_list = ctx->swapchain->imageViews->a, }; __auto_type out = Vulkan_CreateRenderPass (ctx, "output", &output, output_draw); ctx->output_renderpass = out; out->order = QFV_rp_output; octx->output = (qfv_output_t) { .extent = ctx->swapchain->extent, .view = out->attachment_views->a[0], .format = VK_FORMAT_R16G16B16A16_SFLOAT,//FIXME .frames = ctx->swapchain->numImages, .view_list = malloc (sizeof (VkImageView) * ctx->swapchain->numImages), }; for (int i = 0; i < ctx->swapchain->numImages; i++) { octx->output.view_list[i] = octx->output.view; } DARRAY_APPEND (&ctx->renderPasses, out); __auto_type pre = Vulkan_CreateFunctionPass (ctx, "preoutput", update_input); pre->order = QFV_rp_preoutput; DARRAY_APPEND (&ctx->renderPasses, pre); } void Vulkan_Output_Init (vulkan_ctx_t *ctx) { qfv_device_t *device = ctx->device; qfvPushDebug (ctx, "output init"); outputctx_t *octx = ctx->output_context; size_t frames = ctx->frames.size; DARRAY_INIT (&octx->frames, frames); DARRAY_RESIZE (&octx->frames, frames); octx->frames.grow = 0; __auto_type pld = ctx->script_context->pipelineDef;//FIXME ctx->script_context->pipelineDef = Vulkan_GetConfig (ctx, "qf_output"); octx->pipeline = Vulkan_CreateGraphicsPipeline (ctx, "output"); octx->layout = Vulkan_CreatePipelineLayout (ctx, "output_layout"); octx->sampler = Vulkan_CreateSampler (ctx, "linear"); __auto_type layouts = QFV_AllocDescriptorSetLayoutSet (frames, alloca); layouts->a[0] = Vulkan_CreateDescriptorSetLayout (ctx, "output_set"); for (size_t i = 0; i < frames; i++) { layouts->a[i] = layouts->a[0]; } __auto_type pool = Vulkan_CreateDescriptorPool (ctx, "output_pool"); __auto_type sets = QFV_AllocateDescriptorSet (device, pool, layouts); __auto_type cmdSet = QFV_AllocCommandBufferSet (1, alloca); for (size_t i = 0; i < frames; i++) { __auto_type oframe = &octx->frames.a[i]; oframe->set = sets->a[i]; QFV_AllocateCommandBuffers (device, ctx->cmdpool, 1, cmdSet); oframe->cmd = cmdSet->a[0]; QFV_duSetObjectName (device, VK_OBJECT_TYPE_COMMAND_BUFFER, oframe->cmd, "cmd:output"); } ctx->script_context->pipelineDef = pld; free (sets); qfvPopDebug (ctx); } void Vulkan_Output_Shutdown (vulkan_ctx_t *ctx) { qfv_device_t *device = ctx->device; qfv_devfuncs_t *dfunc = device->funcs; outputctx_t *octx = ctx->output_context; dfunc->vkDestroyPipeline (device->dev, octx->pipeline, 0); free (octx->frames.a); free (octx); } qfv_output_t * Vulkan_Output_Get (vulkan_ctx_t *ctx) { outputctx_t *octx = ctx->output_context; return &octx->output; }