blob: d98f4fe46f4cbd8893d67c992c37884fc279641a [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
Jamie Madill98de8262017-05-29 13:01:02 -040050 WARN() << "Present mode " << bestChoice << " not available. Falling back to "
51 << presentModes[0];
Jamie Madillf0eafe12017-02-21 15:03:50 -050052 return presentModes[0];
53}
54
Jamie Madill4d0bf552016-12-28 15:45:24 -050055} // namespace
56
Jamie Madille09bd5d2016-11-29 16:20:35 -050057OffscreenSurfaceVk::OffscreenSurfaceVk(const egl::SurfaceState &surfaceState,
58 EGLint width,
59 EGLint height)
60 : SurfaceImpl(surfaceState), mWidth(width), mHeight(height)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040061{
62}
63
Jamie Madille09bd5d2016-11-29 16:20:35 -050064OffscreenSurfaceVk::~OffscreenSurfaceVk()
Jamie Madill9e54b5a2016-05-25 12:57:39 -040065{
66}
67
Jamie Madille09bd5d2016-11-29 16:20:35 -050068egl::Error OffscreenSurfaceVk::initialize(const DisplayImpl *displayImpl)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040069{
Jamie Madille09bd5d2016-11-29 16:20:35 -050070 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040071}
72
Jamie Madille09bd5d2016-11-29 16:20:35 -050073FramebufferImpl *OffscreenSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040074{
Jamie Madill7b57b9d2017-01-13 09:33:38 -050075 // Use a user FBO for an offscreen RT.
76 return FramebufferVk::CreateUserFBO(state);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040077}
78
Jamie Madille09bd5d2016-11-29 16:20:35 -050079egl::Error OffscreenSurfaceVk::swap(const DisplayImpl *displayImpl)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040080{
Jamie Madille09bd5d2016-11-29 16:20:35 -050081 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040082}
83
Jamie Madille09bd5d2016-11-29 16:20:35 -050084egl::Error OffscreenSurfaceVk::postSubBuffer(EGLint /*x*/,
85 EGLint /*y*/,
86 EGLint /*width*/,
87 EGLint /*height*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040088{
Jamie Madille09bd5d2016-11-29 16:20:35 -050089 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040090}
91
Jamie Madille09bd5d2016-11-29 16:20:35 -050092egl::Error OffscreenSurfaceVk::querySurfacePointerANGLE(EGLint /*attribute*/, void ** /*value*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040093{
Jamie Madille09bd5d2016-11-29 16:20:35 -050094 UNREACHABLE();
95 return egl::Error(EGL_BAD_CURRENT_SURFACE);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040096}
97
Jamie Madille09bd5d2016-11-29 16:20:35 -050098egl::Error OffscreenSurfaceVk::bindTexImage(gl::Texture * /*texture*/, EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040099{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500100 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400101}
102
Jamie Madille09bd5d2016-11-29 16:20:35 -0500103egl::Error OffscreenSurfaceVk::releaseTexImage(EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400104{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500105 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400106}
107
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700108egl::Error OffscreenSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
109 EGLuint64KHR * /*msc*/,
110 EGLuint64KHR * /*sbc*/)
111{
112 UNIMPLEMENTED();
113 return egl::Error(EGL_BAD_ACCESS);
114}
115
Jamie Madille09bd5d2016-11-29 16:20:35 -0500116void OffscreenSurfaceVk::setSwapInterval(EGLint /*interval*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400117{
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400118}
119
Jamie Madille09bd5d2016-11-29 16:20:35 -0500120EGLint OffscreenSurfaceVk::getWidth() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400121{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500122 return mWidth;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400123}
124
Jamie Madille09bd5d2016-11-29 16:20:35 -0500125EGLint OffscreenSurfaceVk::getHeight() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400126{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500127 return mHeight;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400128}
129
Jamie Madille09bd5d2016-11-29 16:20:35 -0500130EGLint OffscreenSurfaceVk::isPostSubBufferSupported() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400131{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500132 return EGL_FALSE;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400133}
134
Jamie Madille09bd5d2016-11-29 16:20:35 -0500135EGLint OffscreenSurfaceVk::getSwapBehavior() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400136{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500137 return EGL_BUFFER_PRESERVED;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400138}
139
Jamie Madille09bd5d2016-11-29 16:20:35 -0500140gl::Error OffscreenSurfaceVk::getAttachmentRenderTarget(
Jamie Madill4fd95d52017-04-05 11:22:18 -0400141 GLenum /*binding*/,
142 const gl::ImageIndex & /*imageIndex*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -0500143 FramebufferAttachmentRenderTarget ** /*rtOut*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400144{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500145 UNREACHABLE();
146 return gl::Error(GL_INVALID_OPERATION);
147}
148
Jamie Madill4d0bf552016-12-28 15:45:24 -0500149WindowSurfaceVk::WindowSurfaceVk(const egl::SurfaceState &surfaceState,
150 EGLNativeWindowType window,
151 EGLint width,
152 EGLint height)
153 : SurfaceImpl(surfaceState),
154 mNativeWindowType(window),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500155 mSurface(VK_NULL_HANDLE),
Frank Henigman29f148b2016-11-23 21:05:36 -0500156 mInstance(VK_NULL_HANDLE),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500157 mSwapchain(VK_NULL_HANDLE),
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500158 mRenderTarget(),
159 mCurrentSwapchainImageIndex(0)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500160{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500161 mRenderTarget.extents.width = static_cast<GLint>(width);
162 mRenderTarget.extents.height = static_cast<GLint>(height);
163 mRenderTarget.extents.depth = 1;
Jamie Madill4c26fc22017-02-24 11:04:10 -0500164 mRenderTarget.resource = this;
Jamie Madille09bd5d2016-11-29 16:20:35 -0500165}
166
167WindowSurfaceVk::~WindowSurfaceVk()
168{
Jamie Madill70ee0f62017-02-06 16:04:20 -0500169 ASSERT(mSurface == VK_NULL_HANDLE);
170 ASSERT(mSwapchain == VK_NULL_HANDLE);
Jamie Madill70ee0f62017-02-06 16:04:20 -0500171}
172
173void WindowSurfaceVk::destroy(const DisplayImpl *displayImpl)
174{
175 const DisplayVk *displayVk = GetAs<DisplayVk>(displayImpl);
176 RendererVk *rendererVk = displayVk->getRenderer();
177 VkDevice device = rendererVk->getDevice();
178 VkInstance instance = rendererVk->getInstance();
179
Jamie Madille918de22017-04-12 10:21:11 -0400180 rendererVk->finish();
181
182 mImageAvailableSemaphore.destroy(device);
183 mRenderingCompleteSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500184
Jamie Madill70ee0f62017-02-06 16:04:20 -0500185 for (auto &imageView : mSwapchainImageViews)
186 {
187 imageView.destroy(device);
188 }
189
Jamie Madill70ee0f62017-02-06 16:04:20 -0500190 // Although we don't own the swapchain image handles, we need to keep our shutdown clean.
191 for (auto &image : mSwapchainImages)
192 {
193 image.reset();
194 }
195
Jamie Madill5deea722017-02-16 10:44:46 -0500196 for (auto &framebuffer : mSwapchainFramebuffers)
197 {
198 framebuffer.destroy(device);
199 }
Jamie Madill4d0bf552016-12-28 15:45:24 -0500200
201 if (mSwapchain)
202 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500203 vkDestroySwapchainKHR(device, mSwapchain, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500204 mSwapchain = VK_NULL_HANDLE;
205 }
206
207 if (mSurface)
208 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500209 vkDestroySurfaceKHR(instance, mSurface, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500210 mSurface = VK_NULL_HANDLE;
211 }
Jamie Madille09bd5d2016-11-29 16:20:35 -0500212}
213
214egl::Error WindowSurfaceVk::initialize(const DisplayImpl *displayImpl)
215{
Jamie Madill4d0bf552016-12-28 15:45:24 -0500216 const DisplayVk *displayVk = GetAs<DisplayVk>(displayImpl);
217 return initializeImpl(displayVk->getRenderer()).toEGL(EGL_BAD_SURFACE);
218}
219
220vk::Error WindowSurfaceVk::initializeImpl(RendererVk *renderer)
221{
Frank Henigman29f148b2016-11-23 21:05:36 -0500222 gl::Extents windowSize;
223 ANGLE_TRY_RESULT(createSurfaceVk(renderer), windowSize);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500224
225 uint32_t presentQueue = 0;
226 ANGLE_TRY_RESULT(renderer->selectPresentQueueForSurface(mSurface), presentQueue);
227
228 const auto &physicalDevice = renderer->getPhysicalDevice();
229
230 VkSurfaceCapabilitiesKHR surfaceCaps;
231 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physicalDevice, mSurface, &surfaceCaps));
232
233 // Adjust width and height to the swapchain if necessary.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500234 uint32_t width = surfaceCaps.currentExtent.width;
235 uint32_t height = surfaceCaps.currentExtent.height;
236
237 // TODO(jmadill): Support devices which don't support copy. We use this for ReadPixels.
238 ANGLE_VK_CHECK((surfaceCaps.supportedUsageFlags & VK_IMAGE_USAGE_TRANSFER_SRC_BIT) != 0,
239 VK_ERROR_INITIALIZATION_FAILED);
240
241 if (surfaceCaps.currentExtent.width == 0xFFFFFFFFu)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500242 {
243 ASSERT(surfaceCaps.currentExtent.height == 0xFFFFFFFFu);
244
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500245 if (mRenderTarget.extents.width == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500246 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500247 width = windowSize.width;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500248 }
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500249 if (mRenderTarget.extents.height == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500250 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500251 height = windowSize.height;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500252 }
253 }
254
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500255 mRenderTarget.extents.width = static_cast<int>(width);
256 mRenderTarget.extents.height = static_cast<int>(height);
257
Jamie Madill4d0bf552016-12-28 15:45:24 -0500258 uint32_t presentModeCount = 0;
259 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
260 &presentModeCount, nullptr));
261 ASSERT(presentModeCount > 0);
262
263 std::vector<VkPresentModeKHR> presentModes(presentModeCount);
264 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
265 &presentModeCount, presentModes.data()));
266
Jamie Madillf0eafe12017-02-21 15:03:50 -0500267 // Select appropriate present mode based on vsync parameter.
268 // TODO(jmadill): More complete implementation, which allows for changing and more values.
269 const EGLint minSwapInterval = mState.config->minSwapInterval;
270 const EGLint maxSwapInterval = mState.config->maxSwapInterval;
271 ASSERT(minSwapInterval == 0 || minSwapInterval == 1);
272 ASSERT(maxSwapInterval == 0 || maxSwapInterval == 1);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500273
Jamie Madillf0eafe12017-02-21 15:03:50 -0500274 VkPresentModeKHR swapchainPresentMode =
275 GetDesiredPresentMode(presentModes, minSwapInterval, maxSwapInterval);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500276
277 // Determine number of swapchain images. Aim for one more than the minimum.
278 uint32_t minImageCount = surfaceCaps.minImageCount + 1;
279 if (surfaceCaps.maxImageCount > 0 && minImageCount > surfaceCaps.maxImageCount)
280 {
281 minImageCount = surfaceCaps.maxImageCount;
282 }
283
284 // Default to identity transform.
285 VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
286 if ((surfaceCaps.supportedTransforms & preTransform) == 0)
287 {
288 preTransform = surfaceCaps.currentTransform;
289 }
290
Jamie Madill4d0bf552016-12-28 15:45:24 -0500291 uint32_t surfaceFormatCount = 0;
292 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
293 nullptr));
294
295 std::vector<VkSurfaceFormatKHR> surfaceFormats(surfaceFormatCount);
296 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
297 surfaceFormats.data()));
298
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500299 mRenderTarget.format = &GetVkFormatFromConfig(*mState.config);
300 auto nativeFormat = mRenderTarget.format->native;
301
Jamie Madill4d0bf552016-12-28 15:45:24 -0500302 if (surfaceFormatCount == 1u && surfaceFormats[0].format == VK_FORMAT_UNDEFINED)
303 {
304 // This is fine.
305 }
306 else
307 {
308 bool foundFormat = false;
309 for (const auto &surfaceFormat : surfaceFormats)
310 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500311 if (surfaceFormat.format == nativeFormat)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500312 {
313 foundFormat = true;
314 break;
315 }
316 }
317
318 ANGLE_VK_CHECK(foundFormat, VK_ERROR_INITIALIZATION_FAILED);
319 }
320
321 VkSwapchainCreateInfoKHR swapchainInfo;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500322 swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
323 swapchainInfo.pNext = nullptr;
324 swapchainInfo.flags = 0;
325 swapchainInfo.surface = mSurface;
326 swapchainInfo.minImageCount = minImageCount;
327 swapchainInfo.imageFormat = nativeFormat;
328 swapchainInfo.imageColorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
329 swapchainInfo.imageExtent.width = width;
330 swapchainInfo.imageExtent.height = height;
331 swapchainInfo.imageArrayLayers = 1;
332 swapchainInfo.imageUsage = (VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
333 VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500334 swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
335 swapchainInfo.queueFamilyIndexCount = 0;
336 swapchainInfo.pQueueFamilyIndices = nullptr;
337 swapchainInfo.preTransform = preTransform;
338 swapchainInfo.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
339 swapchainInfo.presentMode = swapchainPresentMode;
340 swapchainInfo.clipped = VK_TRUE;
341 swapchainInfo.oldSwapchain = VK_NULL_HANDLE;
342
343 const auto &device = renderer->getDevice();
344 ANGLE_VK_TRY(vkCreateSwapchainKHR(device, &swapchainInfo, nullptr, &mSwapchain));
345
346 // Intialize the swapchain image views.
347 uint32_t imageCount = 0;
348 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, nullptr));
349
350 std::vector<VkImage> swapchainImages(imageCount);
351 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, swapchainImages.data()));
352
353 // CommandBuffer is a singleton in the Renderer.
Jamie Madill0c0dc342017-03-24 14:18:51 -0400354 vk::CommandBuffer *commandBuffer = nullptr;
355 ANGLE_TRY(renderer->getStartedCommandBuffer(&commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500356
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500357 VkClearColorValue transparentBlack;
358 transparentBlack.float32[0] = 0.0f;
359 transparentBlack.float32[1] = 0.0f;
360 transparentBlack.float32[2] = 0.0f;
361 transparentBlack.float32[3] = 0.0f;
362
Jamie Madill5deea722017-02-16 10:44:46 -0500363 mSwapchainImages.resize(imageCount);
364 mSwapchainImageViews.resize(imageCount);
365
366 for (uint32_t imageIndex = 0; imageIndex < imageCount; ++imageIndex)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500367 {
Jamie Madill5deea722017-02-16 10:44:46 -0500368 VkImage swapchainImage = swapchainImages[imageIndex];
369
Jamie Madill4d0bf552016-12-28 15:45:24 -0500370 VkImageViewCreateInfo imageViewInfo;
371 imageViewInfo.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
372 imageViewInfo.pNext = nullptr;
373 imageViewInfo.flags = 0;
374 imageViewInfo.image = swapchainImage;
375 imageViewInfo.viewType = VK_IMAGE_VIEW_TYPE_2D;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500376 imageViewInfo.format = nativeFormat;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500377 imageViewInfo.components.r = VK_COMPONENT_SWIZZLE_R;
378 imageViewInfo.components.g = VK_COMPONENT_SWIZZLE_G;
379 imageViewInfo.components.b = VK_COMPONENT_SWIZZLE_B;
380 imageViewInfo.components.a = VK_COMPONENT_SWIZZLE_A;
381 imageViewInfo.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
382 imageViewInfo.subresourceRange.baseMipLevel = 0;
383 imageViewInfo.subresourceRange.levelCount = 1;
384 imageViewInfo.subresourceRange.baseArrayLayer = 0;
385 imageViewInfo.subresourceRange.layerCount = 1;
386
387 vk::Image image(swapchainImage);
Jamie Madill5deea722017-02-16 10:44:46 -0500388 vk::ImageView imageView;
389 ANGLE_TRY(imageView.init(device, imageViewInfo));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500390
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500391 // Set transfer dest layout, and clear the image to black.
392 image.changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
393 commandBuffer);
394 commandBuffer->clearSingleColorImage(image, transparentBlack);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500395
Jamie Madill5deea722017-02-16 10:44:46 -0500396 mSwapchainImages[imageIndex].retain(device, std::move(image));
397 mSwapchainImageViews[imageIndex].retain(device, std::move(imageView));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500398 }
399
Jamie Madill0c0dc342017-03-24 14:18:51 -0400400 ANGLE_TRY(renderer->submitAndFinishCommandBuffer(commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500401
Jamie Madille918de22017-04-12 10:21:11 -0400402 ANGLE_TRY(mImageAvailableSemaphore.init(device));
403 ANGLE_TRY(mRenderingCompleteSemaphore.init(device));
404
405 // Get the first available swapchain iamge.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500406 ANGLE_TRY(nextSwapchainImage(renderer));
407
Jamie Madill4d0bf552016-12-28 15:45:24 -0500408 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500409}
410
411FramebufferImpl *WindowSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
412{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500413 return FramebufferVk::CreateDefaultFBO(state, this);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500414}
415
416egl::Error WindowSurfaceVk::swap(const DisplayImpl *displayImpl)
417{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500418 const DisplayVk *displayVk = GetAs<DisplayVk>(displayImpl);
419 return swapImpl(displayVk->getRenderer()).toEGL(EGL_BAD_ALLOC);
420}
421
422vk::Error WindowSurfaceVk::swapImpl(RendererVk *renderer)
423{
Jamie Madill0c0dc342017-03-24 14:18:51 -0400424 vk::CommandBuffer *currentCB = nullptr;
425 ANGLE_TRY(renderer->getStartedCommandBuffer(&currentCB));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500426
427 auto *image = &mSwapchainImages[mCurrentSwapchainImageIndex];
428
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500429 image->changeLayoutWithStages(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
430 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
431 VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, currentCB);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500432
Jamie Madill0c0dc342017-03-24 14:18:51 -0400433 ANGLE_TRY(renderer->submitCommandsWithSync(currentCB, mImageAvailableSemaphore,
Jamie Madille918de22017-04-12 10:21:11 -0400434 mRenderingCompleteSemaphore));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500435
436 VkPresentInfoKHR presentInfo;
437 presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
438 presentInfo.pNext = nullptr;
Jamie Madille918de22017-04-12 10:21:11 -0400439 presentInfo.waitSemaphoreCount = 1;
440 presentInfo.pWaitSemaphores = mRenderingCompleteSemaphore.ptr();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500441 presentInfo.swapchainCount = 1;
442 presentInfo.pSwapchains = &mSwapchain;
443 presentInfo.pImageIndices = &mCurrentSwapchainImageIndex;
444 presentInfo.pResults = nullptr;
445
446 ANGLE_VK_TRY(vkQueuePresentKHR(renderer->getQueue(), &presentInfo));
447
Jamie Madillf0eafe12017-02-21 15:03:50 -0500448 // Get the next available swapchain image.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500449 ANGLE_TRY(nextSwapchainImage(renderer));
450
451 return vk::NoError();
452}
453
454vk::Error WindowSurfaceVk::nextSwapchainImage(RendererVk *renderer)
455{
456 VkDevice device = renderer->getDevice();
457
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500458 ANGLE_VK_TRY(vkAcquireNextImageKHR(device, mSwapchain, std::numeric_limits<uint64_t>::max(),
Jamie Madille918de22017-04-12 10:21:11 -0400459 mImageAvailableSemaphore.getHandle(), VK_NULL_HANDLE,
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500460 &mCurrentSwapchainImageIndex));
461
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500462 // Update RenderTarget pointers.
463 mRenderTarget.image = &mSwapchainImages[mCurrentSwapchainImageIndex];
464 mRenderTarget.imageView = &mSwapchainImageViews[mCurrentSwapchainImageIndex];
465
466 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500467}
468
469egl::Error WindowSurfaceVk::postSubBuffer(EGLint x, EGLint y, EGLint width, EGLint height)
470{
471 // TODO(jmadill)
472 return egl::Error(EGL_SUCCESS);
473}
474
475egl::Error WindowSurfaceVk::querySurfacePointerANGLE(EGLint attribute, void **value)
476{
477 UNREACHABLE();
478 return egl::Error(EGL_BAD_CURRENT_SURFACE);
479}
480
481egl::Error WindowSurfaceVk::bindTexImage(gl::Texture *texture, EGLint buffer)
482{
483 return egl::Error(EGL_SUCCESS);
484}
485
486egl::Error WindowSurfaceVk::releaseTexImage(EGLint buffer)
487{
488 return egl::Error(EGL_SUCCESS);
489}
490
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700491egl::Error WindowSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
492 EGLuint64KHR * /*msc*/,
493 EGLuint64KHR * /*sbc*/)
494{
495 UNIMPLEMENTED();
496 return egl::Error(EGL_BAD_ACCESS);
497}
498
Jamie Madille09bd5d2016-11-29 16:20:35 -0500499void WindowSurfaceVk::setSwapInterval(EGLint interval)
500{
501}
502
503EGLint WindowSurfaceVk::getWidth() const
504{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500505 return static_cast<EGLint>(mRenderTarget.extents.width);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500506}
507
508EGLint WindowSurfaceVk::getHeight() const
509{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500510 return static_cast<EGLint>(mRenderTarget.extents.height);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500511}
512
513EGLint WindowSurfaceVk::isPostSubBufferSupported() const
514{
515 // TODO(jmadill)
516 return EGL_FALSE;
517}
518
519EGLint WindowSurfaceVk::getSwapBehavior() const
520{
521 // TODO(jmadill)
522 return EGL_BUFFER_DESTROYED;
523}
524
Jamie Madill4fd95d52017-04-05 11:22:18 -0400525gl::Error WindowSurfaceVk::getAttachmentRenderTarget(GLenum /*binding*/,
526 const gl::ImageIndex & /*target*/,
527 FramebufferAttachmentRenderTarget **rtOut)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500528{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500529 *rtOut = &mRenderTarget;
530 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400531}
532
Jamie Madillab9f9c32017-01-17 17:47:34 -0500533gl::ErrorOrResult<vk::Framebuffer *> WindowSurfaceVk::getCurrentFramebuffer(
534 VkDevice device,
535 const vk::RenderPass &compatibleRenderPass)
536{
537 if (!mSwapchainFramebuffers.empty())
538 {
539 // Validation layers should detect if the render pass is really compatible.
540 return &mSwapchainFramebuffers[mCurrentSwapchainImageIndex];
541 }
542
543 VkFramebufferCreateInfo framebufferInfo;
544
545 // TODO(jmadill): Depth/Stencil attachments.
546 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
547 framebufferInfo.pNext = nullptr;
548 framebufferInfo.flags = 0;
549 framebufferInfo.renderPass = compatibleRenderPass.getHandle();
550 framebufferInfo.attachmentCount = 1u;
551 framebufferInfo.pAttachments = nullptr;
552 framebufferInfo.width = static_cast<uint32_t>(mRenderTarget.extents.width);
553 framebufferInfo.height = static_cast<uint32_t>(mRenderTarget.extents.height);
554 framebufferInfo.layers = 1;
555
Jamie Madill5deea722017-02-16 10:44:46 -0500556 mSwapchainFramebuffers.resize(mSwapchainImageViews.size());
557 for (size_t imageIndex = 0; imageIndex < mSwapchainFramebuffers.size(); ++imageIndex)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500558 {
Jamie Madill5deea722017-02-16 10:44:46 -0500559 const auto &imageView = mSwapchainImageViews[imageIndex];
Jamie Madillab9f9c32017-01-17 17:47:34 -0500560 VkImageView imageViewHandle = imageView.getHandle();
561 framebufferInfo.pAttachments = &imageViewHandle;
562
Jamie Madill5deea722017-02-16 10:44:46 -0500563 vk::Framebuffer framebuffer;
564 ANGLE_TRY(framebuffer.init(device, framebufferInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500565
Jamie Madill5deea722017-02-16 10:44:46 -0500566 mSwapchainFramebuffers[imageIndex].retain(device, std::move(framebuffer));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500567 }
568
569 // We should only initialize framebuffers on the first swap.
570 ASSERT(mCurrentSwapchainImageIndex == 0u);
571 return &mSwapchainFramebuffers[mCurrentSwapchainImageIndex];
572}
573
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400574} // namespace rx