blob: cd9013bcc945d2f455af6474916b6ff4956efa44 [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 Madill1b038242017-11-01 15:14:36 -040084 : FramebufferImpl(state), mBackbuffer(nullptr), mRenderPass(), mFramebuffer()
Jamie Madill9e54b5a2016-05-25 12:57:39 -040085{
86}
87
Jamie Madill7b57b9d2017-01-13 09:33:38 -050088FramebufferVk::FramebufferVk(const gl::FramebufferState &state, WindowSurfaceVk *backbuffer)
Jamie Madill1b038242017-11-01 15:14:36 -040089 : FramebufferImpl(state), mBackbuffer(backbuffer), mRenderPass(), mFramebuffer()
Jamie Madill7b57b9d2017-01-13 09:33:38 -050090{
91}
92
Jamie Madill9e54b5a2016-05-25 12:57:39 -040093FramebufferVk::~FramebufferVk()
94{
95}
96
Jamie Madillc564c072017-06-01 12:45:42 -040097void FramebufferVk::destroy(const gl::Context *context)
Jamie Madill5deea722017-02-16 10:44:46 -050098{
Jamie Madillc564c072017-06-01 12:45:42 -040099 VkDevice device = GetImplAs<ContextVk>(context)->getDevice();
Jamie Madill5deea722017-02-16 10:44:46 -0500100
Jamie Madill1b038242017-11-01 15:14:36 -0400101 // TODO(jmadill): Deferred deletion.
Jamie Madill5deea722017-02-16 10:44:46 -0500102 mRenderPass.destroy(device);
103 mFramebuffer.destroy(device);
104}
105
Jamie Madillc564c072017-06-01 12:45:42 -0400106void FramebufferVk::destroyDefault(const egl::Display *display)
Jamie Madill5deea722017-02-16 10:44:46 -0500107{
Jamie Madillc564c072017-06-01 12:45:42 -0400108 VkDevice device = GetImplAs<DisplayVk>(display)->getRenderer()->getDevice();
Jamie Madill5deea722017-02-16 10:44:46 -0500109
110 mRenderPass.destroy(device);
111 mFramebuffer.destroy(device);
112}
113
Jamie Madill4928b7c2017-06-20 12:57:39 -0400114gl::Error FramebufferVk::discard(const gl::Context *context,
115 size_t count,
116 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400117{
118 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500119 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400120}
121
Jamie Madill4928b7c2017-06-20 12:57:39 -0400122gl::Error FramebufferVk::invalidate(const gl::Context *context,
123 size_t count,
124 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400125{
126 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500127 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400128}
129
Jamie Madill4928b7c2017-06-20 12:57:39 -0400130gl::Error FramebufferVk::invalidateSub(const gl::Context *context,
131 size_t count,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400132 const GLenum *attachments,
133 const gl::Rectangle &area)
134{
135 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500136 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400137}
138
Jamie Madillc564c072017-06-01 12:45:42 -0400139gl::Error FramebufferVk::clear(const gl::Context *context, GLbitfield mask)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400140{
Jamie Madillc564c072017-06-01 12:45:42 -0400141 ContextVk *contextVk = GetImplAs<ContextVk>(context);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500142
143 if (mState.getDepthAttachment() && (mask & GL_DEPTH_BUFFER_BIT) != 0)
144 {
145 // TODO(jmadill): Depth clear
146 UNIMPLEMENTED();
147 }
148
149 if (mState.getStencilAttachment() && (mask & GL_STENCIL_BUFFER_BIT) != 0)
150 {
151 // TODO(jmadill): Stencil clear
152 UNIMPLEMENTED();
153 }
154
155 if ((mask & GL_COLOR_BUFFER_BIT) == 0)
156 {
157 return gl::NoError();
158 }
159
160 const auto &glState = context->getGLState();
161 const auto &clearColor = glState.getColorClearValue();
162 VkClearColorValue clearColorValue;
163 clearColorValue.float32[0] = clearColor.red;
164 clearColorValue.float32[1] = clearColor.green;
165 clearColorValue.float32[2] = clearColor.blue;
166 clearColorValue.float32[3] = clearColor.alpha;
167
168 // TODO(jmadill): Scissored clears.
169 const auto *attachment = mState.getFirstNonNullAttachment();
170 ASSERT(attachment && attachment->isAttached());
171 const auto &size = attachment->getSize();
172 const gl::Rectangle renderArea(0, 0, size.width, size.height);
173
Jamie Madill0c0dc342017-03-24 14:18:51 -0400174 vk::CommandBuffer *commandBuffer = nullptr;
175 ANGLE_TRY(contextVk->getStartedCommandBuffer(&commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500176
177 for (const auto &colorAttachment : mState.getColorAttachments())
178 {
179 if (colorAttachment.isAttached())
180 {
181 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400182 ANGLE_TRY(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500183 renderTarget->image->changeLayoutTop(
184 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, commandBuffer);
185 commandBuffer->clearSingleColorImage(*renderTarget->image, clearColorValue);
186 }
187 }
188
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500189 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400190}
191
Jamie Madillc564c072017-06-01 12:45:42 -0400192gl::Error FramebufferVk::clearBufferfv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400193 GLenum buffer,
194 GLint drawbuffer,
195 const GLfloat *values)
196{
197 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500198 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400199}
200
Jamie Madillc564c072017-06-01 12:45:42 -0400201gl::Error FramebufferVk::clearBufferuiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400202 GLenum buffer,
203 GLint drawbuffer,
204 const GLuint *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::clearBufferiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400211 GLenum buffer,
212 GLint drawbuffer,
213 const GLint *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::clearBufferfi(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400220 GLenum buffer,
221 GLint drawbuffer,
222 GLfloat depth,
223 GLint stencil)
224{
225 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500226 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400227}
228
Jamie Madill4928b7c2017-06-20 12:57:39 -0400229GLenum FramebufferVk::getImplementationColorReadFormat(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400230{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400231 auto errOrResult = GetReadAttachmentInfo(context, mState.getReadAttachment());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500232
233 // TODO(jmadill): Handle getRenderTarget error.
234 if (errOrResult.isError())
235 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500236 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500237 return GL_NONE;
238 }
239
240 return errOrResult.getResult()->format;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400241}
242
Jamie Madill4928b7c2017-06-20 12:57:39 -0400243GLenum FramebufferVk::getImplementationColorReadType(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400244{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400245 auto errOrResult = GetReadAttachmentInfo(context, mState.getReadAttachment());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500246
247 // TODO(jmadill): Handle getRenderTarget error.
248 if (errOrResult.isError())
249 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500250 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500251 return GL_NONE;
252 }
253
254 return errOrResult.getResult()->type;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400255}
256
Jamie Madillc564c072017-06-01 12:45:42 -0400257gl::Error FramebufferVk::readPixels(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400258 const gl::Rectangle &area,
259 GLenum format,
260 GLenum type,
Jamie Madilld4826152017-09-21 11:18:59 -0400261 void *pixels)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400262{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500263 const auto &glState = context->getGLState();
264 const auto *readFramebuffer = glState.getReadFramebuffer();
265 const auto *readAttachment = readFramebuffer->getReadColorbuffer();
266
267 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400268 ANGLE_TRY(readAttachment->getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500269
Jamie Madillc564c072017-06-01 12:45:42 -0400270 ContextVk *contextVk = GetImplAs<ContextVk>(context);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500271 RendererVk *renderer = contextVk->getRenderer();
Jamie Madill5deea722017-02-16 10:44:46 -0500272 VkDevice device = renderer->getDevice();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500273
274 vk::Image *readImage = renderTarget->image;
275 vk::StagingImage stagingImage;
Jamie Madill5deea722017-02-16 10:44:46 -0500276 ANGLE_TRY(renderer->createStagingImage(TextureDimension::TEX_2D, *renderTarget->format,
Jamie Madill035fd6b2017-10-03 15:43:22 -0400277 renderTarget->extents, vk::StagingUsage::Read,
278 &stagingImage));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500279
Jamie Madill0c0dc342017-03-24 14:18:51 -0400280 vk::CommandBuffer *commandBuffer = nullptr;
281 ANGLE_TRY(contextVk->getStartedCommandBuffer(&commandBuffer));
282
Jamie Madilld4826152017-09-21 11:18:59 -0400283 // End render pass if we're in one.
Jamie Madill1b038242017-11-01 15:14:36 -0400284 renderer->endRenderPass();
Jamie Madilld4826152017-09-21 11:18:59 -0400285
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500286 stagingImage.getImage().changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_GENERAL,
287 commandBuffer);
288
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500289 readImage->changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
290 commandBuffer);
Jamie Madill815a6c92017-10-21 14:33:04 -0400291
292 VkImageCopy region;
293 region.srcSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
294 region.srcSubresource.mipLevel = 0;
295 region.srcSubresource.baseArrayLayer = 0;
296 region.srcSubresource.layerCount = 1;
297 region.srcOffset.x = area.x;
298 region.srcOffset.y = area.y;
299 region.srcOffset.z = 0;
300 region.dstSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
301 region.dstSubresource.mipLevel = 0;
302 region.dstSubresource.baseArrayLayer = 0;
303 region.dstSubresource.layerCount = 1;
304 region.dstOffset.x = 0;
305 region.dstOffset.y = 0;
306 region.dstOffset.z = 0;
307 region.extent.width = area.width;
308 region.extent.height = area.height;
309 region.extent.depth = 1;
310
311 commandBuffer->copyImage(*readImage, stagingImage.getImage(), 1, &region);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500312
Jamie Madill0c0dc342017-03-24 14:18:51 -0400313 ANGLE_TRY(renderer->submitAndFinishCommandBuffer(commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500314
315 // TODO(jmadill): parameters
316 uint8_t *mapPointer = nullptr;
Jamie Madill5deea722017-02-16 10:44:46 -0500317 ANGLE_TRY(
318 stagingImage.getDeviceMemory().map(device, 0, stagingImage.getSize(), 0, &mapPointer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500319
320 const auto &angleFormat = renderTarget->format->format();
321
322 // TODO(jmadill): Use pixel bytes from the ANGLE format directly.
Geoff Langca271392017-04-05 12:30:00 -0400323 const auto &glFormat = gl::GetSizedInternalFormatInfo(angleFormat.glInternalFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500324 int inputPitch = glFormat.pixelBytes * area.width;
325
326 PackPixelsParams params;
327 params.area = area;
328 params.format = format;
329 params.type = type;
330 params.outputPitch = inputPitch;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400331 params.pack.copyFrom(context, glState.getPackState());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500332
333 PackPixels(params, angleFormat, inputPitch, mapPointer, reinterpret_cast<uint8_t *>(pixels));
334
Jamie Madill5deea722017-02-16 10:44:46 -0500335 stagingImage.getDeviceMemory().unmap(device);
Jamie Madille88ec8e2017-10-31 17:18:14 -0400336 renderer->releaseObject(renderer->getCurrentQueueSerial(), &stagingImage);
Jamie Madillf651c772017-02-21 15:03:51 -0500337
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500338 return vk::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400339}
340
Jamie Madillc564c072017-06-01 12:45:42 -0400341gl::Error FramebufferVk::blit(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400342 const gl::Rectangle &sourceArea,
343 const gl::Rectangle &destArea,
344 GLbitfield mask,
345 GLenum filter)
346{
347 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500348 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400349}
350
Kenneth Russellce8602a2017-10-03 18:23:08 -0700351bool FramebufferVk::checkStatus(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400352{
Jamie Madillb79e7bb2017-10-24 13:55:50 -0400353 return true;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400354}
355
Jamie Madillc564c072017-06-01 12:45:42 -0400356void FramebufferVk::syncState(const gl::Context *context,
357 const gl::Framebuffer::DirtyBits &dirtyBits)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400358{
Jamie Madillc564c072017-06-01 12:45:42 -0400359 auto contextVk = GetImplAs<ContextVk>(context);
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400360
361 ASSERT(dirtyBits.any());
362
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500363 // TODO(jmadill): Smarter update.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400364 mRenderPass.destroy(contextVk->getDevice());
365 mFramebuffer.destroy(contextVk->getDevice());
Jamie Madill72106562017-03-24 14:18:50 -0400366
367 // TODO(jmadill): Use pipeline cache.
368 contextVk->invalidateCurrentPipeline();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500369}
370
Jamie Madill4928b7c2017-06-20 12:57:39 -0400371gl::ErrorOrResult<vk::RenderPass *> FramebufferVk::getRenderPass(const gl::Context *context,
372 VkDevice device)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500373{
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400374 if (mRenderPass.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500375 {
376 return &mRenderPass;
377 }
378
379 // TODO(jmadill): Can we use stack-only memory?
380 std::vector<VkAttachmentDescription> attachmentDescs;
381 std::vector<VkAttachmentReference> colorAttachmentRefs;
382
383 const auto &colorAttachments = mState.getColorAttachments();
384 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
385 {
386 const auto &colorAttachment = colorAttachments[attachmentIndex];
387 if (colorAttachment.isAttached())
388 {
389 VkAttachmentDescription colorDesc;
390 VkAttachmentReference colorRef;
391
392 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400393 ANGLE_TRY(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500394
395 // TODO(jmadill): We would only need this flag for duplicated attachments.
396 colorDesc.flags = VK_ATTACHMENT_DESCRIPTION_MAY_ALIAS_BIT;
397 colorDesc.format = renderTarget->format->native;
398 colorDesc.samples = ConvertSamples(colorAttachment.getSamples());
399
400 // The load op controls the prior existing depth/color attachment data.
401 // TODO(jmadill): Proper load ops. Should not be hard coded to clear.
402 colorDesc.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
403 colorDesc.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
404 colorDesc.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
405 colorDesc.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
406 colorDesc.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500407
408 // We might want to transition directly to PRESENT_SRC for Surface attachments.
409 colorDesc.finalLayout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500410
411 colorRef.attachment = static_cast<uint32_t>(colorAttachments.size()) - 1u;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500412 colorRef.layout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500413
414 attachmentDescs.push_back(colorDesc);
415 colorAttachmentRefs.push_back(colorRef);
416 }
417 }
418
419 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
420 VkAttachmentReference depthStencilAttachmentRef;
421 bool useDepth = depthStencilAttachment && depthStencilAttachment->isAttached();
422
423 if (useDepth)
424 {
425 VkAttachmentDescription depthStencilDesc;
426
427 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400428 ANGLE_TRY(depthStencilAttachment->getRenderTarget(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500429
430 depthStencilDesc.flags = 0;
431 depthStencilDesc.format = renderTarget->format->native;
432 depthStencilDesc.samples = ConvertSamples(depthStencilAttachment->getSamples());
433 depthStencilDesc.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
434 depthStencilDesc.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
435 depthStencilDesc.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
436 depthStencilDesc.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
437 depthStencilDesc.initialLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
438 depthStencilDesc.finalLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
439
440 depthStencilAttachmentRef.attachment = static_cast<uint32_t>(attachmentDescs.size());
441 depthStencilAttachmentRef.layout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
442
443 attachmentDescs.push_back(depthStencilDesc);
444 }
445
446 ASSERT(!attachmentDescs.empty());
447
448 VkSubpassDescription subpassDesc;
449
450 subpassDesc.flags = 0;
451 subpassDesc.pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS;
452 subpassDesc.inputAttachmentCount = 0;
453 subpassDesc.pInputAttachments = nullptr;
454 subpassDesc.colorAttachmentCount = static_cast<uint32_t>(colorAttachmentRefs.size());
455 subpassDesc.pColorAttachments = colorAttachmentRefs.data();
456 subpassDesc.pResolveAttachments = nullptr;
457 subpassDesc.pDepthStencilAttachment = (useDepth ? &depthStencilAttachmentRef : nullptr);
458 subpassDesc.preserveAttachmentCount = 0;
459 subpassDesc.pPreserveAttachments = nullptr;
460
461 VkRenderPassCreateInfo renderPassInfo;
462
463 renderPassInfo.sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO;
464 renderPassInfo.pNext = nullptr;
465 renderPassInfo.flags = 0;
466 renderPassInfo.attachmentCount = static_cast<uint32_t>(attachmentDescs.size());
467 renderPassInfo.pAttachments = attachmentDescs.data();
468 renderPassInfo.subpassCount = 1;
469 renderPassInfo.pSubpasses = &subpassDesc;
470 renderPassInfo.dependencyCount = 0;
471 renderPassInfo.pDependencies = nullptr;
472
Jamie Madill25301b62017-10-28 20:59:31 -0400473 ANGLE_TRY(mRenderPass.init(device, renderPassInfo));
Jamie Madill5deea722017-02-16 10:44:46 -0500474
Jamie Madillab9f9c32017-01-17 17:47:34 -0500475 return &mRenderPass;
476}
477
Jamie Madill4928b7c2017-06-20 12:57:39 -0400478gl::ErrorOrResult<vk::Framebuffer *> FramebufferVk::getFramebuffer(const gl::Context *context,
479 VkDevice device)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500480{
481 // If we've already created our cached Framebuffer, return it.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400482 if (mFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500483 {
484 return &mFramebuffer;
485 }
486
487 vk::RenderPass *renderPass = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400488 ANGLE_TRY_RESULT(getRenderPass(context, device), renderPass);
Jamie Madillab9f9c32017-01-17 17:47:34 -0500489
490 // If we've a Framebuffer provided by a Surface (default FBO/backbuffer), query it.
491 if (mBackbuffer)
492 {
493 return mBackbuffer->getCurrentFramebuffer(device, *renderPass);
494 }
495
496 // Gather VkImageViews over all FBO attachments, also size of attached region.
497 std::vector<VkImageView> attachments;
498 gl::Extents attachmentsSize;
499
500 const auto &colorAttachments = mState.getColorAttachments();
501 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
502 {
503 const auto &colorAttachment = colorAttachments[attachmentIndex];
504 if (colorAttachment.isAttached())
505 {
506 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400507 ANGLE_TRY(colorAttachment.getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500508 attachments.push_back(renderTarget->imageView->getHandle());
509
510 ASSERT(attachmentsSize.empty() || attachmentsSize == colorAttachment.getSize());
511 attachmentsSize = colorAttachment.getSize();
512 }
513 }
514
515 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
516 if (depthStencilAttachment && depthStencilAttachment->isAttached())
517 {
518 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400519 ANGLE_TRY(depthStencilAttachment->getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500520 attachments.push_back(renderTarget->imageView->getHandle());
521
522 ASSERT(attachmentsSize.empty() || attachmentsSize == depthStencilAttachment->getSize());
523 attachmentsSize = depthStencilAttachment->getSize();
524 }
525
526 ASSERT(!attachments.empty());
527
528 VkFramebufferCreateInfo framebufferInfo;
529
530 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
531 framebufferInfo.pNext = nullptr;
532 framebufferInfo.flags = 0;
533 framebufferInfo.renderPass = mRenderPass.getHandle();
534 framebufferInfo.attachmentCount = static_cast<uint32_t>(attachments.size());
535 framebufferInfo.pAttachments = attachments.data();
536 framebufferInfo.width = static_cast<uint32_t>(attachmentsSize.width);
537 framebufferInfo.height = static_cast<uint32_t>(attachmentsSize.height);
538 framebufferInfo.layers = 1;
539
Jamie Madill25301b62017-10-28 20:59:31 -0400540 ANGLE_TRY(mFramebuffer.init(device, framebufferInfo));
Jamie Madill5deea722017-02-16 10:44:46 -0500541
Jamie Madillab9f9c32017-01-17 17:47:34 -0500542 return &mFramebuffer;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400543}
544
JiangYizhoubddc46b2016-12-09 09:50:51 +0800545gl::Error FramebufferVk::getSamplePosition(size_t index, GLfloat *xy) const
546{
547 UNIMPLEMENTED();
548 return gl::InternalError() << "getSamplePosition is unimplemented.";
549}
550
Jamie Madill1b038242017-11-01 15:14:36 -0400551gl::Error FramebufferVk::beginRenderPass(const gl::Context *context,
552 VkDevice device,
553 vk::CommandBuffer *commandBuffer,
554 Serial queueSerial)
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500555{
Jamie Madill4c26fc22017-02-24 11:04:10 -0500556 // TODO(jmadill): Cache render targets.
557 for (const auto &colorAttachment : mState.getColorAttachments())
558 {
559 if (colorAttachment.isAttached())
560 {
561 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400562 ANGLE_TRY(colorAttachment.getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madill4c26fc22017-02-24 11:04:10 -0500563 renderTarget->resource->setQueueSerial(queueSerial);
564 }
565 }
566
567 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
568 if (depthStencilAttachment && depthStencilAttachment->isAttached())
569 {
570 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400571 ANGLE_TRY(depthStencilAttachment->getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madill4c26fc22017-02-24 11:04:10 -0500572 renderTarget->resource->setQueueSerial(queueSerial);
573 }
574
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500575 vk::Framebuffer *framebuffer = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400576 ANGLE_TRY_RESULT(getFramebuffer(context, device), framebuffer);
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500577 ASSERT(framebuffer && framebuffer->valid());
578
579 vk::RenderPass *renderPass = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400580 ANGLE_TRY_RESULT(getRenderPass(context, device), renderPass);
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500581 ASSERT(renderPass && renderPass->valid());
582
583 // TODO(jmadill): Proper clear value implementation.
Jamie Madill1b038242017-11-01 15:14:36 -0400584 const gl::State &glState = context->getGLState();
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500585 VkClearColorValue colorClear;
586 memset(&colorClear, 0, sizeof(VkClearColorValue));
587 colorClear.float32[0] = glState.getColorClearValue().red;
588 colorClear.float32[1] = glState.getColorClearValue().green;
589 colorClear.float32[2] = glState.getColorClearValue().blue;
590 colorClear.float32[3] = glState.getColorClearValue().alpha;
591
592 std::vector<VkClearValue> attachmentClearValues;
593 attachmentClearValues.push_back({colorClear});
594
595 // Updated the cached image layout of the attachments in this FBO.
596 // For a default FBO, we need to call through to the WindowSurfaceVk
597 // TODO(jmadill): Iterate over all attachments.
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500598 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400599 ANGLE_TRY(mState.getFirstColorAttachment()->getRenderTarget(context, &renderTarget));
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500600 renderTarget->image->updateLayout(VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
601
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500602 commandBuffer->beginRenderPass(*renderPass, *framebuffer, glState.getViewport(),
603 attachmentClearValues);
Jamie Madill4c26fc22017-02-24 11:04:10 -0500604
605 setQueueSerial(queueSerial);
606 if (mBackbuffer)
607 {
608 mBackbuffer->setQueueSerial(queueSerial);
609 }
610
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500611 return gl::NoError();
612}
613
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400614} // namespace rx