blob: 636825b30ee1b2c407e2f4385e87c0d5b8c76eb2 [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 Madill49ac74b2017-12-21 14:42:33 -050021#include "libANGLE/renderer/vulkan/CommandBufferNode.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"
27#include "libANGLE/renderer/vulkan/formatutilsvk.h"
Jamie Madill9e54b5a2016-05-25 12:57:39 -040028
29namespace rx
30{
31
Jamie Madill7b57b9d2017-01-13 09:33:38 -050032namespace
33{
Jamie Madill7b57b9d2017-01-13 09:33:38 -050034gl::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
Jamie Madill1d7be502017-10-29 18:06:50 -040041 GLenum implFormat = renderTarget->format->textureFormat().fboImplementationInternalFormat;
Geoff Langca271392017-04-05 12:30:00 -040042 return &gl::GetSizedInternalFormatInfo(implFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -050043}
Jamie Madill7b57b9d2017-01-13 09:33:38 -050044} // anonymous namespace
45
46// static
47FramebufferVk *FramebufferVk::CreateUserFBO(const gl::FramebufferState &state)
48{
49 return new FramebufferVk(state);
50}
51
52// static
53FramebufferVk *FramebufferVk::CreateDefaultFBO(const gl::FramebufferState &state,
54 WindowSurfaceVk *backbuffer)
55{
56 return new FramebufferVk(state, backbuffer);
57}
58
Jamie Madillab9f9c32017-01-17 17:47:34 -050059FramebufferVk::FramebufferVk(const gl::FramebufferState &state)
Jamie Madill49ac74b2017-12-21 14:42:33 -050060 : FramebufferImpl(state),
61 mBackbuffer(nullptr),
62 mRenderPassDesc(),
63 mFramebuffer(),
64 mRenderNodeDirty(false)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040065{
66}
67
Jamie Madill7b57b9d2017-01-13 09:33:38 -050068FramebufferVk::FramebufferVk(const gl::FramebufferState &state, WindowSurfaceVk *backbuffer)
Jamie Madill49ac74b2017-12-21 14:42:33 -050069 : FramebufferImpl(state),
70 mBackbuffer(backbuffer),
71 mRenderPassDesc(),
72 mFramebuffer(),
73 mRenderNodeDirty(false)
Jamie Madill7b57b9d2017-01-13 09:33:38 -050074{
75}
76
Jamie Madill9e54b5a2016-05-25 12:57:39 -040077FramebufferVk::~FramebufferVk()
78{
79}
80
Jamie Madillc564c072017-06-01 12:45:42 -040081void FramebufferVk::destroy(const gl::Context *context)
Jamie Madill5deea722017-02-16 10:44:46 -050082{
Jamie Madille1f3ad42017-10-28 23:00:42 -040083 RendererVk *renderer = vk::GetImpl(context)->getRenderer();
Jamie Madill5deea722017-02-16 10:44:46 -050084
Jamie Madill526543c2017-10-28 10:59:16 -040085 renderer->releaseResource(*this, &mFramebuffer);
Jamie Madill5deea722017-02-16 10:44:46 -050086}
87
Jamie Madillc564c072017-06-01 12:45:42 -040088void FramebufferVk::destroyDefault(const egl::Display *display)
Jamie Madill5deea722017-02-16 10:44:46 -050089{
Jamie Madille1f3ad42017-10-28 23:00:42 -040090 VkDevice device = vk::GetImpl(display)->getRenderer()->getDevice();
Jamie Madill5deea722017-02-16 10:44:46 -050091
Jamie Madill5deea722017-02-16 10:44:46 -050092 mFramebuffer.destroy(device);
93}
94
Jamie Madill4928b7c2017-06-20 12:57:39 -040095gl::Error FramebufferVk::discard(const gl::Context *context,
96 size_t count,
97 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040098{
99 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500100 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400101}
102
Jamie Madill4928b7c2017-06-20 12:57:39 -0400103gl::Error FramebufferVk::invalidate(const gl::Context *context,
104 size_t count,
105 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400106{
107 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500108 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400109}
110
Jamie Madill4928b7c2017-06-20 12:57:39 -0400111gl::Error FramebufferVk::invalidateSub(const gl::Context *context,
112 size_t count,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400113 const GLenum *attachments,
114 const gl::Rectangle &area)
115{
116 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500117 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400118}
119
Jamie Madillc564c072017-06-01 12:45:42 -0400120gl::Error FramebufferVk::clear(const gl::Context *context, GLbitfield mask)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400121{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500122 if (mState.getDepthAttachment() && (mask & GL_DEPTH_BUFFER_BIT) != 0)
123 {
124 // TODO(jmadill): Depth clear
125 UNIMPLEMENTED();
126 }
127
128 if (mState.getStencilAttachment() && (mask & GL_STENCIL_BUFFER_BIT) != 0)
129 {
130 // TODO(jmadill): Stencil clear
131 UNIMPLEMENTED();
132 }
133
134 if ((mask & GL_COLOR_BUFFER_BIT) == 0)
135 {
136 return gl::NoError();
137 }
138
139 const auto &glState = context->getGLState();
140 const auto &clearColor = glState.getColorClearValue();
141 VkClearColorValue clearColorValue;
142 clearColorValue.float32[0] = clearColor.red;
143 clearColorValue.float32[1] = clearColor.green;
144 clearColorValue.float32[2] = clearColor.blue;
145 clearColorValue.float32[3] = clearColor.alpha;
146
147 // TODO(jmadill): Scissored clears.
148 const auto *attachment = mState.getFirstNonNullAttachment();
149 ASSERT(attachment && attachment->isAttached());
150 const auto &size = attachment->getSize();
151 const gl::Rectangle renderArea(0, 0, size.width, size.height);
152
Jamie Madill49ac74b2017-12-21 14:42:33 -0500153 RendererVk *renderer = vk::GetImpl(context)->getRenderer();
154
155 vk::CommandBuffer *commandBuffer = nullptr;
156 ANGLE_TRY(recordWriteCommands(renderer, &commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500157
158 for (const auto &colorAttachment : mState.getColorAttachments())
159 {
160 if (colorAttachment.isAttached())
161 {
162 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400163 ANGLE_TRY(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500164 renderTarget->image->changeLayoutTop(
165 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, commandBuffer);
166 commandBuffer->clearSingleColorImage(*renderTarget->image, clearColorValue);
167 }
168 }
169
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500170 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400171}
172
Jamie Madillc564c072017-06-01 12:45:42 -0400173gl::Error FramebufferVk::clearBufferfv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400174 GLenum buffer,
175 GLint drawbuffer,
176 const GLfloat *values)
177{
178 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500179 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400180}
181
Jamie Madillc564c072017-06-01 12:45:42 -0400182gl::Error FramebufferVk::clearBufferuiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400183 GLenum buffer,
184 GLint drawbuffer,
185 const GLuint *values)
186{
187 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500188 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400189}
190
Jamie Madillc564c072017-06-01 12:45:42 -0400191gl::Error FramebufferVk::clearBufferiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400192 GLenum buffer,
193 GLint drawbuffer,
194 const GLint *values)
195{
196 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500197 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400198}
199
Jamie Madillc564c072017-06-01 12:45:42 -0400200gl::Error FramebufferVk::clearBufferfi(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400201 GLenum buffer,
202 GLint drawbuffer,
203 GLfloat depth,
204 GLint stencil)
205{
206 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500207 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400208}
209
Jamie Madill4928b7c2017-06-20 12:57:39 -0400210GLenum FramebufferVk::getImplementationColorReadFormat(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400211{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400212 auto errOrResult = GetReadAttachmentInfo(context, mState.getReadAttachment());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500213
214 // TODO(jmadill): Handle getRenderTarget error.
215 if (errOrResult.isError())
216 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500217 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500218 return GL_NONE;
219 }
220
221 return errOrResult.getResult()->format;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400222}
223
Jamie Madill4928b7c2017-06-20 12:57:39 -0400224GLenum FramebufferVk::getImplementationColorReadType(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400225{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400226 auto errOrResult = GetReadAttachmentInfo(context, mState.getReadAttachment());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500227
228 // TODO(jmadill): Handle getRenderTarget error.
229 if (errOrResult.isError())
230 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500231 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500232 return GL_NONE;
233 }
234
235 return errOrResult.getResult()->type;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400236}
237
Jamie Madillc564c072017-06-01 12:45:42 -0400238gl::Error FramebufferVk::readPixels(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400239 const gl::Rectangle &area,
240 GLenum format,
241 GLenum type,
Jamie Madilld4826152017-09-21 11:18:59 -0400242 void *pixels)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400243{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500244 const auto &glState = context->getGLState();
245 const auto *readFramebuffer = glState.getReadFramebuffer();
246 const auto *readAttachment = readFramebuffer->getReadColorbuffer();
247
248 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400249 ANGLE_TRY(readAttachment->getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500250
Jamie Madille1f3ad42017-10-28 23:00:42 -0400251 ContextVk *contextVk = vk::GetImpl(context);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500252 RendererVk *renderer = contextVk->getRenderer();
Jamie Madill5deea722017-02-16 10:44:46 -0500253 VkDevice device = renderer->getDevice();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500254
255 vk::Image *readImage = renderTarget->image;
256 vk::StagingImage stagingImage;
Jamie Madill5deea722017-02-16 10:44:46 -0500257 ANGLE_TRY(renderer->createStagingImage(TextureDimension::TEX_2D, *renderTarget->format,
Jamie Madill035fd6b2017-10-03 15:43:22 -0400258 renderTarget->extents, vk::StagingUsage::Read,
259 &stagingImage));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500260
Jamie Madill49ac74b2017-12-21 14:42:33 -0500261 vk::CommandBuffer *commandBuffer = nullptr;
262 ANGLE_TRY(recordWriteCommands(renderer, &commandBuffer));
Jamie Madilld4826152017-09-21 11:18:59 -0400263
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500264 stagingImage.getImage().changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_GENERAL,
265 commandBuffer);
266
Jamie Madilld33c77c2017-11-09 13:08:30 -0500267 readImage->changeLayoutWithStages(
268 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
269 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT, VK_PIPELINE_STAGE_TRANSFER_BIT, commandBuffer);
Jamie Madill815a6c92017-10-21 14:33:04 -0400270
271 VkImageCopy region;
272 region.srcSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
273 region.srcSubresource.mipLevel = 0;
274 region.srcSubresource.baseArrayLayer = 0;
275 region.srcSubresource.layerCount = 1;
276 region.srcOffset.x = area.x;
277 region.srcOffset.y = area.y;
278 region.srcOffset.z = 0;
279 region.dstSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
280 region.dstSubresource.mipLevel = 0;
281 region.dstSubresource.baseArrayLayer = 0;
282 region.dstSubresource.layerCount = 1;
283 region.dstOffset.x = 0;
284 region.dstOffset.y = 0;
285 region.dstOffset.z = 0;
286 region.extent.width = area.width;
287 region.extent.height = area.height;
288 region.extent.depth = 1;
289
290 commandBuffer->copyImage(*readImage, stagingImage.getImage(), 1, &region);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500291
Jamie Madill49ac74b2017-12-21 14:42:33 -0500292 // Triggers a full finish.
293 // TODO(jmadill): Don't block on asynchronous readback.
294 ANGLE_TRY(renderer->finish(context));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500295
296 // TODO(jmadill): parameters
297 uint8_t *mapPointer = nullptr;
Jamie Madill5deea722017-02-16 10:44:46 -0500298 ANGLE_TRY(
299 stagingImage.getDeviceMemory().map(device, 0, stagingImage.getSize(), 0, &mapPointer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500300
Jamie Madill1d7be502017-10-29 18:06:50 -0400301 const auto &angleFormat = renderTarget->format->textureFormat();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500302
303 // TODO(jmadill): Use pixel bytes from the ANGLE format directly.
Geoff Langca271392017-04-05 12:30:00 -0400304 const auto &glFormat = gl::GetSizedInternalFormatInfo(angleFormat.glInternalFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500305 int inputPitch = glFormat.pixelBytes * area.width;
306
307 PackPixelsParams params;
308 params.area = area;
309 params.format = format;
310 params.type = type;
311 params.outputPitch = inputPitch;
Corentin Wallez336129f2017-10-17 15:55:40 -0400312 params.packBuffer = glState.getTargetBuffer(gl::BufferBinding::PixelPack);
Corentin Wallezcda6af12017-10-30 19:20:37 -0400313 params.pack = glState.getPackState();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500314
315 PackPixels(params, angleFormat, inputPitch, mapPointer, reinterpret_cast<uint8_t *>(pixels));
316
Jamie Madill5deea722017-02-16 10:44:46 -0500317 stagingImage.getDeviceMemory().unmap(device);
Jamie Madille88ec8e2017-10-31 17:18:14 -0400318 renderer->releaseObject(renderer->getCurrentQueueSerial(), &stagingImage);
Jamie Madillf651c772017-02-21 15:03:51 -0500319
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500320 return vk::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400321}
322
Jamie Madillc564c072017-06-01 12:45:42 -0400323gl::Error FramebufferVk::blit(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400324 const gl::Rectangle &sourceArea,
325 const gl::Rectangle &destArea,
326 GLbitfield mask,
327 GLenum filter)
328{
329 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500330 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400331}
332
Kenneth Russellce8602a2017-10-03 18:23:08 -0700333bool FramebufferVk::checkStatus(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400334{
Jamie Madillb79e7bb2017-10-24 13:55:50 -0400335 return true;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400336}
337
Jamie Madillc564c072017-06-01 12:45:42 -0400338void FramebufferVk::syncState(const gl::Context *context,
339 const gl::Framebuffer::DirtyBits &dirtyBits)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400340{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400341 ContextVk *contextVk = vk::GetImpl(context);
Jamie Madill7bd16662017-10-28 19:40:50 -0400342 RendererVk *renderer = contextVk->getRenderer();
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400343
344 ASSERT(dirtyBits.any());
345
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500346 // TODO(jmadill): Smarter update.
Jamie Madill9f2a8612017-11-30 12:43:09 -0500347 mRenderPassDesc.reset();
Jamie Madill7bd16662017-10-28 19:40:50 -0400348 renderer->releaseResource(*this, &mFramebuffer);
Jamie Madill49ac74b2017-12-21 14:42:33 -0500349
350 // Trigger a new set of secondary commands next time we render to this FBO,.
351 mRenderNodeDirty = true;
Jamie Madill72106562017-03-24 14:18:50 -0400352
353 // TODO(jmadill): Use pipeline cache.
354 contextVk->invalidateCurrentPipeline();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500355}
356
Jamie Madill9f2a8612017-11-30 12:43:09 -0500357const vk::RenderPassDesc &FramebufferVk::getRenderPassDesc(const gl::Context *context)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500358{
Jamie Madill9f2a8612017-11-30 12:43:09 -0500359 if (mRenderPassDesc.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500360 {
Jamie Madill9f2a8612017-11-30 12:43:09 -0500361 return mRenderPassDesc.value();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500362 }
363
Jamie Madill0b684ce2017-11-23 12:57:39 -0500364 vk::RenderPassDesc desc;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500365
366 const auto &colorAttachments = mState.getColorAttachments();
367 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
368 {
369 const auto &colorAttachment = colorAttachments[attachmentIndex];
370 if (colorAttachment.isAttached())
371 {
Jamie Madillab9f9c32017-01-17 17:47:34 -0500372 RenderTargetVk *renderTarget = nullptr;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500373 ANGLE_SWALLOW_ERR(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madillbef918c2017-12-13 13:11:30 -0500374 desc.packColorAttachment(*renderTarget->format, colorAttachment.getSamples());
Jamie Madillab9f9c32017-01-17 17:47:34 -0500375 }
376 }
377
378 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500379
Jamie Madill0b684ce2017-11-23 12:57:39 -0500380 if (depthStencilAttachment && depthStencilAttachment->isAttached())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500381 {
Jamie Madillab9f9c32017-01-17 17:47:34 -0500382 RenderTargetVk *renderTarget = nullptr;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500383 ANGLE_SWALLOW_ERR(depthStencilAttachment->getRenderTarget(context, &renderTarget));
Jamie Madillbef918c2017-12-13 13:11:30 -0500384 desc.packDepthStencilAttachment(*renderTarget->format,
385 depthStencilAttachment->getSamples());
Jamie Madillab9f9c32017-01-17 17:47:34 -0500386 }
387
Jamie Madill9f2a8612017-11-30 12:43:09 -0500388 mRenderPassDesc = desc;
389 return mRenderPassDesc.value();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500390}
391
Jamie Madill4928b7c2017-06-20 12:57:39 -0400392gl::ErrorOrResult<vk::Framebuffer *> FramebufferVk::getFramebuffer(const gl::Context *context,
Jamie Madill9f2a8612017-11-30 12:43:09 -0500393 RendererVk *rendererVk)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500394{
395 // If we've already created our cached Framebuffer, return it.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400396 if (mFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500397 {
398 return &mFramebuffer;
399 }
400
Jamie Madill9f2a8612017-11-30 12:43:09 -0500401 const vk::RenderPassDesc &desc = getRenderPassDesc(context);
402
Jamie Madillab9f9c32017-01-17 17:47:34 -0500403 vk::RenderPass *renderPass = nullptr;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500404 ANGLE_TRY(rendererVk->getCompatibleRenderPass(desc, &renderPass));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500405
406 // If we've a Framebuffer provided by a Surface (default FBO/backbuffer), query it.
Jamie Madill9f2a8612017-11-30 12:43:09 -0500407 VkDevice device = rendererVk->getDevice();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500408 if (mBackbuffer)
409 {
410 return mBackbuffer->getCurrentFramebuffer(device, *renderPass);
411 }
412
413 // Gather VkImageViews over all FBO attachments, also size of attached region.
414 std::vector<VkImageView> attachments;
415 gl::Extents attachmentsSize;
416
417 const auto &colorAttachments = mState.getColorAttachments();
418 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
419 {
420 const auto &colorAttachment = colorAttachments[attachmentIndex];
421 if (colorAttachment.isAttached())
422 {
423 RenderTargetVk *renderTarget = nullptr;
Jamie Madill49ac74b2017-12-21 14:42:33 -0500424 ANGLE_TRY(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500425 attachments.push_back(renderTarget->imageView->getHandle());
426
427 ASSERT(attachmentsSize.empty() || attachmentsSize == colorAttachment.getSize());
428 attachmentsSize = colorAttachment.getSize();
429 }
430 }
431
432 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
433 if (depthStencilAttachment && depthStencilAttachment->isAttached())
434 {
435 RenderTargetVk *renderTarget = nullptr;
Jamie Madill49ac74b2017-12-21 14:42:33 -0500436 ANGLE_TRY(depthStencilAttachment->getRenderTarget(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500437 attachments.push_back(renderTarget->imageView->getHandle());
438
439 ASSERT(attachmentsSize.empty() || attachmentsSize == depthStencilAttachment->getSize());
440 attachmentsSize = depthStencilAttachment->getSize();
441 }
442
443 ASSERT(!attachments.empty());
444
445 VkFramebufferCreateInfo framebufferInfo;
446
447 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
448 framebufferInfo.pNext = nullptr;
449 framebufferInfo.flags = 0;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500450 framebufferInfo.renderPass = renderPass->getHandle();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500451 framebufferInfo.attachmentCount = static_cast<uint32_t>(attachments.size());
452 framebufferInfo.pAttachments = attachments.data();
453 framebufferInfo.width = static_cast<uint32_t>(attachmentsSize.width);
454 framebufferInfo.height = static_cast<uint32_t>(attachmentsSize.height);
455 framebufferInfo.layers = 1;
456
Jamie Madill25301b62017-10-28 20:59:31 -0400457 ANGLE_TRY(mFramebuffer.init(device, framebufferInfo));
Jamie Madill5deea722017-02-16 10:44:46 -0500458
Jamie Madillab9f9c32017-01-17 17:47:34 -0500459 return &mFramebuffer;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400460}
461
JiangYizhoubddc46b2016-12-09 09:50:51 +0800462gl::Error FramebufferVk::getSamplePosition(size_t index, GLfloat *xy) const
463{
464 UNIMPLEMENTED();
465 return gl::InternalError() << "getSamplePosition is unimplemented.";
466}
467
Jamie Madill49ac74b2017-12-21 14:42:33 -0500468gl::Error FramebufferVk::getRenderNode(const gl::Context *context, vk::CommandBufferNode **nodeOut)
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500469{
Jamie Madill49ac74b2017-12-21 14:42:33 -0500470 ContextVk *contextVk = vk::GetImpl(context);
471 RendererVk *renderer = contextVk->getRenderer();
472 Serial currentSerial = renderer->getCurrentQueueSerial();
Jamie Madillbef918c2017-12-13 13:11:30 -0500473
Jamie Madill49ac74b2017-12-21 14:42:33 -0500474 if (isCurrentlyRecording(currentSerial) && !mRenderNodeDirty)
Jamie Madill4c26fc22017-02-24 11:04:10 -0500475 {
Jamie Madill49ac74b2017-12-21 14:42:33 -0500476 *nodeOut = getCurrentWriteNode(currentSerial);
477 ASSERT((*nodeOut)->getInsideRenderPassCommands()->valid());
478 return gl::NoError();
Jamie Madill4c26fc22017-02-24 11:04:10 -0500479 }
480
Jamie Madill49ac74b2017-12-21 14:42:33 -0500481 vk::CommandBufferNode *node = getNewWriteNode(renderer);
Jamie Madill4c26fc22017-02-24 11:04:10 -0500482
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500483 vk::Framebuffer *framebuffer = nullptr;
Jamie Madill49ac74b2017-12-21 14:42:33 -0500484 ANGLE_TRY_RESULT(getFramebuffer(context, renderer), framebuffer);
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500485
Jamie Madill1b038242017-11-01 15:14:36 -0400486 const gl::State &glState = context->getGLState();
Jamie Madill49ac74b2017-12-21 14:42:33 -0500487
488 // Hard-code RenderPass to clear the first render target to the current clear value.
489 // TODO(jmadill): Proper clear value implementation.
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500490 VkClearColorValue colorClear;
491 memset(&colorClear, 0, sizeof(VkClearColorValue));
492 colorClear.float32[0] = glState.getColorClearValue().red;
493 colorClear.float32[1] = glState.getColorClearValue().green;
494 colorClear.float32[2] = glState.getColorClearValue().blue;
495 colorClear.float32[3] = glState.getColorClearValue().alpha;
496
497 std::vector<VkClearValue> attachmentClearValues;
498 attachmentClearValues.push_back({colorClear});
499
Jamie Madill49ac74b2017-12-21 14:42:33 -0500500 node->storeRenderPassInfo(*framebuffer, glState.getViewport(), attachmentClearValues);
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500501
Jamie Madill49ac74b2017-12-21 14:42:33 -0500502 // Initialize RenderPass info.
503 // TODO(jmadill): Could cache this info, would require dependent state change messaging.
504 const auto &colorAttachments = mState.getColorAttachments();
505 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
Jamie Madill4c26fc22017-02-24 11:04:10 -0500506 {
Jamie Madill49ac74b2017-12-21 14:42:33 -0500507 const auto &colorAttachment = colorAttachments[attachmentIndex];
508 if (colorAttachment.isAttached())
509 {
510 RenderTargetVk *renderTarget = nullptr;
511 ANGLE_SWALLOW_ERR(colorAttachment.getRenderTarget(context, &renderTarget));
512
513 // TODO(jmadill): May need layout transition.
514 renderTarget->image->updateLayout(VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
515 node->appendColorRenderTarget(currentSerial, renderTarget);
516 }
Jamie Madill4c26fc22017-02-24 11:04:10 -0500517 }
518
Jamie Madill49ac74b2017-12-21 14:42:33 -0500519 const gl::FramebufferAttachment *depthStencilAttachment = mState.getDepthOrStencilAttachment();
520 if (depthStencilAttachment && depthStencilAttachment->isAttached())
521 {
522 RenderTargetVk *renderTarget = nullptr;
523 ANGLE_SWALLOW_ERR(depthStencilAttachment->getRenderTarget(context, &renderTarget));
524
525 // TODO(jmadill): May need layout transition.
526 renderTarget->image->updateLayout(VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL);
527 node->appendDepthStencilRenderTarget(currentSerial, renderTarget);
528 }
529
530 mRenderNodeDirty = false;
531
532 *nodeOut = node;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500533 return gl::NoError();
534}
535
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400536} // namespace rx