blob: fca87e2baea7817d6e40962bf05b952334ceaed1 [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
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 Madillbc543422018-03-30 10:43:19 -0400154WindowSurfaceVk::SwapchainImage::SwapchainImage() = default;
155WindowSurfaceVk::SwapchainImage::~SwapchainImage() = default;
156
157WindowSurfaceVk::SwapchainImage::SwapchainImage(SwapchainImage &&other)
158 : image(std::move(other.image)),
159 imageView(std::move(other.imageView)),
160 framebuffer(std::move(other.framebuffer)),
161 imageAcquiredSemaphore(std::move(other.imageAcquiredSemaphore)),
162 commandsCompleteSemaphore(std::move(other.commandsCompleteSemaphore))
163{
164}
Jamie Madillacf2f3a2017-11-21 19:22:44 -0500165
Jamie Madill4d0bf552016-12-28 15:45:24 -0500166WindowSurfaceVk::WindowSurfaceVk(const egl::SurfaceState &surfaceState,
167 EGLNativeWindowType window,
168 EGLint width,
169 EGLint height)
170 : SurfaceImpl(surfaceState),
171 mNativeWindowType(window),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500172 mSurface(VK_NULL_HANDLE),
Frank Henigman29f148b2016-11-23 21:05:36 -0500173 mInstance(VK_NULL_HANDLE),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500174 mSwapchain(VK_NULL_HANDLE),
Jamie Madillf618c9e2018-02-15 14:45:40 -0500175 mColorRenderTarget(),
176 mDepthStencilRenderTarget(),
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500177 mCurrentSwapchainImageIndex(0)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500178{
Jamie Madillbc543422018-03-30 10:43:19 -0400179 mColorRenderTarget.resource = this;
Jamie Madille09bd5d2016-11-29 16:20:35 -0500180}
181
182WindowSurfaceVk::~WindowSurfaceVk()
183{
Jamie Madill70ee0f62017-02-06 16:04:20 -0500184 ASSERT(mSurface == VK_NULL_HANDLE);
185 ASSERT(mSwapchain == VK_NULL_HANDLE);
Jamie Madill70ee0f62017-02-06 16:04:20 -0500186}
187
Jamie Madillc564c072017-06-01 12:45:42 -0400188void WindowSurfaceVk::destroy(const egl::Display *display)
Jamie Madill70ee0f62017-02-06 16:04:20 -0500189{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400190 const DisplayVk *displayVk = vk::GetImpl(display);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500191 RendererVk *renderer = displayVk->getRenderer();
192 VkDevice device = renderer->getDevice();
193 VkInstance instance = renderer->getInstance();
Jamie Madill70ee0f62017-02-06 16:04:20 -0500194
Jamie Madillf618c9e2018-02-15 14:45:40 -0500195 // We might not need to flush the pipe here.
196 renderer->finish(display->getProxyContext());
Jamie Madille918de22017-04-12 10:21:11 -0400197
Jamie Madilla9c60e92017-09-28 19:06:39 -0400198 mAcquireNextImageSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500199
Jamie Madillbc543422018-03-30 10:43:19 -0400200 mDepthStencilImage.release(renderer->getCurrentQueueSerial(), renderer);
201 mDepthStencilImageView.destroy(device);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500202
Jamie Madillbc543422018-03-30 10:43:19 -0400203 for (SwapchainImage &swapchainImage : mSwapchainImages)
Jamie Madill70ee0f62017-02-06 16:04:20 -0500204 {
Jamie Madilla9c60e92017-09-28 19:06:39 -0400205 // Although we don't own the swapchain image handles, we need to keep our shutdown clean.
Jamie Madillbc543422018-03-30 10:43:19 -0400206 swapchainImage.image.getImage().reset();
207 swapchainImage.image.destroy(device);
Jamie Madilla9c60e92017-09-28 19:06:39 -0400208 swapchainImage.imageView.destroy(device);
209 swapchainImage.framebuffer.destroy(device);
210 swapchainImage.imageAcquiredSemaphore.destroy(device);
211 swapchainImage.commandsCompleteSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500212 }
Jamie Madill4d0bf552016-12-28 15:45:24 -0500213
214 if (mSwapchain)
215 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500216 vkDestroySwapchainKHR(device, mSwapchain, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500217 mSwapchain = VK_NULL_HANDLE;
218 }
219
220 if (mSurface)
221 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500222 vkDestroySurfaceKHR(instance, mSurface, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500223 mSurface = VK_NULL_HANDLE;
224 }
Jamie Madille09bd5d2016-11-29 16:20:35 -0500225}
226
Jamie Madillc564c072017-06-01 12:45:42 -0400227egl::Error WindowSurfaceVk::initialize(const egl::Display *display)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500228{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400229 const DisplayVk *displayVk = vk::GetImpl(display);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500230 return initializeImpl(displayVk->getRenderer()).toEGL(EGL_BAD_SURFACE);
231}
232
233vk::Error WindowSurfaceVk::initializeImpl(RendererVk *renderer)
234{
Frank Henigman29f148b2016-11-23 21:05:36 -0500235 gl::Extents windowSize;
236 ANGLE_TRY_RESULT(createSurfaceVk(renderer), windowSize);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500237
238 uint32_t presentQueue = 0;
239 ANGLE_TRY_RESULT(renderer->selectPresentQueueForSurface(mSurface), presentQueue);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500240 UNUSED_VARIABLE(presentQueue);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500241
Jamie Madillacf2f3a2017-11-21 19:22:44 -0500242 const VkPhysicalDevice &physicalDevice = renderer->getPhysicalDevice();
Jamie Madill4d0bf552016-12-28 15:45:24 -0500243
244 VkSurfaceCapabilitiesKHR surfaceCaps;
245 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physicalDevice, mSurface, &surfaceCaps));
246
247 // Adjust width and height to the swapchain if necessary.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500248 uint32_t width = surfaceCaps.currentExtent.width;
249 uint32_t height = surfaceCaps.currentExtent.height;
250
251 // TODO(jmadill): Support devices which don't support copy. We use this for ReadPixels.
252 ANGLE_VK_CHECK((surfaceCaps.supportedUsageFlags & VK_IMAGE_USAGE_TRANSFER_SRC_BIT) != 0,
253 VK_ERROR_INITIALIZATION_FAILED);
254
Jamie Madillbc543422018-03-30 10:43:19 -0400255 EGLAttrib attribWidth = mState.attributes.get(EGL_WIDTH, 0);
256 EGLAttrib attribHeight = mState.attributes.get(EGL_HEIGHT, 0);
257
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500258 if (surfaceCaps.currentExtent.width == 0xFFFFFFFFu)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500259 {
260 ASSERT(surfaceCaps.currentExtent.height == 0xFFFFFFFFu);
261
Jamie Madillbc543422018-03-30 10:43:19 -0400262 if (attribWidth == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500263 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500264 width = windowSize.width;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500265 }
Jamie Madillbc543422018-03-30 10:43:19 -0400266 if (attribHeight == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500267 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500268 height = windowSize.height;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500269 }
270 }
271
Jamie Madillbc543422018-03-30 10:43:19 -0400272 gl::Extents extents(static_cast<int>(width), static_cast<int>(height), 1);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500273
Jamie Madill4d0bf552016-12-28 15:45:24 -0500274 uint32_t presentModeCount = 0;
275 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
276 &presentModeCount, nullptr));
277 ASSERT(presentModeCount > 0);
278
279 std::vector<VkPresentModeKHR> presentModes(presentModeCount);
280 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
281 &presentModeCount, presentModes.data()));
282
Jamie Madillf0eafe12017-02-21 15:03:50 -0500283 // Select appropriate present mode based on vsync parameter.
284 // TODO(jmadill): More complete implementation, which allows for changing and more values.
285 const EGLint minSwapInterval = mState.config->minSwapInterval;
286 const EGLint maxSwapInterval = mState.config->maxSwapInterval;
287 ASSERT(minSwapInterval == 0 || minSwapInterval == 1);
288 ASSERT(maxSwapInterval == 0 || maxSwapInterval == 1);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500289
Jamie Madillf0eafe12017-02-21 15:03:50 -0500290 VkPresentModeKHR swapchainPresentMode =
291 GetDesiredPresentMode(presentModes, minSwapInterval, maxSwapInterval);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500292
293 // Determine number of swapchain images. Aim for one more than the minimum.
294 uint32_t minImageCount = surfaceCaps.minImageCount + 1;
295 if (surfaceCaps.maxImageCount > 0 && minImageCount > surfaceCaps.maxImageCount)
296 {
297 minImageCount = surfaceCaps.maxImageCount;
298 }
299
300 // Default to identity transform.
301 VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
302 if ((surfaceCaps.supportedTransforms & preTransform) == 0)
303 {
304 preTransform = surfaceCaps.currentTransform;
305 }
306
Jamie Madill4d0bf552016-12-28 15:45:24 -0500307 uint32_t surfaceFormatCount = 0;
308 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
309 nullptr));
310
311 std::vector<VkSurfaceFormatKHR> surfaceFormats(surfaceFormatCount);
312 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
313 surfaceFormats.data()));
314
Jamie Madillbc543422018-03-30 10:43:19 -0400315 const vk::Format &format = renderer->getFormat(mState.config->renderTargetFormat);
316 VkFormat nativeFormat = format.vkTextureFormat;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500317
Jamie Madill4d0bf552016-12-28 15:45:24 -0500318 if (surfaceFormatCount == 1u && surfaceFormats[0].format == VK_FORMAT_UNDEFINED)
319 {
320 // This is fine.
321 }
322 else
323 {
324 bool foundFormat = false;
Jamie Madillbc543422018-03-30 10:43:19 -0400325 for (const VkSurfaceFormatKHR &surfaceFormat : surfaceFormats)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500326 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500327 if (surfaceFormat.format == nativeFormat)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500328 {
329 foundFormat = true;
330 break;
331 }
332 }
333
334 ANGLE_VK_CHECK(foundFormat, VK_ERROR_INITIALIZATION_FAILED);
335 }
336
Yuly Novikov12da5e72018-01-23 18:34:53 -0500337 VkCompositeAlphaFlagBitsKHR compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
338 if ((surfaceCaps.supportedCompositeAlpha & compositeAlpha) == 0)
339 {
340 compositeAlpha = VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR;
341 }
342 ANGLE_VK_CHECK((surfaceCaps.supportedCompositeAlpha & compositeAlpha) != 0,
343 VK_ERROR_INITIALIZATION_FAILED);
344
Jamie Madillf618c9e2018-02-15 14:45:40 -0500345 // We need transfer src for reading back from the backbuffer.
346 VkImageUsageFlags imageUsageFlags =
347 (VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_SRC_BIT |
348 VK_IMAGE_USAGE_TRANSFER_DST_BIT);
349
Jamie Madill4d0bf552016-12-28 15:45:24 -0500350 VkSwapchainCreateInfoKHR swapchainInfo;
Jamie Madillf618c9e2018-02-15 14:45:40 -0500351 swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
352 swapchainInfo.pNext = nullptr;
353 swapchainInfo.flags = 0;
354 swapchainInfo.surface = mSurface;
355 swapchainInfo.minImageCount = minImageCount;
356 swapchainInfo.imageFormat = nativeFormat;
357 swapchainInfo.imageColorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
358 swapchainInfo.imageExtent.width = width;
359 swapchainInfo.imageExtent.height = height;
360 swapchainInfo.imageArrayLayers = 1;
361 swapchainInfo.imageUsage = imageUsageFlags;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500362 swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
363 swapchainInfo.queueFamilyIndexCount = 0;
364 swapchainInfo.pQueueFamilyIndices = nullptr;
365 swapchainInfo.preTransform = preTransform;
Yuly Novikov12da5e72018-01-23 18:34:53 -0500366 swapchainInfo.compositeAlpha = compositeAlpha;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500367 swapchainInfo.presentMode = swapchainPresentMode;
368 swapchainInfo.clipped = VK_TRUE;
369 swapchainInfo.oldSwapchain = VK_NULL_HANDLE;
370
Jamie Madill6a89d222017-11-02 11:59:51 -0400371 VkDevice device = renderer->getDevice();
Jamie Madill4d0bf552016-12-28 15:45:24 -0500372 ANGLE_VK_TRY(vkCreateSwapchainKHR(device, &swapchainInfo, nullptr, &mSwapchain));
373
374 // Intialize the swapchain image views.
375 uint32_t imageCount = 0;
376 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, nullptr));
377
378 std::vector<VkImage> swapchainImages(imageCount);
379 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, swapchainImages.data()));
380
Jamie Madill49ac74b2017-12-21 14:42:33 -0500381 // Allocate a command buffer for clearing our images to black.
382 vk::CommandBuffer *commandBuffer = nullptr;
Jamie Madill1f46bc12018-02-20 16:09:43 -0500383 ANGLE_TRY(beginWriteResource(renderer, &commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500384
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500385 VkClearColorValue transparentBlack;
386 transparentBlack.float32[0] = 0.0f;
387 transparentBlack.float32[1] = 0.0f;
388 transparentBlack.float32[2] = 0.0f;
389 transparentBlack.float32[3] = 0.0f;
390
Jamie Madill5deea722017-02-16 10:44:46 -0500391 mSwapchainImages.resize(imageCount);
Jamie Madilla9c60e92017-09-28 19:06:39 -0400392
393 ANGLE_TRY(mAcquireNextImageSemaphore.init(device));
Jamie Madill5deea722017-02-16 10:44:46 -0500394
395 for (uint32_t imageIndex = 0; imageIndex < imageCount; ++imageIndex)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500396 {
Jamie Madillbc543422018-03-30 10:43:19 -0400397 SwapchainImage &member = mSwapchainImages[imageIndex];
398 member.image.init2DWeakReference(swapchainImages[imageIndex], extents, format, 1);
399 member.image.initImageView(device, VK_IMAGE_ASPECT_COLOR_BIT, gl::SwizzleState(),
400 &member.imageView);
Jamie Madill25301b62017-10-28 20:59:31 -0400401
402 // Set transfer dest layout, and clear the image to black.
Jamie Madillbc543422018-03-30 10:43:19 -0400403 member.image.getImage().changeLayoutWithStages(
Jamie Madilld33c77c2017-11-09 13:08:30 -0500404 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
405 VK_PIPELINE_STAGE_ALL_GRAPHICS_BIT, VK_PIPELINE_STAGE_TRANSFER_BIT, commandBuffer);
Jamie Madillbc543422018-03-30 10:43:19 -0400406 commandBuffer->clearSingleColorImage(member.image.getImage(), transparentBlack);
Jamie Madill25301b62017-10-28 20:59:31 -0400407
Jamie Madilla9c60e92017-09-28 19:06:39 -0400408 ANGLE_TRY(member.imageAcquiredSemaphore.init(device));
409 ANGLE_TRY(member.commandsCompleteSemaphore.init(device));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500410 }
411
Jamie Madille918de22017-04-12 10:21:11 -0400412 // Get the first available swapchain iamge.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500413 ANGLE_TRY(nextSwapchainImage(renderer));
414
Jamie Madillf618c9e2018-02-15 14:45:40 -0500415 // Initialize depth/stencil if requested.
416 if (mState.config->depthStencilFormat != GL_NONE)
417 {
418 const vk::Format &dsFormat = renderer->getFormat(mState.config->depthStencilFormat);
419
420 const VkImageUsageFlags usage =
421 (VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT |
422 VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_SAMPLED_BIT);
423
Jamie Madillbc543422018-03-30 10:43:19 -0400424 ANGLE_TRY(mDepthStencilImage.init2D(device, extents, dsFormat, 1, usage));
425 ANGLE_TRY(mDepthStencilImage.initMemory(device, renderer->getMemoryProperties(),
426 VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT));
Jamie Madillf618c9e2018-02-15 14:45:40 -0500427
428 const VkImageAspectFlags aspect =
429 (dsFormat.textureFormat().depthBits > 0 ? VK_IMAGE_ASPECT_DEPTH_BIT : 0) |
430 (dsFormat.textureFormat().stencilBits > 0 ? VK_IMAGE_ASPECT_STENCIL_BIT : 0);
431
432 VkClearDepthStencilValue depthStencilClearValue = {1.0f, 0};
433
434 // Set transfer dest layout, and clear the image.
Jamie Madillbc543422018-03-30 10:43:19 -0400435 mDepthStencilImage.getImage().changeLayoutWithStages(
436 aspect, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, VK_PIPELINE_STAGE_ALL_GRAPHICS_BIT,
437 VK_PIPELINE_STAGE_TRANSFER_BIT, commandBuffer);
438 commandBuffer->clearSingleDepthStencilImage(mDepthStencilImage.getImage(), aspect,
Jamie Madillf618c9e2018-02-15 14:45:40 -0500439 depthStencilClearValue);
440
Jamie Madillbc543422018-03-30 10:43:19 -0400441 ANGLE_TRY(mDepthStencilImage.initImageView(device, aspect, gl::SwizzleState(),
442 &mDepthStencilImageView));
Jamie Madillf618c9e2018-02-15 14:45:40 -0500443
Jamie Madillbc543422018-03-30 10:43:19 -0400444 mDepthStencilRenderTarget.resource = this;
445 mDepthStencilRenderTarget.image = &mDepthStencilImage;
446 mDepthStencilRenderTarget.imageView = &mDepthStencilImageView;
Jamie Madillf618c9e2018-02-15 14:45:40 -0500447
448 // TODO(jmadill): Figure out how to pass depth/stencil image views to the RenderTargetVk.
449 }
450
Jamie Madill4d0bf552016-12-28 15:45:24 -0500451 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500452}
453
454FramebufferImpl *WindowSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
455{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500456 return FramebufferVk::CreateDefaultFBO(state, this);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500457}
458
Jamie Madillfe548342017-06-19 11:13:24 -0400459egl::Error WindowSurfaceVk::swap(const gl::Context *context)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500460{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400461 const DisplayVk *displayVk = vk::GetImpl(context->getCurrentDisplay());
Jamie Madilld4826152017-09-21 11:18:59 -0400462 RendererVk *renderer = displayVk->getRenderer();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500463
Jamie Madill49ac74b2017-12-21 14:42:33 -0500464 vk::CommandBuffer *swapCommands = nullptr;
Jamie Madill1f46bc12018-02-20 16:09:43 -0500465 ANGLE_TRY(beginWriteResource(renderer, &swapCommands));
Jamie Madilld4826152017-09-21 11:18:59 -0400466
Jamie Madillbc543422018-03-30 10:43:19 -0400467 SwapchainImage &image = mSwapchainImages[mCurrentSwapchainImageIndex];
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500468
Jamie Madillbc543422018-03-30 10:43:19 -0400469 image.image.getImage().changeLayoutWithStages(
470 VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
471 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT, VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, swapCommands);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500472
Jamie Madill49ac74b2017-12-21 14:42:33 -0500473 ANGLE_TRY(
474 renderer->flush(context, image.imageAcquiredSemaphore, image.commandsCompleteSemaphore));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500475
476 VkPresentInfoKHR presentInfo;
477 presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
478 presentInfo.pNext = nullptr;
Jamie Madille918de22017-04-12 10:21:11 -0400479 presentInfo.waitSemaphoreCount = 1;
Jamie Madilla9c60e92017-09-28 19:06:39 -0400480 presentInfo.pWaitSemaphores = image.commandsCompleteSemaphore.ptr();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500481 presentInfo.swapchainCount = 1;
482 presentInfo.pSwapchains = &mSwapchain;
483 presentInfo.pImageIndices = &mCurrentSwapchainImageIndex;
484 presentInfo.pResults = nullptr;
485
486 ANGLE_VK_TRY(vkQueuePresentKHR(renderer->getQueue(), &presentInfo));
487
Jamie Madillf0eafe12017-02-21 15:03:50 -0500488 // Get the next available swapchain image.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500489 ANGLE_TRY(nextSwapchainImage(renderer));
490
491 return vk::NoError();
492}
493
494vk::Error WindowSurfaceVk::nextSwapchainImage(RendererVk *renderer)
495{
496 VkDevice device = renderer->getDevice();
497
jchen10191a84a2018-01-31 16:09:45 +0800498 ANGLE_VK_TRY(vkAcquireNextImageKHR(device, mSwapchain, UINT64_MAX,
Jamie Madilla9c60e92017-09-28 19:06:39 -0400499 mAcquireNextImageSemaphore.getHandle(), VK_NULL_HANDLE,
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500500 &mCurrentSwapchainImageIndex));
501
Jamie Madillbc543422018-03-30 10:43:19 -0400502 SwapchainImage &image = mSwapchainImages[mCurrentSwapchainImageIndex];
Jamie Madilla9c60e92017-09-28 19:06:39 -0400503
504 // Swap the unused swapchain semaphore and the now active spare semaphore.
505 std::swap(image.imageAcquiredSemaphore, mAcquireNextImageSemaphore);
506
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500507 // Update RenderTarget pointers.
Jamie Madillf618c9e2018-02-15 14:45:40 -0500508 mColorRenderTarget.image = &image.image;
509 mColorRenderTarget.imageView = &image.imageView;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500510
511 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500512}
513
Jamie Madillfe548342017-06-19 11:13:24 -0400514egl::Error WindowSurfaceVk::postSubBuffer(const gl::Context *context,
515 EGLint x,
516 EGLint y,
517 EGLint width,
518 EGLint height)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500519{
520 // TODO(jmadill)
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500521 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500522}
523
524egl::Error WindowSurfaceVk::querySurfacePointerANGLE(EGLint attribute, void **value)
525{
526 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500527 return egl::EglBadCurrentSurface();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500528}
529
530egl::Error WindowSurfaceVk::bindTexImage(gl::Texture *texture, EGLint buffer)
531{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500532 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500533}
534
535egl::Error WindowSurfaceVk::releaseTexImage(EGLint buffer)
536{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500537 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500538}
539
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700540egl::Error WindowSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
541 EGLuint64KHR * /*msc*/,
542 EGLuint64KHR * /*sbc*/)
543{
544 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500545 return egl::EglBadAccess();
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700546}
547
Jamie Madille09bd5d2016-11-29 16:20:35 -0500548void WindowSurfaceVk::setSwapInterval(EGLint interval)
549{
550}
551
552EGLint WindowSurfaceVk::getWidth() const
553{
Jamie Madillbc543422018-03-30 10:43:19 -0400554 return static_cast<EGLint>(mColorRenderTarget.image->getExtents().width);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500555}
556
557EGLint WindowSurfaceVk::getHeight() const
558{
Jamie Madillbc543422018-03-30 10:43:19 -0400559 return static_cast<EGLint>(mColorRenderTarget.image->getExtents().height);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500560}
561
562EGLint WindowSurfaceVk::isPostSubBufferSupported() const
563{
564 // TODO(jmadill)
565 return EGL_FALSE;
566}
567
568EGLint WindowSurfaceVk::getSwapBehavior() const
569{
570 // TODO(jmadill)
571 return EGL_BUFFER_DESTROYED;
572}
573
Jamie Madill4928b7c2017-06-20 12:57:39 -0400574gl::Error WindowSurfaceVk::getAttachmentRenderTarget(const gl::Context * /*context*/,
Jamie Madillf618c9e2018-02-15 14:45:40 -0500575 GLenum binding,
Jamie Madill4fd95d52017-04-05 11:22:18 -0400576 const gl::ImageIndex & /*target*/,
577 FramebufferAttachmentRenderTarget **rtOut)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500578{
Jamie Madillf618c9e2018-02-15 14:45:40 -0500579 if (binding == GL_BACK)
580 {
581 *rtOut = &mColorRenderTarget;
582 }
583 else
584 {
585 ASSERT(binding == GL_DEPTH || binding == GL_STENCIL || binding == GL_DEPTH_STENCIL);
586 *rtOut = &mDepthStencilRenderTarget;
587 }
588
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500589 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400590}
591
Jamie Madillab9f9c32017-01-17 17:47:34 -0500592gl::ErrorOrResult<vk::Framebuffer *> WindowSurfaceVk::getCurrentFramebuffer(
593 VkDevice device,
594 const vk::RenderPass &compatibleRenderPass)
595{
Jamie Madillbc543422018-03-30 10:43:19 -0400596 vk::Framebuffer &currentFramebuffer = mSwapchainImages[mCurrentSwapchainImageIndex].framebuffer;
Jamie Madilla9c60e92017-09-28 19:06:39 -0400597
598 if (currentFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500599 {
600 // Validation layers should detect if the render pass is really compatible.
Jamie Madilla9c60e92017-09-28 19:06:39 -0400601 return &currentFramebuffer;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500602 }
603
604 VkFramebufferCreateInfo framebufferInfo;
605
Jamie Madillbc543422018-03-30 10:43:19 -0400606 const gl::Extents &extents = mColorRenderTarget.image->getExtents();
Jamie Madillf618c9e2018-02-15 14:45:40 -0500607 std::array<VkImageView, 2> imageViews = {{VK_NULL_HANDLE, mDepthStencilImageView.getHandle()}};
608
Jamie Madillab9f9c32017-01-17 17:47:34 -0500609 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
610 framebufferInfo.pNext = nullptr;
611 framebufferInfo.flags = 0;
612 framebufferInfo.renderPass = compatibleRenderPass.getHandle();
Jamie Madillbc543422018-03-30 10:43:19 -0400613 framebufferInfo.attachmentCount = (mDepthStencilImage.valid() ? 2u : 1u);
Jamie Madillf618c9e2018-02-15 14:45:40 -0500614 framebufferInfo.pAttachments = imageViews.data();
Jamie Madillbc543422018-03-30 10:43:19 -0400615 framebufferInfo.width = static_cast<uint32_t>(extents.width);
616 framebufferInfo.height = static_cast<uint32_t>(extents.height);
Jamie Madillab9f9c32017-01-17 17:47:34 -0500617 framebufferInfo.layers = 1;
618
Jamie Madillbc543422018-03-30 10:43:19 -0400619 for (SwapchainImage &swapchainImage : mSwapchainImages)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500620 {
Jamie Madillf618c9e2018-02-15 14:45:40 -0500621 imageViews[0] = swapchainImage.imageView.getHandle();
Jamie Madill25301b62017-10-28 20:59:31 -0400622 ANGLE_TRY(swapchainImage.framebuffer.init(device, framebufferInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500623 }
624
Jamie Madilla9c60e92017-09-28 19:06:39 -0400625 ASSERT(currentFramebuffer.valid());
626 return &currentFramebuffer;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500627}
628
Jamie Madill05b35b22017-10-03 09:01:44 -0400629gl::Error WindowSurfaceVk::initializeContents(const gl::Context *context,
630 const gl::ImageIndex &imageIndex)
631{
632 UNIMPLEMENTED();
633 return gl::NoError();
634}
635
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400636} // namespace rx