blob: ca9f7246c0aa8909a2f74986e7dd2adc67cc072b [file] [log] [blame]
Jamie Madill9e54b5a2016-05-25 12:57:39 -04001//
2// Copyright 2016 The ANGLE Project Authors. All rights reserved.
3// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6// FramebufferVk.cpp:
7// Implements the class methods for FramebufferVk.
8//
9
10#include "libANGLE/renderer/vulkan/FramebufferVk.h"
11
Jamie Madill7b57b9d2017-01-13 09:33:38 -050012#include <vulkan/vulkan.h>
Jamie Madill231c7f52017-04-26 13:45:37 -040013#include <array>
Jamie Madill7b57b9d2017-01-13 09:33:38 -050014
Jamie Madill9e54b5a2016-05-25 12:57:39 -040015#include "common/debug.h"
Jamie Madill7b57b9d2017-01-13 09:33:38 -050016#include "image_util/imageformats.h"
Jamie Madillc564c072017-06-01 12:45:42 -040017#include "libANGLE/Context.h"
18#include "libANGLE/Display.h"
Jamie Madill7b57b9d2017-01-13 09:33:38 -050019#include "libANGLE/formatutils.h"
20#include "libANGLE/renderer/renderer_utils.h"
Jamie Madill1f46bc12018-02-20 16:09:43 -050021#include "libANGLE/renderer/vulkan/CommandGraph.h"
Jamie Madill7b57b9d2017-01-13 09:33:38 -050022#include "libANGLE/renderer/vulkan/ContextVk.h"
Jamie Madill5deea722017-02-16 10:44:46 -050023#include "libANGLE/renderer/vulkan/DisplayVk.h"
Jamie Madill7b57b9d2017-01-13 09:33:38 -050024#include "libANGLE/renderer/vulkan/RenderTargetVk.h"
25#include "libANGLE/renderer/vulkan/RendererVk.h"
26#include "libANGLE/renderer/vulkan/SurfaceVk.h"
Jamie Madill3c424b42018-01-19 12:35:09 -050027#include "libANGLE/renderer/vulkan/vk_format_utils.h"
Jamie Madill9e54b5a2016-05-25 12:57:39 -040028
29namespace rx
30{
31
Jamie Madill7b57b9d2017-01-13 09:33:38 -050032namespace
33{
Jamie Madill66546be2018-03-08 09:47:20 -050034const gl::InternalFormat &GetReadAttachmentInfo(const gl::Context *context,
35 RenderTargetVk *renderTarget)
Jamie Madill7b57b9d2017-01-13 09:33:38 -050036{
Jamie Madill1d7be502017-10-29 18:06:50 -040037 GLenum implFormat = renderTarget->format->textureFormat().fboImplementationInternalFormat;
Jamie Madill66546be2018-03-08 09:47:20 -050038 return gl::GetSizedInternalFormatInfo(implFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -050039}
Jamie Madill7b57b9d2017-01-13 09:33:38 -050040} // anonymous namespace
41
42// static
43FramebufferVk *FramebufferVk::CreateUserFBO(const gl::FramebufferState &state)
44{
45 return new FramebufferVk(state);
46}
47
48// static
49FramebufferVk *FramebufferVk::CreateDefaultFBO(const gl::FramebufferState &state,
50 WindowSurfaceVk *backbuffer)
51{
52 return new FramebufferVk(state, backbuffer);
53}
54
Jamie Madillab9f9c32017-01-17 17:47:34 -050055FramebufferVk::FramebufferVk(const gl::FramebufferState &state)
Jamie Madill49ac74b2017-12-21 14:42:33 -050056 : FramebufferImpl(state),
57 mBackbuffer(nullptr),
58 mRenderPassDesc(),
59 mFramebuffer(),
Jamie Madill47c8ea32018-01-03 18:27:59 -050060 mLastRenderNodeSerial()
Jamie Madill9e54b5a2016-05-25 12:57:39 -040061{
62}
63
Jamie Madill7b57b9d2017-01-13 09:33:38 -050064FramebufferVk::FramebufferVk(const gl::FramebufferState &state, WindowSurfaceVk *backbuffer)
Jamie Madill49ac74b2017-12-21 14:42:33 -050065 : FramebufferImpl(state),
66 mBackbuffer(backbuffer),
67 mRenderPassDesc(),
68 mFramebuffer(),
Jamie Madill47c8ea32018-01-03 18:27:59 -050069 mLastRenderNodeSerial()
Jamie Madill7b57b9d2017-01-13 09:33:38 -050070{
71}
72
Jamie Madill9e54b5a2016-05-25 12:57:39 -040073FramebufferVk::~FramebufferVk()
74{
75}
76
Jamie Madillc564c072017-06-01 12:45:42 -040077void FramebufferVk::destroy(const gl::Context *context)
Jamie Madill5deea722017-02-16 10:44:46 -050078{
Jamie Madille1f3ad42017-10-28 23:00:42 -040079 RendererVk *renderer = vk::GetImpl(context)->getRenderer();
Jamie Madill5deea722017-02-16 10:44:46 -050080
Jamie Madill526543c2017-10-28 10:59:16 -040081 renderer->releaseResource(*this, &mFramebuffer);
Jamie Madill5deea722017-02-16 10:44:46 -050082}
83
Jamie Madillc564c072017-06-01 12:45:42 -040084void FramebufferVk::destroyDefault(const egl::Display *display)
Jamie Madill5deea722017-02-16 10:44:46 -050085{
Jamie Madille1f3ad42017-10-28 23:00:42 -040086 VkDevice device = vk::GetImpl(display)->getRenderer()->getDevice();
Jamie Madill5deea722017-02-16 10:44:46 -050087
Jamie Madill5deea722017-02-16 10:44:46 -050088 mFramebuffer.destroy(device);
89}
90
Jamie Madill4928b7c2017-06-20 12:57:39 -040091gl::Error FramebufferVk::discard(const gl::Context *context,
92 size_t count,
93 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040094{
95 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050096 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040097}
98
Jamie Madill4928b7c2017-06-20 12:57:39 -040099gl::Error FramebufferVk::invalidate(const gl::Context *context,
100 size_t count,
101 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400102{
103 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500104 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400105}
106
Jamie Madill4928b7c2017-06-20 12:57:39 -0400107gl::Error FramebufferVk::invalidateSub(const gl::Context *context,
108 size_t count,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400109 const GLenum *attachments,
110 const gl::Rectangle &area)
111{
112 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500113 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400114}
115
Jamie Madillc564c072017-06-01 12:45:42 -0400116gl::Error FramebufferVk::clear(const gl::Context *context, GLbitfield mask)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400117{
Jamie Madill0cec82a2018-03-14 09:21:07 -0400118 ContextVk *contextVk = vk::GetImpl(context);
119 RendererVk *renderer = contextVk->getRenderer();
120 Serial currentSerial = renderer->getCurrentQueueSerial();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500121
Jamie Madill0cec82a2018-03-14 09:21:07 -0400122 // This command buffer is only started once.
123 vk::CommandBuffer *commandBuffer = nullptr;
124 vk::CommandGraphNode *writingNode = nullptr;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500125
Jamie Madill0cec82a2018-03-14 09:21:07 -0400126 const gl::FramebufferAttachment *depthAttachment = mState.getDepthAttachment();
127 bool clearDepth = (depthAttachment && (mask & GL_DEPTH_BUFFER_BIT) != 0);
128 ASSERT(!clearDepth || depthAttachment->isAttached());
129
130 const gl::FramebufferAttachment *stencilAttachment = mState.getStencilAttachment();
131 bool clearStencil = (stencilAttachment && (mask & GL_STENCIL_BUFFER_BIT) != 0);
132 ASSERT(!clearStencil || stencilAttachment->isAttached());
133
134 // Depth/stencil clear.
135 if (clearDepth || clearStencil)
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500136 {
Jamie Madill0cec82a2018-03-14 09:21:07 -0400137 ANGLE_TRY(beginWriteResource(renderer, &commandBuffer));
138 writingNode = getCurrentWritingNode(currentSerial);
139
140 const VkClearDepthStencilValue &clearDepthStencilValue =
141 contextVk->getClearDepthStencilValue().depthStencil;
142
143 // We only support packed depth/stencil, not separate.
144 ASSERT(!(clearDepth && clearStencil) || mState.getDepthStencilAttachment());
145
Luc Ferrone6a40d02018-03-22 10:30:57 -0400146 const VkImageAspectFlags aspectFlags =
147 (depthAttachment ? VK_IMAGE_ASPECT_DEPTH_BIT : 0) |
148 (stencilAttachment ? VK_IMAGE_ASPECT_STENCIL_BIT : 0);
Jamie Madill0cec82a2018-03-14 09:21:07 -0400149
150 RenderTargetVk *renderTarget = mRenderTargetCache.getDepthStencil();
151 renderTarget->resource->onWriteResource(writingNode, currentSerial);
152 renderTarget->image->changeLayoutWithStages(
153 aspectFlags, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
154 VK_PIPELINE_STAGE_TRANSFER_BIT, commandBuffer);
155
156 commandBuffer->clearSingleDepthStencilImage(*renderTarget->image, aspectFlags,
157 clearDepthStencilValue);
158
159 if ((mask & GL_COLOR_BUFFER_BIT) == 0)
160 {
161 return gl::NoError();
162 }
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500163 }
164
Luc Ferron5242d5b2018-02-15 07:14:35 -0500165 if (context->getGLState().isScissorTestEnabled())
166 {
167 // With scissor test enabled, we clear very differently and we don't need to access
168 // the image inside each attachment we can just use clearCmdAttachments with our
169 // scissor region instead.
170 ANGLE_TRY(clearColorAttachmentsWithScissorRegion(context));
171 return gl::NoError();
172 }
173
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500174 const auto *attachment = mState.getFirstNonNullAttachment();
175 ASSERT(attachment && attachment->isAttached());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500176
Jamie Madill0cec82a2018-03-14 09:21:07 -0400177 if (!commandBuffer)
178 {
179 ANGLE_TRY(beginWriteResource(renderer, &commandBuffer));
180 writingNode = getCurrentWritingNode(currentSerial);
181 }
Jamie Madillefb5a5c2018-01-29 15:56:59 -0500182
Jamie Madill66546be2018-03-08 09:47:20 -0500183 // TODO(jmadill): Support gaps in RenderTargets. http://anglebug.com/2394
184 const auto &colorRenderTargets = mRenderTargetCache.getColors();
185 for (size_t colorIndex : mState.getEnabledDrawBuffers())
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500186 {
Jamie Madill66546be2018-03-08 09:47:20 -0500187 RenderTargetVk *colorRenderTarget = colorRenderTargets[colorIndex];
188 ASSERT(colorRenderTarget);
Jamie Madill0cec82a2018-03-14 09:21:07 -0400189 colorRenderTarget->resource->onWriteResource(writingNode, currentSerial);
Jamie Madill47c8ea32018-01-03 18:27:59 -0500190
Jamie Madill66546be2018-03-08 09:47:20 -0500191 colorRenderTarget->image->changeLayoutWithStages(
192 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
193 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT, VK_PIPELINE_STAGE_TRANSFER_BIT, commandBuffer);
Jamie Madillefb5a5c2018-01-29 15:56:59 -0500194
Jamie Madill66546be2018-03-08 09:47:20 -0500195 commandBuffer->clearSingleColorImage(*colorRenderTarget->image,
196 contextVk->getClearColorValue().color);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500197 }
198
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500199 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400200}
201
Jamie Madillc564c072017-06-01 12:45:42 -0400202gl::Error FramebufferVk::clearBufferfv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400203 GLenum buffer,
204 GLint drawbuffer,
205 const GLfloat *values)
206{
207 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500208 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400209}
210
Jamie Madillc564c072017-06-01 12:45:42 -0400211gl::Error FramebufferVk::clearBufferuiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400212 GLenum buffer,
213 GLint drawbuffer,
214 const GLuint *values)
215{
216 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500217 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400218}
219
Jamie Madillc564c072017-06-01 12:45:42 -0400220gl::Error FramebufferVk::clearBufferiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400221 GLenum buffer,
222 GLint drawbuffer,
223 const GLint *values)
224{
225 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500226 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400227}
228
Jamie Madillc564c072017-06-01 12:45:42 -0400229gl::Error FramebufferVk::clearBufferfi(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400230 GLenum buffer,
231 GLint drawbuffer,
232 GLfloat depth,
233 GLint stencil)
234{
235 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500236 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400237}
238
Jamie Madill4928b7c2017-06-20 12:57:39 -0400239GLenum FramebufferVk::getImplementationColorReadFormat(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400240{
Jamie Madill66546be2018-03-08 09:47:20 -0500241 return GetReadAttachmentInfo(context, mRenderTargetCache.getColorRead(mState)).format;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400242}
243
Jamie Madill4928b7c2017-06-20 12:57:39 -0400244GLenum FramebufferVk::getImplementationColorReadType(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400245{
Jamie Madill66546be2018-03-08 09:47:20 -0500246 return GetReadAttachmentInfo(context, mRenderTargetCache.getColorRead(mState)).type;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400247}
248
Jamie Madillc564c072017-06-01 12:45:42 -0400249gl::Error FramebufferVk::readPixels(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400250 const gl::Rectangle &area,
251 GLenum format,
252 GLenum type,
Jamie Madilld4826152017-09-21 11:18:59 -0400253 void *pixels)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400254{
Jamie Madill66546be2018-03-08 09:47:20 -0500255 const gl::State &glState = context->getGLState();
Jamie Madille1f3ad42017-10-28 23:00:42 -0400256 ContextVk *contextVk = vk::GetImpl(context);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500257 RendererVk *renderer = contextVk->getRenderer();
Jamie Madill5deea722017-02-16 10:44:46 -0500258 VkDevice device = renderer->getDevice();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500259
Jamie Madill66546be2018-03-08 09:47:20 -0500260 RenderTargetVk *renderTarget = mRenderTargetCache.getColorRead(mState);
261 ASSERT(renderTarget);
262
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500263 vk::Image *readImage = renderTarget->image;
264 vk::StagingImage stagingImage;
Jamie Madill57dd97a2018-02-06 17:10:49 -0500265 ANGLE_TRY(stagingImage.init(contextVk, TextureDimension::TEX_2D, *renderTarget->format,
Luc Ferron33140402018-03-08 13:57:52 -0500266 gl::Extents(area.width, area.height, 1), vk::StagingUsage::Read));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500267
Jamie Madill49ac74b2017-12-21 14:42:33 -0500268 vk::CommandBuffer *commandBuffer = nullptr;
Jamie Madill1f46bc12018-02-20 16:09:43 -0500269 ANGLE_TRY(beginWriteResource(renderer, &commandBuffer));
Jamie Madilld4826152017-09-21 11:18:59 -0400270
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500271 stagingImage.getImage().changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_GENERAL,
272 commandBuffer);
273
Jamie Madilld33c77c2017-11-09 13:08:30 -0500274 readImage->changeLayoutWithStages(
275 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
276 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT, VK_PIPELINE_STAGE_TRANSFER_BIT, commandBuffer);
Jamie Madill815a6c92017-10-21 14:33:04 -0400277
278 VkImageCopy region;
279 region.srcSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
280 region.srcSubresource.mipLevel = 0;
281 region.srcSubresource.baseArrayLayer = 0;
282 region.srcSubresource.layerCount = 1;
283 region.srcOffset.x = area.x;
284 region.srcOffset.y = area.y;
285 region.srcOffset.z = 0;
286 region.dstSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
287 region.dstSubresource.mipLevel = 0;
288 region.dstSubresource.baseArrayLayer = 0;
289 region.dstSubresource.layerCount = 1;
290 region.dstOffset.x = 0;
291 region.dstOffset.y = 0;
292 region.dstOffset.z = 0;
293 region.extent.width = area.width;
294 region.extent.height = area.height;
295 region.extent.depth = 1;
296
297 commandBuffer->copyImage(*readImage, stagingImage.getImage(), 1, &region);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500298
Jamie Madill49ac74b2017-12-21 14:42:33 -0500299 // Triggers a full finish.
300 // TODO(jmadill): Don't block on asynchronous readback.
301 ANGLE_TRY(renderer->finish(context));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500302
303 // TODO(jmadill): parameters
304 uint8_t *mapPointer = nullptr;
Jamie Madill5deea722017-02-16 10:44:46 -0500305 ANGLE_TRY(
306 stagingImage.getDeviceMemory().map(device, 0, stagingImage.getSize(), 0, &mapPointer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500307
Jamie Madill1d7be502017-10-29 18:06:50 -0400308 const auto &angleFormat = renderTarget->format->textureFormat();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500309
310 // TODO(jmadill): Use pixel bytes from the ANGLE format directly.
Geoff Langca271392017-04-05 12:30:00 -0400311 const auto &glFormat = gl::GetSizedInternalFormatInfo(angleFormat.glInternalFormat);
Luc Ferron60284222018-03-20 16:01:44 -0400312 int outputPitch = glFormat.pixelBytes * area.width;
313
314 // Get the staging image pitch and use it to pack the pixels later.
315 VkSubresourceLayout subresourceLayout;
316 stagingImage.getImage().getSubresourceLayout(device, VK_IMAGE_ASPECT_COLOR_BIT, 0, 0,
317 &subresourceLayout);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500318
319 PackPixelsParams params;
320 params.area = area;
321 params.format = format;
322 params.type = type;
Luc Ferron60284222018-03-20 16:01:44 -0400323 params.outputPitch = outputPitch;
Corentin Wallez336129f2017-10-17 15:55:40 -0400324 params.packBuffer = glState.getTargetBuffer(gl::BufferBinding::PixelPack);
Corentin Wallezcda6af12017-10-30 19:20:37 -0400325 params.pack = glState.getPackState();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500326
Luc Ferron60284222018-03-20 16:01:44 -0400327 PackPixels(params, angleFormat, static_cast<int>(subresourceLayout.rowPitch), mapPointer,
328 reinterpret_cast<uint8_t *>(pixels));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500329
Jamie Madill5deea722017-02-16 10:44:46 -0500330 stagingImage.getDeviceMemory().unmap(device);
Jamie Madille88ec8e2017-10-31 17:18:14 -0400331 renderer->releaseObject(renderer->getCurrentQueueSerial(), &stagingImage);
Jamie Madillf651c772017-02-21 15:03:51 -0500332
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500333 return vk::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400334}
335
Jamie Madillc564c072017-06-01 12:45:42 -0400336gl::Error FramebufferVk::blit(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400337 const gl::Rectangle &sourceArea,
338 const gl::Rectangle &destArea,
339 GLbitfield mask,
340 GLenum filter)
341{
342 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500343 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400344}
345
Kenneth Russellce8602a2017-10-03 18:23:08 -0700346bool FramebufferVk::checkStatus(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400347{
Jamie Madillb79e7bb2017-10-24 13:55:50 -0400348 return true;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400349}
350
Jamie Madill19fa1c62018-03-08 09:47:21 -0500351gl::Error FramebufferVk::syncState(const gl::Context *context,
352 const gl::Framebuffer::DirtyBits &dirtyBits)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400353{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400354 ContextVk *contextVk = vk::GetImpl(context);
Jamie Madill7bd16662017-10-28 19:40:50 -0400355 RendererVk *renderer = contextVk->getRenderer();
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400356
357 ASSERT(dirtyBits.any());
Jamie Madill19fa1c62018-03-08 09:47:21 -0500358 ANGLE_TRY(mRenderTargetCache.update(context, mState, dirtyBits));
Jamie Madill66546be2018-03-08 09:47:20 -0500359
Jamie Madill9f2a8612017-11-30 12:43:09 -0500360 mRenderPassDesc.reset();
Jamie Madill7bd16662017-10-28 19:40:50 -0400361 renderer->releaseResource(*this, &mFramebuffer);
Jamie Madill49ac74b2017-12-21 14:42:33 -0500362
363 // Trigger a new set of secondary commands next time we render to this FBO,.
Jamie Madill47c8ea32018-01-03 18:27:59 -0500364 mLastRenderNodeSerial = Serial();
Jamie Madill72106562017-03-24 14:18:50 -0400365
Jamie Madill72106562017-03-24 14:18:50 -0400366 contextVk->invalidateCurrentPipeline();
Jamie Madill19fa1c62018-03-08 09:47:21 -0500367
368 return gl::NoError();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500369}
370
Jamie Madill9f2a8612017-11-30 12:43:09 -0500371const vk::RenderPassDesc &FramebufferVk::getRenderPassDesc(const gl::Context *context)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500372{
Jamie Madill9f2a8612017-11-30 12:43:09 -0500373 if (mRenderPassDesc.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500374 {
Jamie Madill9f2a8612017-11-30 12:43:09 -0500375 return mRenderPassDesc.value();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500376 }
377
Jamie Madill0b684ce2017-11-23 12:57:39 -0500378 vk::RenderPassDesc desc;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500379
Jamie Madill66546be2018-03-08 09:47:20 -0500380 // TODO(jmadill): Support gaps in RenderTargets. http://anglebug.com/2394
381 const auto &colorRenderTargets = mRenderTargetCache.getColors();
382 for (size_t colorIndex : mState.getEnabledDrawBuffers())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500383 {
Jamie Madill66546be2018-03-08 09:47:20 -0500384 RenderTargetVk *colorRenderTarget = colorRenderTargets[colorIndex];
385 ASSERT(colorRenderTarget);
386 desc.packColorAttachment(*colorRenderTarget->format, colorRenderTarget->samples);
Jamie Madillab9f9c32017-01-17 17:47:34 -0500387 }
388
Jamie Madill66546be2018-03-08 09:47:20 -0500389 RenderTargetVk *depthStencilRenderTarget = mRenderTargetCache.getDepthStencil();
390 if (depthStencilRenderTarget)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500391 {
Jamie Madill66546be2018-03-08 09:47:20 -0500392 desc.packDepthStencilAttachment(*depthStencilRenderTarget->format,
393 depthStencilRenderTarget->samples);
Jamie Madillab9f9c32017-01-17 17:47:34 -0500394 }
395
Jamie Madill9f2a8612017-11-30 12:43:09 -0500396 mRenderPassDesc = desc;
397 return mRenderPassDesc.value();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500398}
399
Jamie Madill4928b7c2017-06-20 12:57:39 -0400400gl::ErrorOrResult<vk::Framebuffer *> FramebufferVk::getFramebuffer(const gl::Context *context,
Jamie Madill9f2a8612017-11-30 12:43:09 -0500401 RendererVk *rendererVk)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500402{
403 // If we've already created our cached Framebuffer, return it.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400404 if (mFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500405 {
406 return &mFramebuffer;
407 }
408
Jamie Madill9f2a8612017-11-30 12:43:09 -0500409 const vk::RenderPassDesc &desc = getRenderPassDesc(context);
410
Jamie Madillab9f9c32017-01-17 17:47:34 -0500411 vk::RenderPass *renderPass = nullptr;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500412 ANGLE_TRY(rendererVk->getCompatibleRenderPass(desc, &renderPass));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500413
414 // If we've a Framebuffer provided by a Surface (default FBO/backbuffer), query it.
Jamie Madill9f2a8612017-11-30 12:43:09 -0500415 VkDevice device = rendererVk->getDevice();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500416 if (mBackbuffer)
417 {
418 return mBackbuffer->getCurrentFramebuffer(device, *renderPass);
419 }
420
421 // Gather VkImageViews over all FBO attachments, also size of attached region.
422 std::vector<VkImageView> attachments;
423 gl::Extents attachmentsSize;
424
Jamie Madill66546be2018-03-08 09:47:20 -0500425 // TODO(jmadill): Support gaps in RenderTargets. http://anglebug.com/2394
426 const auto &colorRenderTargets = mRenderTargetCache.getColors();
427 for (size_t colorIndex : mState.getEnabledDrawBuffers())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500428 {
Jamie Madill66546be2018-03-08 09:47:20 -0500429 RenderTargetVk *colorRenderTarget = colorRenderTargets[colorIndex];
430 ASSERT(colorRenderTarget);
431 attachments.push_back(colorRenderTarget->imageView->getHandle());
Jamie Madillab9f9c32017-01-17 17:47:34 -0500432
Jamie Madill66546be2018-03-08 09:47:20 -0500433 ASSERT(attachmentsSize.empty() || attachmentsSize == colorRenderTarget->extents);
434 attachmentsSize = colorRenderTarget->extents;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500435 }
436
Jamie Madill66546be2018-03-08 09:47:20 -0500437 RenderTargetVk *depthStencilRenderTarget = mRenderTargetCache.getDepthStencil();
438 if (depthStencilRenderTarget)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500439 {
Jamie Madill66546be2018-03-08 09:47:20 -0500440 attachments.push_back(depthStencilRenderTarget->imageView->getHandle());
Jamie Madillab9f9c32017-01-17 17:47:34 -0500441
Jamie Madill66546be2018-03-08 09:47:20 -0500442 ASSERT(attachmentsSize.empty() || attachmentsSize == depthStencilRenderTarget->extents);
443 attachmentsSize = depthStencilRenderTarget->extents;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500444 }
445
446 ASSERT(!attachments.empty());
447
448 VkFramebufferCreateInfo framebufferInfo;
449
450 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
451 framebufferInfo.pNext = nullptr;
452 framebufferInfo.flags = 0;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500453 framebufferInfo.renderPass = renderPass->getHandle();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500454 framebufferInfo.attachmentCount = static_cast<uint32_t>(attachments.size());
455 framebufferInfo.pAttachments = attachments.data();
456 framebufferInfo.width = static_cast<uint32_t>(attachmentsSize.width);
457 framebufferInfo.height = static_cast<uint32_t>(attachmentsSize.height);
458 framebufferInfo.layers = 1;
459
Jamie Madill25301b62017-10-28 20:59:31 -0400460 ANGLE_TRY(mFramebuffer.init(device, framebufferInfo));
Jamie Madill5deea722017-02-16 10:44:46 -0500461
Jamie Madillab9f9c32017-01-17 17:47:34 -0500462 return &mFramebuffer;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400463}
464
Luc Ferron5242d5b2018-02-15 07:14:35 -0500465gl::Error FramebufferVk::clearColorAttachmentsWithScissorRegion(const gl::Context *context)
466{
467 ContextVk *contextVk = vk::GetImpl(context);
468 RendererVk *renderer = contextVk->getRenderer();
469
470 // This command can only happen inside a render pass, so obtain one if its already happening
471 // or create a new one if not.
472 vk::CommandGraphNode *node = nullptr;
473 vk::CommandBuffer *commandBuffer = nullptr;
474 ANGLE_TRY(getCommandGraphNodeForDraw(context, &node));
475 if (node->getInsideRenderPassCommands()->valid())
476 {
477 commandBuffer = node->getInsideRenderPassCommands();
478 }
479 else
480 {
481 ANGLE_TRY(node->beginInsideRenderPassRecording(renderer, &commandBuffer));
482 }
483
484 const std::vector<gl::FramebufferAttachment> &colorAttachments = mState.getColorAttachments();
485 gl::AttachmentArray<VkClearAttachment> clearAttachments;
486 int clearAttachmentIndex = 0;
487 for (auto colorIndex : mState.getEnabledDrawBuffers())
488 {
489 VkClearAttachment &clearAttachment = clearAttachments[clearAttachmentIndex];
490 clearAttachment.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
491 clearAttachment.colorAttachment = static_cast<uint32_t>(colorIndex);
492 clearAttachment.clearValue = contextVk->getClearColorValue();
493 ++clearAttachmentIndex;
494 }
495
496 // We assume for now that we always need to clear only 1 layer starting at the
497 // baseArrayLayer 0, this might need to change depending how we'll implement
498 // cube maps, 3d textures and array textures.
499 VkClearRect clearRect;
500 clearRect.baseArrayLayer = 0;
501 clearRect.layerCount = 1;
502 clearRect.rect = contextVk->getScissor();
503
504 commandBuffer->clearAttachments(static_cast<uint32_t>(colorAttachments.size()),
505 clearAttachments.data(), 1, &clearRect);
506 return gl::NoError();
507}
508
JiangYizhoubddc46b2016-12-09 09:50:51 +0800509gl::Error FramebufferVk::getSamplePosition(size_t index, GLfloat *xy) const
510{
511 UNIMPLEMENTED();
512 return gl::InternalError() << "getSamplePosition is unimplemented.";
513}
514
Jamie Madille4c5a232018-03-02 21:00:31 -0500515gl::Error FramebufferVk::getCommandGraphNodeForDraw(const gl::Context *context,
516 vk::CommandGraphNode **nodeOut)
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500517{
Jamie Madill49ac74b2017-12-21 14:42:33 -0500518 ContextVk *contextVk = vk::GetImpl(context);
519 RendererVk *renderer = contextVk->getRenderer();
520 Serial currentSerial = renderer->getCurrentQueueSerial();
Jamie Madillbef918c2017-12-13 13:11:30 -0500521
Jamie Madill1f46bc12018-02-20 16:09:43 -0500522 if (hasCurrentWritingNode(currentSerial) && mLastRenderNodeSerial == currentSerial)
Jamie Madill4c26fc22017-02-24 11:04:10 -0500523 {
Jamie Madill1f46bc12018-02-20 16:09:43 -0500524 *nodeOut = getCurrentWritingNode(currentSerial);
Jamie Madill49ac74b2017-12-21 14:42:33 -0500525 ASSERT((*nodeOut)->getInsideRenderPassCommands()->valid());
526 return gl::NoError();
Jamie Madill4c26fc22017-02-24 11:04:10 -0500527 }
528
Jamie Madill1f46bc12018-02-20 16:09:43 -0500529 vk::CommandGraphNode *node = getNewWritingNode(renderer);
Jamie Madill4c26fc22017-02-24 11:04:10 -0500530
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500531 vk::Framebuffer *framebuffer = nullptr;
Jamie Madill49ac74b2017-12-21 14:42:33 -0500532 ANGLE_TRY_RESULT(getFramebuffer(context, renderer), framebuffer);
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500533
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500534 std::vector<VkClearValue> attachmentClearValues;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500535
Jamie Madille4c5a232018-03-02 21:00:31 -0500536 vk::CommandBuffer *commandBuffer = nullptr;
537 ANGLE_TRY(node->beginOutsideRenderPassRecording(renderer->getDevice(),
538 renderer->getCommandPool(), &commandBuffer));
539
Jamie Madill49ac74b2017-12-21 14:42:33 -0500540 // Initialize RenderPass info.
Jamie Madill66546be2018-03-08 09:47:20 -0500541 // TODO(jmadill): Support gaps in RenderTargets. http://anglebug.com/2394
542 const auto &colorRenderTargets = mRenderTargetCache.getColors();
543 for (size_t colorIndex : mState.getEnabledDrawBuffers())
Jamie Madill4c26fc22017-02-24 11:04:10 -0500544 {
Jamie Madill66546be2018-03-08 09:47:20 -0500545 RenderTargetVk *colorRenderTarget = colorRenderTargets[colorIndex];
546 ASSERT(colorRenderTarget);
Jamie Madill49ac74b2017-12-21 14:42:33 -0500547
Jamie Madille4c5a232018-03-02 21:00:31 -0500548 // TODO(jmadill): Use automatic layout transition. http://anglebug.com/2361
Jamie Madill66546be2018-03-08 09:47:20 -0500549 colorRenderTarget->image->changeLayoutWithStages(
550 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
551 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT, VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT,
552 commandBuffer);
553 node->appendColorRenderTarget(currentSerial, colorRenderTarget);
554 attachmentClearValues.emplace_back(contextVk->getClearColorValue());
555 }
556
557 RenderTargetVk *depthStencilRenderTarget = mRenderTargetCache.getDepthStencil();
558 if (depthStencilRenderTarget)
559 {
560 // TODO(jmadill): Use automatic layout transition. http://anglebug.com/2361
561 const angle::Format &format = depthStencilRenderTarget->format->textureFormat();
Jamie Madille4c5a232018-03-02 21:00:31 -0500562 VkImageAspectFlags aspectFlags = (format.depthBits > 0 ? VK_IMAGE_ASPECT_DEPTH_BIT : 0) |
563 (format.stencilBits > 0 ? VK_IMAGE_ASPECT_STENCIL_BIT : 0);
564
Jamie Madill66546be2018-03-08 09:47:20 -0500565 depthStencilRenderTarget->image->changeLayoutWithStages(
Jamie Madille4c5a232018-03-02 21:00:31 -0500566 aspectFlags, VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL,
567 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT, VK_PIPELINE_STAGE_LATE_FRAGMENT_TESTS_BIT,
568 commandBuffer);
Jamie Madill66546be2018-03-08 09:47:20 -0500569 node->appendDepthStencilRenderTarget(currentSerial, depthStencilRenderTarget);
Jamie Madillf4d693c2018-02-14 16:38:16 -0500570 attachmentClearValues.emplace_back(contextVk->getClearDepthStencilValue());
Jamie Madill49ac74b2017-12-21 14:42:33 -0500571 }
572
Jamie Madillf4d693c2018-02-14 16:38:16 -0500573 // Hard-code RenderPass to clear the first render target to the current clear value.
574 // TODO(jmadill): Proper clear value implementation. http://anglebug.com/2361
575 const gl::State &glState = context->getGLState();
576 node->storeRenderPassInfo(*framebuffer, glState.getViewport(), attachmentClearValues);
Jamie Madill47c8ea32018-01-03 18:27:59 -0500577 mLastRenderNodeSerial = currentSerial;
Jamie Madill49ac74b2017-12-21 14:42:33 -0500578
579 *nodeOut = node;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500580 return gl::NoError();
581}
582
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400583} // namespace rx