blob: c124d270e3a643d7caeecdccc1920ad4c9434abe [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 Madilldd43e6c2017-03-24 14:18:49 -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 Madilldd43e6c2017-03-24 14:18:49 -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
101 mRenderPass.destroy(device);
102 mFramebuffer.destroy(device);
103}
104
Jamie Madillc564c072017-06-01 12:45:42 -0400105void FramebufferVk::destroyDefault(const egl::Display *display)
Jamie Madill5deea722017-02-16 10:44:46 -0500106{
Jamie Madillc564c072017-06-01 12:45:42 -0400107 VkDevice device = GetImplAs<DisplayVk>(display)->getRenderer()->getDevice();
Jamie Madill5deea722017-02-16 10:44:46 -0500108
109 mRenderPass.destroy(device);
110 mFramebuffer.destroy(device);
111}
112
Jamie Madill4928b7c2017-06-20 12:57:39 -0400113gl::Error FramebufferVk::discard(const gl::Context *context,
114 size_t count,
115 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400116{
117 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500118 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400119}
120
Jamie Madill4928b7c2017-06-20 12:57:39 -0400121gl::Error FramebufferVk::invalidate(const gl::Context *context,
122 size_t count,
123 const GLenum *attachments)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400124{
125 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500126 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400127}
128
Jamie Madill4928b7c2017-06-20 12:57:39 -0400129gl::Error FramebufferVk::invalidateSub(const gl::Context *context,
130 size_t count,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400131 const GLenum *attachments,
132 const gl::Rectangle &area)
133{
134 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500135 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400136}
137
Jamie Madillc564c072017-06-01 12:45:42 -0400138gl::Error FramebufferVk::clear(const gl::Context *context, GLbitfield mask)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400139{
Jamie Madillc564c072017-06-01 12:45:42 -0400140 ContextVk *contextVk = GetImplAs<ContextVk>(context);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500141
142 if (mState.getDepthAttachment() && (mask & GL_DEPTH_BUFFER_BIT) != 0)
143 {
144 // TODO(jmadill): Depth clear
145 UNIMPLEMENTED();
146 }
147
148 if (mState.getStencilAttachment() && (mask & GL_STENCIL_BUFFER_BIT) != 0)
149 {
150 // TODO(jmadill): Stencil clear
151 UNIMPLEMENTED();
152 }
153
154 if ((mask & GL_COLOR_BUFFER_BIT) == 0)
155 {
156 return gl::NoError();
157 }
158
159 const auto &glState = context->getGLState();
160 const auto &clearColor = glState.getColorClearValue();
161 VkClearColorValue clearColorValue;
162 clearColorValue.float32[0] = clearColor.red;
163 clearColorValue.float32[1] = clearColor.green;
164 clearColorValue.float32[2] = clearColor.blue;
165 clearColorValue.float32[3] = clearColor.alpha;
166
167 // TODO(jmadill): Scissored clears.
168 const auto *attachment = mState.getFirstNonNullAttachment();
169 ASSERT(attachment && attachment->isAttached());
170 const auto &size = attachment->getSize();
171 const gl::Rectangle renderArea(0, 0, size.width, size.height);
172
Jamie Madill0c0dc342017-03-24 14:18:51 -0400173 vk::CommandBuffer *commandBuffer = nullptr;
174 ANGLE_TRY(contextVk->getStartedCommandBuffer(&commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500175
176 for (const auto &colorAttachment : mState.getColorAttachments())
177 {
178 if (colorAttachment.isAttached())
179 {
180 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400181 ANGLE_TRY(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500182 renderTarget->image->changeLayoutTop(
183 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, commandBuffer);
184 commandBuffer->clearSingleColorImage(*renderTarget->image, clearColorValue);
185 }
186 }
187
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500188 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400189}
190
Jamie Madillc564c072017-06-01 12:45:42 -0400191gl::Error FramebufferVk::clearBufferfv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400192 GLenum buffer,
193 GLint drawbuffer,
194 const GLfloat *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::clearBufferuiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400201 GLenum buffer,
202 GLint drawbuffer,
203 const GLuint *values)
204{
205 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500206 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400207}
208
Jamie Madillc564c072017-06-01 12:45:42 -0400209gl::Error FramebufferVk::clearBufferiv(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400210 GLenum buffer,
211 GLint drawbuffer,
212 const GLint *values)
213{
214 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500215 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400216}
217
Jamie Madillc564c072017-06-01 12:45:42 -0400218gl::Error FramebufferVk::clearBufferfi(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400219 GLenum buffer,
220 GLint drawbuffer,
221 GLfloat depth,
222 GLint stencil)
223{
224 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500225 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400226}
227
Jamie Madill4928b7c2017-06-20 12:57:39 -0400228GLenum FramebufferVk::getImplementationColorReadFormat(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400229{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400230 auto errOrResult = GetReadAttachmentInfo(context, mState.getReadAttachment());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500231
232 // TODO(jmadill): Handle getRenderTarget error.
233 if (errOrResult.isError())
234 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500235 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500236 return GL_NONE;
237 }
238
239 return errOrResult.getResult()->format;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400240}
241
Jamie Madill4928b7c2017-06-20 12:57:39 -0400242GLenum FramebufferVk::getImplementationColorReadType(const gl::Context *context) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400243{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400244 auto errOrResult = GetReadAttachmentInfo(context, mState.getReadAttachment());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500245
246 // TODO(jmadill): Handle getRenderTarget error.
247 if (errOrResult.isError())
248 {
Yuly Novikovd73f8522017-01-13 17:48:57 -0500249 ERR() << "Internal error in FramebufferVk::getImplementationColorReadFormat.";
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500250 return GL_NONE;
251 }
252
253 return errOrResult.getResult()->type;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400254}
255
Jamie Madillc564c072017-06-01 12:45:42 -0400256gl::Error FramebufferVk::readPixels(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400257 const gl::Rectangle &area,
258 GLenum format,
259 GLenum type,
Jamie Madill876429b2017-04-20 15:46:24 -0400260 void *pixels) const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400261{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500262 const auto &glState = context->getGLState();
263 const auto *readFramebuffer = glState.getReadFramebuffer();
264 const auto *readAttachment = readFramebuffer->getReadColorbuffer();
265
266 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400267 ANGLE_TRY(readAttachment->getRenderTarget(context, &renderTarget));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500268
Jamie Madillc564c072017-06-01 12:45:42 -0400269 ContextVk *contextVk = GetImplAs<ContextVk>(context);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500270 RendererVk *renderer = contextVk->getRenderer();
Jamie Madill5deea722017-02-16 10:44:46 -0500271 VkDevice device = renderer->getDevice();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500272
273 vk::Image *readImage = renderTarget->image;
274 vk::StagingImage stagingImage;
Jamie Madill5deea722017-02-16 10:44:46 -0500275 ANGLE_TRY(renderer->createStagingImage(TextureDimension::TEX_2D, *renderTarget->format,
276 renderTarget->extents, &stagingImage));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500277
Jamie Madill0c0dc342017-03-24 14:18:51 -0400278 vk::CommandBuffer *commandBuffer = nullptr;
279 ANGLE_TRY(contextVk->getStartedCommandBuffer(&commandBuffer));
280
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500281 stagingImage.getImage().changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_GENERAL,
282 commandBuffer);
283
284 gl::Box copyRegion;
285 copyRegion.x = area.x;
286 copyRegion.y = area.y;
287 copyRegion.z = 0;
288 copyRegion.width = area.width;
289 copyRegion.height = area.height;
290 copyRegion.depth = 1;
291
292 readImage->changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
293 commandBuffer);
294 commandBuffer->copySingleImage(*readImage, stagingImage.getImage(), copyRegion,
295 VK_IMAGE_ASPECT_COLOR_BIT);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500296
Jamie Madill0c0dc342017-03-24 14:18:51 -0400297 ANGLE_TRY(renderer->submitAndFinishCommandBuffer(commandBuffer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500298
299 // TODO(jmadill): parameters
300 uint8_t *mapPointer = nullptr;
Jamie Madill5deea722017-02-16 10:44:46 -0500301 ANGLE_TRY(
302 stagingImage.getDeviceMemory().map(device, 0, stagingImage.getSize(), 0, &mapPointer));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500303
304 const auto &angleFormat = renderTarget->format->format();
305
306 // TODO(jmadill): Use pixel bytes from the ANGLE format directly.
Geoff Langca271392017-04-05 12:30:00 -0400307 const auto &glFormat = gl::GetSizedInternalFormatInfo(angleFormat.glInternalFormat);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500308 int inputPitch = glFormat.pixelBytes * area.width;
309
310 PackPixelsParams params;
311 params.area = area;
312 params.format = format;
313 params.type = type;
314 params.outputPitch = inputPitch;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400315 params.pack.copyFrom(context, glState.getPackState());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500316
317 PackPixels(params, angleFormat, inputPitch, mapPointer, reinterpret_cast<uint8_t *>(pixels));
318
Jamie Madill5deea722017-02-16 10:44:46 -0500319 stagingImage.getDeviceMemory().unmap(device);
Jamie Madillf651c772017-02-21 15:03:51 -0500320 renderer->enqueueGarbage(renderer->getCurrentQueueSerial(), std::move(stagingImage));
321
322 stagingImage.getImage().destroy(renderer->getDevice());
Jamie Madill5deea722017-02-16 10:44:46 -0500323
324 stagingImage.destroy(device);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500325
326 return vk::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400327}
328
Jamie Madillc564c072017-06-01 12:45:42 -0400329gl::Error FramebufferVk::blit(const gl::Context *context,
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400330 const gl::Rectangle &sourceArea,
331 const gl::Rectangle &destArea,
332 GLbitfield mask,
333 GLenum filter)
334{
335 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500336 return gl::InternalError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400337}
338
339bool FramebufferVk::checkStatus() const
340{
341 UNIMPLEMENTED();
342 return bool();
343}
344
Jamie Madillc564c072017-06-01 12:45:42 -0400345void FramebufferVk::syncState(const gl::Context *context,
346 const gl::Framebuffer::DirtyBits &dirtyBits)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400347{
Jamie Madillc564c072017-06-01 12:45:42 -0400348 auto contextVk = GetImplAs<ContextVk>(context);
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400349
350 ASSERT(dirtyBits.any());
351
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500352 // TODO(jmadill): Smarter update.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400353 mRenderPass.destroy(contextVk->getDevice());
354 mFramebuffer.destroy(contextVk->getDevice());
Jamie Madill72106562017-03-24 14:18:50 -0400355
356 // TODO(jmadill): Use pipeline cache.
357 contextVk->invalidateCurrentPipeline();
Jamie Madillab9f9c32017-01-17 17:47:34 -0500358}
359
Jamie Madill4928b7c2017-06-20 12:57:39 -0400360gl::ErrorOrResult<vk::RenderPass *> FramebufferVk::getRenderPass(const gl::Context *context,
361 VkDevice device)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500362{
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400363 if (mRenderPass.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500364 {
365 return &mRenderPass;
366 }
367
368 // TODO(jmadill): Can we use stack-only memory?
369 std::vector<VkAttachmentDescription> attachmentDescs;
370 std::vector<VkAttachmentReference> colorAttachmentRefs;
371
372 const auto &colorAttachments = mState.getColorAttachments();
373 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
374 {
375 const auto &colorAttachment = colorAttachments[attachmentIndex];
376 if (colorAttachment.isAttached())
377 {
378 VkAttachmentDescription colorDesc;
379 VkAttachmentReference colorRef;
380
381 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400382 ANGLE_TRY(colorAttachment.getRenderTarget(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500383
384 // TODO(jmadill): We would only need this flag for duplicated attachments.
385 colorDesc.flags = VK_ATTACHMENT_DESCRIPTION_MAY_ALIAS_BIT;
386 colorDesc.format = renderTarget->format->native;
387 colorDesc.samples = ConvertSamples(colorAttachment.getSamples());
388
389 // The load op controls the prior existing depth/color attachment data.
390 // TODO(jmadill): Proper load ops. Should not be hard coded to clear.
391 colorDesc.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
392 colorDesc.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
393 colorDesc.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
394 colorDesc.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
395 colorDesc.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500396
397 // We might want to transition directly to PRESENT_SRC for Surface attachments.
398 colorDesc.finalLayout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500399
400 colorRef.attachment = static_cast<uint32_t>(colorAttachments.size()) - 1u;
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500401 colorRef.layout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500402
403 attachmentDescs.push_back(colorDesc);
404 colorAttachmentRefs.push_back(colorRef);
405 }
406 }
407
408 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
409 VkAttachmentReference depthStencilAttachmentRef;
410 bool useDepth = depthStencilAttachment && depthStencilAttachment->isAttached();
411
412 if (useDepth)
413 {
414 VkAttachmentDescription depthStencilDesc;
415
416 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400417 ANGLE_TRY(depthStencilAttachment->getRenderTarget(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500418
419 depthStencilDesc.flags = 0;
420 depthStencilDesc.format = renderTarget->format->native;
421 depthStencilDesc.samples = ConvertSamples(depthStencilAttachment->getSamples());
422 depthStencilDesc.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
423 depthStencilDesc.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
424 depthStencilDesc.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
425 depthStencilDesc.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
426 depthStencilDesc.initialLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
427 depthStencilDesc.finalLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
428
429 depthStencilAttachmentRef.attachment = static_cast<uint32_t>(attachmentDescs.size());
430 depthStencilAttachmentRef.layout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
431
432 attachmentDescs.push_back(depthStencilDesc);
433 }
434
435 ASSERT(!attachmentDescs.empty());
436
437 VkSubpassDescription subpassDesc;
438
439 subpassDesc.flags = 0;
440 subpassDesc.pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS;
441 subpassDesc.inputAttachmentCount = 0;
442 subpassDesc.pInputAttachments = nullptr;
443 subpassDesc.colorAttachmentCount = static_cast<uint32_t>(colorAttachmentRefs.size());
444 subpassDesc.pColorAttachments = colorAttachmentRefs.data();
445 subpassDesc.pResolveAttachments = nullptr;
446 subpassDesc.pDepthStencilAttachment = (useDepth ? &depthStencilAttachmentRef : nullptr);
447 subpassDesc.preserveAttachmentCount = 0;
448 subpassDesc.pPreserveAttachments = nullptr;
449
450 VkRenderPassCreateInfo renderPassInfo;
451
452 renderPassInfo.sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO;
453 renderPassInfo.pNext = nullptr;
454 renderPassInfo.flags = 0;
455 renderPassInfo.attachmentCount = static_cast<uint32_t>(attachmentDescs.size());
456 renderPassInfo.pAttachments = attachmentDescs.data();
457 renderPassInfo.subpassCount = 1;
458 renderPassInfo.pSubpasses = &subpassDesc;
459 renderPassInfo.dependencyCount = 0;
460 renderPassInfo.pDependencies = nullptr;
461
Jamie Madill5deea722017-02-16 10:44:46 -0500462 vk::RenderPass renderPass;
463 ANGLE_TRY(renderPass.init(device, renderPassInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500464
Jamie Madill5deea722017-02-16 10:44:46 -0500465 mRenderPass.retain(device, std::move(renderPass));
466
Jamie Madillab9f9c32017-01-17 17:47:34 -0500467 return &mRenderPass;
468}
469
Jamie Madill4928b7c2017-06-20 12:57:39 -0400470gl::ErrorOrResult<vk::Framebuffer *> FramebufferVk::getFramebuffer(const gl::Context *context,
471 VkDevice device)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500472{
473 // If we've already created our cached Framebuffer, return it.
Jamie Madilldd43e6c2017-03-24 14:18:49 -0400474 if (mFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500475 {
476 return &mFramebuffer;
477 }
478
479 vk::RenderPass *renderPass = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400480 ANGLE_TRY_RESULT(getRenderPass(context, device), renderPass);
Jamie Madillab9f9c32017-01-17 17:47:34 -0500481
482 // If we've a Framebuffer provided by a Surface (default FBO/backbuffer), query it.
483 if (mBackbuffer)
484 {
485 return mBackbuffer->getCurrentFramebuffer(device, *renderPass);
486 }
487
488 // Gather VkImageViews over all FBO attachments, also size of attached region.
489 std::vector<VkImageView> attachments;
490 gl::Extents attachmentsSize;
491
492 const auto &colorAttachments = mState.getColorAttachments();
493 for (size_t attachmentIndex = 0; attachmentIndex < colorAttachments.size(); ++attachmentIndex)
494 {
495 const auto &colorAttachment = colorAttachments[attachmentIndex];
496 if (colorAttachment.isAttached())
497 {
498 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400499 ANGLE_TRY(colorAttachment.getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500500 attachments.push_back(renderTarget->imageView->getHandle());
501
502 ASSERT(attachmentsSize.empty() || attachmentsSize == colorAttachment.getSize());
503 attachmentsSize = colorAttachment.getSize();
504 }
505 }
506
507 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
508 if (depthStencilAttachment && depthStencilAttachment->isAttached())
509 {
510 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400511 ANGLE_TRY(depthStencilAttachment->getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500512 attachments.push_back(renderTarget->imageView->getHandle());
513
514 ASSERT(attachmentsSize.empty() || attachmentsSize == depthStencilAttachment->getSize());
515 attachmentsSize = depthStencilAttachment->getSize();
516 }
517
518 ASSERT(!attachments.empty());
519
520 VkFramebufferCreateInfo framebufferInfo;
521
522 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
523 framebufferInfo.pNext = nullptr;
524 framebufferInfo.flags = 0;
525 framebufferInfo.renderPass = mRenderPass.getHandle();
526 framebufferInfo.attachmentCount = static_cast<uint32_t>(attachments.size());
527 framebufferInfo.pAttachments = attachments.data();
528 framebufferInfo.width = static_cast<uint32_t>(attachmentsSize.width);
529 framebufferInfo.height = static_cast<uint32_t>(attachmentsSize.height);
530 framebufferInfo.layers = 1;
531
Jamie Madill5deea722017-02-16 10:44:46 -0500532 vk::Framebuffer framebuffer;
Jamie Madill71c88b32017-09-14 22:20:29 -0400533 ANGLE_TRY(framebuffer.init(device, framebufferInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500534
Jamie Madill5deea722017-02-16 10:44:46 -0500535 mFramebuffer.retain(device, std::move(framebuffer));
536
Jamie Madillab9f9c32017-01-17 17:47:34 -0500537 return &mFramebuffer;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400538}
539
JiangYizhoubddc46b2016-12-09 09:50:51 +0800540gl::Error FramebufferVk::getSamplePosition(size_t index, GLfloat *xy) const
541{
542 UNIMPLEMENTED();
543 return gl::InternalError() << "getSamplePosition is unimplemented.";
544}
545
Jamie Madill4928b7c2017-06-20 12:57:39 -0400546gl::Error FramebufferVk::beginRenderPass(const gl::Context *context,
547 VkDevice device,
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500548 vk::CommandBuffer *commandBuffer,
Jamie Madill4c26fc22017-02-24 11:04:10 -0500549 Serial queueSerial,
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500550 const gl::State &glState)
551{
Jamie Madill4c26fc22017-02-24 11:04:10 -0500552 // TODO(jmadill): Cache render targets.
553 for (const auto &colorAttachment : mState.getColorAttachments())
554 {
555 if (colorAttachment.isAttached())
556 {
557 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400558 ANGLE_TRY(colorAttachment.getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madill4c26fc22017-02-24 11:04:10 -0500559 renderTarget->resource->setQueueSerial(queueSerial);
560 }
561 }
562
563 const auto *depthStencilAttachment = mState.getDepthStencilAttachment();
564 if (depthStencilAttachment && depthStencilAttachment->isAttached())
565 {
566 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400567 ANGLE_TRY(depthStencilAttachment->getRenderTarget<RenderTargetVk>(context, &renderTarget));
Jamie Madill4c26fc22017-02-24 11:04:10 -0500568 renderTarget->resource->setQueueSerial(queueSerial);
569 }
570
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500571 vk::Framebuffer *framebuffer = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400572 ANGLE_TRY_RESULT(getFramebuffer(context, device), framebuffer);
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500573 ASSERT(framebuffer && framebuffer->valid());
574
575 vk::RenderPass *renderPass = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400576 ANGLE_TRY_RESULT(getRenderPass(context, device), renderPass);
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500577 ASSERT(renderPass && renderPass->valid());
578
579 // TODO(jmadill): Proper clear value implementation.
580 VkClearColorValue colorClear;
581 memset(&colorClear, 0, sizeof(VkClearColorValue));
582 colorClear.float32[0] = glState.getColorClearValue().red;
583 colorClear.float32[1] = glState.getColorClearValue().green;
584 colorClear.float32[2] = glState.getColorClearValue().blue;
585 colorClear.float32[3] = glState.getColorClearValue().alpha;
586
587 std::vector<VkClearValue> attachmentClearValues;
588 attachmentClearValues.push_back({colorClear});
589
590 // Updated the cached image layout of the attachments in this FBO.
591 // For a default FBO, we need to call through to the WindowSurfaceVk
592 // TODO(jmadill): Iterate over all attachments.
593 ASSERT(mBackbuffer);
594 RenderTargetVk *renderTarget = nullptr;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400595 ANGLE_TRY(mState.getFirstColorAttachment()->getRenderTarget(context, &renderTarget));
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500596 renderTarget->image->updateLayout(VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
597
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500598 commandBuffer->beginRenderPass(*renderPass, *framebuffer, glState.getViewport(),
599 attachmentClearValues);
Jamie Madill4c26fc22017-02-24 11:04:10 -0500600
601 setQueueSerial(queueSerial);
602 if (mBackbuffer)
603 {
604 mBackbuffer->setQueueSerial(queueSerial);
605 }
606
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500607 return gl::NoError();
608}
609
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400610} // namespace rx