blob: 763adc27bf1a218aeace8963a80cafc508621587 [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
38 for (auto presentMode : presentModes)
39 {
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
Jamie Madille09bd5d2016-11-29 16:20:35 -050069FramebufferImpl *OffscreenSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040070{
Jamie Madill7b57b9d2017-01-13 09:33:38 -050071 // Use a user FBO for an offscreen RT.
72 return FramebufferVk::CreateUserFBO(state);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040073}
74
Jamie Madillfe548342017-06-19 11:13:24 -040075egl::Error OffscreenSurfaceVk::swap(const gl::Context *context)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040076{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050077 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040078}
79
Jamie Madillfe548342017-06-19 11:13:24 -040080egl::Error OffscreenSurfaceVk::postSubBuffer(const gl::Context * /*context*/,
81 EGLint /*x*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -050082 EGLint /*y*/,
83 EGLint /*width*/,
84 EGLint /*height*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040085{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050086 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040087}
88
Jamie Madille09bd5d2016-11-29 16:20:35 -050089egl::Error OffscreenSurfaceVk::querySurfacePointerANGLE(EGLint /*attribute*/, void ** /*value*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040090{
Jamie Madille09bd5d2016-11-29 16:20:35 -050091 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050092 return egl::EglBadCurrentSurface();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040093}
94
Jamie Madille09bd5d2016-11-29 16:20:35 -050095egl::Error OffscreenSurfaceVk::bindTexImage(gl::Texture * /*texture*/, EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040096{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050097 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040098}
99
Jamie Madille09bd5d2016-11-29 16:20:35 -0500100egl::Error OffscreenSurfaceVk::releaseTexImage(EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400101{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500102 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400103}
104
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700105egl::Error OffscreenSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
106 EGLuint64KHR * /*msc*/,
107 EGLuint64KHR * /*sbc*/)
108{
109 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500110 return egl::EglBadAccess();
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700111}
112
Jamie Madille09bd5d2016-11-29 16:20:35 -0500113void OffscreenSurfaceVk::setSwapInterval(EGLint /*interval*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400114{
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400115}
116
Jamie Madille09bd5d2016-11-29 16:20:35 -0500117EGLint OffscreenSurfaceVk::getWidth() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400118{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500119 return mWidth;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400120}
121
Jamie Madille09bd5d2016-11-29 16:20:35 -0500122EGLint OffscreenSurfaceVk::getHeight() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400123{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500124 return mHeight;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400125}
126
Jamie Madille09bd5d2016-11-29 16:20:35 -0500127EGLint OffscreenSurfaceVk::isPostSubBufferSupported() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400128{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500129 return EGL_FALSE;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400130}
131
Jamie Madille09bd5d2016-11-29 16:20:35 -0500132EGLint OffscreenSurfaceVk::getSwapBehavior() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400133{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500134 return EGL_BUFFER_PRESERVED;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400135}
136
Jamie Madille09bd5d2016-11-29 16:20:35 -0500137gl::Error OffscreenSurfaceVk::getAttachmentRenderTarget(
Jamie Madill4928b7c2017-06-20 12:57:39 -0400138 const gl::Context * /*context*/,
Jamie Madill4fd95d52017-04-05 11:22:18 -0400139 GLenum /*binding*/,
140 const gl::ImageIndex & /*imageIndex*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -0500141 FramebufferAttachmentRenderTarget ** /*rtOut*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400142{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500143 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500144 return gl::InternalError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500145}
146
Jamie Madill05b35b22017-10-03 09:01:44 -0400147gl::Error OffscreenSurfaceVk::initializeContents(const gl::Context *context,
148 const gl::ImageIndex &imageIndex)
149{
150 UNIMPLEMENTED();
151 return gl::NoError();
152}
153
Jamie Madillacf2f3a2017-11-21 19:22:44 -0500154WindowSurfaceVk::SwapchainImage::SwapchainImage() = default;
155WindowSurfaceVk::SwapchainImage::SwapchainImage(WindowSurfaceVk::SwapchainImage &&other) = default;
156WindowSurfaceVk::SwapchainImage::~SwapchainImage() = default;
157
Jamie Madill4d0bf552016-12-28 15:45:24 -0500158WindowSurfaceVk::WindowSurfaceVk(const egl::SurfaceState &surfaceState,
159 EGLNativeWindowType window,
160 EGLint width,
161 EGLint height)
162 : SurfaceImpl(surfaceState),
163 mNativeWindowType(window),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500164 mSurface(VK_NULL_HANDLE),
Frank Henigman29f148b2016-11-23 21:05:36 -0500165 mInstance(VK_NULL_HANDLE),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500166 mSwapchain(VK_NULL_HANDLE),
Jamie Madillf618c9e2018-02-15 14:45:40 -0500167 mColorRenderTarget(),
168 mDepthStencilRenderTarget(),
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500169 mCurrentSwapchainImageIndex(0)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500170{
Jamie Madillf618c9e2018-02-15 14:45:40 -0500171 mColorRenderTarget.extents.width = static_cast<GLint>(width);
172 mColorRenderTarget.extents.height = static_cast<GLint>(height);
173 mColorRenderTarget.extents.depth = 1;
174 mColorRenderTarget.resource = this;
Jamie Madille09bd5d2016-11-29 16:20:35 -0500175}
176
177WindowSurfaceVk::~WindowSurfaceVk()
178{
Jamie Madill70ee0f62017-02-06 16:04:20 -0500179 ASSERT(mSurface == VK_NULL_HANDLE);
180 ASSERT(mSwapchain == VK_NULL_HANDLE);
Jamie Madill70ee0f62017-02-06 16:04:20 -0500181}
182
Jamie Madillc564c072017-06-01 12:45:42 -0400183void WindowSurfaceVk::destroy(const egl::Display *display)
Jamie Madill70ee0f62017-02-06 16:04:20 -0500184{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400185 const DisplayVk *displayVk = vk::GetImpl(display);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500186 RendererVk *renderer = displayVk->getRenderer();
187 VkDevice device = renderer->getDevice();
188 VkInstance instance = renderer->getInstance();
Jamie Madill70ee0f62017-02-06 16:04:20 -0500189
Jamie Madillf618c9e2018-02-15 14:45:40 -0500190 // We might not need to flush the pipe here.
191 renderer->finish(display->getProxyContext());
Jamie Madille918de22017-04-12 10:21:11 -0400192
Jamie Madilla9c60e92017-09-28 19:06:39 -0400193 mAcquireNextImageSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500194
Jamie Madillf618c9e2018-02-15 14:45:40 -0500195 renderer->releaseResource(*this, &mDepthStencilImage);
196 renderer->releaseResource(*this, &mDepthStencilDeviceMemory);
197 renderer->releaseResource(*this, &mDepthStencilImageView);
198
Jamie Madilla9c60e92017-09-28 19:06:39 -0400199 for (auto &swapchainImage : mSwapchainImages)
Jamie Madill70ee0f62017-02-06 16:04:20 -0500200 {
Jamie Madilla9c60e92017-09-28 19:06:39 -0400201 // Although we don't own the swapchain image handles, we need to keep our shutdown clean.
202 swapchainImage.image.reset();
Jamie Madill70ee0f62017-02-06 16:04:20 -0500203
Jamie Madilla9c60e92017-09-28 19:06:39 -0400204 swapchainImage.imageView.destroy(device);
205 swapchainImage.framebuffer.destroy(device);
206 swapchainImage.imageAcquiredSemaphore.destroy(device);
207 swapchainImage.commandsCompleteSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500208 }
Jamie Madill4d0bf552016-12-28 15:45:24 -0500209
210 if (mSwapchain)
211 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500212 vkDestroySwapchainKHR(device, mSwapchain, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500213 mSwapchain = VK_NULL_HANDLE;
214 }
215
216 if (mSurface)
217 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500218 vkDestroySurfaceKHR(instance, mSurface, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500219 mSurface = VK_NULL_HANDLE;
220 }
Jamie Madille09bd5d2016-11-29 16:20:35 -0500221}
222
Jamie Madillc564c072017-06-01 12:45:42 -0400223egl::Error WindowSurfaceVk::initialize(const egl::Display *display)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500224{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400225 const DisplayVk *displayVk = vk::GetImpl(display);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500226 return initializeImpl(displayVk->getRenderer()).toEGL(EGL_BAD_SURFACE);
227}
228
229vk::Error WindowSurfaceVk::initializeImpl(RendererVk *renderer)
230{
Frank Henigman29f148b2016-11-23 21:05:36 -0500231 gl::Extents windowSize;
232 ANGLE_TRY_RESULT(createSurfaceVk(renderer), windowSize);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500233
234 uint32_t presentQueue = 0;
235 ANGLE_TRY_RESULT(renderer->selectPresentQueueForSurface(mSurface), presentQueue);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500236 UNUSED_VARIABLE(presentQueue);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500237
Jamie Madillacf2f3a2017-11-21 19:22:44 -0500238 const VkPhysicalDevice &physicalDevice = renderer->getPhysicalDevice();
Jamie Madill4d0bf552016-12-28 15:45:24 -0500239
240 VkSurfaceCapabilitiesKHR surfaceCaps;
241 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physicalDevice, mSurface, &surfaceCaps));
242
243 // Adjust width and height to the swapchain if necessary.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500244 uint32_t width = surfaceCaps.currentExtent.width;
245 uint32_t height = surfaceCaps.currentExtent.height;
246
247 // TODO(jmadill): Support devices which don't support copy. We use this for ReadPixels.
248 ANGLE_VK_CHECK((surfaceCaps.supportedUsageFlags & VK_IMAGE_USAGE_TRANSFER_SRC_BIT) != 0,
249 VK_ERROR_INITIALIZATION_FAILED);
250
251 if (surfaceCaps.currentExtent.width == 0xFFFFFFFFu)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500252 {
253 ASSERT(surfaceCaps.currentExtent.height == 0xFFFFFFFFu);
254
Jamie Madillf618c9e2018-02-15 14:45:40 -0500255 if (mColorRenderTarget.extents.width == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500256 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500257 width = windowSize.width;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500258 }
Jamie Madillf618c9e2018-02-15 14:45:40 -0500259 if (mColorRenderTarget.extents.height == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500260 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500261 height = windowSize.height;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500262 }
263 }
264
Jamie Madillf618c9e2018-02-15 14:45:40 -0500265 mColorRenderTarget.extents.width = static_cast<int>(width);
266 mColorRenderTarget.extents.height = static_cast<int>(height);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500267
Jamie Madill4d0bf552016-12-28 15:45:24 -0500268 uint32_t presentModeCount = 0;
269 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
270 &presentModeCount, nullptr));
271 ASSERT(presentModeCount > 0);
272
273 std::vector<VkPresentModeKHR> presentModes(presentModeCount);
274 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
275 &presentModeCount, presentModes.data()));
276
Jamie Madillf0eafe12017-02-21 15:03:50 -0500277 // Select appropriate present mode based on vsync parameter.
278 // TODO(jmadill): More complete implementation, which allows for changing and more values.
279 const EGLint minSwapInterval = mState.config->minSwapInterval;
280 const EGLint maxSwapInterval = mState.config->maxSwapInterval;
281 ASSERT(minSwapInterval == 0 || minSwapInterval == 1);
282 ASSERT(maxSwapInterval == 0 || maxSwapInterval == 1);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500283
Jamie Madillf0eafe12017-02-21 15:03:50 -0500284 VkPresentModeKHR swapchainPresentMode =
285 GetDesiredPresentMode(presentModes, minSwapInterval, maxSwapInterval);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500286
287 // Determine number of swapchain images. Aim for one more than the minimum.
288 uint32_t minImageCount = surfaceCaps.minImageCount + 1;
289 if (surfaceCaps.maxImageCount > 0 && minImageCount > surfaceCaps.maxImageCount)
290 {
291 minImageCount = surfaceCaps.maxImageCount;
292 }
293
294 // Default to identity transform.
295 VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
296 if ((surfaceCaps.supportedTransforms & preTransform) == 0)
297 {
298 preTransform = surfaceCaps.currentTransform;
299 }
300
Jamie Madill4d0bf552016-12-28 15:45:24 -0500301 uint32_t surfaceFormatCount = 0;
302 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
303 nullptr));
304
305 std::vector<VkSurfaceFormatKHR> surfaceFormats(surfaceFormatCount);
306 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
307 surfaceFormats.data()));
308
Jamie Madillf618c9e2018-02-15 14:45:40 -0500309 mColorRenderTarget.format = &renderer->getFormat(mState.config->renderTargetFormat);
310 VkFormat nativeFormat = mColorRenderTarget.format->vkTextureFormat;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500311
Jamie Madill4d0bf552016-12-28 15:45:24 -0500312 if (surfaceFormatCount == 1u && surfaceFormats[0].format == VK_FORMAT_UNDEFINED)
313 {
314 // This is fine.
315 }
316 else
317 {
318 bool foundFormat = false;
319 for (const auto &surfaceFormat : surfaceFormats)
320 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500321 if (surfaceFormat.format == nativeFormat)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500322 {
323 foundFormat = true;
324 break;
325 }
326 }
327
328 ANGLE_VK_CHECK(foundFormat, VK_ERROR_INITIALIZATION_FAILED);
329 }
330
Yuly Novikov12da5e72018-01-23 18:34:53 -0500331 VkCompositeAlphaFlagBitsKHR compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
332 if ((surfaceCaps.supportedCompositeAlpha & compositeAlpha) == 0)
333 {
334 compositeAlpha = VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR;
335 }
336 ANGLE_VK_CHECK((surfaceCaps.supportedCompositeAlpha & compositeAlpha) != 0,
337 VK_ERROR_INITIALIZATION_FAILED);
338
Jamie Madillf618c9e2018-02-15 14:45:40 -0500339 // We need transfer src for reading back from the backbuffer.
340 VkImageUsageFlags imageUsageFlags =
341 (VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_SRC_BIT |
342 VK_IMAGE_USAGE_TRANSFER_DST_BIT);
343
Jamie Madill4d0bf552016-12-28 15:45:24 -0500344 VkSwapchainCreateInfoKHR swapchainInfo;
Jamie Madillf618c9e2018-02-15 14:45:40 -0500345 swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
346 swapchainInfo.pNext = nullptr;
347 swapchainInfo.flags = 0;
348 swapchainInfo.surface = mSurface;
349 swapchainInfo.minImageCount = minImageCount;
350 swapchainInfo.imageFormat = nativeFormat;
351 swapchainInfo.imageColorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
352 swapchainInfo.imageExtent.width = width;
353 swapchainInfo.imageExtent.height = height;
354 swapchainInfo.imageArrayLayers = 1;
355 swapchainInfo.imageUsage = imageUsageFlags;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500356 swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
357 swapchainInfo.queueFamilyIndexCount = 0;
358 swapchainInfo.pQueueFamilyIndices = nullptr;
359 swapchainInfo.preTransform = preTransform;
Yuly Novikov12da5e72018-01-23 18:34:53 -0500360 swapchainInfo.compositeAlpha = compositeAlpha;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500361 swapchainInfo.presentMode = swapchainPresentMode;
362 swapchainInfo.clipped = VK_TRUE;
363 swapchainInfo.oldSwapchain = VK_NULL_HANDLE;
364
Jamie Madill6a89d222017-11-02 11:59:51 -0400365 VkDevice device = renderer->getDevice();
Jamie Madill4d0bf552016-12-28 15:45:24 -0500366 ANGLE_VK_TRY(vkCreateSwapchainKHR(device, &swapchainInfo, nullptr, &mSwapchain));
367
368 // Intialize the swapchain image views.
369 uint32_t imageCount = 0;
370 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, nullptr));
371
372 std::vector<VkImage> swapchainImages(imageCount);
373 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, swapchainImages.data()));
374
Jamie Madill49ac74b2017-12-21 14:42:33 -0500375 // Allocate a command buffer for clearing our images to black.
376 vk::CommandBuffer *commandBuffer = nullptr;
Jamie Madill1f46bc12018-02-20 16:09:43 -0500377 ANGLE_TRY(beginWriteResource(renderer, &commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500378
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500379 VkClearColorValue transparentBlack;
380 transparentBlack.float32[0] = 0.0f;
381 transparentBlack.float32[1] = 0.0f;
382 transparentBlack.float32[2] = 0.0f;
383 transparentBlack.float32[3] = 0.0f;
384
Jamie Madill5deea722017-02-16 10:44:46 -0500385 mSwapchainImages.resize(imageCount);
Jamie Madilla9c60e92017-09-28 19:06:39 -0400386
387 ANGLE_TRY(mAcquireNextImageSemaphore.init(device));
Jamie Madill5deea722017-02-16 10:44:46 -0500388
389 for (uint32_t imageIndex = 0; imageIndex < imageCount; ++imageIndex)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500390 {
Jamie Madill5deea722017-02-16 10:44:46 -0500391 VkImage swapchainImage = swapchainImages[imageIndex];
392
Jamie Madill4d0bf552016-12-28 15:45:24 -0500393 VkImageViewCreateInfo imageViewInfo;
394 imageViewInfo.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
395 imageViewInfo.pNext = nullptr;
396 imageViewInfo.flags = 0;
397 imageViewInfo.image = swapchainImage;
398 imageViewInfo.viewType = VK_IMAGE_VIEW_TYPE_2D;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500399 imageViewInfo.format = nativeFormat;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500400 imageViewInfo.components.r = VK_COMPONENT_SWIZZLE_R;
401 imageViewInfo.components.g = VK_COMPONENT_SWIZZLE_G;
402 imageViewInfo.components.b = VK_COMPONENT_SWIZZLE_B;
403 imageViewInfo.components.a = VK_COMPONENT_SWIZZLE_A;
404 imageViewInfo.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
405 imageViewInfo.subresourceRange.baseMipLevel = 0;
406 imageViewInfo.subresourceRange.levelCount = 1;
407 imageViewInfo.subresourceRange.baseArrayLayer = 0;
408 imageViewInfo.subresourceRange.layerCount = 1;
409
Jamie Madilla9c60e92017-09-28 19:06:39 -0400410 auto &member = mSwapchainImages[imageIndex];
411
Jamie Madill25301b62017-10-28 20:59:31 -0400412 member.image.setHandle(swapchainImage);
413 ANGLE_TRY(member.imageView.init(device, imageViewInfo));
414
415 // Set transfer dest layout, and clear the image to black.
Jamie Madilld33c77c2017-11-09 13:08:30 -0500416 member.image.changeLayoutWithStages(
417 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
418 VK_PIPELINE_STAGE_ALL_GRAPHICS_BIT, VK_PIPELINE_STAGE_TRANSFER_BIT, commandBuffer);
Jamie Madill25301b62017-10-28 20:59:31 -0400419 commandBuffer->clearSingleColorImage(member.image, transparentBlack);
420
Jamie Madilla9c60e92017-09-28 19:06:39 -0400421 ANGLE_TRY(member.imageAcquiredSemaphore.init(device));
422 ANGLE_TRY(member.commandsCompleteSemaphore.init(device));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500423 }
424
Jamie Madille918de22017-04-12 10:21:11 -0400425 // Get the first available swapchain iamge.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500426 ANGLE_TRY(nextSwapchainImage(renderer));
427
Jamie Madillf618c9e2018-02-15 14:45:40 -0500428 // Initialize depth/stencil if requested.
429 if (mState.config->depthStencilFormat != GL_NONE)
430 {
431 const vk::Format &dsFormat = renderer->getFormat(mState.config->depthStencilFormat);
432
433 const VkImageUsageFlags usage =
434 (VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT |
435 VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_SAMPLED_BIT);
436
437 VkImageCreateInfo imageInfo;
438 imageInfo.sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO;
439 imageInfo.pNext = nullptr;
440 imageInfo.flags = 0;
441 imageInfo.imageType = VK_IMAGE_TYPE_2D;
442 imageInfo.format = dsFormat.vkTextureFormat;
443 imageInfo.extent.width = static_cast<uint32_t>(width);
444 imageInfo.extent.height = static_cast<uint32_t>(height);
445 imageInfo.extent.depth = 1;
446 imageInfo.mipLevels = 1;
447 imageInfo.arrayLayers = 1;
448 imageInfo.samples = VK_SAMPLE_COUNT_1_BIT;
449 imageInfo.tiling = VK_IMAGE_TILING_OPTIMAL;
450 imageInfo.usage = usage;
451 imageInfo.sharingMode = VK_SHARING_MODE_EXCLUSIVE;
452 imageInfo.queueFamilyIndexCount = 0;
453 imageInfo.pQueueFamilyIndices = nullptr;
454 imageInfo.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
455
456 ANGLE_TRY(mDepthStencilImage.init(device, imageInfo));
457
458 // TODO(jmadill): Memory sub-allocation. http://anglebug.com/2162
459 size_t requiredSize;
460 ANGLE_TRY(vk::AllocateImageMemory(renderer, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT,
461 &mDepthStencilImage, &mDepthStencilDeviceMemory,
462 &requiredSize));
463
464 const VkImageAspectFlags aspect =
465 (dsFormat.textureFormat().depthBits > 0 ? VK_IMAGE_ASPECT_DEPTH_BIT : 0) |
466 (dsFormat.textureFormat().stencilBits > 0 ? VK_IMAGE_ASPECT_STENCIL_BIT : 0);
467
468 VkClearDepthStencilValue depthStencilClearValue = {1.0f, 0};
469
470 // Set transfer dest layout, and clear the image.
471 mDepthStencilImage.changeLayoutWithStages(aspect, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
472 VK_PIPELINE_STAGE_ALL_GRAPHICS_BIT,
473 VK_PIPELINE_STAGE_TRANSFER_BIT, commandBuffer);
474 commandBuffer->clearSingleDepthStencilImage(mDepthStencilImage, aspect,
475 depthStencilClearValue);
476
477 // Depth/Stencil image views.
478 VkImageViewCreateInfo imageViewInfo;
479 imageViewInfo.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
480 imageViewInfo.pNext = nullptr;
481 imageViewInfo.flags = 0;
482 imageViewInfo.image = mDepthStencilImage.getHandle();
483 imageViewInfo.viewType = VK_IMAGE_VIEW_TYPE_2D;
484 imageViewInfo.format = dsFormat.vkTextureFormat;
485 imageViewInfo.components.r = VK_COMPONENT_SWIZZLE_R;
486 imageViewInfo.components.g = VK_COMPONENT_SWIZZLE_G;
487 imageViewInfo.components.b = VK_COMPONENT_SWIZZLE_B;
488 imageViewInfo.components.a = VK_COMPONENT_SWIZZLE_A;
489 imageViewInfo.subresourceRange.aspectMask = aspect;
490 imageViewInfo.subresourceRange.baseMipLevel = 0;
491 imageViewInfo.subresourceRange.levelCount = 1;
492 imageViewInfo.subresourceRange.baseArrayLayer = 0;
493 imageViewInfo.subresourceRange.layerCount = 1;
494
495 ANGLE_TRY(mDepthStencilImageView.init(device, imageViewInfo));
496
497 mDepthStencilRenderTarget.extents.width = static_cast<GLint>(width);
498 mDepthStencilRenderTarget.extents.height = static_cast<GLint>(height);
499 mDepthStencilRenderTarget.extents.depth = 1;
500 mDepthStencilRenderTarget.resource = this;
501 mDepthStencilRenderTarget.image = &mDepthStencilImage;
502 mDepthStencilRenderTarget.format = &dsFormat;
503
504 // TODO(jmadill): Figure out how to pass depth/stencil image views to the RenderTargetVk.
505 }
506
Jamie Madill4d0bf552016-12-28 15:45:24 -0500507 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500508}
509
510FramebufferImpl *WindowSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
511{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500512 return FramebufferVk::CreateDefaultFBO(state, this);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500513}
514
Jamie Madillfe548342017-06-19 11:13:24 -0400515egl::Error WindowSurfaceVk::swap(const gl::Context *context)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500516{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400517 const DisplayVk *displayVk = vk::GetImpl(context->getCurrentDisplay());
Jamie Madilld4826152017-09-21 11:18:59 -0400518 RendererVk *renderer = displayVk->getRenderer();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500519
Jamie Madill49ac74b2017-12-21 14:42:33 -0500520 vk::CommandBuffer *swapCommands = nullptr;
Jamie Madill1f46bc12018-02-20 16:09:43 -0500521 ANGLE_TRY(beginWriteResource(renderer, &swapCommands));
Jamie Madilld4826152017-09-21 11:18:59 -0400522
Jamie Madilla9c60e92017-09-28 19:06:39 -0400523 auto &image = mSwapchainImages[mCurrentSwapchainImageIndex];
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500524
Jamie Madilla9c60e92017-09-28 19:06:39 -0400525 image.image.changeLayoutWithStages(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
526 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
Jamie Madill49ac74b2017-12-21 14:42:33 -0500527 VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, swapCommands);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500528
Jamie Madill49ac74b2017-12-21 14:42:33 -0500529 ANGLE_TRY(
530 renderer->flush(context, image.imageAcquiredSemaphore, image.commandsCompleteSemaphore));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500531
532 VkPresentInfoKHR presentInfo;
533 presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
534 presentInfo.pNext = nullptr;
Jamie Madille918de22017-04-12 10:21:11 -0400535 presentInfo.waitSemaphoreCount = 1;
Jamie Madilla9c60e92017-09-28 19:06:39 -0400536 presentInfo.pWaitSemaphores = image.commandsCompleteSemaphore.ptr();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500537 presentInfo.swapchainCount = 1;
538 presentInfo.pSwapchains = &mSwapchain;
539 presentInfo.pImageIndices = &mCurrentSwapchainImageIndex;
540 presentInfo.pResults = nullptr;
541
542 ANGLE_VK_TRY(vkQueuePresentKHR(renderer->getQueue(), &presentInfo));
543
Jamie Madillf0eafe12017-02-21 15:03:50 -0500544 // Get the next available swapchain image.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500545 ANGLE_TRY(nextSwapchainImage(renderer));
546
547 return vk::NoError();
548}
549
550vk::Error WindowSurfaceVk::nextSwapchainImage(RendererVk *renderer)
551{
552 VkDevice device = renderer->getDevice();
553
jchen10191a84a2018-01-31 16:09:45 +0800554 ANGLE_VK_TRY(vkAcquireNextImageKHR(device, mSwapchain, UINT64_MAX,
Jamie Madilla9c60e92017-09-28 19:06:39 -0400555 mAcquireNextImageSemaphore.getHandle(), VK_NULL_HANDLE,
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500556 &mCurrentSwapchainImageIndex));
557
Jamie Madilla9c60e92017-09-28 19:06:39 -0400558 auto &image = mSwapchainImages[mCurrentSwapchainImageIndex];
559
560 // Swap the unused swapchain semaphore and the now active spare semaphore.
561 std::swap(image.imageAcquiredSemaphore, mAcquireNextImageSemaphore);
562
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500563 // Update RenderTarget pointers.
Jamie Madillf618c9e2018-02-15 14:45:40 -0500564 mColorRenderTarget.image = &image.image;
565 mColorRenderTarget.imageView = &image.imageView;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500566
567 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500568}
569
Jamie Madillfe548342017-06-19 11:13:24 -0400570egl::Error WindowSurfaceVk::postSubBuffer(const gl::Context *context,
571 EGLint x,
572 EGLint y,
573 EGLint width,
574 EGLint height)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500575{
576 // TODO(jmadill)
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500577 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500578}
579
580egl::Error WindowSurfaceVk::querySurfacePointerANGLE(EGLint attribute, void **value)
581{
582 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500583 return egl::EglBadCurrentSurface();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500584}
585
586egl::Error WindowSurfaceVk::bindTexImage(gl::Texture *texture, EGLint buffer)
587{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500588 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500589}
590
591egl::Error WindowSurfaceVk::releaseTexImage(EGLint buffer)
592{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500593 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500594}
595
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700596egl::Error WindowSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
597 EGLuint64KHR * /*msc*/,
598 EGLuint64KHR * /*sbc*/)
599{
600 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500601 return egl::EglBadAccess();
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700602}
603
Jamie Madille09bd5d2016-11-29 16:20:35 -0500604void WindowSurfaceVk::setSwapInterval(EGLint interval)
605{
606}
607
608EGLint WindowSurfaceVk::getWidth() const
609{
Jamie Madillf618c9e2018-02-15 14:45:40 -0500610 return static_cast<EGLint>(mColorRenderTarget.extents.width);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500611}
612
613EGLint WindowSurfaceVk::getHeight() const
614{
Jamie Madillf618c9e2018-02-15 14:45:40 -0500615 return static_cast<EGLint>(mColorRenderTarget.extents.height);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500616}
617
618EGLint WindowSurfaceVk::isPostSubBufferSupported() const
619{
620 // TODO(jmadill)
621 return EGL_FALSE;
622}
623
624EGLint WindowSurfaceVk::getSwapBehavior() const
625{
626 // TODO(jmadill)
627 return EGL_BUFFER_DESTROYED;
628}
629
Jamie Madill4928b7c2017-06-20 12:57:39 -0400630gl::Error WindowSurfaceVk::getAttachmentRenderTarget(const gl::Context * /*context*/,
Jamie Madillf618c9e2018-02-15 14:45:40 -0500631 GLenum binding,
Jamie Madill4fd95d52017-04-05 11:22:18 -0400632 const gl::ImageIndex & /*target*/,
633 FramebufferAttachmentRenderTarget **rtOut)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500634{
Jamie Madillf618c9e2018-02-15 14:45:40 -0500635 if (binding == GL_BACK)
636 {
637 *rtOut = &mColorRenderTarget;
638 }
639 else
640 {
641 ASSERT(binding == GL_DEPTH || binding == GL_STENCIL || binding == GL_DEPTH_STENCIL);
642 *rtOut = &mDepthStencilRenderTarget;
643 }
644
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500645 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400646}
647
Jamie Madillab9f9c32017-01-17 17:47:34 -0500648gl::ErrorOrResult<vk::Framebuffer *> WindowSurfaceVk::getCurrentFramebuffer(
649 VkDevice device,
650 const vk::RenderPass &compatibleRenderPass)
651{
Jamie Madilla9c60e92017-09-28 19:06:39 -0400652 auto &currentFramebuffer = mSwapchainImages[mCurrentSwapchainImageIndex].framebuffer;
653
654 if (currentFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500655 {
656 // Validation layers should detect if the render pass is really compatible.
Jamie Madilla9c60e92017-09-28 19:06:39 -0400657 return &currentFramebuffer;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500658 }
659
660 VkFramebufferCreateInfo framebufferInfo;
661
Jamie Madillf618c9e2018-02-15 14:45:40 -0500662 std::array<VkImageView, 2> imageViews = {{VK_NULL_HANDLE, mDepthStencilImageView.getHandle()}};
663
Jamie Madillab9f9c32017-01-17 17:47:34 -0500664 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
665 framebufferInfo.pNext = nullptr;
666 framebufferInfo.flags = 0;
667 framebufferInfo.renderPass = compatibleRenderPass.getHandle();
Jamie Madillf618c9e2018-02-15 14:45:40 -0500668 framebufferInfo.attachmentCount = (mDepthStencilImageView.valid() ? 2u : 1u);
669 framebufferInfo.pAttachments = imageViews.data();
670 framebufferInfo.width = static_cast<uint32_t>(mColorRenderTarget.extents.width);
671 framebufferInfo.height = static_cast<uint32_t>(mColorRenderTarget.extents.height);
Jamie Madillab9f9c32017-01-17 17:47:34 -0500672 framebufferInfo.layers = 1;
673
Jamie Madilla9c60e92017-09-28 19:06:39 -0400674 for (auto &swapchainImage : mSwapchainImages)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500675 {
Jamie Madillf618c9e2018-02-15 14:45:40 -0500676 imageViews[0] = swapchainImage.imageView.getHandle();
Jamie Madill25301b62017-10-28 20:59:31 -0400677 ANGLE_TRY(swapchainImage.framebuffer.init(device, framebufferInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500678 }
679
Jamie Madilla9c60e92017-09-28 19:06:39 -0400680 ASSERT(currentFramebuffer.valid());
681 return &currentFramebuffer;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500682}
683
Jamie Madill05b35b22017-10-03 09:01:44 -0400684gl::Error WindowSurfaceVk::initializeContents(const gl::Context *context,
685 const gl::ImageIndex &imageIndex)
686{
687 UNIMPLEMENTED();
688 return gl::NoError();
689}
690
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400691} // namespace rx