Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2016 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
| 7 | |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 8 | #include "src/gpu/vk/GrVkFramebuffer.h" |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 9 | |
Greg Daniel | 38517c2 | 2021-03-29 16:01:19 -0400 | [diff] [blame] | 10 | #include "src/gpu/vk/GrVkAttachment.h" |
Greg Daniel | de4bbdb | 2021-04-13 14:23:23 -0400 | [diff] [blame^] | 11 | #include "src/gpu/vk/GrVkCommandBuffer.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 12 | #include "src/gpu/vk/GrVkGpu.h" |
| 13 | #include "src/gpu/vk/GrVkImageView.h" |
| 14 | #include "src/gpu/vk/GrVkRenderPass.h" |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 15 | |
Greg Daniel | a92d787 | 2021-04-07 11:08:36 -0400 | [diff] [blame] | 16 | GrVkFramebuffer* GrVkFramebuffer::Create( |
| 17 | GrVkGpu* gpu, |
| 18 | int width, int height, |
| 19 | const GrVkRenderPass* renderPass, |
Greg Daniel | de4bbdb | 2021-04-13 14:23:23 -0400 | [diff] [blame^] | 20 | GrVkAttachment* colorAttachment, |
| 21 | GrVkAttachment* resolveAttachment, |
| 22 | GrVkAttachment* stencilAttachment, |
Greg Daniel | a92d787 | 2021-04-07 11:08:36 -0400 | [diff] [blame] | 23 | GrVkResourceProvider::CompatibleRPHandle compatibleRenderPassHandle) { |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 24 | // At the very least we need a renderPass and a colorAttachment |
| 25 | SkASSERT(renderPass); |
| 26 | SkASSERT(colorAttachment); |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 27 | |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 28 | VkImageView attachments[3]; |
Greg Daniel | 38517c2 | 2021-03-29 16:01:19 -0400 | [diff] [blame] | 29 | attachments[0] = colorAttachment->framebufferView()->imageView(); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 30 | int numAttachments = 1; |
Greg Daniel | 7acddf5 | 2020-12-16 15:15:51 -0500 | [diff] [blame] | 31 | if (resolveAttachment) { |
Greg Daniel | 38517c2 | 2021-03-29 16:01:19 -0400 | [diff] [blame] | 32 | attachments[numAttachments++] = resolveAttachment->framebufferView()->imageView(); |
Greg Daniel | 7acddf5 | 2020-12-16 15:15:51 -0500 | [diff] [blame] | 33 | } |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 34 | if (stencilAttachment) { |
Greg Daniel | 38517c2 | 2021-03-29 16:01:19 -0400 | [diff] [blame] | 35 | attachments[numAttachments++] = stencilAttachment->framebufferView()->imageView(); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 36 | } |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 37 | |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 38 | VkFramebufferCreateInfo createInfo; |
| 39 | memset(&createInfo, 0, sizeof(VkFramebufferCreateInfo)); |
| 40 | createInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO; |
| 41 | createInfo.pNext = nullptr; |
| 42 | createInfo.flags = 0; |
| 43 | createInfo.renderPass = renderPass->vkRenderPass(); |
| 44 | createInfo.attachmentCount = numAttachments; |
| 45 | createInfo.pAttachments = attachments; |
| 46 | createInfo.width = width; |
| 47 | createInfo.height = height; |
| 48 | createInfo.layers = 1; |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 49 | |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 50 | VkFramebuffer framebuffer; |
Greg Daniel | fa3adf7 | 2019-11-07 09:53:41 -0500 | [diff] [blame] | 51 | VkResult err; |
| 52 | GR_VK_CALL_RESULT(gpu, err, CreateFramebuffer(gpu->device(), &createInfo, nullptr, |
| 53 | &framebuffer)); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 54 | if (err) { |
| 55 | return nullptr; |
| 56 | } |
| 57 | |
Greg Daniel | 38517c2 | 2021-03-29 16:01:19 -0400 | [diff] [blame] | 58 | return new GrVkFramebuffer(gpu, framebuffer, sk_ref_sp(colorAttachment), |
Greg Daniel | a92d787 | 2021-04-07 11:08:36 -0400 | [diff] [blame] | 59 | sk_ref_sp(resolveAttachment), sk_ref_sp(stencilAttachment), |
| 60 | compatibleRenderPassHandle); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 61 | } |
| 62 | |
Greg Daniel | de4bbdb | 2021-04-13 14:23:23 -0400 | [diff] [blame^] | 63 | GrVkFramebuffer::GrVkFramebuffer(const GrVkGpu* gpu, |
| 64 | VkFramebuffer framebuffer, |
| 65 | sk_sp<GrVkAttachment> colorAttachment, |
| 66 | sk_sp<GrVkAttachment> resolveAttachment, |
| 67 | sk_sp<GrVkAttachment> stencilAttachment, |
| 68 | GrVkResourceProvider::CompatibleRPHandle compatibleRPHandle) |
| 69 | : GrVkManagedResource(gpu) |
| 70 | , fFramebuffer(framebuffer) |
| 71 | , fColorAttachment(std::move(colorAttachment)) |
| 72 | , fResolveAttachment(std::move(resolveAttachment)) |
| 73 | , fStencilAttachment(std::move(stencilAttachment)) |
| 74 | , fCompatibleRenderPassHandle(compatibleRPHandle) {} |
| 75 | |
| 76 | GrVkFramebuffer::GrVkFramebuffer(const GrVkGpu* gpu, |
| 77 | sk_sp<GrVkAttachment> colorAttachment, |
| 78 | sk_sp<const GrVkRenderPass> renderPass, |
| 79 | std::unique_ptr<GrVkSecondaryCommandBuffer> externalCommandBuffer) |
| 80 | : GrVkManagedResource(gpu) |
| 81 | , fColorAttachment(std::move(colorAttachment)) |
| 82 | , fExternalRenderPass(std::move(renderPass)) |
| 83 | , fExternalCommandBuffer(std::move(externalCommandBuffer)) {} |
| 84 | |
Greg Daniel | 38517c2 | 2021-03-29 16:01:19 -0400 | [diff] [blame] | 85 | GrVkFramebuffer::~GrVkFramebuffer() {} |
| 86 | |
Jim Van Verth | 5082df1 | 2020-03-11 16:14:51 -0400 | [diff] [blame] | 87 | void GrVkFramebuffer::freeGPUData() const { |
Greg Daniel | de4bbdb | 2021-04-13 14:23:23 -0400 | [diff] [blame^] | 88 | SkASSERT(this->isExternal() || fFramebuffer != VK_NULL_HANDLE); |
| 89 | if (!this->isExternal()) { |
| 90 | GR_VK_CALL(fGpu->vkInterface(), DestroyFramebuffer(fGpu->device(), fFramebuffer, nullptr)); |
| 91 | } |
| 92 | |
| 93 | // TODO: having freeGPUData virtual on GrManagedResource be const seems like a bad restriction |
| 94 | // since we are changing the internal objects of these classes when it is called. We should go |
| 95 | // back a revisit how much of a headache it would be to make this function non-const |
| 96 | GrVkFramebuffer* nonConstThis = const_cast<GrVkFramebuffer*>(this); |
| 97 | nonConstThis->releaseResources(); |
| 98 | } |
| 99 | |
| 100 | void GrVkFramebuffer::releaseResources() { |
| 101 | if (fExternalCommandBuffer) { |
| 102 | fExternalCommandBuffer->releaseResources(); |
| 103 | fExternalCommandBuffer.reset(); |
| 104 | } |
| 105 | } |
| 106 | |
| 107 | void GrVkFramebuffer::returnExternalGrSecondaryCommandBuffer( |
| 108 | std::unique_ptr<GrVkSecondaryCommandBuffer> cmdBuffer) { |
| 109 | SkASSERT(!fExternalCommandBuffer); |
| 110 | fExternalCommandBuffer = std::move(cmdBuffer); |
| 111 | } |
| 112 | |
| 113 | std::unique_ptr<GrVkSecondaryCommandBuffer> GrVkFramebuffer::externalCommandBuffer() { |
| 114 | SkASSERT(fExternalCommandBuffer); |
| 115 | return std::move(fExternalCommandBuffer); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 116 | } |