blob: 5a955ab6c3fea3ebade18be20913dcb7bae2e808 [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
Jamie Madill4d0bf552016-12-28 15:45:24 -050051} // namespace
52
Jamie Madille09bd5d2016-11-29 16:20:35 -050053OffscreenSurfaceVk::OffscreenSurfaceVk(const egl::SurfaceState &surfaceState,
54 EGLint width,
55 EGLint height)
56 : SurfaceImpl(surfaceState), mWidth(width), mHeight(height)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040057{
58}
59
Jamie Madille09bd5d2016-11-29 16:20:35 -050060OffscreenSurfaceVk::~OffscreenSurfaceVk()
Jamie Madill9e54b5a2016-05-25 12:57:39 -040061{
62}
63
Jamie Madillc564c072017-06-01 12:45:42 -040064egl::Error OffscreenSurfaceVk::initialize(const egl::Display *display)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040065{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050066 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040067}
68
Geoff Langbf7b95d2018-05-01 16:48:21 -040069FramebufferImpl *OffscreenSurfaceVk::createDefaultFramebuffer(const gl::Context *context,
70 const gl::FramebufferState &state)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040071{
Jamie Madill7b57b9d2017-01-13 09:33:38 -050072 // Use a user FBO for an offscreen RT.
73 return FramebufferVk::CreateUserFBO(state);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040074}
75
Jamie Madillfe548342017-06-19 11:13:24 -040076egl::Error OffscreenSurfaceVk::swap(const gl::Context *context)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040077{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050078 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040079}
80
Jamie Madillfe548342017-06-19 11:13:24 -040081egl::Error OffscreenSurfaceVk::postSubBuffer(const gl::Context * /*context*/,
82 EGLint /*x*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -050083 EGLint /*y*/,
84 EGLint /*width*/,
85 EGLint /*height*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040086{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050087 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040088}
89
Jamie Madille09bd5d2016-11-29 16:20:35 -050090egl::Error OffscreenSurfaceVk::querySurfacePointerANGLE(EGLint /*attribute*/, void ** /*value*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040091{
Jamie Madille09bd5d2016-11-29 16:20:35 -050092 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050093 return egl::EglBadCurrentSurface();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040094}
95
Geoff Langccafa622018-05-02 13:07:53 -040096egl::Error OffscreenSurfaceVk::bindTexImage(const gl::Context * /*context*/,
97 gl::Texture * /*texture*/,
98 EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040099{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500100 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400101}
102
Geoff Langccafa622018-05-02 13:07:53 -0400103egl::Error OffscreenSurfaceVk::releaseTexImage(const gl::Context * /*context*/, EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400104{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500105 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400106}
107
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700108egl::Error OffscreenSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
109 EGLuint64KHR * /*msc*/,
110 EGLuint64KHR * /*sbc*/)
111{
112 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500113 return egl::EglBadAccess();
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700114}
115
Jamie Madille09bd5d2016-11-29 16:20:35 -0500116void OffscreenSurfaceVk::setSwapInterval(EGLint /*interval*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400117{
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400118}
119
Jamie Madille09bd5d2016-11-29 16:20:35 -0500120EGLint OffscreenSurfaceVk::getWidth() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400121{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500122 return mWidth;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400123}
124
Jamie Madille09bd5d2016-11-29 16:20:35 -0500125EGLint OffscreenSurfaceVk::getHeight() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400126{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500127 return mHeight;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400128}
129
Jamie Madille09bd5d2016-11-29 16:20:35 -0500130EGLint OffscreenSurfaceVk::isPostSubBufferSupported() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400131{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500132 return EGL_FALSE;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400133}
134
Jamie Madille09bd5d2016-11-29 16:20:35 -0500135EGLint OffscreenSurfaceVk::getSwapBehavior() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400136{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500137 return EGL_BUFFER_PRESERVED;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400138}
139
Jamie Madille09bd5d2016-11-29 16:20:35 -0500140gl::Error OffscreenSurfaceVk::getAttachmentRenderTarget(
Jamie Madill4928b7c2017-06-20 12:57:39 -0400141 const gl::Context * /*context*/,
Jamie Madill4fd95d52017-04-05 11:22:18 -0400142 GLenum /*binding*/,
143 const gl::ImageIndex & /*imageIndex*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -0500144 FramebufferAttachmentRenderTarget ** /*rtOut*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400145{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500146 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500147 return gl::InternalError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500148}
149
Jamie Madill05b35b22017-10-03 09:01:44 -0400150gl::Error OffscreenSurfaceVk::initializeContents(const gl::Context *context,
151 const gl::ImageIndex &imageIndex)
152{
153 UNIMPLEMENTED();
154 return gl::NoError();
155}
156
Jamie Madillbc543422018-03-30 10:43:19 -0400157WindowSurfaceVk::SwapchainImage::SwapchainImage() = default;
158WindowSurfaceVk::SwapchainImage::~SwapchainImage() = default;
159
160WindowSurfaceVk::SwapchainImage::SwapchainImage(SwapchainImage &&other)
161 : image(std::move(other.image)),
162 imageView(std::move(other.imageView)),
163 framebuffer(std::move(other.framebuffer)),
164 imageAcquiredSemaphore(std::move(other.imageAcquiredSemaphore)),
165 commandsCompleteSemaphore(std::move(other.commandsCompleteSemaphore))
166{
167}
Jamie Madillacf2f3a2017-11-21 19:22:44 -0500168
Jamie Madill4d0bf552016-12-28 15:45:24 -0500169WindowSurfaceVk::WindowSurfaceVk(const egl::SurfaceState &surfaceState,
170 EGLNativeWindowType window,
171 EGLint width,
172 EGLint height)
173 : SurfaceImpl(surfaceState),
174 mNativeWindowType(window),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500175 mSurface(VK_NULL_HANDLE),
Frank Henigman29f148b2016-11-23 21:05:36 -0500176 mInstance(VK_NULL_HANDLE),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500177 mSwapchain(VK_NULL_HANDLE),
Jamie Madillbcf467f2018-05-23 09:46:00 -0400178 mColorRenderTarget(nullptr, nullptr, this),
179 mDepthStencilRenderTarget(&mDepthStencilImage, &mDepthStencilImageView, this),
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500180 mCurrentSwapchainImageIndex(0)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500181{
182}
183
184WindowSurfaceVk::~WindowSurfaceVk()
185{
Jamie Madill70ee0f62017-02-06 16:04:20 -0500186 ASSERT(mSurface == VK_NULL_HANDLE);
187 ASSERT(mSwapchain == VK_NULL_HANDLE);
Jamie Madill70ee0f62017-02-06 16:04:20 -0500188}
189
Jamie Madillc564c072017-06-01 12:45:42 -0400190void WindowSurfaceVk::destroy(const egl::Display *display)
Jamie Madill70ee0f62017-02-06 16:04:20 -0500191{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400192 const DisplayVk *displayVk = vk::GetImpl(display);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500193 RendererVk *renderer = displayVk->getRenderer();
194 VkDevice device = renderer->getDevice();
195 VkInstance instance = renderer->getInstance();
Jamie Madill70ee0f62017-02-06 16:04:20 -0500196
Jamie Madillf618c9e2018-02-15 14:45:40 -0500197 // We might not need to flush the pipe here.
198 renderer->finish(display->getProxyContext());
Jamie Madille918de22017-04-12 10:21:11 -0400199
Jamie Madilla9c60e92017-09-28 19:06:39 -0400200 mAcquireNextImageSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500201
Jamie Madillbc543422018-03-30 10:43:19 -0400202 mDepthStencilImage.release(renderer->getCurrentQueueSerial(), renderer);
203 mDepthStencilImageView.destroy(device);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500204
Jamie Madillbc543422018-03-30 10:43:19 -0400205 for (SwapchainImage &swapchainImage : mSwapchainImages)
Jamie Madill70ee0f62017-02-06 16:04:20 -0500206 {
Jamie Madilla9c60e92017-09-28 19:06:39 -0400207 // Although we don't own the swapchain image handles, we need to keep our shutdown clean.
Jamie Madill858c1cc2018-03-31 14:19:13 -0400208 swapchainImage.image.resetImageWeakReference();
Jamie Madillbc543422018-03-30 10:43:19 -0400209 swapchainImage.image.destroy(device);
Jamie Madilla9c60e92017-09-28 19:06:39 -0400210 swapchainImage.imageView.destroy(device);
211 swapchainImage.framebuffer.destroy(device);
212 swapchainImage.imageAcquiredSemaphore.destroy(device);
213 swapchainImage.commandsCompleteSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500214 }
Jamie Madill4d0bf552016-12-28 15:45:24 -0500215
216 if (mSwapchain)
217 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500218 vkDestroySwapchainKHR(device, mSwapchain, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500219 mSwapchain = VK_NULL_HANDLE;
220 }
221
222 if (mSurface)
223 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500224 vkDestroySurfaceKHR(instance, mSurface, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500225 mSurface = VK_NULL_HANDLE;
226 }
Jamie Madille09bd5d2016-11-29 16:20:35 -0500227}
228
Jamie Madillc564c072017-06-01 12:45:42 -0400229egl::Error WindowSurfaceVk::initialize(const egl::Display *display)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500230{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400231 const DisplayVk *displayVk = vk::GetImpl(display);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500232 return initializeImpl(displayVk->getRenderer()).toEGL(EGL_BAD_SURFACE);
233}
234
235vk::Error WindowSurfaceVk::initializeImpl(RendererVk *renderer)
236{
Frank Henigman29f148b2016-11-23 21:05:36 -0500237 gl::Extents windowSize;
238 ANGLE_TRY_RESULT(createSurfaceVk(renderer), windowSize);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500239
240 uint32_t presentQueue = 0;
241 ANGLE_TRY_RESULT(renderer->selectPresentQueueForSurface(mSurface), presentQueue);
Jamie Madillc6dbc252018-04-30 19:07:56 -0400242 ANGLE_UNUSED_VARIABLE(presentQueue);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500243
Jamie Madillacf2f3a2017-11-21 19:22:44 -0500244 const VkPhysicalDevice &physicalDevice = renderer->getPhysicalDevice();
Jamie Madill4d0bf552016-12-28 15:45:24 -0500245
246 VkSurfaceCapabilitiesKHR surfaceCaps;
247 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physicalDevice, mSurface, &surfaceCaps));
248
249 // Adjust width and height to the swapchain if necessary.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500250 uint32_t width = surfaceCaps.currentExtent.width;
251 uint32_t height = surfaceCaps.currentExtent.height;
252
253 // TODO(jmadill): Support devices which don't support copy. We use this for ReadPixels.
254 ANGLE_VK_CHECK((surfaceCaps.supportedUsageFlags & VK_IMAGE_USAGE_TRANSFER_SRC_BIT) != 0,
255 VK_ERROR_INITIALIZATION_FAILED);
256
Jamie Madillbc543422018-03-30 10:43:19 -0400257 EGLAttrib attribWidth = mState.attributes.get(EGL_WIDTH, 0);
258 EGLAttrib attribHeight = mState.attributes.get(EGL_HEIGHT, 0);
259
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500260 if (surfaceCaps.currentExtent.width == 0xFFFFFFFFu)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500261 {
262 ASSERT(surfaceCaps.currentExtent.height == 0xFFFFFFFFu);
263
Jamie Madillbc543422018-03-30 10:43:19 -0400264 if (attribWidth == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500265 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500266 width = windowSize.width;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500267 }
Jamie Madillbc543422018-03-30 10:43:19 -0400268 if (attribHeight == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500269 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500270 height = windowSize.height;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500271 }
272 }
273
Jamie Madillbc543422018-03-30 10:43:19 -0400274 gl::Extents extents(static_cast<int>(width), static_cast<int>(height), 1);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500275
Jamie Madill4d0bf552016-12-28 15:45:24 -0500276 uint32_t presentModeCount = 0;
277 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
278 &presentModeCount, nullptr));
279 ASSERT(presentModeCount > 0);
280
281 std::vector<VkPresentModeKHR> presentModes(presentModeCount);
282 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
283 &presentModeCount, presentModes.data()));
284
Jamie Madillf0eafe12017-02-21 15:03:50 -0500285 // Select appropriate present mode based on vsync parameter.
286 // TODO(jmadill): More complete implementation, which allows for changing and more values.
287 const EGLint minSwapInterval = mState.config->minSwapInterval;
288 const EGLint maxSwapInterval = mState.config->maxSwapInterval;
289 ASSERT(minSwapInterval == 0 || minSwapInterval == 1);
290 ASSERT(maxSwapInterval == 0 || maxSwapInterval == 1);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500291
Jamie Madillf0eafe12017-02-21 15:03:50 -0500292 VkPresentModeKHR swapchainPresentMode =
293 GetDesiredPresentMode(presentModes, minSwapInterval, maxSwapInterval);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500294
295 // Determine number of swapchain images. Aim for one more than the minimum.
296 uint32_t minImageCount = surfaceCaps.minImageCount + 1;
297 if (surfaceCaps.maxImageCount > 0 && minImageCount > surfaceCaps.maxImageCount)
298 {
299 minImageCount = surfaceCaps.maxImageCount;
300 }
301
302 // Default to identity transform.
303 VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
304 if ((surfaceCaps.supportedTransforms & preTransform) == 0)
305 {
306 preTransform = surfaceCaps.currentTransform;
307 }
308
Jamie Madill4d0bf552016-12-28 15:45:24 -0500309 uint32_t surfaceFormatCount = 0;
310 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
311 nullptr));
312
313 std::vector<VkSurfaceFormatKHR> surfaceFormats(surfaceFormatCount);
314 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
315 surfaceFormats.data()));
316
Jamie Madillbc543422018-03-30 10:43:19 -0400317 const vk::Format &format = renderer->getFormat(mState.config->renderTargetFormat);
318 VkFormat nativeFormat = format.vkTextureFormat;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500319
Jamie Madill4d0bf552016-12-28 15:45:24 -0500320 if (surfaceFormatCount == 1u && surfaceFormats[0].format == VK_FORMAT_UNDEFINED)
321 {
322 // This is fine.
323 }
324 else
325 {
326 bool foundFormat = false;
Jamie Madillbc543422018-03-30 10:43:19 -0400327 for (const VkSurfaceFormatKHR &surfaceFormat : surfaceFormats)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500328 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500329 if (surfaceFormat.format == nativeFormat)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500330 {
331 foundFormat = true;
332 break;
333 }
334 }
335
336 ANGLE_VK_CHECK(foundFormat, VK_ERROR_INITIALIZATION_FAILED);
337 }
338
Yuly Novikov12da5e72018-01-23 18:34:53 -0500339 VkCompositeAlphaFlagBitsKHR compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
340 if ((surfaceCaps.supportedCompositeAlpha & compositeAlpha) == 0)
341 {
342 compositeAlpha = VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR;
343 }
344 ANGLE_VK_CHECK((surfaceCaps.supportedCompositeAlpha & compositeAlpha) != 0,
345 VK_ERROR_INITIALIZATION_FAILED);
346
Jamie Madillf618c9e2018-02-15 14:45:40 -0500347 // We need transfer src for reading back from the backbuffer.
348 VkImageUsageFlags imageUsageFlags =
349 (VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_SRC_BIT |
350 VK_IMAGE_USAGE_TRANSFER_DST_BIT);
351
Jamie Madill4d0bf552016-12-28 15:45:24 -0500352 VkSwapchainCreateInfoKHR swapchainInfo;
Jamie Madillf618c9e2018-02-15 14:45:40 -0500353 swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
354 swapchainInfo.pNext = nullptr;
355 swapchainInfo.flags = 0;
356 swapchainInfo.surface = mSurface;
357 swapchainInfo.minImageCount = minImageCount;
358 swapchainInfo.imageFormat = nativeFormat;
359 swapchainInfo.imageColorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
360 swapchainInfo.imageExtent.width = width;
361 swapchainInfo.imageExtent.height = height;
362 swapchainInfo.imageArrayLayers = 1;
363 swapchainInfo.imageUsage = imageUsageFlags;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500364 swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
365 swapchainInfo.queueFamilyIndexCount = 0;
366 swapchainInfo.pQueueFamilyIndices = nullptr;
367 swapchainInfo.preTransform = preTransform;
Yuly Novikov12da5e72018-01-23 18:34:53 -0500368 swapchainInfo.compositeAlpha = compositeAlpha;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500369 swapchainInfo.presentMode = swapchainPresentMode;
370 swapchainInfo.clipped = VK_TRUE;
371 swapchainInfo.oldSwapchain = VK_NULL_HANDLE;
372
Jamie Madill6a89d222017-11-02 11:59:51 -0400373 VkDevice device = renderer->getDevice();
Jamie Madill4d0bf552016-12-28 15:45:24 -0500374 ANGLE_VK_TRY(vkCreateSwapchainKHR(device, &swapchainInfo, nullptr, &mSwapchain));
375
376 // Intialize the swapchain image views.
377 uint32_t imageCount = 0;
378 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, nullptr));
379
380 std::vector<VkImage> swapchainImages(imageCount);
381 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, swapchainImages.data()));
382
Jamie Madill49ac74b2017-12-21 14:42:33 -0500383 // Allocate a command buffer for clearing our images to black.
384 vk::CommandBuffer *commandBuffer = nullptr;
Jamie Madill1f46bc12018-02-20 16:09:43 -0500385 ANGLE_TRY(beginWriteResource(renderer, &commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500386
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500387 VkClearColorValue transparentBlack;
388 transparentBlack.float32[0] = 0.0f;
389 transparentBlack.float32[1] = 0.0f;
390 transparentBlack.float32[2] = 0.0f;
391 transparentBlack.float32[3] = 0.0f;
392
Jamie Madill5deea722017-02-16 10:44:46 -0500393 mSwapchainImages.resize(imageCount);
Jamie Madilla9c60e92017-09-28 19:06:39 -0400394
395 ANGLE_TRY(mAcquireNextImageSemaphore.init(device));
Jamie Madill5deea722017-02-16 10:44:46 -0500396
397 for (uint32_t imageIndex = 0; imageIndex < imageCount; ++imageIndex)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500398 {
Jamie Madillbc543422018-03-30 10:43:19 -0400399 SwapchainImage &member = mSwapchainImages[imageIndex];
400 member.image.init2DWeakReference(swapchainImages[imageIndex], extents, format, 1);
Jamie Madill20fa8d52018-04-15 10:09:32 -0400401 member.image.initImageView(device, gl::TextureType::_2D, VK_IMAGE_ASPECT_COLOR_BIT,
Luc Ferron66410532018-04-20 12:47:45 -0400402 gl::SwizzleState(), &member.imageView, 1);
Jamie Madill25301b62017-10-28 20:59:31 -0400403
404 // Set transfer dest layout, and clear the image to black.
Luc Ferronc20b9502018-05-24 09:30:17 -0400405 member.image.clearColor(transparentBlack, 0, 1, commandBuffer);
Jamie Madill25301b62017-10-28 20:59:31 -0400406
Jamie Madilla9c60e92017-09-28 19:06:39 -0400407 ANGLE_TRY(member.imageAcquiredSemaphore.init(device));
408 ANGLE_TRY(member.commandsCompleteSemaphore.init(device));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500409 }
410
Jamie Madille918de22017-04-12 10:21:11 -0400411 // Get the first available swapchain iamge.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500412 ANGLE_TRY(nextSwapchainImage(renderer));
413
Jamie Madillf618c9e2018-02-15 14:45:40 -0500414 // Initialize depth/stencil if requested.
415 if (mState.config->depthStencilFormat != GL_NONE)
416 {
417 const vk::Format &dsFormat = renderer->getFormat(mState.config->depthStencilFormat);
418
419 const VkImageUsageFlags usage =
420 (VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT |
421 VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_SAMPLED_BIT);
422
Jamie Madill20fa8d52018-04-15 10:09:32 -0400423 ANGLE_TRY(
Luc Ferron66410532018-04-20 12:47:45 -0400424 mDepthStencilImage.init(device, gl::TextureType::_2D, extents, dsFormat, 1, usage, 1));
Jamie Madillbc543422018-03-30 10:43:19 -0400425 ANGLE_TRY(mDepthStencilImage.initMemory(device, renderer->getMemoryProperties(),
426 VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT));
Jamie Madillf618c9e2018-02-15 14:45:40 -0500427
Luc Ferron5bdf8bd2018-06-20 09:51:37 -0400428 const VkImageAspectFlags aspect = vk::GetDepthStencilAspectFlags(dsFormat.textureFormat());
Jamie Madillf618c9e2018-02-15 14:45:40 -0500429 VkClearDepthStencilValue depthStencilClearValue = {1.0f, 0};
430
431 // Set transfer dest layout, and clear the image.
Jamie Madill858c1cc2018-03-31 14:19:13 -0400432 mDepthStencilImage.clearDepthStencil(aspect, depthStencilClearValue, commandBuffer);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500433
Jamie Madill20fa8d52018-04-15 10:09:32 -0400434 ANGLE_TRY(mDepthStencilImage.initImageView(device, gl::TextureType::_2D, aspect,
Luc Ferron66410532018-04-20 12:47:45 -0400435 gl::SwizzleState(), &mDepthStencilImageView, 1));
Jamie Madillf618c9e2018-02-15 14:45:40 -0500436
Jamie Madillf618c9e2018-02-15 14:45:40 -0500437 // TODO(jmadill): Figure out how to pass depth/stencil image views to the RenderTargetVk.
438 }
439
Jamie Madill4d0bf552016-12-28 15:45:24 -0500440 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500441}
442
Geoff Langbf7b95d2018-05-01 16:48:21 -0400443FramebufferImpl *WindowSurfaceVk::createDefaultFramebuffer(const gl::Context *context,
444 const gl::FramebufferState &state)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500445{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500446 return FramebufferVk::CreateDefaultFBO(state, this);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500447}
448
Jamie Madillfe548342017-06-19 11:13:24 -0400449egl::Error WindowSurfaceVk::swap(const gl::Context *context)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500450{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400451 const DisplayVk *displayVk = vk::GetImpl(context->getCurrentDisplay());
Jamie Madilld4826152017-09-21 11:18:59 -0400452 RendererVk *renderer = displayVk->getRenderer();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500453
Jamie Madill49ac74b2017-12-21 14:42:33 -0500454 vk::CommandBuffer *swapCommands = nullptr;
Jamie Madill1f46bc12018-02-20 16:09:43 -0500455 ANGLE_TRY(beginWriteResource(renderer, &swapCommands));
Jamie Madilld4826152017-09-21 11:18:59 -0400456
Jamie Madillbc543422018-03-30 10:43:19 -0400457 SwapchainImage &image = mSwapchainImages[mCurrentSwapchainImageIndex];
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500458
Jamie Madill858c1cc2018-03-31 14:19:13 -0400459 image.image.changeLayoutWithStages(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
460 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
461 VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, swapCommands);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500462
Jamie Madill49ac74b2017-12-21 14:42:33 -0500463 ANGLE_TRY(
464 renderer->flush(context, image.imageAcquiredSemaphore, image.commandsCompleteSemaphore));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500465
466 VkPresentInfoKHR presentInfo;
467 presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
468 presentInfo.pNext = nullptr;
Jamie Madille918de22017-04-12 10:21:11 -0400469 presentInfo.waitSemaphoreCount = 1;
Jamie Madilla9c60e92017-09-28 19:06:39 -0400470 presentInfo.pWaitSemaphores = image.commandsCompleteSemaphore.ptr();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500471 presentInfo.swapchainCount = 1;
472 presentInfo.pSwapchains = &mSwapchain;
473 presentInfo.pImageIndices = &mCurrentSwapchainImageIndex;
474 presentInfo.pResults = nullptr;
475
476 ANGLE_VK_TRY(vkQueuePresentKHR(renderer->getQueue(), &presentInfo));
477
Jamie Madillf0eafe12017-02-21 15:03:50 -0500478 // Get the next available swapchain image.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500479 ANGLE_TRY(nextSwapchainImage(renderer));
480
481 return vk::NoError();
482}
483
484vk::Error WindowSurfaceVk::nextSwapchainImage(RendererVk *renderer)
485{
486 VkDevice device = renderer->getDevice();
487
jchen10191a84a2018-01-31 16:09:45 +0800488 ANGLE_VK_TRY(vkAcquireNextImageKHR(device, mSwapchain, UINT64_MAX,
Jamie Madilla9c60e92017-09-28 19:06:39 -0400489 mAcquireNextImageSemaphore.getHandle(), VK_NULL_HANDLE,
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500490 &mCurrentSwapchainImageIndex));
491
Jamie Madillbc543422018-03-30 10:43:19 -0400492 SwapchainImage &image = mSwapchainImages[mCurrentSwapchainImageIndex];
Jamie Madilla9c60e92017-09-28 19:06:39 -0400493
494 // Swap the unused swapchain semaphore and the now active spare semaphore.
495 std::swap(image.imageAcquiredSemaphore, mAcquireNextImageSemaphore);
496
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500497 // Update RenderTarget pointers.
Jamie Madillbcf467f2018-05-23 09:46:00 -0400498 mColorRenderTarget.updateSwapchainImage(&image.image, &image.imageView);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500499
500 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500501}
502
Jamie Madillfe548342017-06-19 11:13:24 -0400503egl::Error WindowSurfaceVk::postSubBuffer(const gl::Context *context,
504 EGLint x,
505 EGLint y,
506 EGLint width,
507 EGLint height)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500508{
509 // TODO(jmadill)
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500510 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500511}
512
513egl::Error WindowSurfaceVk::querySurfacePointerANGLE(EGLint attribute, void **value)
514{
515 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500516 return egl::EglBadCurrentSurface();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500517}
518
Geoff Langccafa622018-05-02 13:07:53 -0400519egl::Error WindowSurfaceVk::bindTexImage(const gl::Context *context,
520 gl::Texture *texture,
521 EGLint buffer)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500522{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500523 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500524}
525
Geoff Langccafa622018-05-02 13:07:53 -0400526egl::Error WindowSurfaceVk::releaseTexImage(const gl::Context *context, EGLint buffer)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500527{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500528 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500529}
530
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700531egl::Error WindowSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
532 EGLuint64KHR * /*msc*/,
533 EGLuint64KHR * /*sbc*/)
534{
535 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500536 return egl::EglBadAccess();
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700537}
538
Jamie Madille09bd5d2016-11-29 16:20:35 -0500539void WindowSurfaceVk::setSwapInterval(EGLint interval)
540{
541}
542
543EGLint WindowSurfaceVk::getWidth() const
544{
Jamie Madillbcf467f2018-05-23 09:46:00 -0400545 return static_cast<EGLint>(mColorRenderTarget.getImageExtents().width);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500546}
547
548EGLint WindowSurfaceVk::getHeight() const
549{
Jamie Madillbcf467f2018-05-23 09:46:00 -0400550 return static_cast<EGLint>(mColorRenderTarget.getImageExtents().height);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500551}
552
553EGLint WindowSurfaceVk::isPostSubBufferSupported() const
554{
555 // TODO(jmadill)
556 return EGL_FALSE;
557}
558
559EGLint WindowSurfaceVk::getSwapBehavior() const
560{
561 // TODO(jmadill)
562 return EGL_BUFFER_DESTROYED;
563}
564
Jamie Madill4928b7c2017-06-20 12:57:39 -0400565gl::Error WindowSurfaceVk::getAttachmentRenderTarget(const gl::Context * /*context*/,
Jamie Madillf618c9e2018-02-15 14:45:40 -0500566 GLenum binding,
Jamie Madill4fd95d52017-04-05 11:22:18 -0400567 const gl::ImageIndex & /*target*/,
568 FramebufferAttachmentRenderTarget **rtOut)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500569{
Jamie Madillf618c9e2018-02-15 14:45:40 -0500570 if (binding == GL_BACK)
571 {
572 *rtOut = &mColorRenderTarget;
573 }
574 else
575 {
576 ASSERT(binding == GL_DEPTH || binding == GL_STENCIL || binding == GL_DEPTH_STENCIL);
577 *rtOut = &mDepthStencilRenderTarget;
578 }
579
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500580 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400581}
582
Jamie Madillab9f9c32017-01-17 17:47:34 -0500583gl::ErrorOrResult<vk::Framebuffer *> WindowSurfaceVk::getCurrentFramebuffer(
584 VkDevice device,
585 const vk::RenderPass &compatibleRenderPass)
586{
Jamie Madillbc543422018-03-30 10:43:19 -0400587 vk::Framebuffer &currentFramebuffer = mSwapchainImages[mCurrentSwapchainImageIndex].framebuffer;
Jamie Madilla9c60e92017-09-28 19:06:39 -0400588
589 if (currentFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500590 {
591 // Validation layers should detect if the render pass is really compatible.
Jamie Madilla9c60e92017-09-28 19:06:39 -0400592 return &currentFramebuffer;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500593 }
594
595 VkFramebufferCreateInfo framebufferInfo;
596
Jamie Madillbcf467f2018-05-23 09:46:00 -0400597 const gl::Extents &extents = mColorRenderTarget.getImageExtents();
Jamie Madillf618c9e2018-02-15 14:45:40 -0500598 std::array<VkImageView, 2> imageViews = {{VK_NULL_HANDLE, mDepthStencilImageView.getHandle()}};
599
Jamie Madillab9f9c32017-01-17 17:47:34 -0500600 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
601 framebufferInfo.pNext = nullptr;
602 framebufferInfo.flags = 0;
603 framebufferInfo.renderPass = compatibleRenderPass.getHandle();
Jamie Madillbc543422018-03-30 10:43:19 -0400604 framebufferInfo.attachmentCount = (mDepthStencilImage.valid() ? 2u : 1u);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500605 framebufferInfo.pAttachments = imageViews.data();
Jamie Madillbc543422018-03-30 10:43:19 -0400606 framebufferInfo.width = static_cast<uint32_t>(extents.width);
607 framebufferInfo.height = static_cast<uint32_t>(extents.height);
Jamie Madillab9f9c32017-01-17 17:47:34 -0500608 framebufferInfo.layers = 1;
609
Jamie Madillbc543422018-03-30 10:43:19 -0400610 for (SwapchainImage &swapchainImage : mSwapchainImages)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500611 {
Jamie Madillf618c9e2018-02-15 14:45:40 -0500612 imageViews[0] = swapchainImage.imageView.getHandle();
Jamie Madill25301b62017-10-28 20:59:31 -0400613 ANGLE_TRY(swapchainImage.framebuffer.init(device, framebufferInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500614 }
615
Jamie Madilla9c60e92017-09-28 19:06:39 -0400616 ASSERT(currentFramebuffer.valid());
617 return &currentFramebuffer;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500618}
619
Jamie Madill05b35b22017-10-03 09:01:44 -0400620gl::Error WindowSurfaceVk::initializeContents(const gl::Context *context,
621 const gl::ImageIndex &imageIndex)
622{
623 UNIMPLEMENTED();
624 return gl::NoError();
625}
626
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400627} // namespace rx