blob: ed2ed9508cb437d62d142bd4d935d3efffb4cca3 [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 Madillabaab232017-01-10 12:37:37 -050019#include "libANGLE/renderer/vulkan/formatutilsvk.h"
Jamie Madill9e54b5a2016-05-25 12:57:39 -040020
21namespace rx
22{
23
Jamie Madill4d0bf552016-12-28 15:45:24 -050024namespace
25{
26
Jamie Madill6a89d222017-11-02 11:59:51 -040027const vk::Format &GetVkFormatFromConfig(RendererVk *renderer, const egl::Config &config)
Jamie Madill4d0bf552016-12-28 15:45:24 -050028{
29 // TODO(jmadill): Properly handle format interpretation.
Jamie Madill6a89d222017-11-02 11:59:51 -040030 return renderer->getFormat(GL_BGRA8_EXT);
Jamie Madill4d0bf552016-12-28 15:45:24 -050031}
32
Jamie Madillf0eafe12017-02-21 15:03:50 -050033VkPresentModeKHR GetDesiredPresentMode(const std::vector<VkPresentModeKHR> &presentModes,
34 EGLint minSwapInterval,
35 EGLint maxSwapInterval)
36{
37 ASSERT(!presentModes.empty());
38
39 // Use FIFO mode for v-sync, since it throttles you to the display rate. Mailbox is more
Jamie Madilla9c60e92017-09-28 19:06:39 -040040 // similar to triple-buffering. For now we hard-code Mailbox for perf testing.
Jamie Madillf0eafe12017-02-21 15:03:50 -050041 // TODO(jmadill): Properly select present mode and re-create display if changed.
42 VkPresentModeKHR bestChoice = VK_PRESENT_MODE_MAILBOX_KHR;
43
44 for (auto presentMode : presentModes)
45 {
46 if (presentMode == bestChoice)
47 {
48 return bestChoice;
49 }
50 }
51
Jamie Madill98de8262017-05-29 13:01:02 -040052 WARN() << "Present mode " << bestChoice << " not available. Falling back to "
53 << presentModes[0];
Jamie Madillf0eafe12017-02-21 15:03:50 -050054 return presentModes[0];
55}
56
Jamie Madill4d0bf552016-12-28 15:45:24 -050057} // namespace
58
Jamie Madille09bd5d2016-11-29 16:20:35 -050059OffscreenSurfaceVk::OffscreenSurfaceVk(const egl::SurfaceState &surfaceState,
60 EGLint width,
61 EGLint height)
62 : SurfaceImpl(surfaceState), mWidth(width), mHeight(height)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040063{
64}
65
Jamie Madille09bd5d2016-11-29 16:20:35 -050066OffscreenSurfaceVk::~OffscreenSurfaceVk()
Jamie Madill9e54b5a2016-05-25 12:57:39 -040067{
68}
69
Jamie Madillc564c072017-06-01 12:45:42 -040070egl::Error OffscreenSurfaceVk::initialize(const egl::Display *display)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040071{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050072 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040073}
74
Jamie Madille09bd5d2016-11-29 16:20:35 -050075FramebufferImpl *OffscreenSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040076{
Jamie Madill7b57b9d2017-01-13 09:33:38 -050077 // Use a user FBO for an offscreen RT.
78 return FramebufferVk::CreateUserFBO(state);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040079}
80
Jamie Madillfe548342017-06-19 11:13:24 -040081egl::Error OffscreenSurfaceVk::swap(const gl::Context *context)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040082{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050083 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040084}
85
Jamie Madillfe548342017-06-19 11:13:24 -040086egl::Error OffscreenSurfaceVk::postSubBuffer(const gl::Context * /*context*/,
87 EGLint /*x*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -050088 EGLint /*y*/,
89 EGLint /*width*/,
90 EGLint /*height*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040091{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050092 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040093}
94
Jamie Madille09bd5d2016-11-29 16:20:35 -050095egl::Error OffscreenSurfaceVk::querySurfacePointerANGLE(EGLint /*attribute*/, void ** /*value*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040096{
Jamie Madille09bd5d2016-11-29 16:20:35 -050097 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -050098 return egl::EglBadCurrentSurface();
Jamie Madill9e54b5a2016-05-25 12:57:39 -040099}
100
Jamie Madille09bd5d2016-11-29 16:20:35 -0500101egl::Error OffscreenSurfaceVk::bindTexImage(gl::Texture * /*texture*/, EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400102{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500103 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400104}
105
Jamie Madille09bd5d2016-11-29 16:20:35 -0500106egl::Error OffscreenSurfaceVk::releaseTexImage(EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400107{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500108 return egl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400109}
110
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700111egl::Error OffscreenSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
112 EGLuint64KHR * /*msc*/,
113 EGLuint64KHR * /*sbc*/)
114{
115 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500116 return egl::EglBadAccess();
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700117}
118
Jamie Madille09bd5d2016-11-29 16:20:35 -0500119void OffscreenSurfaceVk::setSwapInterval(EGLint /*interval*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400120{
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400121}
122
Jamie Madille09bd5d2016-11-29 16:20:35 -0500123EGLint OffscreenSurfaceVk::getWidth() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400124{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500125 return mWidth;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400126}
127
Jamie Madille09bd5d2016-11-29 16:20:35 -0500128EGLint OffscreenSurfaceVk::getHeight() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400129{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500130 return mHeight;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400131}
132
Jamie Madille09bd5d2016-11-29 16:20:35 -0500133EGLint OffscreenSurfaceVk::isPostSubBufferSupported() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400134{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500135 return EGL_FALSE;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400136}
137
Jamie Madille09bd5d2016-11-29 16:20:35 -0500138EGLint OffscreenSurfaceVk::getSwapBehavior() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400139{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500140 return EGL_BUFFER_PRESERVED;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400141}
142
Jamie Madille09bd5d2016-11-29 16:20:35 -0500143gl::Error OffscreenSurfaceVk::getAttachmentRenderTarget(
Jamie Madill4928b7c2017-06-20 12:57:39 -0400144 const gl::Context * /*context*/,
Jamie Madill4fd95d52017-04-05 11:22:18 -0400145 GLenum /*binding*/,
146 const gl::ImageIndex & /*imageIndex*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -0500147 FramebufferAttachmentRenderTarget ** /*rtOut*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400148{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500149 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500150 return gl::InternalError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500151}
152
Jamie Madill05b35b22017-10-03 09:01:44 -0400153gl::Error OffscreenSurfaceVk::initializeContents(const gl::Context *context,
154 const gl::ImageIndex &imageIndex)
155{
156 UNIMPLEMENTED();
157 return gl::NoError();
158}
159
Jamie Madillacf2f3a2017-11-21 19:22:44 -0500160WindowSurfaceVk::SwapchainImage::SwapchainImage() = default;
161WindowSurfaceVk::SwapchainImage::SwapchainImage(WindowSurfaceVk::SwapchainImage &&other) = default;
162WindowSurfaceVk::SwapchainImage::~SwapchainImage() = default;
163
Jamie Madill4d0bf552016-12-28 15:45:24 -0500164WindowSurfaceVk::WindowSurfaceVk(const egl::SurfaceState &surfaceState,
165 EGLNativeWindowType window,
166 EGLint width,
167 EGLint height)
168 : SurfaceImpl(surfaceState),
169 mNativeWindowType(window),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500170 mSurface(VK_NULL_HANDLE),
Frank Henigman29f148b2016-11-23 21:05:36 -0500171 mInstance(VK_NULL_HANDLE),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500172 mSwapchain(VK_NULL_HANDLE),
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500173 mRenderTarget(),
174 mCurrentSwapchainImageIndex(0)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500175{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500176 mRenderTarget.extents.width = static_cast<GLint>(width);
177 mRenderTarget.extents.height = static_cast<GLint>(height);
178 mRenderTarget.extents.depth = 1;
Jamie Madill4c26fc22017-02-24 11:04:10 -0500179 mRenderTarget.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 Madill70ee0f62017-02-06 16:04:20 -0500191 RendererVk *rendererVk = displayVk->getRenderer();
192 VkDevice device = rendererVk->getDevice();
193 VkInstance instance = rendererVk->getInstance();
194
Jamie Madille918de22017-04-12 10:21:11 -0400195 rendererVk->finish();
196
Jamie Madilla9c60e92017-09-28 19:06:39 -0400197 mAcquireNextImageSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500198
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);
236
Jamie Madillacf2f3a2017-11-21 19:22:44 -0500237 const VkPhysicalDevice &physicalDevice = renderer->getPhysicalDevice();
Jamie Madill4d0bf552016-12-28 15:45:24 -0500238
239 VkSurfaceCapabilitiesKHR surfaceCaps;
240 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physicalDevice, mSurface, &surfaceCaps));
241
242 // Adjust width and height to the swapchain if necessary.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500243 uint32_t width = surfaceCaps.currentExtent.width;
244 uint32_t height = surfaceCaps.currentExtent.height;
245
246 // TODO(jmadill): Support devices which don't support copy. We use this for ReadPixels.
247 ANGLE_VK_CHECK((surfaceCaps.supportedUsageFlags & VK_IMAGE_USAGE_TRANSFER_SRC_BIT) != 0,
248 VK_ERROR_INITIALIZATION_FAILED);
249
250 if (surfaceCaps.currentExtent.width == 0xFFFFFFFFu)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500251 {
252 ASSERT(surfaceCaps.currentExtent.height == 0xFFFFFFFFu);
253
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500254 if (mRenderTarget.extents.width == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500255 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500256 width = windowSize.width;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500257 }
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500258 if (mRenderTarget.extents.height == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500259 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500260 height = windowSize.height;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500261 }
262 }
263
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500264 mRenderTarget.extents.width = static_cast<int>(width);
265 mRenderTarget.extents.height = static_cast<int>(height);
266
Jamie Madill4d0bf552016-12-28 15:45:24 -0500267 uint32_t presentModeCount = 0;
268 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
269 &presentModeCount, nullptr));
270 ASSERT(presentModeCount > 0);
271
272 std::vector<VkPresentModeKHR> presentModes(presentModeCount);
273 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
274 &presentModeCount, presentModes.data()));
275
Jamie Madillf0eafe12017-02-21 15:03:50 -0500276 // Select appropriate present mode based on vsync parameter.
277 // TODO(jmadill): More complete implementation, which allows for changing and more values.
278 const EGLint minSwapInterval = mState.config->minSwapInterval;
279 const EGLint maxSwapInterval = mState.config->maxSwapInterval;
280 ASSERT(minSwapInterval == 0 || minSwapInterval == 1);
281 ASSERT(maxSwapInterval == 0 || maxSwapInterval == 1);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500282
Jamie Madillf0eafe12017-02-21 15:03:50 -0500283 VkPresentModeKHR swapchainPresentMode =
284 GetDesiredPresentMode(presentModes, minSwapInterval, maxSwapInterval);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500285
286 // Determine number of swapchain images. Aim for one more than the minimum.
287 uint32_t minImageCount = surfaceCaps.minImageCount + 1;
288 if (surfaceCaps.maxImageCount > 0 && minImageCount > surfaceCaps.maxImageCount)
289 {
290 minImageCount = surfaceCaps.maxImageCount;
291 }
292
293 // Default to identity transform.
294 VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
295 if ((surfaceCaps.supportedTransforms & preTransform) == 0)
296 {
297 preTransform = surfaceCaps.currentTransform;
298 }
299
Jamie Madill4d0bf552016-12-28 15:45:24 -0500300 uint32_t surfaceFormatCount = 0;
301 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
302 nullptr));
303
304 std::vector<VkSurfaceFormatKHR> surfaceFormats(surfaceFormatCount);
305 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
306 surfaceFormats.data()));
307
Jamie Madill1d7be502017-10-29 18:06:50 -0400308 mRenderTarget.format = &GetVkFormatFromConfig(renderer, *mState.config);
309 VkFormat nativeFormat = mRenderTarget.format->vkTextureFormat;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500310
Jamie Madill4d0bf552016-12-28 15:45:24 -0500311 if (surfaceFormatCount == 1u && surfaceFormats[0].format == VK_FORMAT_UNDEFINED)
312 {
313 // This is fine.
314 }
315 else
316 {
317 bool foundFormat = false;
318 for (const auto &surfaceFormat : surfaceFormats)
319 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500320 if (surfaceFormat.format == nativeFormat)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500321 {
322 foundFormat = true;
323 break;
324 }
325 }
326
327 ANGLE_VK_CHECK(foundFormat, VK_ERROR_INITIALIZATION_FAILED);
328 }
329
330 VkSwapchainCreateInfoKHR swapchainInfo;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500331 swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
332 swapchainInfo.pNext = nullptr;
333 swapchainInfo.flags = 0;
334 swapchainInfo.surface = mSurface;
335 swapchainInfo.minImageCount = minImageCount;
336 swapchainInfo.imageFormat = nativeFormat;
337 swapchainInfo.imageColorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
338 swapchainInfo.imageExtent.width = width;
339 swapchainInfo.imageExtent.height = height;
340 swapchainInfo.imageArrayLayers = 1;
341 swapchainInfo.imageUsage = (VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
342 VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500343 swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
344 swapchainInfo.queueFamilyIndexCount = 0;
345 swapchainInfo.pQueueFamilyIndices = nullptr;
346 swapchainInfo.preTransform = preTransform;
347 swapchainInfo.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
348 swapchainInfo.presentMode = swapchainPresentMode;
349 swapchainInfo.clipped = VK_TRUE;
350 swapchainInfo.oldSwapchain = VK_NULL_HANDLE;
351
Jamie Madill6a89d222017-11-02 11:59:51 -0400352 VkDevice device = renderer->getDevice();
Jamie Madill4d0bf552016-12-28 15:45:24 -0500353 ANGLE_VK_TRY(vkCreateSwapchainKHR(device, &swapchainInfo, nullptr, &mSwapchain));
354
355 // Intialize the swapchain image views.
356 uint32_t imageCount = 0;
357 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, nullptr));
358
359 std::vector<VkImage> swapchainImages(imageCount);
360 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, swapchainImages.data()));
361
362 // CommandBuffer is a singleton in the Renderer.
Jamie Madill7f738d42017-11-20 17:06:27 -0500363 vk::CommandBufferAndState *commandBuffer = nullptr;
Jamie Madill0c0dc342017-03-24 14:18:51 -0400364 ANGLE_TRY(renderer->getStartedCommandBuffer(&commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500365
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500366 VkClearColorValue transparentBlack;
367 transparentBlack.float32[0] = 0.0f;
368 transparentBlack.float32[1] = 0.0f;
369 transparentBlack.float32[2] = 0.0f;
370 transparentBlack.float32[3] = 0.0f;
371
Jamie Madill5deea722017-02-16 10:44:46 -0500372 mSwapchainImages.resize(imageCount);
Jamie Madilla9c60e92017-09-28 19:06:39 -0400373
374 ANGLE_TRY(mAcquireNextImageSemaphore.init(device));
Jamie Madill5deea722017-02-16 10:44:46 -0500375
376 for (uint32_t imageIndex = 0; imageIndex < imageCount; ++imageIndex)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500377 {
Jamie Madill5deea722017-02-16 10:44:46 -0500378 VkImage swapchainImage = swapchainImages[imageIndex];
379
Jamie Madill4d0bf552016-12-28 15:45:24 -0500380 VkImageViewCreateInfo imageViewInfo;
381 imageViewInfo.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
382 imageViewInfo.pNext = nullptr;
383 imageViewInfo.flags = 0;
384 imageViewInfo.image = swapchainImage;
385 imageViewInfo.viewType = VK_IMAGE_VIEW_TYPE_2D;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500386 imageViewInfo.format = nativeFormat;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500387 imageViewInfo.components.r = VK_COMPONENT_SWIZZLE_R;
388 imageViewInfo.components.g = VK_COMPONENT_SWIZZLE_G;
389 imageViewInfo.components.b = VK_COMPONENT_SWIZZLE_B;
390 imageViewInfo.components.a = VK_COMPONENT_SWIZZLE_A;
391 imageViewInfo.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
392 imageViewInfo.subresourceRange.baseMipLevel = 0;
393 imageViewInfo.subresourceRange.levelCount = 1;
394 imageViewInfo.subresourceRange.baseArrayLayer = 0;
395 imageViewInfo.subresourceRange.layerCount = 1;
396
Jamie Madilla9c60e92017-09-28 19:06:39 -0400397 auto &member = mSwapchainImages[imageIndex];
398
Jamie Madill25301b62017-10-28 20:59:31 -0400399 member.image.setHandle(swapchainImage);
400 ANGLE_TRY(member.imageView.init(device, imageViewInfo));
401
402 // Set transfer dest layout, and clear the image to black.
Jamie Madilld33c77c2017-11-09 13:08:30 -0500403 member.image.changeLayoutWithStages(
404 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 Madill25301b62017-10-28 20:59:31 -0400406 commandBuffer->clearSingleColorImage(member.image, transparentBlack);
407
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 Madill0c0dc342017-03-24 14:18:51 -0400412 ANGLE_TRY(renderer->submitAndFinishCommandBuffer(commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500413
Jamie Madille918de22017-04-12 10:21:11 -0400414 // Get the first available swapchain iamge.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500415 ANGLE_TRY(nextSwapchainImage(renderer));
416
Jamie Madill4d0bf552016-12-28 15:45:24 -0500417 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500418}
419
420FramebufferImpl *WindowSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
421{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500422 return FramebufferVk::CreateDefaultFBO(state, this);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500423}
424
Jamie Madillfe548342017-06-19 11:13:24 -0400425egl::Error WindowSurfaceVk::swap(const gl::Context *context)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500426{
Jamie Madille1f3ad42017-10-28 23:00:42 -0400427 const DisplayVk *displayVk = vk::GetImpl(context->getCurrentDisplay());
Jamie Madilld4826152017-09-21 11:18:59 -0400428 RendererVk *renderer = displayVk->getRenderer();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500429
Jamie Madill7f738d42017-11-20 17:06:27 -0500430 vk::CommandBufferAndState *currentCB = nullptr;
Jamie Madill0c0dc342017-03-24 14:18:51 -0400431 ANGLE_TRY(renderer->getStartedCommandBuffer(&currentCB));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500432
Jamie Madilld4826152017-09-21 11:18:59 -0400433 // End render pass
Jamie Madill1b038242017-11-01 15:14:36 -0400434 renderer->endRenderPass();
Jamie Madilld4826152017-09-21 11:18:59 -0400435
Jamie Madilla9c60e92017-09-28 19:06:39 -0400436 auto &image = mSwapchainImages[mCurrentSwapchainImageIndex];
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500437
Jamie Madilla9c60e92017-09-28 19:06:39 -0400438 image.image.changeLayoutWithStages(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
439 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
440 VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, currentCB);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500441
Jamie Madilla9c60e92017-09-28 19:06:39 -0400442 ANGLE_TRY(renderer->submitCommandsWithSync(currentCB, image.imageAcquiredSemaphore,
443 image.commandsCompleteSemaphore));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500444
445 VkPresentInfoKHR presentInfo;
446 presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
447 presentInfo.pNext = nullptr;
Jamie Madille918de22017-04-12 10:21:11 -0400448 presentInfo.waitSemaphoreCount = 1;
Jamie Madilla9c60e92017-09-28 19:06:39 -0400449 presentInfo.pWaitSemaphores = image.commandsCompleteSemaphore.ptr();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500450 presentInfo.swapchainCount = 1;
451 presentInfo.pSwapchains = &mSwapchain;
452 presentInfo.pImageIndices = &mCurrentSwapchainImageIndex;
453 presentInfo.pResults = nullptr;
454
455 ANGLE_VK_TRY(vkQueuePresentKHR(renderer->getQueue(), &presentInfo));
456
Jamie Madillf0eafe12017-02-21 15:03:50 -0500457 // Get the next available swapchain image.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500458 ANGLE_TRY(nextSwapchainImage(renderer));
459
460 return vk::NoError();
461}
462
463vk::Error WindowSurfaceVk::nextSwapchainImage(RendererVk *renderer)
464{
465 VkDevice device = renderer->getDevice();
466
Jamie Madilla9c60e92017-09-28 19:06:39 -0400467 // Use a timeout of zero for AcquireNextImage so we don't actually block.
468 // TODO(jmadill): We should handle VK_NOT_READY and block until we can acquire the image.
469 ANGLE_VK_TRY(vkAcquireNextImageKHR(device, mSwapchain, 0,
470 mAcquireNextImageSemaphore.getHandle(), VK_NULL_HANDLE,
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500471 &mCurrentSwapchainImageIndex));
472
Jamie Madilla9c60e92017-09-28 19:06:39 -0400473 auto &image = mSwapchainImages[mCurrentSwapchainImageIndex];
474
475 // Swap the unused swapchain semaphore and the now active spare semaphore.
476 std::swap(image.imageAcquiredSemaphore, mAcquireNextImageSemaphore);
477
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500478 // Update RenderTarget pointers.
Jamie Madilla9c60e92017-09-28 19:06:39 -0400479 mRenderTarget.image = &image.image;
480 mRenderTarget.imageView = &image.imageView;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500481
482 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500483}
484
Jamie Madillfe548342017-06-19 11:13:24 -0400485egl::Error WindowSurfaceVk::postSubBuffer(const gl::Context *context,
486 EGLint x,
487 EGLint y,
488 EGLint width,
489 EGLint height)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500490{
491 // TODO(jmadill)
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500492 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500493}
494
495egl::Error WindowSurfaceVk::querySurfacePointerANGLE(EGLint attribute, void **value)
496{
497 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500498 return egl::EglBadCurrentSurface();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500499}
500
501egl::Error WindowSurfaceVk::bindTexImage(gl::Texture *texture, EGLint buffer)
502{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500503 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500504}
505
506egl::Error WindowSurfaceVk::releaseTexImage(EGLint buffer)
507{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500508 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500509}
510
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700511egl::Error WindowSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
512 EGLuint64KHR * /*msc*/,
513 EGLuint64KHR * /*sbc*/)
514{
515 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500516 return egl::EglBadAccess();
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700517}
518
Jamie Madille09bd5d2016-11-29 16:20:35 -0500519void WindowSurfaceVk::setSwapInterval(EGLint interval)
520{
521}
522
523EGLint WindowSurfaceVk::getWidth() const
524{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500525 return static_cast<EGLint>(mRenderTarget.extents.width);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500526}
527
528EGLint WindowSurfaceVk::getHeight() const
529{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500530 return static_cast<EGLint>(mRenderTarget.extents.height);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500531}
532
533EGLint WindowSurfaceVk::isPostSubBufferSupported() const
534{
535 // TODO(jmadill)
536 return EGL_FALSE;
537}
538
539EGLint WindowSurfaceVk::getSwapBehavior() const
540{
541 // TODO(jmadill)
542 return EGL_BUFFER_DESTROYED;
543}
544
Jamie Madill4928b7c2017-06-20 12:57:39 -0400545gl::Error WindowSurfaceVk::getAttachmentRenderTarget(const gl::Context * /*context*/,
546 GLenum /*binding*/,
Jamie Madill4fd95d52017-04-05 11:22:18 -0400547 const gl::ImageIndex & /*target*/,
548 FramebufferAttachmentRenderTarget **rtOut)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500549{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500550 *rtOut = &mRenderTarget;
551 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400552}
553
Jamie Madillab9f9c32017-01-17 17:47:34 -0500554gl::ErrorOrResult<vk::Framebuffer *> WindowSurfaceVk::getCurrentFramebuffer(
555 VkDevice device,
556 const vk::RenderPass &compatibleRenderPass)
557{
Jamie Madilla9c60e92017-09-28 19:06:39 -0400558 auto &currentFramebuffer = mSwapchainImages[mCurrentSwapchainImageIndex].framebuffer;
559
560 if (currentFramebuffer.valid())
Jamie Madillab9f9c32017-01-17 17:47:34 -0500561 {
562 // Validation layers should detect if the render pass is really compatible.
Jamie Madilla9c60e92017-09-28 19:06:39 -0400563 return &currentFramebuffer;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500564 }
565
566 VkFramebufferCreateInfo framebufferInfo;
567
568 // TODO(jmadill): Depth/Stencil attachments.
569 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
570 framebufferInfo.pNext = nullptr;
571 framebufferInfo.flags = 0;
572 framebufferInfo.renderPass = compatibleRenderPass.getHandle();
573 framebufferInfo.attachmentCount = 1u;
574 framebufferInfo.pAttachments = nullptr;
575 framebufferInfo.width = static_cast<uint32_t>(mRenderTarget.extents.width);
576 framebufferInfo.height = static_cast<uint32_t>(mRenderTarget.extents.height);
577 framebufferInfo.layers = 1;
578
Jamie Madilla9c60e92017-09-28 19:06:39 -0400579 for (auto &swapchainImage : mSwapchainImages)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500580 {
Jamie Madilla9c60e92017-09-28 19:06:39 -0400581 framebufferInfo.pAttachments = swapchainImage.imageView.ptr();
Jamie Madill25301b62017-10-28 20:59:31 -0400582 ANGLE_TRY(swapchainImage.framebuffer.init(device, framebufferInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500583 }
584
Jamie Madilla9c60e92017-09-28 19:06:39 -0400585 ASSERT(currentFramebuffer.valid());
586 return &currentFramebuffer;
Jamie Madillab9f9c32017-01-17 17:47:34 -0500587}
588
Jamie Madill05b35b22017-10-03 09:01:44 -0400589gl::Error WindowSurfaceVk::initializeContents(const gl::Context *context,
590 const gl::ImageIndex &imageIndex)
591{
592 UNIMPLEMENTED();
593 return gl::NoError();
594}
595
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400596} // namespace rx