blob: c868ce922eaf4ef61cb6731673873c66774fd28e [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"
21#include "libANGLE/renderer/vulkan/ContextVk.h"
Jamie Madill5deea722017-02-16 10:44:46 -050022#include "libANGLE/renderer/vulkan/DisplayVk.h"
Jamie Madill7b57b9d2017-01-13 09:33:38 -050023#include "libANGLE/renderer/vulkan/RenderTargetVk.h"
24#include "libANGLE/renderer/vulkan/RendererVk.h"
25#include "libANGLE/renderer/vulkan/SurfaceVk.h"
26#include "libANGLE/renderer/vulkan/formatutilsvk.h"
Jamie Madill9e54b5a2016-05-25 12:57:39 -040027
28namespace rx
29{
30
Jamie Madill7b57b9d2017-01-13 09:33:38 -050031namespace
32{
33
34gl::ErrorOrResult<const gl::InternalFormat *> GetReadAttachmentInfo(
Jamie Madill4928b7c2017-06-20 12:57:39 -040035 const gl::Context *context,
Jamie Madill7b57b9d2017-01-13 09:33:38 -050036 const gl::FramebufferAttachment *readAttachment)
37{
38 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -040039 ANGLE_TRY(readAttachment->getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -050040
41 GLenum implFormat = renderTarget->format->format().fboImplementationInternalFormat;
Geoff Langca271392017-04-05 12:30:00 -040042 return &gl::GetSizedInternalFormatInfo(implFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -050043}
44
Jamie Madillab9f9c32017-01-17 17:47:34 -050045VkSampleCountFlagBits ConvertSamples(GLint sampleCount)
46{
47 switch (sampleCount)
48 {
49 case 0:
50 case 1:
51 return VK_SAMPLE_COUNT_1_BIT;
52 case 2:
53 return VK_SAMPLE_COUNT_2_BIT;
54 case 4:
55 return VK_SAMPLE_COUNT_4_BIT;
56 case 8:
57 return VK_SAMPLE_COUNT_8_BIT;
58 case 16:
59 return VK_SAMPLE_COUNT_16_BIT;
60 case 32:
61 return VK_SAMPLE_COUNT_32_BIT;
62 default:
63 UNREACHABLE();
64 return VK_SAMPLE_COUNT_FLAG_BITS_MAX_ENUM;
65 }
66}
67
Jamie Madill7b57b9d2017-01-13 09:33:38 -050068} // anonymous namespace
69
70// static
71FramebufferVk *FramebufferVk::CreateUserFBO(const gl::FramebufferState &state)
72{
73 return new FramebufferVk(state);
74}
75
76// static
77FramebufferVk *FramebufferVk::CreateDefaultFBO(const gl::FramebufferState &state,
78 WindowSurfaceVk *backbuffer)
79{
80 return new FramebufferVk(state, backbuffer);
81}
82
Jamie Madillab9f9c32017-01-17 17:47:34 -050083FramebufferVk::FramebufferVk(const gl::FramebufferState &state)
Jamie Madilld4826152017-09-21 11:18:59 -040084 : FramebufferImpl(state),
85 mBackbuffer(nullptr),
86 mRenderPass(),
87 mFramebuffer(),
88 mInRenderPass(false)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040089{
90}
91
Jamie Madill7b57b9d2017-01-13 09:33:38 -050092FramebufferVk::FramebufferVk(const gl::FramebufferState &state, WindowSurfaceVk *backbuffer)
Jamie Madilld4826152017-09-21 11:18:59 -040093 : FramebufferImpl(state),
94 mBackbuffer(backbuffer),
95 mRenderPass(),
96 mFramebuffer(),
97 mInRenderPass(false)
Jamie Madill7b57b9d2017-01-13 09:33:38 -050098{
99}
100
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400101FramebufferVk::~FramebufferVk()
102{
103}
104
Jamie Madillc564c072017-06-01 12:45:42 -0400105void FramebufferVk::destroy(const gl::Context *context)
Jamie Madill5deea722017-02-16 10:44:46 -0500106{
Jamie Madilld4826152017-09-21 11:18:59 -0400107 ASSERT(!mInRenderPass);
108
Jamie Madillc564c072017-06-01 12:45:42 -0400109 VkDevice device = GetImplAs<ContextVk>(context)->getDevice();
Jamie Madill5deea722017-02-16 10:44:46 -0500110
111 mRenderPass.destroy(device);
112 mFramebuffer.destroy(device);
113}
114
Jamie Madillc564c072017-06-01 12:45:42 -0400115void FramebufferVk::destroyDefault(const egl::Display *display)
Jamie Madill5deea722017-02-16 10:44:46 -0500116{
Jamie Madillc564c072017-06-01 12:45:42 -0400117 VkDevice device = GetImplAs<DisplayVk>(display)->getRenderer()->getDevice();
Jamie Madill5deea722017-02-16 10:44:46 -0500118
119 mRenderPass.destroy(device);
120 mFramebuffer.destroy(device);
121}
122
Jamie Madill4928b7c2017-06-20 12:57:39 -0400123gl::Error FramebufferVk::discard(const gl::Context *context,
124 size_t count,
125 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400126{
127 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500128 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400129}
130
Jamie Madill4928b7c2017-06-20 12:57:39 -0400131gl::Error FramebufferVk::invalidate(const gl::Context *context,
132 size_t count,
133 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400134{
135 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500136 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400137}
138
Jamie Madill4928b7c2017-06-20 12:57:39 -0400139gl::Error FramebufferVk::invalidateSub(const gl::Context *context,
140 size_t count,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400141 const GLenum *attachments,
142 const gl::Rectangle &area)
143{
144 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500145 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400146}
147
Jamie Madillc564c072017-06-01 12:45:42 -0400148gl::Error FramebufferVk::clear(const gl::Context *context, GLbitfield mask)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400149{
Jamie Madillc564c072017-06-01 12:45:42 -0400150 ContextVk *contextVk = GetImplAs<ContextVk>(context);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500151
152 if (mState.getDepthAttachment() && (mask & GL_DEPTH_BUFFER_BIT) != 0)
153 {
154 // TODO(jmadill): Depth clear
155 UNIMPLEMENTED();
156 }
157
158 if (mState.getStencilAttachment() && (mask & GL_STENCIL_BUFFER_BIT) != 0)
159 {
160 // TODO(jmadill): Stencil clear
161 UNIMPLEMENTED();
162 }
163
164 if ((mask & GL_COLOR_BUFFER_BIT) == 0)
165 {
166 return gl::NoError();
167 }
168
169 const auto &glState = context->getGLState();
170 const auto &clearColor = glState.getColorClearValue();
171 VkClearColorValue clearColorValue;
172 clearColorValue.float32[0] = clearColor.red;
173 clearColorValue.float32[1] = clearColor.green;
174 clearColorValue.float32[2] = clearColor.blue;
175 clearColorValue.float32[3] = clearColor.alpha;
176
177 // TODO(jmadill): Scissored clears.
178 const auto *attachment = mState.getFirstNonNullAttachment();
179 ASSERT(attachment && attachment->isAttached());
180 const auto &size = attachment->getSize();
181 const gl::Rectangle renderArea(0, 0, size.width, size.height);
182
Jamie Madill0c0dc342017-03-24 14:18:51 -0400183 vk::CommandBuffer *commandBuffer = nullptr;
184 ANGLE_TRY(contextVk->getStartedCommandBuffer(&commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500185
186 for (const auto &colorAttachment : mState.getColorAttachments())
187 {
188 if (colorAttachment.isAttached())
189 {
190 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400191 ANGLE_TRY(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500192 renderTarget->image->changeLayoutTop(
193 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, commandBuffer);
194 commandBuffer->clearSingleColorImage(*renderTarget->image, clearColorValue);
195 }
196 }
197
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500198 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400199}
200
Jamie Madillc564c072017-06-01 12:45:42 -0400201gl::Error FramebufferVk::clearBufferfv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400202 GLenum buffer,
203 GLint drawbuffer,
204 const GLfloat *values)
205{
206 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500207 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400208}
209
Jamie Madillc564c072017-06-01 12:45:42 -0400210gl::Error FramebufferVk::clearBufferuiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400211 GLenum buffer,
212 GLint drawbuffer,
213 const GLuint *values)
214{
215 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500216 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400217}
218
Jamie Madillc564c072017-06-01 12:45:42 -0400219gl::Error FramebufferVk::clearBufferiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400220 GLenum buffer,
221 GLint drawbuffer,
222 const GLint *values)
223{
224 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500225 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400226}
227
Jamie Madillc564c072017-06-01 12:45:42 -0400228gl::Error FramebufferVk::clearBufferfi(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400229 GLenum buffer,
230 GLint drawbuffer,
231 GLfloat depth,
232 GLint stencil)
233{
234 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500235 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400236}
237
Jamie Madill4928b7c2017-06-20 12:57:39 -0400238GLenum FramebufferVk::getImplementationColorReadFormat(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400239{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400240 auto errOrResult = GetReadAttachmentInfo(context, mState.getReadAttachment());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500241
242 // TODO(jmadill): Handle getRenderTarget error.
243 if (errOrResult.isError())
244 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500245 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500246 return GL_NONE;
247 }
248
249 return errOrResult.getResult()->format;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400250}
251
Jamie Madill4928b7c2017-06-20 12:57:39 -0400252GLenum FramebufferVk::getImplementationColorReadType(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400253{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400254 auto errOrResult = GetReadAttachmentInfo(context, mState.getReadAttachment());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500255
256 // TODO(jmadill): Handle getRenderTarget error.
257 if (errOrResult.isError())
258 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500259 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500260 return GL_NONE;
261 }
262
263 return errOrResult.getResult()->type;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400264}
265
Jamie Madillc564c072017-06-01 12:45:42 -0400266gl::Error FramebufferVk::readPixels(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400267 const gl::Rectangle &area,
268 GLenum format,
269 GLenum type,
Jamie Madilld4826152017-09-21 11:18:59 -0400270 void *pixels)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400271{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500272 const auto &glState = context->getGLState();
273 const auto *readFramebuffer = glState.getReadFramebuffer();
274 const auto *readAttachment = readFramebuffer->getReadColorbuffer();
275
276 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400277 ANGLE_TRY(readAttachment->getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500278
Jamie Madillc564c072017-06-01 12:45:42 -0400279 ContextVk *contextVk = GetImplAs<ContextVk>(context);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500280 RendererVk *renderer = contextVk->getRenderer();
Jamie Madill5deea722017-02-16 10:44:46 -0500281 VkDevice device = renderer->getDevice();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500282
283 vk::Image *readImage = renderTarget->image;
284 vk::StagingImage stagingImage;
Jamie Madill5deea722017-02-16 10:44:46 -0500285 ANGLE_TRY(renderer->createStagingImage(TextureDimension::TEX_2D, *renderTarget->format,
Jamie Madill035fd6b2017-10-03 15:43:22 -0400286 renderTarget->extents, vk::StagingUsage::Read,
287 &stagingImage));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500288
Jamie Madill0c0dc342017-03-24 14:18:51 -0400289 vk::CommandBuffer *commandBuffer = nullptr;
290 ANGLE_TRY(contextVk->getStartedCommandBuffer(&commandBuffer));
291
Jamie Madilld4826152017-09-21 11:18:59 -0400292 // End render pass if we're in one.
293 endRenderPass(commandBuffer);
294
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500295 stagingImage.getImage().changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_GENERAL,
296 commandBuffer);
297
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500298 readImage->changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
299 commandBuffer);
Jamie Madill815a6c92017-10-21 14:33:04 -0400300
301 VkImageCopy region;
302 region.srcSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
303 region.srcSubresource.mipLevel = 0;
304 region.srcSubresource.baseArrayLayer = 0;
305 region.srcSubresource.layerCount = 1;
306 region.srcOffset.x = area.x;
307 region.srcOffset.y = area.y;
308 region.srcOffset.z = 0;
309 region.dstSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
310 region.dstSubresource.mipLevel = 0;
311 region.dstSubresource.baseArrayLayer = 0;
312 region.dstSubresource.layerCount = 1;
313 region.dstOffset.x = 0;
314 region.dstOffset.y = 0;
315 region.dstOffset.z = 0;
316 region.extent.width = area.width;
317 region.extent.height = area.height;
318 region.extent.depth = 1;
319
320 commandBuffer->copyImage(*readImage, stagingImage.getImage(), 1, &region);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500321
Jamie Madill0c0dc342017-03-24 14:18:51 -0400322 ANGLE_TRY(renderer->submitAndFinishCommandBuffer(commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500323
324 // TODO(jmadill): parameters
325 uint8_t *mapPointer = nullptr;
Jamie Madill5deea722017-02-16 10:44:46 -0500326 ANGLE_TRY(
327 stagingImage.getDeviceMemory().map(device, 0, stagingImage.getSize(), 0, &mapPointer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500328
329 const auto &angleFormat = renderTarget->format->format();
330
331 // TODO(jmadill): Use pixel bytes from the ANGLE format directly.
Geoff Langca271392017-04-05 12:30:00 -0400332 const auto &glFormat = gl::GetSizedInternalFormatInfo(angleFormat.glInternalFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500333 int inputPitch = glFormat.pixelBytes * area.width;
334
335 PackPixelsParams params;
336 params.area = area;
337 params.format = format;
338 params.type = type;
339 params.outputPitch = inputPitch;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400340 params.pack.copyFrom(context, glState.getPackState());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500341
342 PackPixels(params, angleFormat, inputPitch, mapPointer, reinterpret_cast<uint8_t *>(pixels));
343
Jamie Madill5deea722017-02-16 10:44:46 -0500344 stagingImage.getDeviceMemory().unmap(device);
Jamie Madille88ec8e2017-10-31 17:18:14 -0400345 renderer->releaseObject(renderer->getCurrentQueueSerial(), &stagingImage);
Jamie Madillf651c772017-02-21 15:03:51 -0500346
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500347 return vk::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400348}
349
Jamie Madillc564c072017-06-01 12:45:42 -0400350gl::Error FramebufferVk::blit(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400351 const gl::Rectangle &sourceArea,
352 const gl::Rectangle &destArea,
353 GLbitfield mask,
354 GLenum filter)
355{
356 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500357 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400358}
359
Kenneth Russellce8602a2017-10-03 18:23:08 -0700360bool FramebufferVk::checkStatus(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400361{
Jamie Madillb79e7bb2017-10-24 13:55:50 -0400362 return true;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400363}
364
Jamie Madillc564c072017-06-01 12:45:42 -0400365void FramebufferVk::syncState(const gl::Context *context,
366 const gl::Framebuffer::DirtyBits &dirtyBits)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400367{
Jamie Madillc564c072017-06-01 12:45:42 -0400368 auto contextVk = GetImplAs<ContextVk>(context);
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400369
370 ASSERT(dirtyBits.any());
371
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500372 // TODO(jmadill): Smarter update.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400373 mRenderPass.destroy(contextVk->getDevice());
374 mFramebuffer.destroy(contextVk->getDevice());
Jamie Madill72106562017-03-24 14:18:50 -0400375
376 // TODO(jmadill): Use pipeline cache.
377 contextVk->invalidateCurrentPipeline();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500378}
379
Jamie Madill4928b7c2017-06-20 12:57:39 -0400380gl::ErrorOrResult<vk::RenderPass *> FramebufferVk::getRenderPass(const gl::Context *context,
381 VkDevice device)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500382{
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400383 if (mRenderPass.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500384 {
385 return &mRenderPass;
386 }
387
388 // TODO(jmadill): Can we use stack-only memory?
389 std::vector<VkAttachmentDescription> attachmentDescs;
390 std::vector<VkAttachmentReference> colorAttachmentRefs;
391
392 const auto &colorAttachments = mState.getColorAttachments();
393 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
394 {
395 const auto &colorAttachment = colorAttachments[attachmentIndex];
396 if (colorAttachment.isAttached())
397 {
398 VkAttachmentDescription colorDesc;
399 VkAttachmentReference colorRef;
400
401 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400402 ANGLE_TRY(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500403
404 // TODO(jmadill): We would only need this flag for duplicated attachments.
405 colorDesc.flags = VK_ATTACHMENT_DESCRIPTION_MAY_ALIAS_BIT;
406 colorDesc.format = renderTarget->format->native;
407 colorDesc.samples = ConvertSamples(colorAttachment.getSamples());
408
409 // The load op controls the prior existing depth/color attachment data.
410 // TODO(jmadill): Proper load ops. Should not be hard coded to clear.
411 colorDesc.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
412 colorDesc.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
413 colorDesc.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
414 colorDesc.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
415 colorDesc.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500416
417 // We might want to transition directly to PRESENT_SRC for Surface attachments.
418 colorDesc.finalLayout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500419
420 colorRef.attachment = static_cast<uint32_t>(colorAttachments.size()) - 1u;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500421 colorRef.layout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500422
423 attachmentDescs.push_back(colorDesc);
424 colorAttachmentRefs.push_back(colorRef);
425 }
426 }
427
428 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
429 VkAttachmentReference depthStencilAttachmentRef;
430 bool useDepth = depthStencilAttachment && depthStencilAttachment->isAttached();
431
432 if (useDepth)
433 {
434 VkAttachmentDescription depthStencilDesc;
435
436 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400437 ANGLE_TRY(depthStencilAttachment->getRenderTarget(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500438
439 depthStencilDesc.flags = 0;
440 depthStencilDesc.format = renderTarget->format->native;
441 depthStencilDesc.samples = ConvertSamples(depthStencilAttachment->getSamples());
442 depthStencilDesc.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
443 depthStencilDesc.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
444 depthStencilDesc.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
445 depthStencilDesc.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
446 depthStencilDesc.initialLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
447 depthStencilDesc.finalLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
448
449 depthStencilAttachmentRef.attachment = static_cast<uint32_t>(attachmentDescs.size());
450 depthStencilAttachmentRef.layout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
451
452 attachmentDescs.push_back(depthStencilDesc);
453 }
454
455 ASSERT(!attachmentDescs.empty());
456
457 VkSubpassDescription subpassDesc;
458
459 subpassDesc.flags = 0;
460 subpassDesc.pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS;
461 subpassDesc.inputAttachmentCount = 0;
462 subpassDesc.pInputAttachments = nullptr;
463 subpassDesc.colorAttachmentCount = static_cast<uint32_t>(colorAttachmentRefs.size());
464 subpassDesc.pColorAttachments = colorAttachmentRefs.data();
465 subpassDesc.pResolveAttachments = nullptr;
466 subpassDesc.pDepthStencilAttachment = (useDepth ? &depthStencilAttachmentRef : nullptr);
467 subpassDesc.preserveAttachmentCount = 0;
468 subpassDesc.pPreserveAttachments = nullptr;
469
470 VkRenderPassCreateInfo renderPassInfo;
471
472 renderPassInfo.sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO;
473 renderPassInfo.pNext = nullptr;
474 renderPassInfo.flags = 0;
475 renderPassInfo.attachmentCount = static_cast<uint32_t>(attachmentDescs.size());
476 renderPassInfo.pAttachments = attachmentDescs.data();
477 renderPassInfo.subpassCount = 1;
478 renderPassInfo.pSubpasses = &subpassDesc;
479 renderPassInfo.dependencyCount = 0;
480 renderPassInfo.pDependencies = nullptr;
481
Jamie Madill25301b62017-10-28 20:59:31 -0400482 ANGLE_TRY(mRenderPass.init(device, renderPassInfo));
Jamie Madill5deea722017-02-16 10:44:46 -0500483
Jamie Madillab9f9c32017-01-17 17:47:34 -0500484 return &mRenderPass;
485}
486
Jamie Madill4928b7c2017-06-20 12:57:39 -0400487gl::ErrorOrResult<vk::Framebuffer *> FramebufferVk::getFramebuffer(const gl::Context *context,
488 VkDevice device)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500489{
490 // If we've already created our cached Framebuffer, return it.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400491 if (mFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500492 {
493 return &mFramebuffer;
494 }
495
496 vk::RenderPass *renderPass = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400497 ANGLE_TRY_RESULT(getRenderPass(context, device), renderPass);
Jamie Madillab9f9c32017-01-17 17:47:34 -0500498
499 // If we've a Framebuffer provided by a Surface (default FBO/backbuffer), query it.
500 if (mBackbuffer)
501 {
502 return mBackbuffer->getCurrentFramebuffer(device, *renderPass);
503 }
504
505 // Gather VkImageViews over all FBO attachments, also size of attached region.
506 std::vector<VkImageView> attachments;
507 gl::Extents attachmentsSize;
508
509 const auto &colorAttachments = mState.getColorAttachments();
510 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
511 {
512 const auto &colorAttachment = colorAttachments[attachmentIndex];
513 if (colorAttachment.isAttached())
514 {
515 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400516 ANGLE_TRY(colorAttachment.getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500517 attachments.push_back(renderTarget->imageView->getHandle());
518
519 ASSERT(attachmentsSize.empty() || attachmentsSize == colorAttachment.getSize());
520 attachmentsSize = colorAttachment.getSize();
521 }
522 }
523
524 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
525 if (depthStencilAttachment && depthStencilAttachment->isAttached())
526 {
527 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400528 ANGLE_TRY(depthStencilAttachment->getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500529 attachments.push_back(renderTarget->imageView->getHandle());
530
531 ASSERT(attachmentsSize.empty() || attachmentsSize == depthStencilAttachment->getSize());
532 attachmentsSize = depthStencilAttachment->getSize();
533 }
534
535 ASSERT(!attachments.empty());
536
537 VkFramebufferCreateInfo framebufferInfo;
538
539 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
540 framebufferInfo.pNext = nullptr;
541 framebufferInfo.flags = 0;
542 framebufferInfo.renderPass = mRenderPass.getHandle();
543 framebufferInfo.attachmentCount = static_cast<uint32_t>(attachments.size());
544 framebufferInfo.pAttachments = attachments.data();
545 framebufferInfo.width = static_cast<uint32_t>(attachmentsSize.width);
546 framebufferInfo.height = static_cast<uint32_t>(attachmentsSize.height);
547 framebufferInfo.layers = 1;
548
Jamie Madill25301b62017-10-28 20:59:31 -0400549 ANGLE_TRY(mFramebuffer.init(device, framebufferInfo));
Jamie Madill5deea722017-02-16 10:44:46 -0500550
Jamie Madillab9f9c32017-01-17 17:47:34 -0500551 return &mFramebuffer;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400552}
553
JiangYizhoubddc46b2016-12-09 09:50:51 +0800554gl::Error FramebufferVk::getSamplePosition(size_t index, GLfloat *xy) const
555{
556 UNIMPLEMENTED();
557 return gl::InternalError() << "getSamplePosition is unimplemented.";
558}
559
Jamie Madilld4826152017-09-21 11:18:59 -0400560gl::Error FramebufferVk::ensureInRenderPass(const gl::Context *context,
561 VkDevice device,
562 vk::CommandBuffer *commandBuffer,
563 Serial queueSerial,
564 const gl::State &glState)
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500565{
Jamie Madilld4826152017-09-21 11:18:59 -0400566 if (mInRenderPass)
567 {
568 return gl::NoError();
569 }
570
571 mInRenderPass = true;
572
Jamie Madill4c26fc22017-02-24 11:04:10 -0500573 // TODO(jmadill): Cache render targets.
574 for (const auto &colorAttachment : mState.getColorAttachments())
575 {
576 if (colorAttachment.isAttached())
577 {
578 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400579 ANGLE_TRY(colorAttachment.getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madill4c26fc22017-02-24 11:04:10 -0500580 renderTarget->resource->setQueueSerial(queueSerial);
581 }
582 }
583
584 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
585 if (depthStencilAttachment && depthStencilAttachment->isAttached())
586 {
587 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400588 ANGLE_TRY(depthStencilAttachment->getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madill4c26fc22017-02-24 11:04:10 -0500589 renderTarget->resource->setQueueSerial(queueSerial);
590 }
591
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500592 vk::Framebuffer *framebuffer = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400593 ANGLE_TRY_RESULT(getFramebuffer(context, device), framebuffer);
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500594 ASSERT(framebuffer && framebuffer->valid());
595
596 vk::RenderPass *renderPass = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400597 ANGLE_TRY_RESULT(getRenderPass(context, device), renderPass);
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500598 ASSERT(renderPass && renderPass->valid());
599
600 // TODO(jmadill): Proper clear value implementation.
601 VkClearColorValue colorClear;
602 memset(&colorClear, 0, sizeof(VkClearColorValue));
603 colorClear.float32[0] = glState.getColorClearValue().red;
604 colorClear.float32[1] = glState.getColorClearValue().green;
605 colorClear.float32[2] = glState.getColorClearValue().blue;
606 colorClear.float32[3] = glState.getColorClearValue().alpha;
607
608 std::vector<VkClearValue> attachmentClearValues;
609 attachmentClearValues.push_back({colorClear});
610
611 // Updated the cached image layout of the attachments in this FBO.
612 // For a default FBO, we need to call through to the WindowSurfaceVk
613 // TODO(jmadill): Iterate over all attachments.
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500614 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400615 ANGLE_TRY(mState.getFirstColorAttachment()->getRenderTarget(context, &renderTarget));
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500616 renderTarget->image->updateLayout(VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
617
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500618 commandBuffer->beginRenderPass(*renderPass, *framebuffer, glState.getViewport(),
619 attachmentClearValues);
Jamie Madill4c26fc22017-02-24 11:04:10 -0500620
621 setQueueSerial(queueSerial);
622 if (mBackbuffer)
623 {
624 mBackbuffer->setQueueSerial(queueSerial);
625 }
626
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500627 return gl::NoError();
628}
629
Jamie Madilld4826152017-09-21 11:18:59 -0400630void FramebufferVk::endRenderPass(vk::CommandBuffer *commandBuffer)
631{
632 if (mInRenderPass)
633 {
634 commandBuffer->endRenderPass();
635 mInRenderPass = false;
636 }
637}
638
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400639} // namespace rx