blob: 0f12ba9022db67d88d0df9ea65887f12c01aac8c [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// SurfaceVk.cpp:
7// Implements the class methods for SurfaceVk.
8//
9
10#include "libANGLE/renderer/vulkan/SurfaceVk.h"
11
12#include "common/debug.h"
Jamie Madillfe548342017-06-19 11:13:24 -040013#include "libANGLE/Context.h"
Jamie Madillc564c072017-06-01 12:45:42 -040014#include "libANGLE/Display.h"
Jamie Madill4d0bf552016-12-28 15:45:24 -050015#include "libANGLE/Surface.h"
16#include "libANGLE/renderer/vulkan/DisplayVk.h"
Jamie Madille09bd5d2016-11-29 16:20:35 -050017#include "libANGLE/renderer/vulkan/FramebufferVk.h"
18#include "libANGLE/renderer/vulkan/RendererVk.h"
Jamie Madill3c424b42018-01-19 12:35:09 -050019#include "libANGLE/renderer/vulkan/vk_format_utils.h"
Jamie Madill9e54b5a2016-05-25 12:57:39 -040020
21namespace rx
22{
23
Jamie Madill4d0bf552016-12-28 15:45:24 -050024namespace
25{
26
Jamie Madillf0eafe12017-02-21 15:03:50 -050027VkPresentModeKHR GetDesiredPresentMode(const std::vector<VkPresentModeKHR> &presentModes,
28 EGLint minSwapInterval,
29 EGLint maxSwapInterval)
30{
31 ASSERT(!presentModes.empty());
32
33 // Use FIFO mode for v-sync, since it throttles you to the display rate. Mailbox is more
Jamie Madilla9c60e92017-09-28 19:06:39 -040034 // similar to triple-buffering. For now we hard-code Mailbox for perf testing.
Jamie Madillf0eafe12017-02-21 15:03:50 -050035 // TODO(jmadill): Properly select present mode and re-create display if changed.
36 VkPresentModeKHR bestChoice = VK_PRESENT_MODE_MAILBOX_KHR;
37
Jamie Madillbc543422018-03-30 10:43:19 -040038 for (VkPresentModeKHR presentMode : presentModes)
Jamie Madillf0eafe12017-02-21 15:03:50 -050039 {
40 if (presentMode == bestChoice)
41 {
42 return bestChoice;
43 }
44 }
45
Jamie Madill98de8262017-05-29 13:01:02 -040046 WARN() << "Present mode " << bestChoice << " not available. Falling back to "
47 << presentModes[0];
Jamie Madillf0eafe12017-02-21 15:03:50 -050048 return presentModes[0];
49}
50
Geoff Lang9e141642018-06-27 11:43:18 -040051constexpr VkImageUsageFlags kSurfaceVKImageUsageFlags =
52 VK_IMAGE_USAGE_TRANSFER_DST_BIT | VK_IMAGE_USAGE_TRANSFER_SRC_BIT;
53constexpr VkImageUsageFlags kSurfaceVKColorImageUsageFlags =
54 kSurfaceVKImageUsageFlags | VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
55constexpr VkImageUsageFlags kSurfaceVKDepthStencilImageUsageFlags =
56 kSurfaceVKImageUsageFlags | VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT;
57
Jamie Madill4d0bf552016-12-28 15:45:24 -050058} // namespace
59
Geoff Lang9e141642018-06-27 11:43:18 -040060OffscreenSurfaceVk::AttachmentImage::AttachmentImage(vk::CommandGraphResource *commandGraphResource)
61 : renderTarget(&image, &imageView, commandGraphResource)
62{
63}
64
65OffscreenSurfaceVk::AttachmentImage::~AttachmentImage() = default;
66
67egl::Error OffscreenSurfaceVk::AttachmentImage::initialize(const egl::Display *display,
68 EGLint width,
69 EGLint height,
70 const vk::Format &vkFormat)
71{
72 const DisplayVk *displayVk = vk::GetImpl(display);
73 RendererVk *renderer = displayVk->getRenderer();
74 VkDevice device = renderer->getDevice();
75
76 const angle::Format &textureFormat = vkFormat.textureFormat();
77 bool isDepthOrStencilFormat = textureFormat.depthBits > 0 || textureFormat.stencilBits > 0;
78 const VkImageUsageFlags usage = isDepthOrStencilFormat ? kSurfaceVKDepthStencilImageUsageFlags
79 : kSurfaceVKColorImageUsageFlags;
80
81 gl::Extents extents(static_cast<int>(width), static_cast<int>(height), 1);
82 ANGLE_TRY(image.init(device, gl::TextureType::_2D, extents, vkFormat, 1, usage, 1));
83
84 VkMemoryPropertyFlags flags = VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT;
85 ANGLE_TRY(image.initMemory(device, renderer->getMemoryProperties(), flags));
86
87 VkImageAspectFlags aspect = vk::GetFormatAspectFlags(textureFormat);
88
89 ANGLE_TRY(image.initImageView(device, gl::TextureType::_2D, aspect, gl::SwizzleState(),
90 &imageView, 1));
91
92 return egl::NoError();
93}
94
95void OffscreenSurfaceVk::AttachmentImage::destroy(const egl::Display *display,
96 Serial storedQueueSerial)
97{
98 const DisplayVk *displayVk = vk::GetImpl(display);
99 RendererVk *renderer = displayVk->getRenderer();
100
101 image.release(renderer->getCurrentQueueSerial(), renderer);
102 renderer->releaseObject(storedQueueSerial, &imageView);
103}
104
Jamie Madille09bd5d2016-11-29 16:20:35 -0500105OffscreenSurfaceVk::OffscreenSurfaceVk(const egl::SurfaceState &surfaceState,
106 EGLint width,
107 EGLint height)
Geoff Lang9e141642018-06-27 11:43:18 -0400108 : SurfaceImpl(surfaceState),
109 mWidth(width),
110 mHeight(height),
111 mColorAttachment(this),
112 mDepthStencilAttachment(this)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400113{
114}
115
Jamie Madille09bd5d2016-11-29 16:20:35 -0500116OffscreenSurfaceVk::~OffscreenSurfaceVk()
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400117{
118}
119
Jamie Madillc564c072017-06-01 12:45:42 -0400120egl::Error OffscreenSurfaceVk::initialize(const egl::Display *display)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400121{
Geoff Lang9e141642018-06-27 11:43:18 -0400122 const DisplayVk *displayVk = vk::GetImpl(display);
123 RendererVk *renderer = displayVk->getRenderer();
124
125 const egl::Config *config = mState.config;
126
127 if (config->renderTargetFormat != GL_NONE)
128 {
129 ANGLE_TRY(mColorAttachment.initialize(display, mWidth, mHeight,
130 renderer->getFormat(config->renderTargetFormat)));
131 }
132
133 if (config->depthStencilFormat != GL_NONE)
134 {
135 ANGLE_TRY(mDepthStencilAttachment.initialize(
136 display, mWidth, mHeight, renderer->getFormat(config->depthStencilFormat)));
137 }
138
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500139 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400140}
141
Geoff Lang9e141642018-06-27 11:43:18 -0400142void OffscreenSurfaceVk::destroy(const egl::Display *display)
143{
144 mColorAttachment.destroy(display, getStoredQueueSerial());
145 mDepthStencilAttachment.destroy(display, getStoredQueueSerial());
146}
147
Geoff Langbf7b95d2018-05-01 16:48:21 -0400148FramebufferImpl *OffscreenSurfaceVk::createDefaultFramebuffer(const gl::Context *context,
149 const gl::FramebufferState &state)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400150{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500151 // Use a user FBO for an offscreen RT.
152 return FramebufferVk::CreateUserFBO(state);
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400153}
154
Jamie Madillfe548342017-06-19 11:13:24 -0400155egl::Error OffscreenSurfaceVk::swap(const gl::Context *context)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400156{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500157 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400158}
159
Jamie Madillfe548342017-06-19 11:13:24 -0400160egl::Error OffscreenSurfaceVk::postSubBuffer(const gl::Context * /*context*/,
161 EGLint /*x*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -0500162 EGLint /*y*/,
163 EGLint /*width*/,
164 EGLint /*height*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400165{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500166 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400167}
168
Jamie Madille09bd5d2016-11-29 16:20:35 -0500169egl::Error OffscreenSurfaceVk::querySurfacePointerANGLE(EGLint /*attribute*/, void ** /*value*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400170{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500171 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500172 return egl::EglBadCurrentSurface();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400173}
174
Geoff Langccafa622018-05-02 13:07:53 -0400175egl::Error OffscreenSurfaceVk::bindTexImage(const gl::Context * /*context*/,
176 gl::Texture * /*texture*/,
177 EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400178{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500179 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400180}
181
Geoff Langccafa622018-05-02 13:07:53 -0400182egl::Error OffscreenSurfaceVk::releaseTexImage(const gl::Context * /*context*/, EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400183{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500184 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400185}
186
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700187egl::Error OffscreenSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
188 EGLuint64KHR * /*msc*/,
189 EGLuint64KHR * /*sbc*/)
190{
191 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500192 return egl::EglBadAccess();
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700193}
194
Jamie Madille09bd5d2016-11-29 16:20:35 -0500195void OffscreenSurfaceVk::setSwapInterval(EGLint /*interval*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400196{
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400197}
198
Jamie Madille09bd5d2016-11-29 16:20:35 -0500199EGLint OffscreenSurfaceVk::getWidth() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400200{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500201 return mWidth;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400202}
203
Jamie Madille09bd5d2016-11-29 16:20:35 -0500204EGLint OffscreenSurfaceVk::getHeight() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400205{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500206 return mHeight;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400207}
208
Jamie Madille09bd5d2016-11-29 16:20:35 -0500209EGLint OffscreenSurfaceVk::isPostSubBufferSupported() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400210{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500211 return EGL_FALSE;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400212}
213
Jamie Madille09bd5d2016-11-29 16:20:35 -0500214EGLint OffscreenSurfaceVk::getSwapBehavior() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400215{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500216 return EGL_BUFFER_PRESERVED;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400217}
218
Geoff Lang9e141642018-06-27 11:43:18 -0400219gl::Error OffscreenSurfaceVk::getAttachmentRenderTarget(const gl::Context * /*context*/,
220 GLenum binding,
221 const gl::ImageIndex & /*imageIndex*/,
222 FramebufferAttachmentRenderTarget **rtOut)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400223{
Geoff Lang9e141642018-06-27 11:43:18 -0400224 if (binding == GL_BACK)
225 {
226 *rtOut = &mColorAttachment.renderTarget;
227 }
228 else
229 {
230 ASSERT(binding == GL_DEPTH || binding == GL_STENCIL || binding == GL_DEPTH_STENCIL);
231 *rtOut = &mDepthStencilAttachment.renderTarget;
232 }
233
234 return gl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500235}
236
Jamie Madill05b35b22017-10-03 09:01:44 -0400237gl::Error OffscreenSurfaceVk::initializeContents(const gl::Context *context,
238 const gl::ImageIndex &imageIndex)
239{
240 UNIMPLEMENTED();
241 return gl::NoError();
242}
243
Jamie Madillbc543422018-03-30 10:43:19 -0400244WindowSurfaceVk::SwapchainImage::SwapchainImage() = default;
245WindowSurfaceVk::SwapchainImage::~SwapchainImage() = default;
246
247WindowSurfaceVk::SwapchainImage::SwapchainImage(SwapchainImage &&other)
248 : image(std::move(other.image)),
249 imageView(std::move(other.imageView)),
250 framebuffer(std::move(other.framebuffer)),
251 imageAcquiredSemaphore(std::move(other.imageAcquiredSemaphore)),
252 commandsCompleteSemaphore(std::move(other.commandsCompleteSemaphore))
253{
254}
Jamie Madillacf2f3a2017-11-21 19:22:44 -0500255
Jamie Madill4d0bf552016-12-28 15:45:24 -0500256WindowSurfaceVk::WindowSurfaceVk(const egl::SurfaceState &surfaceState,
257 EGLNativeWindowType window,
258 EGLint width,
259 EGLint height)
260 : SurfaceImpl(surfaceState),
261 mNativeWindowType(window),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500262 mSurface(VK_NULL_HANDLE),
Frank Henigman29f148b2016-11-23 21:05:36 -0500263 mInstance(VK_NULL_HANDLE),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500264 mSwapchain(VK_NULL_HANDLE),
Jamie Madillbcf467f2018-05-23 09:46:00 -0400265 mColorRenderTarget(nullptr, nullptr, this),
266 mDepthStencilRenderTarget(&mDepthStencilImage, &mDepthStencilImageView, this),
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500267 mCurrentSwapchainImageIndex(0)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500268{
269}
270
271WindowSurfaceVk::~WindowSurfaceVk()
272{
Jamie Madill70ee0f62017-02-06 16:04:20 -0500273 ASSERT(mSurface == VK_NULL_HANDLE);
274 ASSERT(mSwapchain == VK_NULL_HANDLE);
Jamie Madill70ee0f62017-02-06 16:04:20 -0500275}
276
Jamie Madillc564c072017-06-01 12:45:42 -0400277void WindowSurfaceVk::destroy(const egl::Display *display)
Jamie Madill70ee0f62017-02-06 16:04:20 -0500278{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400279 const DisplayVk *displayVk = vk::GetImpl(display);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500280 RendererVk *renderer = displayVk->getRenderer();
281 VkDevice device = renderer->getDevice();
282 VkInstance instance = renderer->getInstance();
Jamie Madill70ee0f62017-02-06 16:04:20 -0500283
Jamie Madillf618c9e2018-02-15 14:45:40 -0500284 // We might not need to flush the pipe here.
Jamie Madilleebe2192018-07-11 09:01:18 -0400285 ANGLE_SWALLOW_ERR(renderer->finish(display->getProxyContext()));
Jamie Madille918de22017-04-12 10:21:11 -0400286
Jamie Madilla9c60e92017-09-28 19:06:39 -0400287 mAcquireNextImageSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500288
Jamie Madillbc543422018-03-30 10:43:19 -0400289 mDepthStencilImage.release(renderer->getCurrentQueueSerial(), renderer);
290 mDepthStencilImageView.destroy(device);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500291
Jamie Madillbc543422018-03-30 10:43:19 -0400292 for (SwapchainImage &swapchainImage : mSwapchainImages)
Jamie Madill70ee0f62017-02-06 16:04:20 -0500293 {
Jamie Madilla9c60e92017-09-28 19:06:39 -0400294 // Although we don't own the swapchain image handles, we need to keep our shutdown clean.
Jamie Madill858c1cc2018-03-31 14:19:13 -0400295 swapchainImage.image.resetImageWeakReference();
Jamie Madillbc543422018-03-30 10:43:19 -0400296 swapchainImage.image.destroy(device);
Jamie Madilla9c60e92017-09-28 19:06:39 -0400297 swapchainImage.imageView.destroy(device);
298 swapchainImage.framebuffer.destroy(device);
299 swapchainImage.imageAcquiredSemaphore.destroy(device);
300 swapchainImage.commandsCompleteSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500301 }
Jamie Madill4d0bf552016-12-28 15:45:24 -0500302
303 if (mSwapchain)
304 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500305 vkDestroySwapchainKHR(device, mSwapchain, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500306 mSwapchain = VK_NULL_HANDLE;
307 }
308
309 if (mSurface)
310 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500311 vkDestroySurfaceKHR(instance, mSurface, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500312 mSurface = VK_NULL_HANDLE;
313 }
Jamie Madille09bd5d2016-11-29 16:20:35 -0500314}
315
Jamie Madillc564c072017-06-01 12:45:42 -0400316egl::Error WindowSurfaceVk::initialize(const egl::Display *display)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500317{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400318 const DisplayVk *displayVk = vk::GetImpl(display);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500319 return initializeImpl(displayVk->getRenderer()).toEGL(EGL_BAD_SURFACE);
320}
321
322vk::Error WindowSurfaceVk::initializeImpl(RendererVk *renderer)
323{
Frank Henigman29f148b2016-11-23 21:05:36 -0500324 gl::Extents windowSize;
Jamie Madill6cad7732018-07-11 09:01:17 -0400325 ANGLE_TRY(createSurfaceVk(renderer, &windowSize));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500326
327 uint32_t presentQueue = 0;
Jamie Madill6cad7732018-07-11 09:01:17 -0400328 ANGLE_TRY(renderer->selectPresentQueueForSurface(mSurface, &presentQueue));
Jamie Madillc6dbc252018-04-30 19:07:56 -0400329 ANGLE_UNUSED_VARIABLE(presentQueue);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500330
Jamie Madillacf2f3a2017-11-21 19:22:44 -0500331 const VkPhysicalDevice &physicalDevice = renderer->getPhysicalDevice();
Jamie Madill4d0bf552016-12-28 15:45:24 -0500332
333 VkSurfaceCapabilitiesKHR surfaceCaps;
334 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physicalDevice, mSurface, &surfaceCaps));
335
336 // Adjust width and height to the swapchain if necessary.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500337 uint32_t width = surfaceCaps.currentExtent.width;
338 uint32_t height = surfaceCaps.currentExtent.height;
339
340 // TODO(jmadill): Support devices which don't support copy. We use this for ReadPixels.
341 ANGLE_VK_CHECK((surfaceCaps.supportedUsageFlags & VK_IMAGE_USAGE_TRANSFER_SRC_BIT) != 0,
342 VK_ERROR_INITIALIZATION_FAILED);
343
Jamie Madillbc543422018-03-30 10:43:19 -0400344 EGLAttrib attribWidth = mState.attributes.get(EGL_WIDTH, 0);
345 EGLAttrib attribHeight = mState.attributes.get(EGL_HEIGHT, 0);
346
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500347 if (surfaceCaps.currentExtent.width == 0xFFFFFFFFu)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500348 {
349 ASSERT(surfaceCaps.currentExtent.height == 0xFFFFFFFFu);
350
Jamie Madillbc543422018-03-30 10:43:19 -0400351 if (attribWidth == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500352 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500353 width = windowSize.width;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500354 }
Jamie Madillbc543422018-03-30 10:43:19 -0400355 if (attribHeight == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500356 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500357 height = windowSize.height;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500358 }
359 }
360
Jamie Madillbc543422018-03-30 10:43:19 -0400361 gl::Extents extents(static_cast<int>(width), static_cast<int>(height), 1);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500362
Jamie Madill4d0bf552016-12-28 15:45:24 -0500363 uint32_t presentModeCount = 0;
364 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
365 &presentModeCount, nullptr));
366 ASSERT(presentModeCount > 0);
367
368 std::vector<VkPresentModeKHR> presentModes(presentModeCount);
369 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
370 &presentModeCount, presentModes.data()));
371
Jamie Madillf0eafe12017-02-21 15:03:50 -0500372 // Select appropriate present mode based on vsync parameter.
373 // TODO(jmadill): More complete implementation, which allows for changing and more values.
374 const EGLint minSwapInterval = mState.config->minSwapInterval;
375 const EGLint maxSwapInterval = mState.config->maxSwapInterval;
376 ASSERT(minSwapInterval == 0 || minSwapInterval == 1);
377 ASSERT(maxSwapInterval == 0 || maxSwapInterval == 1);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500378
Jamie Madillf0eafe12017-02-21 15:03:50 -0500379 VkPresentModeKHR swapchainPresentMode =
380 GetDesiredPresentMode(presentModes, minSwapInterval, maxSwapInterval);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500381
382 // Determine number of swapchain images. Aim for one more than the minimum.
383 uint32_t minImageCount = surfaceCaps.minImageCount + 1;
384 if (surfaceCaps.maxImageCount > 0 && minImageCount > surfaceCaps.maxImageCount)
385 {
386 minImageCount = surfaceCaps.maxImageCount;
387 }
388
389 // Default to identity transform.
390 VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
391 if ((surfaceCaps.supportedTransforms & preTransform) == 0)
392 {
393 preTransform = surfaceCaps.currentTransform;
394 }
395
Jamie Madill4d0bf552016-12-28 15:45:24 -0500396 uint32_t surfaceFormatCount = 0;
397 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
398 nullptr));
399
400 std::vector<VkSurfaceFormatKHR> surfaceFormats(surfaceFormatCount);
401 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
402 surfaceFormats.data()));
403
Jamie Madillbc543422018-03-30 10:43:19 -0400404 const vk::Format &format = renderer->getFormat(mState.config->renderTargetFormat);
405 VkFormat nativeFormat = format.vkTextureFormat;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500406
Jamie Madill4d0bf552016-12-28 15:45:24 -0500407 if (surfaceFormatCount == 1u && surfaceFormats[0].format == VK_FORMAT_UNDEFINED)
408 {
409 // This is fine.
410 }
411 else
412 {
413 bool foundFormat = false;
Jamie Madillbc543422018-03-30 10:43:19 -0400414 for (const VkSurfaceFormatKHR &surfaceFormat : surfaceFormats)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500415 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500416 if (surfaceFormat.format == nativeFormat)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500417 {
418 foundFormat = true;
419 break;
420 }
421 }
422
423 ANGLE_VK_CHECK(foundFormat, VK_ERROR_INITIALIZATION_FAILED);
424 }
425
Yuly Novikov12da5e72018-01-23 18:34:53 -0500426 VkCompositeAlphaFlagBitsKHR compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
427 if ((surfaceCaps.supportedCompositeAlpha & compositeAlpha) == 0)
428 {
429 compositeAlpha = VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR;
430 }
431 ANGLE_VK_CHECK((surfaceCaps.supportedCompositeAlpha & compositeAlpha) != 0,
432 VK_ERROR_INITIALIZATION_FAILED);
433
Jamie Madillf618c9e2018-02-15 14:45:40 -0500434 // We need transfer src for reading back from the backbuffer.
Geoff Lang9e141642018-06-27 11:43:18 -0400435 VkImageUsageFlags imageUsageFlags = kSurfaceVKColorImageUsageFlags;
Jamie Madillf618c9e2018-02-15 14:45:40 -0500436
Jamie Madill4d0bf552016-12-28 15:45:24 -0500437 VkSwapchainCreateInfoKHR swapchainInfo;
Jamie Madillf618c9e2018-02-15 14:45:40 -0500438 swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
439 swapchainInfo.pNext = nullptr;
440 swapchainInfo.flags = 0;
441 swapchainInfo.surface = mSurface;
442 swapchainInfo.minImageCount = minImageCount;
443 swapchainInfo.imageFormat = nativeFormat;
444 swapchainInfo.imageColorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
445 swapchainInfo.imageExtent.width = width;
446 swapchainInfo.imageExtent.height = height;
447 swapchainInfo.imageArrayLayers = 1;
448 swapchainInfo.imageUsage = imageUsageFlags;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500449 swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
450 swapchainInfo.queueFamilyIndexCount = 0;
451 swapchainInfo.pQueueFamilyIndices = nullptr;
452 swapchainInfo.preTransform = preTransform;
Yuly Novikov12da5e72018-01-23 18:34:53 -0500453 swapchainInfo.compositeAlpha = compositeAlpha;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500454 swapchainInfo.presentMode = swapchainPresentMode;
455 swapchainInfo.clipped = VK_TRUE;
456 swapchainInfo.oldSwapchain = VK_NULL_HANDLE;
457
Jamie Madill6a89d222017-11-02 11:59:51 -0400458 VkDevice device = renderer->getDevice();
Jamie Madill4d0bf552016-12-28 15:45:24 -0500459 ANGLE_VK_TRY(vkCreateSwapchainKHR(device, &swapchainInfo, nullptr, &mSwapchain));
460
461 // Intialize the swapchain image views.
462 uint32_t imageCount = 0;
463 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, nullptr));
464
465 std::vector<VkImage> swapchainImages(imageCount);
466 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, swapchainImages.data()));
467
Jamie Madill49ac74b2017-12-21 14:42:33 -0500468 // Allocate a command buffer for clearing our images to black.
469 vk::CommandBuffer *commandBuffer = nullptr;
Jamie Madill1f46bc12018-02-20 16:09:43 -0500470 ANGLE_TRY(beginWriteResource(renderer, &commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500471
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500472 VkClearColorValue transparentBlack;
473 transparentBlack.float32[0] = 0.0f;
474 transparentBlack.float32[1] = 0.0f;
475 transparentBlack.float32[2] = 0.0f;
476 transparentBlack.float32[3] = 0.0f;
477
Jamie Madill5deea722017-02-16 10:44:46 -0500478 mSwapchainImages.resize(imageCount);
Jamie Madilla9c60e92017-09-28 19:06:39 -0400479
480 ANGLE_TRY(mAcquireNextImageSemaphore.init(device));
Jamie Madill5deea722017-02-16 10:44:46 -0500481
482 for (uint32_t imageIndex = 0; imageIndex < imageCount; ++imageIndex)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500483 {
Jamie Madillbc543422018-03-30 10:43:19 -0400484 SwapchainImage &member = mSwapchainImages[imageIndex];
485 member.image.init2DWeakReference(swapchainImages[imageIndex], extents, format, 1);
Jamie Madilleebe2192018-07-11 09:01:18 -0400486 ANGLE_TRY(member.image.initImageView(device, gl::TextureType::_2D,
487 VK_IMAGE_ASPECT_COLOR_BIT, gl::SwizzleState(),
488 &member.imageView, 1));
Jamie Madill25301b62017-10-28 20:59:31 -0400489
490 // Set transfer dest layout, and clear the image to black.
Luc Ferronc20b9502018-05-24 09:30:17 -0400491 member.image.clearColor(transparentBlack, 0, 1, commandBuffer);
Jamie Madill25301b62017-10-28 20:59:31 -0400492
Jamie Madilla9c60e92017-09-28 19:06:39 -0400493 ANGLE_TRY(member.imageAcquiredSemaphore.init(device));
494 ANGLE_TRY(member.commandsCompleteSemaphore.init(device));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500495 }
496
Jamie Madille918de22017-04-12 10:21:11 -0400497 // Get the first available swapchain iamge.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500498 ANGLE_TRY(nextSwapchainImage(renderer));
499
Jamie Madillf618c9e2018-02-15 14:45:40 -0500500 // Initialize depth/stencil if requested.
501 if (mState.config->depthStencilFormat != GL_NONE)
502 {
503 const vk::Format &dsFormat = renderer->getFormat(mState.config->depthStencilFormat);
504
Geoff Lang9e141642018-06-27 11:43:18 -0400505 const VkImageUsageFlags dsUsage = kSurfaceVKDepthStencilImageUsageFlags;
Jamie Madillf618c9e2018-02-15 14:45:40 -0500506
Geoff Lang9e141642018-06-27 11:43:18 -0400507 ANGLE_TRY(mDepthStencilImage.init(device, gl::TextureType::_2D, extents, dsFormat, 1,
508 dsUsage, 1));
Jamie Madillbc543422018-03-30 10:43:19 -0400509 ANGLE_TRY(mDepthStencilImage.initMemory(device, renderer->getMemoryProperties(),
510 VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT));
Jamie Madillf618c9e2018-02-15 14:45:40 -0500511
Luc Ferron5bdf8bd2018-06-20 09:51:37 -0400512 const VkImageAspectFlags aspect = vk::GetDepthStencilAspectFlags(dsFormat.textureFormat());
Jamie Madillf618c9e2018-02-15 14:45:40 -0500513 VkClearDepthStencilValue depthStencilClearValue = {1.0f, 0};
514
515 // Set transfer dest layout, and clear the image.
Jamie Madill858c1cc2018-03-31 14:19:13 -0400516 mDepthStencilImage.clearDepthStencil(aspect, depthStencilClearValue, commandBuffer);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500517
Jamie Madill20fa8d52018-04-15 10:09:32 -0400518 ANGLE_TRY(mDepthStencilImage.initImageView(device, gl::TextureType::_2D, aspect,
Luc Ferron66410532018-04-20 12:47:45 -0400519 gl::SwizzleState(), &mDepthStencilImageView, 1));
Jamie Madillf618c9e2018-02-15 14:45:40 -0500520
Jamie Madillf618c9e2018-02-15 14:45:40 -0500521 // TODO(jmadill): Figure out how to pass depth/stencil image views to the RenderTargetVk.
522 }
523
Jamie Madill4d0bf552016-12-28 15:45:24 -0500524 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500525}
526
Geoff Langbf7b95d2018-05-01 16:48:21 -0400527FramebufferImpl *WindowSurfaceVk::createDefaultFramebuffer(const gl::Context *context,
528 const gl::FramebufferState &state)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500529{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500530 return FramebufferVk::CreateDefaultFBO(state, this);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500531}
532
Jamie Madillfe548342017-06-19 11:13:24 -0400533egl::Error WindowSurfaceVk::swap(const gl::Context *context)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500534{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400535 const DisplayVk *displayVk = vk::GetImpl(context->getCurrentDisplay());
Jamie Madilld4826152017-09-21 11:18:59 -0400536 RendererVk *renderer = displayVk->getRenderer();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500537
Jamie Madill49ac74b2017-12-21 14:42:33 -0500538 vk::CommandBuffer *swapCommands = nullptr;
Jamie Madill1f46bc12018-02-20 16:09:43 -0500539 ANGLE_TRY(beginWriteResource(renderer, &swapCommands));
Jamie Madilld4826152017-09-21 11:18:59 -0400540
Jamie Madillbc543422018-03-30 10:43:19 -0400541 SwapchainImage &image = mSwapchainImages[mCurrentSwapchainImageIndex];
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500542
Jamie Madill858c1cc2018-03-31 14:19:13 -0400543 image.image.changeLayoutWithStages(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
544 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
545 VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, swapCommands);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500546
Jamie Madill49ac74b2017-12-21 14:42:33 -0500547 ANGLE_TRY(
548 renderer->flush(context, image.imageAcquiredSemaphore, image.commandsCompleteSemaphore));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500549
550 VkPresentInfoKHR presentInfo;
551 presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
552 presentInfo.pNext = nullptr;
Jamie Madille918de22017-04-12 10:21:11 -0400553 presentInfo.waitSemaphoreCount = 1;
Jamie Madilla9c60e92017-09-28 19:06:39 -0400554 presentInfo.pWaitSemaphores = image.commandsCompleteSemaphore.ptr();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500555 presentInfo.swapchainCount = 1;
556 presentInfo.pSwapchains = &mSwapchain;
557 presentInfo.pImageIndices = &mCurrentSwapchainImageIndex;
558 presentInfo.pResults = nullptr;
559
560 ANGLE_VK_TRY(vkQueuePresentKHR(renderer->getQueue(), &presentInfo));
561
Jamie Madillf0eafe12017-02-21 15:03:50 -0500562 // Get the next available swapchain image.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500563 ANGLE_TRY(nextSwapchainImage(renderer));
564
565 return vk::NoError();
566}
567
568vk::Error WindowSurfaceVk::nextSwapchainImage(RendererVk *renderer)
569{
570 VkDevice device = renderer->getDevice();
571
jchen10191a84a2018-01-31 16:09:45 +0800572 ANGLE_VK_TRY(vkAcquireNextImageKHR(device, mSwapchain, UINT64_MAX,
Jamie Madilla9c60e92017-09-28 19:06:39 -0400573 mAcquireNextImageSemaphore.getHandle(), VK_NULL_HANDLE,
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500574 &mCurrentSwapchainImageIndex));
575
Jamie Madillbc543422018-03-30 10:43:19 -0400576 SwapchainImage &image = mSwapchainImages[mCurrentSwapchainImageIndex];
Jamie Madilla9c60e92017-09-28 19:06:39 -0400577
578 // Swap the unused swapchain semaphore and the now active spare semaphore.
579 std::swap(image.imageAcquiredSemaphore, mAcquireNextImageSemaphore);
580
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500581 // Update RenderTarget pointers.
Jamie Madillbcf467f2018-05-23 09:46:00 -0400582 mColorRenderTarget.updateSwapchainImage(&image.image, &image.imageView);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500583
584 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500585}
586
Jamie Madillfe548342017-06-19 11:13:24 -0400587egl::Error WindowSurfaceVk::postSubBuffer(const gl::Context *context,
588 EGLint x,
589 EGLint y,
590 EGLint width,
591 EGLint height)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500592{
593 // TODO(jmadill)
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500594 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500595}
596
597egl::Error WindowSurfaceVk::querySurfacePointerANGLE(EGLint attribute, void **value)
598{
599 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500600 return egl::EglBadCurrentSurface();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500601}
602
Geoff Langccafa622018-05-02 13:07:53 -0400603egl::Error WindowSurfaceVk::bindTexImage(const gl::Context *context,
604 gl::Texture *texture,
605 EGLint buffer)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500606{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500607 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500608}
609
Geoff Langccafa622018-05-02 13:07:53 -0400610egl::Error WindowSurfaceVk::releaseTexImage(const gl::Context *context, EGLint buffer)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500611{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500612 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500613}
614
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700615egl::Error WindowSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
616 EGLuint64KHR * /*msc*/,
617 EGLuint64KHR * /*sbc*/)
618{
619 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500620 return egl::EglBadAccess();
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700621}
622
Jamie Madille09bd5d2016-11-29 16:20:35 -0500623void WindowSurfaceVk::setSwapInterval(EGLint interval)
624{
625}
626
627EGLint WindowSurfaceVk::getWidth() const
628{
Jamie Madillbcf467f2018-05-23 09:46:00 -0400629 return static_cast<EGLint>(mColorRenderTarget.getImageExtents().width);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500630}
631
632EGLint WindowSurfaceVk::getHeight() const
633{
Jamie Madillbcf467f2018-05-23 09:46:00 -0400634 return static_cast<EGLint>(mColorRenderTarget.getImageExtents().height);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500635}
636
637EGLint WindowSurfaceVk::isPostSubBufferSupported() const
638{
639 // TODO(jmadill)
640 return EGL_FALSE;
641}
642
643EGLint WindowSurfaceVk::getSwapBehavior() const
644{
645 // TODO(jmadill)
646 return EGL_BUFFER_DESTROYED;
647}
648
Jamie Madill4928b7c2017-06-20 12:57:39 -0400649gl::Error WindowSurfaceVk::getAttachmentRenderTarget(const gl::Context * /*context*/,
Jamie Madillf618c9e2018-02-15 14:45:40 -0500650 GLenum binding,
Jamie Madill4fd95d52017-04-05 11:22:18 -0400651 const gl::ImageIndex & /*target*/,
652 FramebufferAttachmentRenderTarget **rtOut)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500653{
Jamie Madillf618c9e2018-02-15 14:45:40 -0500654 if (binding == GL_BACK)
655 {
656 *rtOut = &mColorRenderTarget;
657 }
658 else
659 {
660 ASSERT(binding == GL_DEPTH || binding == GL_STENCIL || binding == GL_DEPTH_STENCIL);
661 *rtOut = &mDepthStencilRenderTarget;
662 }
663
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500664 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400665}
666
Jamie Madillab9f9c32017-01-17 17:47:34 -0500667gl::ErrorOrResult<vk::Framebuffer *> WindowSurfaceVk::getCurrentFramebuffer(
668 VkDevice device,
669 const vk::RenderPass &compatibleRenderPass)
670{
Jamie Madillbc543422018-03-30 10:43:19 -0400671 vk::Framebuffer &currentFramebuffer = mSwapchainImages[mCurrentSwapchainImageIndex].framebuffer;
Jamie Madilla9c60e92017-09-28 19:06:39 -0400672
673 if (currentFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500674 {
675 // Validation layers should detect if the render pass is really compatible.
Jamie Madilla9c60e92017-09-28 19:06:39 -0400676 return &currentFramebuffer;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500677 }
678
679 VkFramebufferCreateInfo framebufferInfo;
680
Jamie Madillbcf467f2018-05-23 09:46:00 -0400681 const gl::Extents &extents = mColorRenderTarget.getImageExtents();
Jamie Madillf618c9e2018-02-15 14:45:40 -0500682 std::array<VkImageView, 2> imageViews = {{VK_NULL_HANDLE, mDepthStencilImageView.getHandle()}};
683
Jamie Madillab9f9c32017-01-17 17:47:34 -0500684 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
685 framebufferInfo.pNext = nullptr;
686 framebufferInfo.flags = 0;
687 framebufferInfo.renderPass = compatibleRenderPass.getHandle();
Jamie Madillbc543422018-03-30 10:43:19 -0400688 framebufferInfo.attachmentCount = (mDepthStencilImage.valid() ? 2u : 1u);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500689 framebufferInfo.pAttachments = imageViews.data();
Jamie Madillbc543422018-03-30 10:43:19 -0400690 framebufferInfo.width = static_cast<uint32_t>(extents.width);
691 framebufferInfo.height = static_cast<uint32_t>(extents.height);
Jamie Madillab9f9c32017-01-17 17:47:34 -0500692 framebufferInfo.layers = 1;
693
Jamie Madillbc543422018-03-30 10:43:19 -0400694 for (SwapchainImage &swapchainImage : mSwapchainImages)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500695 {
Jamie Madillf618c9e2018-02-15 14:45:40 -0500696 imageViews[0] = swapchainImage.imageView.getHandle();
Jamie Madill25301b62017-10-28 20:59:31 -0400697 ANGLE_TRY(swapchainImage.framebuffer.init(device, framebufferInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500698 }
699
Jamie Madilla9c60e92017-09-28 19:06:39 -0400700 ASSERT(currentFramebuffer.valid());
701 return &currentFramebuffer;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500702}
703
Jamie Madill05b35b22017-10-03 09:01:44 -0400704gl::Error WindowSurfaceVk::initializeContents(const gl::Context *context,
705 const gl::ImageIndex &imageIndex)
706{
707 UNIMPLEMENTED();
708 return gl::NoError();
709}
710
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400711} // namespace rx