Path: blob/21.2-virgl/src/gallium/drivers/zink/zink_render_pass.c
4570 views
/*1* Copyright 2018 Collabora Ltd.2*3* Permission is hereby granted, free of charge, to any person obtaining a4* copy of this software and associated documentation files (the "Software"),5* to deal in the Software without restriction, including without limitation6* on the rights to use, copy, modify, merge, publish, distribute, sub7* license, and/or sell copies of the Software, and to permit persons to whom8* the Software is furnished to do so, subject to the following conditions:9*10* The above copyright notice and this permission notice (including the next11* paragraph) shall be included in all copies or substantial portions of the12* Software.13*14* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR15* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,16* FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL17* THE AUTHOR(S) AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM,18* DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR19* OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE20* USE OR OTHER DEALINGS IN THE SOFTWARE.21*/2223#include "zink_render_pass.h"2425#include "zink_screen.h"2627#include "util/u_memory.h"28#include "util/u_string.h"2930static VkRenderPass31create_render_pass(VkDevice dev, struct zink_render_pass_state *state)32{3334VkAttachmentReference color_refs[PIPE_MAX_COLOR_BUFS], zs_ref;35VkAttachmentDescription attachments[PIPE_MAX_COLOR_BUFS + 1];36VkPipelineStageFlags dep_pipeline = 0;37VkAccessFlags dep_access = 0;3839for (int i = 0; i < state->num_cbufs; i++) {40struct zink_rt_attrib *rt = state->rts + i;41attachments[i].flags = 0;42attachments[i].format = rt->format;43attachments[i].samples = rt->samples;44attachments[i].loadOp = rt->clear_color ? VK_ATTACHMENT_LOAD_OP_CLEAR :45state->swapchain_init && rt->swapchain ?46VK_ATTACHMENT_LOAD_OP_DONT_CARE :47VK_ATTACHMENT_LOAD_OP_LOAD;48attachments[i].storeOp = VK_ATTACHMENT_STORE_OP_STORE;49attachments[i].stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;50attachments[i].stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;51/* if layout changes are ever handled here, need VkAttachmentSampleLocationsEXT */52attachments[i].initialLayout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;53attachments[i].finalLayout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;54color_refs[i].attachment = i;55color_refs[i].layout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;56dep_pipeline |= VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;57dep_access |= VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT;58if (attachments[i].loadOp == VK_ATTACHMENT_LOAD_OP_LOAD)59dep_access |= VK_ACCESS_COLOR_ATTACHMENT_READ_BIT;60}6162int num_attachments = state->num_cbufs;63if (state->have_zsbuf) {64struct zink_rt_attrib *rt = state->rts + state->num_cbufs;65bool has_clear = rt->clear_color || rt->clear_stencil;66VkImageLayout layout = rt->needs_write || has_clear ? VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL : VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL;67attachments[num_attachments].flags = 0;68attachments[num_attachments].format = rt->format;69attachments[num_attachments].samples = rt->samples;70attachments[num_attachments].loadOp = rt->clear_color ? VK_ATTACHMENT_LOAD_OP_CLEAR : VK_ATTACHMENT_LOAD_OP_LOAD;71attachments[num_attachments].storeOp = VK_ATTACHMENT_STORE_OP_STORE;72attachments[num_attachments].stencilLoadOp = rt->clear_stencil ? VK_ATTACHMENT_LOAD_OP_CLEAR : VK_ATTACHMENT_LOAD_OP_LOAD;73attachments[num_attachments].stencilStoreOp = VK_ATTACHMENT_STORE_OP_STORE;74/* if layout changes are ever handled here, need VkAttachmentSampleLocationsEXT */75attachments[num_attachments].initialLayout = layout;76attachments[num_attachments].finalLayout = layout;7778dep_pipeline |= VK_PIPELINE_STAGE_EARLY_FRAGMENT_TESTS_BIT | VK_PIPELINE_STAGE_LATE_FRAGMENT_TESTS_BIT;79if (layout == VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL)80dep_access |= VK_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE_BIT;81if (attachments[num_attachments].loadOp == VK_ATTACHMENT_LOAD_OP_LOAD ||82attachments[num_attachments].stencilLoadOp == VK_ATTACHMENT_LOAD_OP_LOAD)83dep_access |= VK_ACCESS_DEPTH_STENCIL_ATTACHMENT_READ_BIT;8485zs_ref.attachment = num_attachments++;86zs_ref.layout = layout;87}8889VkSubpassDependency deps[] = {90[0] = {VK_SUBPASS_EXTERNAL, 0, dep_pipeline, dep_pipeline, 0, dep_access, VK_DEPENDENCY_BY_REGION_BIT},91[1] = {0, VK_SUBPASS_EXTERNAL, dep_pipeline, VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, dep_access, 0, VK_DEPENDENCY_BY_REGION_BIT}92};9394VkSubpassDescription subpass = {0};95subpass.pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS;96subpass.colorAttachmentCount = state->num_cbufs;97subpass.pColorAttachments = color_refs;98subpass.pDepthStencilAttachment = state->have_zsbuf ? &zs_ref : NULL;99100VkRenderPassCreateInfo rpci = {0};101rpci.sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO;102rpci.attachmentCount = num_attachments;103rpci.pAttachments = attachments;104rpci.subpassCount = 1;105rpci.pSubpasses = &subpass;106rpci.dependencyCount = 2;107rpci.pDependencies = deps;108109VkRenderPass render_pass;110if (vkCreateRenderPass(dev, &rpci, NULL, &render_pass) != VK_SUCCESS) {111debug_printf("vkCreateRenderPass failed\n");112return VK_NULL_HANDLE;113}114115return render_pass;116}117118struct zink_render_pass *119zink_create_render_pass(struct zink_screen *screen,120struct zink_render_pass_state *state)121{122struct zink_render_pass *rp = CALLOC_STRUCT(zink_render_pass);123if (!rp)124goto fail;125126rp->render_pass = create_render_pass(screen->dev, state);127if (!rp->render_pass)128goto fail;129memcpy(&rp->state, state, sizeof(struct zink_render_pass_state));130return rp;131132fail:133if (rp)134zink_destroy_render_pass(screen, rp);135return NULL;136}137138void139zink_destroy_render_pass(struct zink_screen *screen,140struct zink_render_pass *rp)141{142vkDestroyRenderPass(screen->dev, rp->render_pass, NULL);143FREE(rp);144}145146VkImageLayout147zink_render_pass_attachment_get_barrier_info(const struct zink_render_pass *rp, unsigned idx,148VkPipelineStageFlags *pipeline, VkAccessFlags *access)149{150*access = 0;151assert(idx < rp->state.num_rts);152const struct zink_rt_attrib *rt = &rp->state.rts[idx];153if (idx < rp->state.num_cbufs) {154*pipeline = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;155*access |= VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT;156if (!rt->clear_color && (!rp->state.swapchain_init || !rt->swapchain))157*access |= VK_ACCESS_COLOR_ATTACHMENT_READ_BIT;158return VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;159}160161assert(rp->state.have_zsbuf);162*pipeline = VK_PIPELINE_STAGE_EARLY_FRAGMENT_TESTS_BIT | VK_PIPELINE_STAGE_LATE_FRAGMENT_TESTS_BIT;163if (!rp->state.rts[idx].clear_color && !rp->state.rts[idx].clear_stencil)164*access |= VK_ACCESS_DEPTH_STENCIL_ATTACHMENT_READ_BIT;165if (!rp->state.rts[idx].clear_color && !rp->state.rts[idx].clear_stencil && !rp->state.rts[idx].needs_write)166return VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL;167*access |= VK_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE_BIT;168return VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;169}170171172