blob: 5f074219a1330739a8048086561eab1a15a41770 [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 Madill4d0bf552016-12-28 15:45:24 -050013#include "libANGLE/Surface.h"
14#include "libANGLE/renderer/vulkan/DisplayVk.h"
Jamie Madille09bd5d2016-11-29 16:20:35 -050015#include "libANGLE/renderer/vulkan/FramebufferVk.h"
16#include "libANGLE/renderer/vulkan/RendererVk.h"
Jamie Madillabaab232017-01-10 12:37:37 -050017#include "libANGLE/renderer/vulkan/formatutilsvk.h"
Jamie Madill9e54b5a2016-05-25 12:57:39 -040018
19namespace rx
20{
21
Jamie Madill4d0bf552016-12-28 15:45:24 -050022namespace
23{
24
Jamie Madillabaab232017-01-10 12:37:37 -050025const vk::Format &GetVkFormatFromConfig(const egl::Config &config)
Jamie Madill4d0bf552016-12-28 15:45:24 -050026{
27 // TODO(jmadill): Properly handle format interpretation.
Jamie Madillabaab232017-01-10 12:37:37 -050028 return vk::Format::Get(GL_BGRA8_EXT);
Jamie Madill4d0bf552016-12-28 15:45:24 -050029}
30
Jamie Madillf0eafe12017-02-21 15:03:50 -050031VkPresentModeKHR GetDesiredPresentMode(const std::vector<VkPresentModeKHR> &presentModes,
32 EGLint minSwapInterval,
33 EGLint maxSwapInterval)
34{
35 ASSERT(!presentModes.empty());
36
37 // Use FIFO mode for v-sync, since it throttles you to the display rate. Mailbox is more
38 // similar to triple-buffering. For now we hard-code Mailbox for perf tseting.
39 // TODO(jmadill): Properly select present mode and re-create display if changed.
40 VkPresentModeKHR bestChoice = VK_PRESENT_MODE_MAILBOX_KHR;
41
42 for (auto presentMode : presentModes)
43 {
44 if (presentMode == bestChoice)
45 {
46 return bestChoice;
47 }
48 }
49
50 ERR() << "Desired present mode not available. Falling back to " << presentModes[0];
51 return presentModes[0];
52}
53
Jamie Madill4d0bf552016-12-28 15:45:24 -050054} // namespace
55
Jamie Madille09bd5d2016-11-29 16:20:35 -050056OffscreenSurfaceVk::OffscreenSurfaceVk(const egl::SurfaceState &surfaceState,
57 EGLint width,
58 EGLint height)
59 : SurfaceImpl(surfaceState), mWidth(width), mHeight(height)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040060{
61}
62
Jamie Madille09bd5d2016-11-29 16:20:35 -050063OffscreenSurfaceVk::~OffscreenSurfaceVk()
Jamie Madill9e54b5a2016-05-25 12:57:39 -040064{
65}
66
Jamie Madille09bd5d2016-11-29 16:20:35 -050067egl::Error OffscreenSurfaceVk::initialize(const DisplayImpl *displayImpl)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040068{
Jamie Madille09bd5d2016-11-29 16:20:35 -050069 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040070}
71
Jamie Madille09bd5d2016-11-29 16:20:35 -050072FramebufferImpl *OffscreenSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040073{
Jamie Madill7b57b9d2017-01-13 09:33:38 -050074 // Use a user FBO for an offscreen RT.
75 return FramebufferVk::CreateUserFBO(state);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040076}
77
Jamie Madille09bd5d2016-11-29 16:20:35 -050078egl::Error OffscreenSurfaceVk::swap(const DisplayImpl *displayImpl)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040079{
Jamie Madille09bd5d2016-11-29 16:20:35 -050080 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040081}
82
Jamie Madille09bd5d2016-11-29 16:20:35 -050083egl::Error OffscreenSurfaceVk::postSubBuffer(EGLint /*x*/,
84 EGLint /*y*/,
85 EGLint /*width*/,
86 EGLint /*height*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040087{
Jamie Madille09bd5d2016-11-29 16:20:35 -050088 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040089}
90
Jamie Madille09bd5d2016-11-29 16:20:35 -050091egl::Error OffscreenSurfaceVk::querySurfacePointerANGLE(EGLint /*attribute*/, void ** /*value*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040092{
Jamie Madille09bd5d2016-11-29 16:20:35 -050093 UNREACHABLE();
94 return egl::Error(EGL_BAD_CURRENT_SURFACE);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040095}
96
Jamie Madille09bd5d2016-11-29 16:20:35 -050097egl::Error OffscreenSurfaceVk::bindTexImage(gl::Texture * /*texture*/, EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040098{
Jamie Madille09bd5d2016-11-29 16:20:35 -050099 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400100}
101
Jamie Madille09bd5d2016-11-29 16:20:35 -0500102egl::Error OffscreenSurfaceVk::releaseTexImage(EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400103{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500104 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400105}
106
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700107egl::Error OffscreenSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
108 EGLuint64KHR * /*msc*/,
109 EGLuint64KHR * /*sbc*/)
110{
111 UNIMPLEMENTED();
112 return egl::Error(EGL_BAD_ACCESS);
113}
114
Jamie Madille09bd5d2016-11-29 16:20:35 -0500115void OffscreenSurfaceVk::setSwapInterval(EGLint /*interval*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400116{
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400117}
118
Jamie Madille09bd5d2016-11-29 16:20:35 -0500119EGLint OffscreenSurfaceVk::getWidth() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400120{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500121 return mWidth;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400122}
123
Jamie Madille09bd5d2016-11-29 16:20:35 -0500124EGLint OffscreenSurfaceVk::getHeight() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400125{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500126 return mHeight;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400127}
128
Jamie Madille09bd5d2016-11-29 16:20:35 -0500129EGLint OffscreenSurfaceVk::isPostSubBufferSupported() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400130{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500131 return EGL_FALSE;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400132}
133
Jamie Madille09bd5d2016-11-29 16:20:35 -0500134EGLint OffscreenSurfaceVk::getSwapBehavior() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400135{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500136 return EGL_BUFFER_PRESERVED;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400137}
138
Jamie Madille09bd5d2016-11-29 16:20:35 -0500139gl::Error OffscreenSurfaceVk::getAttachmentRenderTarget(
Jamie Madill4fd95d52017-04-05 11:22:18 -0400140 GLenum /*binding*/,
141 const gl::ImageIndex & /*imageIndex*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -0500142 FramebufferAttachmentRenderTarget ** /*rtOut*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400143{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500144 UNREACHABLE();
145 return gl::Error(GL_INVALID_OPERATION);
146}
147
Jamie Madill4d0bf552016-12-28 15:45:24 -0500148WindowSurfaceVk::WindowSurfaceVk(const egl::SurfaceState &surfaceState,
149 EGLNativeWindowType window,
150 EGLint width,
151 EGLint height)
152 : SurfaceImpl(surfaceState),
153 mNativeWindowType(window),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500154 mSurface(VK_NULL_HANDLE),
155 mSwapchain(VK_NULL_HANDLE),
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500156 mRenderTarget(),
157 mCurrentSwapchainImageIndex(0)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500158{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500159 mRenderTarget.extents.width = static_cast<GLint>(width);
160 mRenderTarget.extents.height = static_cast<GLint>(height);
161 mRenderTarget.extents.depth = 1;
Jamie Madill4c26fc22017-02-24 11:04:10 -0500162 mRenderTarget.resource = this;
Jamie Madille09bd5d2016-11-29 16:20:35 -0500163}
164
165WindowSurfaceVk::~WindowSurfaceVk()
166{
Jamie Madill70ee0f62017-02-06 16:04:20 -0500167 ASSERT(mSurface == VK_NULL_HANDLE);
168 ASSERT(mSwapchain == VK_NULL_HANDLE);
Jamie Madill70ee0f62017-02-06 16:04:20 -0500169}
170
171void WindowSurfaceVk::destroy(const DisplayImpl *displayImpl)
172{
173 const DisplayVk *displayVk = GetAs<DisplayVk>(displayImpl);
174 RendererVk *rendererVk = displayVk->getRenderer();
175 VkDevice device = rendererVk->getDevice();
176 VkInstance instance = rendererVk->getInstance();
177
Jamie Madille918de22017-04-12 10:21:11 -0400178 rendererVk->finish();
179
180 mImageAvailableSemaphore.destroy(device);
181 mRenderingCompleteSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500182
Jamie Madill70ee0f62017-02-06 16:04:20 -0500183 for (auto &imageView : mSwapchainImageViews)
184 {
185 imageView.destroy(device);
186 }
187
Jamie Madill70ee0f62017-02-06 16:04:20 -0500188 // Although we don't own the swapchain image handles, we need to keep our shutdown clean.
189 for (auto &image : mSwapchainImages)
190 {
191 image.reset();
192 }
193
Jamie Madill5deea722017-02-16 10:44:46 -0500194 for (auto &framebuffer : mSwapchainFramebuffers)
195 {
196 framebuffer.destroy(device);
197 }
Jamie Madill4d0bf552016-12-28 15:45:24 -0500198
199 if (mSwapchain)
200 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500201 vkDestroySwapchainKHR(device, mSwapchain, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500202 mSwapchain = VK_NULL_HANDLE;
203 }
204
205 if (mSurface)
206 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500207 vkDestroySurfaceKHR(instance, mSurface, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500208 mSurface = VK_NULL_HANDLE;
209 }
Jamie Madille09bd5d2016-11-29 16:20:35 -0500210}
211
212egl::Error WindowSurfaceVk::initialize(const DisplayImpl *displayImpl)
213{
Jamie Madill4d0bf552016-12-28 15:45:24 -0500214 const DisplayVk *displayVk = GetAs<DisplayVk>(displayImpl);
215 return initializeImpl(displayVk->getRenderer()).toEGL(EGL_BAD_SURFACE);
216}
217
218vk::Error WindowSurfaceVk::initializeImpl(RendererVk *renderer)
219{
Jamie Madill4d0bf552016-12-28 15:45:24 -0500220 // TODO(jmadill): Make this platform-specific.
221 VkWin32SurfaceCreateInfoKHR createInfo;
222
223 createInfo.sType = VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR;
224 createInfo.pNext = nullptr;
225 createInfo.flags = 0;
226 createInfo.hinstance = GetModuleHandle(nullptr);
227 createInfo.hwnd = mNativeWindowType;
228 ANGLE_VK_TRY(vkCreateWin32SurfaceKHR(renderer->getInstance(), &createInfo, nullptr, &mSurface));
229
230 uint32_t presentQueue = 0;
231 ANGLE_TRY_RESULT(renderer->selectPresentQueueForSurface(mSurface), presentQueue);
232
233 const auto &physicalDevice = renderer->getPhysicalDevice();
234
235 VkSurfaceCapabilitiesKHR surfaceCaps;
236 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physicalDevice, mSurface, &surfaceCaps));
237
238 // Adjust width and height to the swapchain if necessary.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500239 uint32_t width = surfaceCaps.currentExtent.width;
240 uint32_t height = surfaceCaps.currentExtent.height;
241
242 // TODO(jmadill): Support devices which don't support copy. We use this for ReadPixels.
243 ANGLE_VK_CHECK((surfaceCaps.supportedUsageFlags & VK_IMAGE_USAGE_TRANSFER_SRC_BIT) != 0,
244 VK_ERROR_INITIALIZATION_FAILED);
245
246 if (surfaceCaps.currentExtent.width == 0xFFFFFFFFu)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500247 {
248 ASSERT(surfaceCaps.currentExtent.height == 0xFFFFFFFFu);
249
250 RECT rect;
251 ANGLE_VK_CHECK(GetClientRect(mNativeWindowType, &rect) == TRUE,
252 VK_ERROR_INITIALIZATION_FAILED);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500253 if (mRenderTarget.extents.width == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500254 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500255 width = static_cast<uint32_t>(rect.right - rect.left);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500256 }
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500257 if (mRenderTarget.extents.height == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500258 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500259 height = static_cast<uint32_t>(rect.bottom - rect.top);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500260 }
261 }
262
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500263 mRenderTarget.extents.width = static_cast<int>(width);
264 mRenderTarget.extents.height = static_cast<int>(height);
265
Jamie Madill4d0bf552016-12-28 15:45:24 -0500266 uint32_t presentModeCount = 0;
267 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
268 &presentModeCount, nullptr));
269 ASSERT(presentModeCount > 0);
270
271 std::vector<VkPresentModeKHR> presentModes(presentModeCount);
272 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
273 &presentModeCount, presentModes.data()));
274
Jamie Madillf0eafe12017-02-21 15:03:50 -0500275 // Select appropriate present mode based on vsync parameter.
276 // TODO(jmadill): More complete implementation, which allows for changing and more values.
277 const EGLint minSwapInterval = mState.config->minSwapInterval;
278 const EGLint maxSwapInterval = mState.config->maxSwapInterval;
279 ASSERT(minSwapInterval == 0 || minSwapInterval == 1);
280 ASSERT(maxSwapInterval == 0 || maxSwapInterval == 1);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500281
Jamie Madillf0eafe12017-02-21 15:03:50 -0500282 VkPresentModeKHR swapchainPresentMode =
283 GetDesiredPresentMode(presentModes, minSwapInterval, maxSwapInterval);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500284
285 // Determine number of swapchain images. Aim for one more than the minimum.
286 uint32_t minImageCount = surfaceCaps.minImageCount + 1;
287 if (surfaceCaps.maxImageCount > 0 && minImageCount > surfaceCaps.maxImageCount)
288 {
289 minImageCount = surfaceCaps.maxImageCount;
290 }
291
292 // Default to identity transform.
293 VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
294 if ((surfaceCaps.supportedTransforms & preTransform) == 0)
295 {
296 preTransform = surfaceCaps.currentTransform;
297 }
298
Jamie Madill4d0bf552016-12-28 15:45:24 -0500299 uint32_t surfaceFormatCount = 0;
300 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
301 nullptr));
302
303 std::vector<VkSurfaceFormatKHR> surfaceFormats(surfaceFormatCount);
304 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
305 surfaceFormats.data()));
306
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500307 mRenderTarget.format = &GetVkFormatFromConfig(*mState.config);
308 auto nativeFormat = mRenderTarget.format->native;
309
Jamie Madill4d0bf552016-12-28 15:45:24 -0500310 if (surfaceFormatCount == 1u && surfaceFormats[0].format == VK_FORMAT_UNDEFINED)
311 {
312 // This is fine.
313 }
314 else
315 {
316 bool foundFormat = false;
317 for (const auto &surfaceFormat : surfaceFormats)
318 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500319 if (surfaceFormat.format == nativeFormat)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500320 {
321 foundFormat = true;
322 break;
323 }
324 }
325
326 ANGLE_VK_CHECK(foundFormat, VK_ERROR_INITIALIZATION_FAILED);
327 }
328
329 VkSwapchainCreateInfoKHR swapchainInfo;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500330 swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
331 swapchainInfo.pNext = nullptr;
332 swapchainInfo.flags = 0;
333 swapchainInfo.surface = mSurface;
334 swapchainInfo.minImageCount = minImageCount;
335 swapchainInfo.imageFormat = nativeFormat;
336 swapchainInfo.imageColorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
337 swapchainInfo.imageExtent.width = width;
338 swapchainInfo.imageExtent.height = height;
339 swapchainInfo.imageArrayLayers = 1;
340 swapchainInfo.imageUsage = (VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
341 VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500342 swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
343 swapchainInfo.queueFamilyIndexCount = 0;
344 swapchainInfo.pQueueFamilyIndices = nullptr;
345 swapchainInfo.preTransform = preTransform;
346 swapchainInfo.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
347 swapchainInfo.presentMode = swapchainPresentMode;
348 swapchainInfo.clipped = VK_TRUE;
349 swapchainInfo.oldSwapchain = VK_NULL_HANDLE;
350
351 const auto &device = renderer->getDevice();
352 ANGLE_VK_TRY(vkCreateSwapchainKHR(device, &swapchainInfo, nullptr, &mSwapchain));
353
354 // Intialize the swapchain image views.
355 uint32_t imageCount = 0;
356 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, nullptr));
357
358 std::vector<VkImage> swapchainImages(imageCount);
359 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, swapchainImages.data()));
360
361 // CommandBuffer is a singleton in the Renderer.
Jamie Madill0c0dc342017-03-24 14:18:51 -0400362 vk::CommandBuffer *commandBuffer = nullptr;
363 ANGLE_TRY(renderer->getStartedCommandBuffer(&commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500364
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500365 VkClearColorValue transparentBlack;
366 transparentBlack.float32[0] = 0.0f;
367 transparentBlack.float32[1] = 0.0f;
368 transparentBlack.float32[2] = 0.0f;
369 transparentBlack.float32[3] = 0.0f;
370
Jamie Madill5deea722017-02-16 10:44:46 -0500371 mSwapchainImages.resize(imageCount);
372 mSwapchainImageViews.resize(imageCount);
373
374 for (uint32_t imageIndex = 0; imageIndex < imageCount; ++imageIndex)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500375 {
Jamie Madill5deea722017-02-16 10:44:46 -0500376 VkImage swapchainImage = swapchainImages[imageIndex];
377
Jamie Madill4d0bf552016-12-28 15:45:24 -0500378 VkImageViewCreateInfo imageViewInfo;
379 imageViewInfo.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
380 imageViewInfo.pNext = nullptr;
381 imageViewInfo.flags = 0;
382 imageViewInfo.image = swapchainImage;
383 imageViewInfo.viewType = VK_IMAGE_VIEW_TYPE_2D;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500384 imageViewInfo.format = nativeFormat;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500385 imageViewInfo.components.r = VK_COMPONENT_SWIZZLE_R;
386 imageViewInfo.components.g = VK_COMPONENT_SWIZZLE_G;
387 imageViewInfo.components.b = VK_COMPONENT_SWIZZLE_B;
388 imageViewInfo.components.a = VK_COMPONENT_SWIZZLE_A;
389 imageViewInfo.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
390 imageViewInfo.subresourceRange.baseMipLevel = 0;
391 imageViewInfo.subresourceRange.levelCount = 1;
392 imageViewInfo.subresourceRange.baseArrayLayer = 0;
393 imageViewInfo.subresourceRange.layerCount = 1;
394
395 vk::Image image(swapchainImage);
Jamie Madill5deea722017-02-16 10:44:46 -0500396 vk::ImageView imageView;
397 ANGLE_TRY(imageView.init(device, imageViewInfo));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500398
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500399 // Set transfer dest layout, and clear the image to black.
400 image.changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
401 commandBuffer);
402 commandBuffer->clearSingleColorImage(image, transparentBlack);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500403
Jamie Madill5deea722017-02-16 10:44:46 -0500404 mSwapchainImages[imageIndex].retain(device, std::move(image));
405 mSwapchainImageViews[imageIndex].retain(device, std::move(imageView));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500406 }
407
Jamie Madill0c0dc342017-03-24 14:18:51 -0400408 ANGLE_TRY(renderer->submitAndFinishCommandBuffer(commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500409
Jamie Madille918de22017-04-12 10:21:11 -0400410 ANGLE_TRY(mImageAvailableSemaphore.init(device));
411 ANGLE_TRY(mRenderingCompleteSemaphore.init(device));
412
413 // Get the first available swapchain iamge.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500414 ANGLE_TRY(nextSwapchainImage(renderer));
415
Jamie Madill4d0bf552016-12-28 15:45:24 -0500416 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500417}
418
419FramebufferImpl *WindowSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
420{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500421 return FramebufferVk::CreateDefaultFBO(state, this);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500422}
423
424egl::Error WindowSurfaceVk::swap(const DisplayImpl *displayImpl)
425{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500426 const DisplayVk *displayVk = GetAs<DisplayVk>(displayImpl);
427 return swapImpl(displayVk->getRenderer()).toEGL(EGL_BAD_ALLOC);
428}
429
430vk::Error WindowSurfaceVk::swapImpl(RendererVk *renderer)
431{
Jamie Madill0c0dc342017-03-24 14:18:51 -0400432 vk::CommandBuffer *currentCB = nullptr;
433 ANGLE_TRY(renderer->getStartedCommandBuffer(&currentCB));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500434
435 auto *image = &mSwapchainImages[mCurrentSwapchainImageIndex];
436
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500437 image->changeLayoutWithStages(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
438 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
439 VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, currentCB);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500440
Jamie Madill0c0dc342017-03-24 14:18:51 -0400441 ANGLE_TRY(renderer->submitCommandsWithSync(currentCB, mImageAvailableSemaphore,
Jamie Madille918de22017-04-12 10:21:11 -0400442 mRenderingCompleteSemaphore));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500443
444 VkPresentInfoKHR presentInfo;
445 presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
446 presentInfo.pNext = nullptr;
Jamie Madille918de22017-04-12 10:21:11 -0400447 presentInfo.waitSemaphoreCount = 1;
448 presentInfo.pWaitSemaphores = mRenderingCompleteSemaphore.ptr();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500449 presentInfo.swapchainCount = 1;
450 presentInfo.pSwapchains = &mSwapchain;
451 presentInfo.pImageIndices = &mCurrentSwapchainImageIndex;
452 presentInfo.pResults = nullptr;
453
454 ANGLE_VK_TRY(vkQueuePresentKHR(renderer->getQueue(), &presentInfo));
455
Jamie Madillf0eafe12017-02-21 15:03:50 -0500456 // Get the next available swapchain image.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500457 ANGLE_TRY(nextSwapchainImage(renderer));
458
459 return vk::NoError();
460}
461
462vk::Error WindowSurfaceVk::nextSwapchainImage(RendererVk *renderer)
463{
464 VkDevice device = renderer->getDevice();
465
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500466 ANGLE_VK_TRY(vkAcquireNextImageKHR(device, mSwapchain, std::numeric_limits<uint64_t>::max(),
Jamie Madille918de22017-04-12 10:21:11 -0400467 mImageAvailableSemaphore.getHandle(), VK_NULL_HANDLE,
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500468 &mCurrentSwapchainImageIndex));
469
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500470 // Update RenderTarget pointers.
471 mRenderTarget.image = &mSwapchainImages[mCurrentSwapchainImageIndex];
472 mRenderTarget.imageView = &mSwapchainImageViews[mCurrentSwapchainImageIndex];
473
474 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500475}
476
477egl::Error WindowSurfaceVk::postSubBuffer(EGLint x, EGLint y, EGLint width, EGLint height)
478{
479 // TODO(jmadill)
480 return egl::Error(EGL_SUCCESS);
481}
482
483egl::Error WindowSurfaceVk::querySurfacePointerANGLE(EGLint attribute, void **value)
484{
485 UNREACHABLE();
486 return egl::Error(EGL_BAD_CURRENT_SURFACE);
487}
488
489egl::Error WindowSurfaceVk::bindTexImage(gl::Texture *texture, EGLint buffer)
490{
491 return egl::Error(EGL_SUCCESS);
492}
493
494egl::Error WindowSurfaceVk::releaseTexImage(EGLint buffer)
495{
496 return egl::Error(EGL_SUCCESS);
497}
498
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700499egl::Error WindowSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
500 EGLuint64KHR * /*msc*/,
501 EGLuint64KHR * /*sbc*/)
502{
503 UNIMPLEMENTED();
504 return egl::Error(EGL_BAD_ACCESS);
505}
506
Jamie Madille09bd5d2016-11-29 16:20:35 -0500507void WindowSurfaceVk::setSwapInterval(EGLint interval)
508{
509}
510
511EGLint WindowSurfaceVk::getWidth() const
512{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500513 return static_cast<EGLint>(mRenderTarget.extents.width);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500514}
515
516EGLint WindowSurfaceVk::getHeight() const
517{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500518 return static_cast<EGLint>(mRenderTarget.extents.height);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500519}
520
521EGLint WindowSurfaceVk::isPostSubBufferSupported() const
522{
523 // TODO(jmadill)
524 return EGL_FALSE;
525}
526
527EGLint WindowSurfaceVk::getSwapBehavior() const
528{
529 // TODO(jmadill)
530 return EGL_BUFFER_DESTROYED;
531}
532
Jamie Madill4fd95d52017-04-05 11:22:18 -0400533gl::Error WindowSurfaceVk::getAttachmentRenderTarget(GLenum /*binding*/,
534 const gl::ImageIndex & /*target*/,
535 FramebufferAttachmentRenderTarget **rtOut)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500536{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500537 *rtOut = &mRenderTarget;
538 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400539}
540
Jamie Madillab9f9c32017-01-17 17:47:34 -0500541gl::ErrorOrResult<vk::Framebuffer *> WindowSurfaceVk::getCurrentFramebuffer(
542 VkDevice device,
543 const vk::RenderPass &compatibleRenderPass)
544{
545 if (!mSwapchainFramebuffers.empty())
546 {
547 // Validation layers should detect if the render pass is really compatible.
548 return &mSwapchainFramebuffers[mCurrentSwapchainImageIndex];
549 }
550
551 VkFramebufferCreateInfo framebufferInfo;
552
553 // TODO(jmadill): Depth/Stencil attachments.
554 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
555 framebufferInfo.pNext = nullptr;
556 framebufferInfo.flags = 0;
557 framebufferInfo.renderPass = compatibleRenderPass.getHandle();
558 framebufferInfo.attachmentCount = 1u;
559 framebufferInfo.pAttachments = nullptr;
560 framebufferInfo.width = static_cast<uint32_t>(mRenderTarget.extents.width);
561 framebufferInfo.height = static_cast<uint32_t>(mRenderTarget.extents.height);
562 framebufferInfo.layers = 1;
563
Jamie Madill5deea722017-02-16 10:44:46 -0500564 mSwapchainFramebuffers.resize(mSwapchainImageViews.size());
565 for (size_t imageIndex = 0; imageIndex < mSwapchainFramebuffers.size(); ++imageIndex)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500566 {
Jamie Madill5deea722017-02-16 10:44:46 -0500567 const auto &imageView = mSwapchainImageViews[imageIndex];
Jamie Madillab9f9c32017-01-17 17:47:34 -0500568 VkImageView imageViewHandle = imageView.getHandle();
569 framebufferInfo.pAttachments = &imageViewHandle;
570
Jamie Madill5deea722017-02-16 10:44:46 -0500571 vk::Framebuffer framebuffer;
572 ANGLE_TRY(framebuffer.init(device, framebufferInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500573
Jamie Madill5deea722017-02-16 10:44:46 -0500574 mSwapchainFramebuffers[imageIndex].retain(device, std::move(framebuffer));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500575 }
576
577 // We should only initialize framebuffers on the first swap.
578 ASSERT(mCurrentSwapchainImageIndex == 0u);
579 return &mSwapchainFramebuffers[mCurrentSwapchainImageIndex];
580}
581
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400582} // namespace rx