blob: 5bd8b12f3a52f545eac75401737b829530f1a305 [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 <array>
13#include <vulkan/vulkan.h>
14
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"
17#include "libANGLE/formatutils.h"
18#include "libANGLE/renderer/renderer_utils.h"
19#include "libANGLE/renderer/vulkan/ContextVk.h"
Jamie Madill5deea722017-02-16 10:44:46 -050020#include "libANGLE/renderer/vulkan/DisplayVk.h"
Jamie Madill7b57b9d2017-01-13 09:33:38 -050021#include "libANGLE/renderer/vulkan/RenderTargetVk.h"
22#include "libANGLE/renderer/vulkan/RendererVk.h"
23#include "libANGLE/renderer/vulkan/SurfaceVk.h"
24#include "libANGLE/renderer/vulkan/formatutilsvk.h"
Jamie Madill9e54b5a2016-05-25 12:57:39 -040025
26namespace rx
27{
28
Jamie Madill7b57b9d2017-01-13 09:33:38 -050029namespace
30{
31
32gl::ErrorOrResult<const gl::InternalFormat *> GetReadAttachmentInfo(
33 const gl::FramebufferAttachment *readAttachment)
34{
35 RenderTargetVk *renderTarget = nullptr;
36 ANGLE_TRY(readAttachment->getRenderTarget(&renderTarget));
37
38 GLenum implFormat = renderTarget->format->format().fboImplementationInternalFormat;
Geoff Langca271392017-04-05 12:30:00 -040039 return &gl::GetSizedInternalFormatInfo(implFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -050040}
41
Jamie Madillab9f9c32017-01-17 17:47:34 -050042VkSampleCountFlagBits ConvertSamples(GLint sampleCount)
43{
44 switch (sampleCount)
45 {
46 case 0:
47 case 1:
48 return VK_SAMPLE_COUNT_1_BIT;
49 case 2:
50 return VK_SAMPLE_COUNT_2_BIT;
51 case 4:
52 return VK_SAMPLE_COUNT_4_BIT;
53 case 8:
54 return VK_SAMPLE_COUNT_8_BIT;
55 case 16:
56 return VK_SAMPLE_COUNT_16_BIT;
57 case 32:
58 return VK_SAMPLE_COUNT_32_BIT;
59 default:
60 UNREACHABLE();
61 return VK_SAMPLE_COUNT_FLAG_BITS_MAX_ENUM;
62 }
63}
64
Jamie Madill7b57b9d2017-01-13 09:33:38 -050065} // anonymous namespace
66
67// static
68FramebufferVk *FramebufferVk::CreateUserFBO(const gl::FramebufferState &state)
69{
70 return new FramebufferVk(state);
71}
72
73// static
74FramebufferVk *FramebufferVk::CreateDefaultFBO(const gl::FramebufferState &state,
75 WindowSurfaceVk *backbuffer)
76{
77 return new FramebufferVk(state, backbuffer);
78}
79
Jamie Madillab9f9c32017-01-17 17:47:34 -050080FramebufferVk::FramebufferVk(const gl::FramebufferState &state)
Jamie Madilldd43e6c2017-03-24 14:18:49 -040081 : FramebufferImpl(state), mBackbuffer(nullptr), mRenderPass(), mFramebuffer()
Jamie Madill9e54b5a2016-05-25 12:57:39 -040082{
83}
84
Jamie Madill7b57b9d2017-01-13 09:33:38 -050085FramebufferVk::FramebufferVk(const gl::FramebufferState &state, WindowSurfaceVk *backbuffer)
Jamie Madilldd43e6c2017-03-24 14:18:49 -040086 : FramebufferImpl(state), mBackbuffer(backbuffer), mRenderPass(), mFramebuffer()
Jamie Madill7b57b9d2017-01-13 09:33:38 -050087{
88}
89
Jamie Madill9e54b5a2016-05-25 12:57:39 -040090FramebufferVk::~FramebufferVk()
91{
92}
93
Jamie Madill5deea722017-02-16 10:44:46 -050094void FramebufferVk::destroy(ContextImpl *contextImpl)
95{
96 VkDevice device = GetAs<ContextVk>(contextImpl)->getDevice();
97
98 mRenderPass.destroy(device);
99 mFramebuffer.destroy(device);
100}
101
102void FramebufferVk::destroyDefault(DisplayImpl *displayImpl)
103{
104 VkDevice device = GetAs<DisplayVk>(displayImpl)->getRenderer()->getDevice();
105
106 mRenderPass.destroy(device);
107 mFramebuffer.destroy(device);
108}
109
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400110gl::Error FramebufferVk::discard(size_t count, const GLenum *attachments)
111{
112 UNIMPLEMENTED();
113 return gl::Error(GL_INVALID_OPERATION);
114}
115
116gl::Error FramebufferVk::invalidate(size_t count, const GLenum *attachments)
117{
118 UNIMPLEMENTED();
119 return gl::Error(GL_INVALID_OPERATION);
120}
121
122gl::Error FramebufferVk::invalidateSub(size_t count,
123 const GLenum *attachments,
124 const gl::Rectangle &area)
125{
126 UNIMPLEMENTED();
127 return gl::Error(GL_INVALID_OPERATION);
128}
129
130gl::Error FramebufferVk::clear(ContextImpl *context, GLbitfield mask)
131{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500132 ContextVk *contextVk = GetAs<ContextVk>(context);
133
134 if (mState.getDepthAttachment() && (mask & GL_DEPTH_BUFFER_BIT) != 0)
135 {
136 // TODO(jmadill): Depth clear
137 UNIMPLEMENTED();
138 }
139
140 if (mState.getStencilAttachment() && (mask & GL_STENCIL_BUFFER_BIT) != 0)
141 {
142 // TODO(jmadill): Stencil clear
143 UNIMPLEMENTED();
144 }
145
146 if ((mask & GL_COLOR_BUFFER_BIT) == 0)
147 {
148 return gl::NoError();
149 }
150
151 const auto &glState = context->getGLState();
152 const auto &clearColor = glState.getColorClearValue();
153 VkClearColorValue clearColorValue;
154 clearColorValue.float32[0] = clearColor.red;
155 clearColorValue.float32[1] = clearColor.green;
156 clearColorValue.float32[2] = clearColor.blue;
157 clearColorValue.float32[3] = clearColor.alpha;
158
159 // TODO(jmadill): Scissored clears.
160 const auto *attachment = mState.getFirstNonNullAttachment();
161 ASSERT(attachment && attachment->isAttached());
162 const auto &size = attachment->getSize();
163 const gl::Rectangle renderArea(0, 0, size.width, size.height);
164
Jamie Madill0c0dc342017-03-24 14:18:51 -0400165 vk::CommandBuffer *commandBuffer = nullptr;
166 ANGLE_TRY(contextVk->getStartedCommandBuffer(&commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500167
168 for (const auto &colorAttachment : mState.getColorAttachments())
169 {
170 if (colorAttachment.isAttached())
171 {
172 RenderTargetVk *renderTarget = nullptr;
173 ANGLE_TRY(colorAttachment.getRenderTarget(&renderTarget));
174 renderTarget->image->changeLayoutTop(
175 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, commandBuffer);
176 commandBuffer->clearSingleColorImage(*renderTarget->image, clearColorValue);
177 }
178 }
179
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500180 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400181}
182
183gl::Error FramebufferVk::clearBufferfv(ContextImpl *context,
184 GLenum buffer,
185 GLint drawbuffer,
186 const GLfloat *values)
187{
188 UNIMPLEMENTED();
189 return gl::Error(GL_INVALID_OPERATION);
190}
191
192gl::Error FramebufferVk::clearBufferuiv(ContextImpl *context,
193 GLenum buffer,
194 GLint drawbuffer,
195 const GLuint *values)
196{
197 UNIMPLEMENTED();
198 return gl::Error(GL_INVALID_OPERATION);
199}
200
201gl::Error FramebufferVk::clearBufferiv(ContextImpl *context,
202 GLenum buffer,
203 GLint drawbuffer,
204 const GLint *values)
205{
206 UNIMPLEMENTED();
207 return gl::Error(GL_INVALID_OPERATION);
208}
209
210gl::Error FramebufferVk::clearBufferfi(ContextImpl *context,
211 GLenum buffer,
212 GLint drawbuffer,
213 GLfloat depth,
214 GLint stencil)
215{
216 UNIMPLEMENTED();
217 return gl::Error(GL_INVALID_OPERATION);
218}
219
220GLenum FramebufferVk::getImplementationColorReadFormat() const
221{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500222 auto errOrResult = GetReadAttachmentInfo(mState.getReadAttachment());
223
224 // TODO(jmadill): Handle getRenderTarget error.
225 if (errOrResult.isError())
226 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500227 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500228 return GL_NONE;
229 }
230
231 return errOrResult.getResult()->format;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400232}
233
234GLenum FramebufferVk::getImplementationColorReadType() const
235{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500236 auto errOrResult = GetReadAttachmentInfo(mState.getReadAttachment());
237
238 // TODO(jmadill): Handle getRenderTarget error.
239 if (errOrResult.isError())
240 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500241 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500242 return GL_NONE;
243 }
244
245 return errOrResult.getResult()->type;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400246}
247
248gl::Error FramebufferVk::readPixels(ContextImpl *context,
249 const gl::Rectangle &area,
250 GLenum format,
251 GLenum type,
252 GLvoid *pixels) const
253{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500254 const auto &glState = context->getGLState();
255 const auto *readFramebuffer = glState.getReadFramebuffer();
256 const auto *readAttachment = readFramebuffer->getReadColorbuffer();
257
258 RenderTargetVk *renderTarget = nullptr;
259 ANGLE_TRY(readAttachment->getRenderTarget(&renderTarget));
260
261 ContextVk *contextVk = GetAs<ContextVk>(context);
262 RendererVk *renderer = contextVk->getRenderer();
Jamie Madill5deea722017-02-16 10:44:46 -0500263 VkDevice device = renderer->getDevice();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500264
265 vk::Image *readImage = renderTarget->image;
266 vk::StagingImage stagingImage;
Jamie Madill5deea722017-02-16 10:44:46 -0500267 ANGLE_TRY(renderer->createStagingImage(TextureDimension::TEX_2D, *renderTarget->format,
268 renderTarget->extents, &stagingImage));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500269
Jamie Madill0c0dc342017-03-24 14:18:51 -0400270 vk::CommandBuffer *commandBuffer = nullptr;
271 ANGLE_TRY(contextVk->getStartedCommandBuffer(&commandBuffer));
272
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500273 stagingImage.getImage().changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_GENERAL,
274 commandBuffer);
275
276 gl::Box copyRegion;
277 copyRegion.x = area.x;
278 copyRegion.y = area.y;
279 copyRegion.z = 0;
280 copyRegion.width = area.width;
281 copyRegion.height = area.height;
282 copyRegion.depth = 1;
283
284 readImage->changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
285 commandBuffer);
286 commandBuffer->copySingleImage(*readImage, stagingImage.getImage(), copyRegion,
287 VK_IMAGE_ASPECT_COLOR_BIT);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500288
Jamie Madill0c0dc342017-03-24 14:18:51 -0400289 ANGLE_TRY(renderer->submitAndFinishCommandBuffer(commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500290
291 // TODO(jmadill): parameters
292 uint8_t *mapPointer = nullptr;
Jamie Madill5deea722017-02-16 10:44:46 -0500293 ANGLE_TRY(
294 stagingImage.getDeviceMemory().map(device, 0, stagingImage.getSize(), 0, &mapPointer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500295
296 const auto &angleFormat = renderTarget->format->format();
297
298 // TODO(jmadill): Use pixel bytes from the ANGLE format directly.
Geoff Langca271392017-04-05 12:30:00 -0400299 const auto &glFormat = gl::GetSizedInternalFormatInfo(angleFormat.glInternalFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500300 int inputPitch = glFormat.pixelBytes * area.width;
301
302 PackPixelsParams params;
303 params.area = area;
304 params.format = format;
305 params.type = type;
306 params.outputPitch = inputPitch;
307 params.pack = glState.getPackState();
308
309 PackPixels(params, angleFormat, inputPitch, mapPointer, reinterpret_cast<uint8_t *>(pixels));
310
Jamie Madill5deea722017-02-16 10:44:46 -0500311 stagingImage.getDeviceMemory().unmap(device);
Jamie Madillf651c772017-02-21 15:03:51 -0500312 renderer->enqueueGarbage(renderer->getCurrentQueueSerial(), std::move(stagingImage));
313
314 stagingImage.getImage().destroy(renderer->getDevice());
Jamie Madill5deea722017-02-16 10:44:46 -0500315
316 stagingImage.destroy(device);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500317
318 return vk::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400319}
320
321gl::Error FramebufferVk::blit(ContextImpl *context,
322 const gl::Rectangle &sourceArea,
323 const gl::Rectangle &destArea,
324 GLbitfield mask,
325 GLenum filter)
326{
327 UNIMPLEMENTED();
328 return gl::Error(GL_INVALID_OPERATION);
329}
330
331bool FramebufferVk::checkStatus() const
332{
333 UNIMPLEMENTED();
334 return bool();
335}
336
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400337void FramebufferVk::syncState(ContextImpl *contextImpl, const gl::Framebuffer::DirtyBits &dirtyBits)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400338{
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400339 auto contextVk = GetAs<ContextVk>(contextImpl);
340
341 ASSERT(dirtyBits.any());
342
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500343 // TODO(jmadill): Smarter update.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400344 mRenderPass.destroy(contextVk->getDevice());
345 mFramebuffer.destroy(contextVk->getDevice());
Jamie Madill72106562017-03-24 14:18:50 -0400346
347 // TODO(jmadill): Use pipeline cache.
348 contextVk->invalidateCurrentPipeline();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500349}
350
351gl::ErrorOrResult<vk::RenderPass *> FramebufferVk::getRenderPass(VkDevice device)
352{
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400353 if (mRenderPass.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500354 {
355 return &mRenderPass;
356 }
357
358 // TODO(jmadill): Can we use stack-only memory?
359 std::vector<VkAttachmentDescription> attachmentDescs;
360 std::vector<VkAttachmentReference> colorAttachmentRefs;
361
362 const auto &colorAttachments = mState.getColorAttachments();
363 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
364 {
365 const auto &colorAttachment = colorAttachments[attachmentIndex];
366 if (colorAttachment.isAttached())
367 {
368 VkAttachmentDescription colorDesc;
369 VkAttachmentReference colorRef;
370
371 RenderTargetVk *renderTarget = nullptr;
372 ANGLE_TRY(colorAttachment.getRenderTarget(&renderTarget));
373
374 // TODO(jmadill): We would only need this flag for duplicated attachments.
375 colorDesc.flags = VK_ATTACHMENT_DESCRIPTION_MAY_ALIAS_BIT;
376 colorDesc.format = renderTarget->format->native;
377 colorDesc.samples = ConvertSamples(colorAttachment.getSamples());
378
379 // The load op controls the prior existing depth/color attachment data.
380 // TODO(jmadill): Proper load ops. Should not be hard coded to clear.
381 colorDesc.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
382 colorDesc.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
383 colorDesc.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
384 colorDesc.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
385 colorDesc.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500386
387 // We might want to transition directly to PRESENT_SRC for Surface attachments.
388 colorDesc.finalLayout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500389
390 colorRef.attachment = static_cast<uint32_t>(colorAttachments.size()) - 1u;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500391 colorRef.layout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500392
393 attachmentDescs.push_back(colorDesc);
394 colorAttachmentRefs.push_back(colorRef);
395 }
396 }
397
398 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
399 VkAttachmentReference depthStencilAttachmentRef;
400 bool useDepth = depthStencilAttachment && depthStencilAttachment->isAttached();
401
402 if (useDepth)
403 {
404 VkAttachmentDescription depthStencilDesc;
405
406 RenderTargetVk *renderTarget = nullptr;
407 ANGLE_TRY(depthStencilAttachment->getRenderTarget(&renderTarget));
408
409 depthStencilDesc.flags = 0;
410 depthStencilDesc.format = renderTarget->format->native;
411 depthStencilDesc.samples = ConvertSamples(depthStencilAttachment->getSamples());
412 depthStencilDesc.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
413 depthStencilDesc.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
414 depthStencilDesc.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
415 depthStencilDesc.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
416 depthStencilDesc.initialLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
417 depthStencilDesc.finalLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
418
419 depthStencilAttachmentRef.attachment = static_cast<uint32_t>(attachmentDescs.size());
420 depthStencilAttachmentRef.layout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
421
422 attachmentDescs.push_back(depthStencilDesc);
423 }
424
425 ASSERT(!attachmentDescs.empty());
426
427 VkSubpassDescription subpassDesc;
428
429 subpassDesc.flags = 0;
430 subpassDesc.pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS;
431 subpassDesc.inputAttachmentCount = 0;
432 subpassDesc.pInputAttachments = nullptr;
433 subpassDesc.colorAttachmentCount = static_cast<uint32_t>(colorAttachmentRefs.size());
434 subpassDesc.pColorAttachments = colorAttachmentRefs.data();
435 subpassDesc.pResolveAttachments = nullptr;
436 subpassDesc.pDepthStencilAttachment = (useDepth ? &depthStencilAttachmentRef : nullptr);
437 subpassDesc.preserveAttachmentCount = 0;
438 subpassDesc.pPreserveAttachments = nullptr;
439
440 VkRenderPassCreateInfo renderPassInfo;
441
442 renderPassInfo.sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO;
443 renderPassInfo.pNext = nullptr;
444 renderPassInfo.flags = 0;
445 renderPassInfo.attachmentCount = static_cast<uint32_t>(attachmentDescs.size());
446 renderPassInfo.pAttachments = attachmentDescs.data();
447 renderPassInfo.subpassCount = 1;
448 renderPassInfo.pSubpasses = &subpassDesc;
449 renderPassInfo.dependencyCount = 0;
450 renderPassInfo.pDependencies = nullptr;
451
Jamie Madill5deea722017-02-16 10:44:46 -0500452 vk::RenderPass renderPass;
453 ANGLE_TRY(renderPass.init(device, renderPassInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500454
Jamie Madill5deea722017-02-16 10:44:46 -0500455 mRenderPass.retain(device, std::move(renderPass));
456
Jamie Madillab9f9c32017-01-17 17:47:34 -0500457 return &mRenderPass;
458}
459
460gl::ErrorOrResult<vk::Framebuffer *> FramebufferVk::getFramebuffer(VkDevice device)
461{
462 // If we've already created our cached Framebuffer, return it.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400463 if (mFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500464 {
465 return &mFramebuffer;
466 }
467
468 vk::RenderPass *renderPass = nullptr;
469 ANGLE_TRY_RESULT(getRenderPass(device), renderPass);
470
471 // If we've a Framebuffer provided by a Surface (default FBO/backbuffer), query it.
472 if (mBackbuffer)
473 {
474 return mBackbuffer->getCurrentFramebuffer(device, *renderPass);
475 }
476
477 // Gather VkImageViews over all FBO attachments, also size of attached region.
478 std::vector<VkImageView> attachments;
479 gl::Extents attachmentsSize;
480
481 const auto &colorAttachments = mState.getColorAttachments();
482 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
483 {
484 const auto &colorAttachment = colorAttachments[attachmentIndex];
485 if (colorAttachment.isAttached())
486 {
487 RenderTargetVk *renderTarget = nullptr;
488 ANGLE_TRY(colorAttachment.getRenderTarget<RenderTargetVk>(&renderTarget));
489 attachments.push_back(renderTarget->imageView->getHandle());
490
491 ASSERT(attachmentsSize.empty() || attachmentsSize == colorAttachment.getSize());
492 attachmentsSize = colorAttachment.getSize();
493 }
494 }
495
496 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
497 if (depthStencilAttachment && depthStencilAttachment->isAttached())
498 {
499 RenderTargetVk *renderTarget = nullptr;
500 ANGLE_TRY(depthStencilAttachment->getRenderTarget<RenderTargetVk>(&renderTarget));
501 attachments.push_back(renderTarget->imageView->getHandle());
502
503 ASSERT(attachmentsSize.empty() || attachmentsSize == depthStencilAttachment->getSize());
504 attachmentsSize = depthStencilAttachment->getSize();
505 }
506
507 ASSERT(!attachments.empty());
508
509 VkFramebufferCreateInfo framebufferInfo;
510
511 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
512 framebufferInfo.pNext = nullptr;
513 framebufferInfo.flags = 0;
514 framebufferInfo.renderPass = mRenderPass.getHandle();
515 framebufferInfo.attachmentCount = static_cast<uint32_t>(attachments.size());
516 framebufferInfo.pAttachments = attachments.data();
517 framebufferInfo.width = static_cast<uint32_t>(attachmentsSize.width);
518 framebufferInfo.height = static_cast<uint32_t>(attachmentsSize.height);
519 framebufferInfo.layers = 1;
520
Jamie Madill5deea722017-02-16 10:44:46 -0500521 vk::Framebuffer framebuffer;
522 ANGLE_TRY(static_cast<gl::Error>(framebuffer.init(device, framebufferInfo)));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500523
Jamie Madill5deea722017-02-16 10:44:46 -0500524 mFramebuffer.retain(device, std::move(framebuffer));
525
Jamie Madillab9f9c32017-01-17 17:47:34 -0500526 return &mFramebuffer;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400527}
528
JiangYizhoubddc46b2016-12-09 09:50:51 +0800529gl::Error FramebufferVk::getSamplePosition(size_t index, GLfloat *xy) const
530{
531 UNIMPLEMENTED();
532 return gl::InternalError() << "getSamplePosition is unimplemented.";
533}
534
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500535gl::Error FramebufferVk::beginRenderPass(VkDevice device,
536 vk::CommandBuffer *commandBuffer,
Jamie Madill4c26fc22017-02-24 11:04:10 -0500537 Serial queueSerial,
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500538 const gl::State &glState)
539{
Jamie Madill4c26fc22017-02-24 11:04:10 -0500540 // TODO(jmadill): Cache render targets.
541 for (const auto &colorAttachment : mState.getColorAttachments())
542 {
543 if (colorAttachment.isAttached())
544 {
545 RenderTargetVk *renderTarget = nullptr;
546 ANGLE_TRY(colorAttachment.getRenderTarget<RenderTargetVk>(&renderTarget));
547 renderTarget->resource->setQueueSerial(queueSerial);
548 }
549 }
550
551 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
552 if (depthStencilAttachment && depthStencilAttachment->isAttached())
553 {
554 RenderTargetVk *renderTarget = nullptr;
555 ANGLE_TRY(depthStencilAttachment->getRenderTarget<RenderTargetVk>(&renderTarget));
556 renderTarget->resource->setQueueSerial(queueSerial);
557 }
558
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500559 vk::Framebuffer *framebuffer = nullptr;
560 ANGLE_TRY_RESULT(getFramebuffer(device), framebuffer);
561 ASSERT(framebuffer && framebuffer->valid());
562
563 vk::RenderPass *renderPass = nullptr;
564 ANGLE_TRY_RESULT(getRenderPass(device), renderPass);
565 ASSERT(renderPass && renderPass->valid());
566
567 // TODO(jmadill): Proper clear value implementation.
568 VkClearColorValue colorClear;
569 memset(&colorClear, 0, sizeof(VkClearColorValue));
570 colorClear.float32[0] = glState.getColorClearValue().red;
571 colorClear.float32[1] = glState.getColorClearValue().green;
572 colorClear.float32[2] = glState.getColorClearValue().blue;
573 colorClear.float32[3] = glState.getColorClearValue().alpha;
574
575 std::vector<VkClearValue> attachmentClearValues;
576 attachmentClearValues.push_back({colorClear});
577
578 // Updated the cached image layout of the attachments in this FBO.
579 // For a default FBO, we need to call through to the WindowSurfaceVk
580 // TODO(jmadill): Iterate over all attachments.
581 ASSERT(mBackbuffer);
582 RenderTargetVk *renderTarget = nullptr;
583 ANGLE_TRY(mState.getFirstColorAttachment()->getRenderTarget(&renderTarget));
584 renderTarget->image->updateLayout(VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
585
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500586 commandBuffer->beginRenderPass(*renderPass, *framebuffer, glState.getViewport(),
587 attachmentClearValues);
Jamie Madill4c26fc22017-02-24 11:04:10 -0500588
589 setQueueSerial(queueSerial);
590 if (mBackbuffer)
591 {
592 mBackbuffer->setQueueSerial(queueSerial);
593 }
594
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500595 return gl::NoError();
596}
597
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400598} // namespace rx