blob: 64ca87fbc2ef1ec170375ca0e4579e9903901b9d [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{
Jamie Madill7b57b9d2017-01-13 09:33:38 -050033gl::ErrorOrResult<const gl::InternalFormat *> GetReadAttachmentInfo(
Jamie Madill4928b7c2017-06-20 12:57:39 -040034 const gl::Context *context,
Jamie Madill7b57b9d2017-01-13 09:33:38 -050035 const gl::FramebufferAttachment *readAttachment)
36{
37 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -040038 ANGLE_TRY(readAttachment->getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -050039
Jamie Madill1d7be502017-10-29 18:06:50 -040040 GLenum implFormat = renderTarget->format->textureFormat().fboImplementationInternalFormat;
Geoff Langca271392017-04-05 12:30:00 -040041 return &gl::GetSizedInternalFormatInfo(implFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -050042}
Jamie Madill7b57b9d2017-01-13 09:33:38 -050043} // anonymous namespace
44
45// static
46FramebufferVk *FramebufferVk::CreateUserFBO(const gl::FramebufferState &state)
47{
48 return new FramebufferVk(state);
49}
50
51// static
52FramebufferVk *FramebufferVk::CreateDefaultFBO(const gl::FramebufferState &state,
53 WindowSurfaceVk *backbuffer)
54{
55 return new FramebufferVk(state, backbuffer);
56}
57
Jamie Madillab9f9c32017-01-17 17:47:34 -050058FramebufferVk::FramebufferVk(const gl::FramebufferState &state)
Jamie Madill9f2a8612017-11-30 12:43:09 -050059 : FramebufferImpl(state), mBackbuffer(nullptr), mRenderPassDesc(), mFramebuffer()
Jamie Madill9e54b5a2016-05-25 12:57:39 -040060{
61}
62
Jamie Madill7b57b9d2017-01-13 09:33:38 -050063FramebufferVk::FramebufferVk(const gl::FramebufferState &state, WindowSurfaceVk *backbuffer)
Jamie Madill9f2a8612017-11-30 12:43:09 -050064 : FramebufferImpl(state), mBackbuffer(backbuffer), mRenderPassDesc(), mFramebuffer()
Jamie Madill7b57b9d2017-01-13 09:33:38 -050065{
66}
67
Jamie Madill9e54b5a2016-05-25 12:57:39 -040068FramebufferVk::~FramebufferVk()
69{
70}
71
Jamie Madillc564c072017-06-01 12:45:42 -040072void FramebufferVk::destroy(const gl::Context *context)
Jamie Madill5deea722017-02-16 10:44:46 -050073{
Jamie Madille1f3ad42017-10-28 23:00:42 -040074 RendererVk *renderer = vk::GetImpl(context)->getRenderer();
Jamie Madill5deea722017-02-16 10:44:46 -050075
Jamie Madill526543c2017-10-28 10:59:16 -040076 renderer->releaseResource(*this, &mFramebuffer);
Jamie Madill5deea722017-02-16 10:44:46 -050077}
78
Jamie Madillc564c072017-06-01 12:45:42 -040079void FramebufferVk::destroyDefault(const egl::Display *display)
Jamie Madill5deea722017-02-16 10:44:46 -050080{
Jamie Madille1f3ad42017-10-28 23:00:42 -040081 VkDevice device = vk::GetImpl(display)->getRenderer()->getDevice();
Jamie Madill5deea722017-02-16 10:44:46 -050082
Jamie Madill5deea722017-02-16 10:44:46 -050083 mFramebuffer.destroy(device);
84}
85
Jamie Madill4928b7c2017-06-20 12:57:39 -040086gl::Error FramebufferVk::discard(const gl::Context *context,
87 size_t count,
88 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040089{
90 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050091 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040092}
93
Jamie Madill4928b7c2017-06-20 12:57:39 -040094gl::Error FramebufferVk::invalidate(const gl::Context *context,
95 size_t count,
96 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040097{
98 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050099 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400100}
101
Jamie Madill4928b7c2017-06-20 12:57:39 -0400102gl::Error FramebufferVk::invalidateSub(const gl::Context *context,
103 size_t count,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400104 const GLenum *attachments,
105 const gl::Rectangle &area)
106{
107 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500108 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400109}
110
Jamie Madillc564c072017-06-01 12:45:42 -0400111gl::Error FramebufferVk::clear(const gl::Context *context, GLbitfield mask)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400112{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400113 ContextVk *contextVk = vk::GetImpl(context);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500114
115 if (mState.getDepthAttachment() && (mask & GL_DEPTH_BUFFER_BIT) != 0)
116 {
117 // TODO(jmadill): Depth clear
118 UNIMPLEMENTED();
119 }
120
121 if (mState.getStencilAttachment() && (mask & GL_STENCIL_BUFFER_BIT) != 0)
122 {
123 // TODO(jmadill): Stencil clear
124 UNIMPLEMENTED();
125 }
126
127 if ((mask & GL_COLOR_BUFFER_BIT) == 0)
128 {
129 return gl::NoError();
130 }
131
132 const auto &glState = context->getGLState();
133 const auto &clearColor = glState.getColorClearValue();
134 VkClearColorValue clearColorValue;
135 clearColorValue.float32[0] = clearColor.red;
136 clearColorValue.float32[1] = clearColor.green;
137 clearColorValue.float32[2] = clearColor.blue;
138 clearColorValue.float32[3] = clearColor.alpha;
139
140 // TODO(jmadill): Scissored clears.
141 const auto *attachment = mState.getFirstNonNullAttachment();
142 ASSERT(attachment && attachment->isAttached());
143 const auto &size = attachment->getSize();
144 const gl::Rectangle renderArea(0, 0, size.width, size.height);
145
Jamie Madill7f738d42017-11-20 17:06:27 -0500146 vk::CommandBufferAndState *commandBuffer = nullptr;
Jamie Madill0c0dc342017-03-24 14:18:51 -0400147 ANGLE_TRY(contextVk->getStartedCommandBuffer(&commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500148
149 for (const auto &colorAttachment : mState.getColorAttachments())
150 {
151 if (colorAttachment.isAttached())
152 {
153 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400154 ANGLE_TRY(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500155 renderTarget->image->changeLayoutTop(
156 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, commandBuffer);
157 commandBuffer->clearSingleColorImage(*renderTarget->image, clearColorValue);
158 }
159 }
160
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500161 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400162}
163
Jamie Madillc564c072017-06-01 12:45:42 -0400164gl::Error FramebufferVk::clearBufferfv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400165 GLenum buffer,
166 GLint drawbuffer,
167 const GLfloat *values)
168{
169 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500170 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400171}
172
Jamie Madillc564c072017-06-01 12:45:42 -0400173gl::Error FramebufferVk::clearBufferuiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400174 GLenum buffer,
175 GLint drawbuffer,
176 const GLuint *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::clearBufferiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400183 GLenum buffer,
184 GLint drawbuffer,
185 const GLint *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::clearBufferfi(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400192 GLenum buffer,
193 GLint drawbuffer,
194 GLfloat depth,
195 GLint stencil)
196{
197 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500198 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400199}
200
Jamie Madill4928b7c2017-06-20 12:57:39 -0400201GLenum FramebufferVk::getImplementationColorReadFormat(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400202{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400203 auto errOrResult = GetReadAttachmentInfo(context, mState.getReadAttachment());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500204
205 // TODO(jmadill): Handle getRenderTarget error.
206 if (errOrResult.isError())
207 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500208 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500209 return GL_NONE;
210 }
211
212 return errOrResult.getResult()->format;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400213}
214
Jamie Madill4928b7c2017-06-20 12:57:39 -0400215GLenum FramebufferVk::getImplementationColorReadType(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400216{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400217 auto errOrResult = GetReadAttachmentInfo(context, mState.getReadAttachment());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500218
219 // TODO(jmadill): Handle getRenderTarget error.
220 if (errOrResult.isError())
221 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500222 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500223 return GL_NONE;
224 }
225
226 return errOrResult.getResult()->type;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400227}
228
Jamie Madillc564c072017-06-01 12:45:42 -0400229gl::Error FramebufferVk::readPixels(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400230 const gl::Rectangle &area,
231 GLenum format,
232 GLenum type,
Jamie Madilld4826152017-09-21 11:18:59 -0400233 void *pixels)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400234{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500235 const auto &glState = context->getGLState();
236 const auto *readFramebuffer = glState.getReadFramebuffer();
237 const auto *readAttachment = readFramebuffer->getReadColorbuffer();
238
239 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400240 ANGLE_TRY(readAttachment->getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500241
Jamie Madille1f3ad42017-10-28 23:00:42 -0400242 ContextVk *contextVk = vk::GetImpl(context);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500243 RendererVk *renderer = contextVk->getRenderer();
Jamie Madill5deea722017-02-16 10:44:46 -0500244 VkDevice device = renderer->getDevice();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500245
246 vk::Image *readImage = renderTarget->image;
247 vk::StagingImage stagingImage;
Jamie Madill5deea722017-02-16 10:44:46 -0500248 ANGLE_TRY(renderer->createStagingImage(TextureDimension::TEX_2D, *renderTarget->format,
Jamie Madill035fd6b2017-10-03 15:43:22 -0400249 renderTarget->extents, vk::StagingUsage::Read,
250 &stagingImage));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500251
Jamie Madill7f738d42017-11-20 17:06:27 -0500252 vk::CommandBufferAndState *commandBuffer = nullptr;
Jamie Madill0c0dc342017-03-24 14:18:51 -0400253 ANGLE_TRY(contextVk->getStartedCommandBuffer(&commandBuffer));
254
Jamie Madilld4826152017-09-21 11:18:59 -0400255 // End render pass if we're in one.
Jamie Madill1b038242017-11-01 15:14:36 -0400256 renderer->endRenderPass();
Jamie Madilld4826152017-09-21 11:18:59 -0400257
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500258 stagingImage.getImage().changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_GENERAL,
259 commandBuffer);
260
Jamie Madilld33c77c2017-11-09 13:08:30 -0500261 readImage->changeLayoutWithStages(
262 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
263 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT, VK_PIPELINE_STAGE_TRANSFER_BIT, commandBuffer);
Jamie Madill815a6c92017-10-21 14:33:04 -0400264
265 VkImageCopy region;
266 region.srcSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
267 region.srcSubresource.mipLevel = 0;
268 region.srcSubresource.baseArrayLayer = 0;
269 region.srcSubresource.layerCount = 1;
270 region.srcOffset.x = area.x;
271 region.srcOffset.y = area.y;
272 region.srcOffset.z = 0;
273 region.dstSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
274 region.dstSubresource.mipLevel = 0;
275 region.dstSubresource.baseArrayLayer = 0;
276 region.dstSubresource.layerCount = 1;
277 region.dstOffset.x = 0;
278 region.dstOffset.y = 0;
279 region.dstOffset.z = 0;
280 region.extent.width = area.width;
281 region.extent.height = area.height;
282 region.extent.depth = 1;
283
284 commandBuffer->copyImage(*readImage, stagingImage.getImage(), 1, &region);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500285
Jamie Madill0c0dc342017-03-24 14:18:51 -0400286 ANGLE_TRY(renderer->submitAndFinishCommandBuffer(commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500287
288 // TODO(jmadill): parameters
289 uint8_t *mapPointer = nullptr;
Jamie Madill5deea722017-02-16 10:44:46 -0500290 ANGLE_TRY(
291 stagingImage.getDeviceMemory().map(device, 0, stagingImage.getSize(), 0, &mapPointer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500292
Jamie Madill1d7be502017-10-29 18:06:50 -0400293 const auto &angleFormat = renderTarget->format->textureFormat();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500294
295 // TODO(jmadill): Use pixel bytes from the ANGLE format directly.
Geoff Langca271392017-04-05 12:30:00 -0400296 const auto &glFormat = gl::GetSizedInternalFormatInfo(angleFormat.glInternalFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500297 int inputPitch = glFormat.pixelBytes * area.width;
298
299 PackPixelsParams params;
300 params.area = area;
301 params.format = format;
302 params.type = type;
303 params.outputPitch = inputPitch;
Corentin Wallez336129f2017-10-17 15:55:40 -0400304 params.packBuffer = glState.getTargetBuffer(gl::BufferBinding::PixelPack);
Corentin Wallezcda6af12017-10-30 19:20:37 -0400305 params.pack = glState.getPackState();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500306
307 PackPixels(params, angleFormat, inputPitch, mapPointer, reinterpret_cast<uint8_t *>(pixels));
308
Jamie Madill5deea722017-02-16 10:44:46 -0500309 stagingImage.getDeviceMemory().unmap(device);
Jamie Madille88ec8e2017-10-31 17:18:14 -0400310 renderer->releaseObject(renderer->getCurrentQueueSerial(), &stagingImage);
Jamie Madillf651c772017-02-21 15:03:51 -0500311
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500312 return vk::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400313}
314
Jamie Madillc564c072017-06-01 12:45:42 -0400315gl::Error FramebufferVk::blit(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400316 const gl::Rectangle &sourceArea,
317 const gl::Rectangle &destArea,
318 GLbitfield mask,
319 GLenum filter)
320{
321 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500322 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400323}
324
Kenneth Russellce8602a2017-10-03 18:23:08 -0700325bool FramebufferVk::checkStatus(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400326{
Jamie Madillb79e7bb2017-10-24 13:55:50 -0400327 return true;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400328}
329
Jamie Madillc564c072017-06-01 12:45:42 -0400330void FramebufferVk::syncState(const gl::Context *context,
331 const gl::Framebuffer::DirtyBits &dirtyBits)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400332{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400333 ContextVk *contextVk = vk::GetImpl(context);
Jamie Madill7bd16662017-10-28 19:40:50 -0400334 RendererVk *renderer = contextVk->getRenderer();
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400335
336 ASSERT(dirtyBits.any());
337
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500338 // TODO(jmadill): Smarter update.
Jamie Madill9f2a8612017-11-30 12:43:09 -0500339 mRenderPassDesc.reset();
Jamie Madill7bd16662017-10-28 19:40:50 -0400340 renderer->releaseResource(*this, &mFramebuffer);
341 renderer->onReleaseRenderPass(this);
Jamie Madill72106562017-03-24 14:18:50 -0400342
343 // TODO(jmadill): Use pipeline cache.
344 contextVk->invalidateCurrentPipeline();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500345}
346
Jamie Madill9f2a8612017-11-30 12:43:09 -0500347const vk::RenderPassDesc &FramebufferVk::getRenderPassDesc(const gl::Context *context)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500348{
Jamie Madill9f2a8612017-11-30 12:43:09 -0500349 if (mRenderPassDesc.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500350 {
Jamie Madill9f2a8612017-11-30 12:43:09 -0500351 return mRenderPassDesc.value();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500352 }
353
Jamie Madill0b684ce2017-11-23 12:57:39 -0500354 vk::RenderPassDesc desc;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500355
356 const auto &colorAttachments = mState.getColorAttachments();
357 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
358 {
359 const auto &colorAttachment = colorAttachments[attachmentIndex];
360 if (colorAttachment.isAttached())
361 {
Jamie Madillab9f9c32017-01-17 17:47:34 -0500362 RenderTargetVk *renderTarget = nullptr;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500363 ANGLE_SWALLOW_ERR(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madillbef918c2017-12-13 13:11:30 -0500364 desc.packColorAttachment(*renderTarget->format, colorAttachment.getSamples());
Jamie Madillab9f9c32017-01-17 17:47:34 -0500365 }
366 }
367
368 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500369
Jamie Madill0b684ce2017-11-23 12:57:39 -0500370 if (depthStencilAttachment && depthStencilAttachment->isAttached())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500371 {
Jamie Madillab9f9c32017-01-17 17:47:34 -0500372 RenderTargetVk *renderTarget = nullptr;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500373 ANGLE_SWALLOW_ERR(depthStencilAttachment->getRenderTarget(context, &renderTarget));
Jamie Madillbef918c2017-12-13 13:11:30 -0500374 desc.packDepthStencilAttachment(*renderTarget->format,
375 depthStencilAttachment->getSamples());
Jamie Madillab9f9c32017-01-17 17:47:34 -0500376 }
377
Jamie Madill9f2a8612017-11-30 12:43:09 -0500378 mRenderPassDesc = desc;
379 return mRenderPassDesc.value();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500380}
381
Jamie Madill4928b7c2017-06-20 12:57:39 -0400382gl::ErrorOrResult<vk::Framebuffer *> FramebufferVk::getFramebuffer(const gl::Context *context,
Jamie Madill9f2a8612017-11-30 12:43:09 -0500383 RendererVk *rendererVk)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500384{
385 // If we've already created our cached Framebuffer, return it.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400386 if (mFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500387 {
388 return &mFramebuffer;
389 }
390
Jamie Madill9f2a8612017-11-30 12:43:09 -0500391 const vk::RenderPassDesc &desc = getRenderPassDesc(context);
392
Jamie Madillab9f9c32017-01-17 17:47:34 -0500393 vk::RenderPass *renderPass = nullptr;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500394 ANGLE_TRY(rendererVk->getCompatibleRenderPass(desc, &renderPass));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500395
396 // If we've a Framebuffer provided by a Surface (default FBO/backbuffer), query it.
Jamie Madill9f2a8612017-11-30 12:43:09 -0500397 VkDevice device = rendererVk->getDevice();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500398 if (mBackbuffer)
399 {
400 return mBackbuffer->getCurrentFramebuffer(device, *renderPass);
401 }
402
403 // Gather VkImageViews over all FBO attachments, also size of attached region.
404 std::vector<VkImageView> attachments;
405 gl::Extents attachmentsSize;
406
407 const auto &colorAttachments = mState.getColorAttachments();
408 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
409 {
410 const auto &colorAttachment = colorAttachments[attachmentIndex];
411 if (colorAttachment.isAttached())
412 {
413 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400414 ANGLE_TRY(colorAttachment.getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500415 attachments.push_back(renderTarget->imageView->getHandle());
416
417 ASSERT(attachmentsSize.empty() || attachmentsSize == colorAttachment.getSize());
418 attachmentsSize = colorAttachment.getSize();
419 }
420 }
421
422 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
423 if (depthStencilAttachment && depthStencilAttachment->isAttached())
424 {
425 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400426 ANGLE_TRY(depthStencilAttachment->getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500427 attachments.push_back(renderTarget->imageView->getHandle());
428
429 ASSERT(attachmentsSize.empty() || attachmentsSize == depthStencilAttachment->getSize());
430 attachmentsSize = depthStencilAttachment->getSize();
431 }
432
433 ASSERT(!attachments.empty());
434
435 VkFramebufferCreateInfo framebufferInfo;
436
437 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
438 framebufferInfo.pNext = nullptr;
439 framebufferInfo.flags = 0;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500440 framebufferInfo.renderPass = renderPass->getHandle();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500441 framebufferInfo.attachmentCount = static_cast<uint32_t>(attachments.size());
442 framebufferInfo.pAttachments = attachments.data();
443 framebufferInfo.width = static_cast<uint32_t>(attachmentsSize.width);
444 framebufferInfo.height = static_cast<uint32_t>(attachmentsSize.height);
445 framebufferInfo.layers = 1;
446
Jamie Madill25301b62017-10-28 20:59:31 -0400447 ANGLE_TRY(mFramebuffer.init(device, framebufferInfo));
Jamie Madill5deea722017-02-16 10:44:46 -0500448
Jamie Madillab9f9c32017-01-17 17:47:34 -0500449 return &mFramebuffer;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400450}
451
JiangYizhoubddc46b2016-12-09 09:50:51 +0800452gl::Error FramebufferVk::getSamplePosition(size_t index, GLfloat *xy) const
453{
454 UNIMPLEMENTED();
455 return gl::InternalError() << "getSamplePosition is unimplemented.";
456}
457
Jamie Madill1b038242017-11-01 15:14:36 -0400458gl::Error FramebufferVk::beginRenderPass(const gl::Context *context,
Jamie Madill9f2a8612017-11-30 12:43:09 -0500459 RendererVk *rendererVk,
Jamie Madill1b038242017-11-01 15:14:36 -0400460 vk::CommandBuffer *commandBuffer,
461 Serial queueSerial)
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500462{
Jamie Madillbef918c2017-12-13 13:11:30 -0500463 uint32_t attachmentIndex = 0;
464 vk::AttachmentOpsArray attachmentOps;
465
Jamie Madill4c26fc22017-02-24 11:04:10 -0500466 // TODO(jmadill): Cache render targets.
467 for (const auto &colorAttachment : mState.getColorAttachments())
468 {
469 if (colorAttachment.isAttached())
470 {
471 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400472 ANGLE_TRY(colorAttachment.getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madill4c26fc22017-02-24 11:04:10 -0500473 renderTarget->resource->setQueueSerial(queueSerial);
Jamie Madillbef918c2017-12-13 13:11:30 -0500474
475 // Fill in some default load and store ops.
476 attachmentOps.initDummyOp(attachmentIndex++, VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
Jamie Madill4c26fc22017-02-24 11:04:10 -0500477 }
478 }
479
480 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
481 if (depthStencilAttachment && depthStencilAttachment->isAttached())
482 {
483 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400484 ANGLE_TRY(depthStencilAttachment->getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madill4c26fc22017-02-24 11:04:10 -0500485 renderTarget->resource->setQueueSerial(queueSerial);
Jamie Madillbef918c2017-12-13 13:11:30 -0500486
487 // Fill in some default load and store ops.
488 attachmentOps.initDummyOp(attachmentIndex++,
489 VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL);
Jamie Madill4c26fc22017-02-24 11:04:10 -0500490 }
491
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500492 vk::Framebuffer *framebuffer = nullptr;
Jamie Madill9f2a8612017-11-30 12:43:09 -0500493 ANGLE_TRY_RESULT(getFramebuffer(context, rendererVk), framebuffer);
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500494 ASSERT(framebuffer && framebuffer->valid());
495
Jamie Madill9f2a8612017-11-30 12:43:09 -0500496 const vk::RenderPassDesc &desc = getRenderPassDesc(context);
497
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500498 vk::RenderPass *renderPass = nullptr;
Jamie Madillbef918c2017-12-13 13:11:30 -0500499 ANGLE_TRY(rendererVk->getRenderPassWithOps(desc, attachmentOps, &renderPass));
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500500 ASSERT(renderPass && renderPass->valid());
501
502 // TODO(jmadill): Proper clear value implementation.
Jamie Madill1b038242017-11-01 15:14:36 -0400503 const gl::State &glState = context->getGLState();
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500504 VkClearColorValue colorClear;
505 memset(&colorClear, 0, sizeof(VkClearColorValue));
506 colorClear.float32[0] = glState.getColorClearValue().red;
507 colorClear.float32[1] = glState.getColorClearValue().green;
508 colorClear.float32[2] = glState.getColorClearValue().blue;
509 colorClear.float32[3] = glState.getColorClearValue().alpha;
510
511 std::vector<VkClearValue> attachmentClearValues;
512 attachmentClearValues.push_back({colorClear});
513
514 // Updated the cached image layout of the attachments in this FBO.
515 // For a default FBO, we need to call through to the WindowSurfaceVk
516 // TODO(jmadill): Iterate over all attachments.
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500517 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400518 ANGLE_TRY(mState.getFirstColorAttachment()->getRenderTarget(context, &renderTarget));
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500519 renderTarget->image->updateLayout(VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
520
Jamie Madille218f152017-11-30 12:38:50 -0500521 commandBuffer->beginRenderPass(*renderPass, *framebuffer, glState.getViewport(), 1,
522 attachmentClearValues.data());
Jamie Madill4c26fc22017-02-24 11:04:10 -0500523
524 setQueueSerial(queueSerial);
525 if (mBackbuffer)
526 {
527 mBackbuffer->setQueueSerial(queueSerial);
528 }
529
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500530 return gl::NoError();
531}
532
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400533} // namespace rx