blob: cd0ee504e8b8c960060e875586ade5ff6f58d7d9 [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 Madillabaab232017-01-10 12:37:37 -050027const vk::Format &GetVkFormatFromConfig(const egl::Config &config)
Jamie Madill4d0bf552016-12-28 15:45:24 -050028{
29 // TODO(jmadill): Properly handle format interpretation.
Jamie Madillabaab232017-01-10 12:37:37 -050030 return vk::Format::Get(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
40 // similar to triple-buffering. For now we hard-code Mailbox for perf tseting.
41 // 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 Madill4fd95d52017-04-05 11:22:18 -0400144 GLenum /*binding*/,
145 const gl::ImageIndex & /*imageIndex*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -0500146 FramebufferAttachmentRenderTarget ** /*rtOut*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400147{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500148 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500149 return gl::InternalError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500150}
151
Jamie Madill4d0bf552016-12-28 15:45:24 -0500152WindowSurfaceVk::WindowSurfaceVk(const egl::SurfaceState &surfaceState,
153 EGLNativeWindowType window,
154 EGLint width,
155 EGLint height)
156 : SurfaceImpl(surfaceState),
157 mNativeWindowType(window),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500158 mSurface(VK_NULL_HANDLE),
Frank Henigman29f148b2016-11-23 21:05:36 -0500159 mInstance(VK_NULL_HANDLE),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500160 mSwapchain(VK_NULL_HANDLE),
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500161 mRenderTarget(),
162 mCurrentSwapchainImageIndex(0)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500163{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500164 mRenderTarget.extents.width = static_cast<GLint>(width);
165 mRenderTarget.extents.height = static_cast<GLint>(height);
166 mRenderTarget.extents.depth = 1;
Jamie Madill4c26fc22017-02-24 11:04:10 -0500167 mRenderTarget.resource = this;
Jamie Madille09bd5d2016-11-29 16:20:35 -0500168}
169
170WindowSurfaceVk::~WindowSurfaceVk()
171{
Jamie Madill70ee0f62017-02-06 16:04:20 -0500172 ASSERT(mSurface == VK_NULL_HANDLE);
173 ASSERT(mSwapchain == VK_NULL_HANDLE);
Jamie Madill70ee0f62017-02-06 16:04:20 -0500174}
175
Jamie Madillc564c072017-06-01 12:45:42 -0400176void WindowSurfaceVk::destroy(const egl::Display *display)
Jamie Madill70ee0f62017-02-06 16:04:20 -0500177{
Jamie Madillc564c072017-06-01 12:45:42 -0400178 const DisplayVk *displayVk = GetImplAs<DisplayVk>(display);
Jamie Madill70ee0f62017-02-06 16:04:20 -0500179 RendererVk *rendererVk = displayVk->getRenderer();
180 VkDevice device = rendererVk->getDevice();
181 VkInstance instance = rendererVk->getInstance();
182
Jamie Madille918de22017-04-12 10:21:11 -0400183 rendererVk->finish();
184
185 mImageAvailableSemaphore.destroy(device);
186 mRenderingCompleteSemaphore.destroy(device);
Jamie Madill5deea722017-02-16 10:44:46 -0500187
Jamie Madill70ee0f62017-02-06 16:04:20 -0500188 for (auto &imageView : mSwapchainImageViews)
189 {
190 imageView.destroy(device);
191 }
192
Jamie Madill70ee0f62017-02-06 16:04:20 -0500193 // Although we don't own the swapchain image handles, we need to keep our shutdown clean.
194 for (auto &image : mSwapchainImages)
195 {
196 image.reset();
197 }
198
Jamie Madill5deea722017-02-16 10:44:46 -0500199 for (auto &framebuffer : mSwapchainFramebuffers)
200 {
201 framebuffer.destroy(device);
202 }
Jamie Madill4d0bf552016-12-28 15:45:24 -0500203
204 if (mSwapchain)
205 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500206 vkDestroySwapchainKHR(device, mSwapchain, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500207 mSwapchain = VK_NULL_HANDLE;
208 }
209
210 if (mSurface)
211 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500212 vkDestroySurfaceKHR(instance, mSurface, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500213 mSurface = VK_NULL_HANDLE;
214 }
Jamie Madille09bd5d2016-11-29 16:20:35 -0500215}
216
Jamie Madillc564c072017-06-01 12:45:42 -0400217egl::Error WindowSurfaceVk::initialize(const egl::Display *display)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500218{
Jamie Madillc564c072017-06-01 12:45:42 -0400219 const DisplayVk *displayVk = GetImplAs<DisplayVk>(display);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500220 return initializeImpl(displayVk->getRenderer()).toEGL(EGL_BAD_SURFACE);
221}
222
223vk::Error WindowSurfaceVk::initializeImpl(RendererVk *renderer)
224{
Frank Henigman29f148b2016-11-23 21:05:36 -0500225 gl::Extents windowSize;
226 ANGLE_TRY_RESULT(createSurfaceVk(renderer), windowSize);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500227
228 uint32_t presentQueue = 0;
229 ANGLE_TRY_RESULT(renderer->selectPresentQueueForSurface(mSurface), presentQueue);
230
231 const auto &physicalDevice = renderer->getPhysicalDevice();
232
233 VkSurfaceCapabilitiesKHR surfaceCaps;
234 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physicalDevice, mSurface, &surfaceCaps));
235
236 // Adjust width and height to the swapchain if necessary.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500237 uint32_t width = surfaceCaps.currentExtent.width;
238 uint32_t height = surfaceCaps.currentExtent.height;
239
240 // TODO(jmadill): Support devices which don't support copy. We use this for ReadPixels.
241 ANGLE_VK_CHECK((surfaceCaps.supportedUsageFlags & VK_IMAGE_USAGE_TRANSFER_SRC_BIT) != 0,
242 VK_ERROR_INITIALIZATION_FAILED);
243
244 if (surfaceCaps.currentExtent.width == 0xFFFFFFFFu)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500245 {
246 ASSERT(surfaceCaps.currentExtent.height == 0xFFFFFFFFu);
247
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500248 if (mRenderTarget.extents.width == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500249 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500250 width = windowSize.width;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500251 }
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500252 if (mRenderTarget.extents.height == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500253 {
Frank Henigman29f148b2016-11-23 21:05:36 -0500254 height = windowSize.height;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500255 }
256 }
257
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500258 mRenderTarget.extents.width = static_cast<int>(width);
259 mRenderTarget.extents.height = static_cast<int>(height);
260
Jamie Madill4d0bf552016-12-28 15:45:24 -0500261 uint32_t presentModeCount = 0;
262 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
263 &presentModeCount, nullptr));
264 ASSERT(presentModeCount > 0);
265
266 std::vector<VkPresentModeKHR> presentModes(presentModeCount);
267 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
268 &presentModeCount, presentModes.data()));
269
Jamie Madillf0eafe12017-02-21 15:03:50 -0500270 // Select appropriate present mode based on vsync parameter.
271 // TODO(jmadill): More complete implementation, which allows for changing and more values.
272 const EGLint minSwapInterval = mState.config->minSwapInterval;
273 const EGLint maxSwapInterval = mState.config->maxSwapInterval;
274 ASSERT(minSwapInterval == 0 || minSwapInterval == 1);
275 ASSERT(maxSwapInterval == 0 || maxSwapInterval == 1);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500276
Jamie Madillf0eafe12017-02-21 15:03:50 -0500277 VkPresentModeKHR swapchainPresentMode =
278 GetDesiredPresentMode(presentModes, minSwapInterval, maxSwapInterval);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500279
280 // Determine number of swapchain images. Aim for one more than the minimum.
281 uint32_t minImageCount = surfaceCaps.minImageCount + 1;
282 if (surfaceCaps.maxImageCount > 0 && minImageCount > surfaceCaps.maxImageCount)
283 {
284 minImageCount = surfaceCaps.maxImageCount;
285 }
286
287 // Default to identity transform.
288 VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
289 if ((surfaceCaps.supportedTransforms & preTransform) == 0)
290 {
291 preTransform = surfaceCaps.currentTransform;
292 }
293
Jamie Madill4d0bf552016-12-28 15:45:24 -0500294 uint32_t surfaceFormatCount = 0;
295 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
296 nullptr));
297
298 std::vector<VkSurfaceFormatKHR> surfaceFormats(surfaceFormatCount);
299 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
300 surfaceFormats.data()));
301
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500302 mRenderTarget.format = &GetVkFormatFromConfig(*mState.config);
303 auto nativeFormat = mRenderTarget.format->native;
304
Jamie Madill4d0bf552016-12-28 15:45:24 -0500305 if (surfaceFormatCount == 1u && surfaceFormats[0].format == VK_FORMAT_UNDEFINED)
306 {
307 // This is fine.
308 }
309 else
310 {
311 bool foundFormat = false;
312 for (const auto &surfaceFormat : surfaceFormats)
313 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500314 if (surfaceFormat.format == nativeFormat)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500315 {
316 foundFormat = true;
317 break;
318 }
319 }
320
321 ANGLE_VK_CHECK(foundFormat, VK_ERROR_INITIALIZATION_FAILED);
322 }
323
324 VkSwapchainCreateInfoKHR swapchainInfo;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500325 swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
326 swapchainInfo.pNext = nullptr;
327 swapchainInfo.flags = 0;
328 swapchainInfo.surface = mSurface;
329 swapchainInfo.minImageCount = minImageCount;
330 swapchainInfo.imageFormat = nativeFormat;
331 swapchainInfo.imageColorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
332 swapchainInfo.imageExtent.width = width;
333 swapchainInfo.imageExtent.height = height;
334 swapchainInfo.imageArrayLayers = 1;
335 swapchainInfo.imageUsage = (VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
336 VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500337 swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
338 swapchainInfo.queueFamilyIndexCount = 0;
339 swapchainInfo.pQueueFamilyIndices = nullptr;
340 swapchainInfo.preTransform = preTransform;
341 swapchainInfo.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
342 swapchainInfo.presentMode = swapchainPresentMode;
343 swapchainInfo.clipped = VK_TRUE;
344 swapchainInfo.oldSwapchain = VK_NULL_HANDLE;
345
346 const auto &device = renderer->getDevice();
347 ANGLE_VK_TRY(vkCreateSwapchainKHR(device, &swapchainInfo, nullptr, &mSwapchain));
348
349 // Intialize the swapchain image views.
350 uint32_t imageCount = 0;
351 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, nullptr));
352
353 std::vector<VkImage> swapchainImages(imageCount);
354 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, swapchainImages.data()));
355
356 // CommandBuffer is a singleton in the Renderer.
Jamie Madill0c0dc342017-03-24 14:18:51 -0400357 vk::CommandBuffer *commandBuffer = nullptr;
358 ANGLE_TRY(renderer->getStartedCommandBuffer(&commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500359
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500360 VkClearColorValue transparentBlack;
361 transparentBlack.float32[0] = 0.0f;
362 transparentBlack.float32[1] = 0.0f;
363 transparentBlack.float32[2] = 0.0f;
364 transparentBlack.float32[3] = 0.0f;
365
Jamie Madill5deea722017-02-16 10:44:46 -0500366 mSwapchainImages.resize(imageCount);
367 mSwapchainImageViews.resize(imageCount);
368
369 for (uint32_t imageIndex = 0; imageIndex < imageCount; ++imageIndex)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500370 {
Jamie Madill5deea722017-02-16 10:44:46 -0500371 VkImage swapchainImage = swapchainImages[imageIndex];
372
Jamie Madill4d0bf552016-12-28 15:45:24 -0500373 VkImageViewCreateInfo imageViewInfo;
374 imageViewInfo.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
375 imageViewInfo.pNext = nullptr;
376 imageViewInfo.flags = 0;
377 imageViewInfo.image = swapchainImage;
378 imageViewInfo.viewType = VK_IMAGE_VIEW_TYPE_2D;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500379 imageViewInfo.format = nativeFormat;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500380 imageViewInfo.components.r = VK_COMPONENT_SWIZZLE_R;
381 imageViewInfo.components.g = VK_COMPONENT_SWIZZLE_G;
382 imageViewInfo.components.b = VK_COMPONENT_SWIZZLE_B;
383 imageViewInfo.components.a = VK_COMPONENT_SWIZZLE_A;
384 imageViewInfo.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
385 imageViewInfo.subresourceRange.baseMipLevel = 0;
386 imageViewInfo.subresourceRange.levelCount = 1;
387 imageViewInfo.subresourceRange.baseArrayLayer = 0;
388 imageViewInfo.subresourceRange.layerCount = 1;
389
390 vk::Image image(swapchainImage);
Jamie Madill5deea722017-02-16 10:44:46 -0500391 vk::ImageView imageView;
392 ANGLE_TRY(imageView.init(device, imageViewInfo));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500393
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500394 // Set transfer dest layout, and clear the image to black.
395 image.changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
396 commandBuffer);
397 commandBuffer->clearSingleColorImage(image, transparentBlack);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500398
Jamie Madill5deea722017-02-16 10:44:46 -0500399 mSwapchainImages[imageIndex].retain(device, std::move(image));
400 mSwapchainImageViews[imageIndex].retain(device, std::move(imageView));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500401 }
402
Jamie Madill0c0dc342017-03-24 14:18:51 -0400403 ANGLE_TRY(renderer->submitAndFinishCommandBuffer(commandBuffer));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500404
Jamie Madille918de22017-04-12 10:21:11 -0400405 ANGLE_TRY(mImageAvailableSemaphore.init(device));
406 ANGLE_TRY(mRenderingCompleteSemaphore.init(device));
407
408 // Get the first available swapchain iamge.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500409 ANGLE_TRY(nextSwapchainImage(renderer));
410
Jamie Madill4d0bf552016-12-28 15:45:24 -0500411 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500412}
413
414FramebufferImpl *WindowSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
415{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500416 return FramebufferVk::CreateDefaultFBO(state, this);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500417}
418
Jamie Madillfe548342017-06-19 11:13:24 -0400419egl::Error WindowSurfaceVk::swap(const gl::Context *context)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500420{
Jamie Madillfe548342017-06-19 11:13:24 -0400421 const DisplayVk *displayVk = GetImplAs<DisplayVk>(context->getCurrentDisplay());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500422 return swapImpl(displayVk->getRenderer()).toEGL(EGL_BAD_ALLOC);
423}
424
425vk::Error WindowSurfaceVk::swapImpl(RendererVk *renderer)
426{
Jamie Madill0c0dc342017-03-24 14:18:51 -0400427 vk::CommandBuffer *currentCB = nullptr;
428 ANGLE_TRY(renderer->getStartedCommandBuffer(&currentCB));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500429
430 auto *image = &mSwapchainImages[mCurrentSwapchainImageIndex];
431
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500432 image->changeLayoutWithStages(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
433 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
434 VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, currentCB);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500435
Jamie Madill0c0dc342017-03-24 14:18:51 -0400436 ANGLE_TRY(renderer->submitCommandsWithSync(currentCB, mImageAvailableSemaphore,
Jamie Madille918de22017-04-12 10:21:11 -0400437 mRenderingCompleteSemaphore));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500438
439 VkPresentInfoKHR presentInfo;
440 presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
441 presentInfo.pNext = nullptr;
Jamie Madille918de22017-04-12 10:21:11 -0400442 presentInfo.waitSemaphoreCount = 1;
443 presentInfo.pWaitSemaphores = mRenderingCompleteSemaphore.ptr();
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500444 presentInfo.swapchainCount = 1;
445 presentInfo.pSwapchains = &mSwapchain;
446 presentInfo.pImageIndices = &mCurrentSwapchainImageIndex;
447 presentInfo.pResults = nullptr;
448
449 ANGLE_VK_TRY(vkQueuePresentKHR(renderer->getQueue(), &presentInfo));
450
Jamie Madillf0eafe12017-02-21 15:03:50 -0500451 // Get the next available swapchain image.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500452 ANGLE_TRY(nextSwapchainImage(renderer));
453
454 return vk::NoError();
455}
456
457vk::Error WindowSurfaceVk::nextSwapchainImage(RendererVk *renderer)
458{
459 VkDevice device = renderer->getDevice();
460
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500461 ANGLE_VK_TRY(vkAcquireNextImageKHR(device, mSwapchain, std::numeric_limits<uint64_t>::max(),
Jamie Madille918de22017-04-12 10:21:11 -0400462 mImageAvailableSemaphore.getHandle(), VK_NULL_HANDLE,
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500463 &mCurrentSwapchainImageIndex));
464
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500465 // Update RenderTarget pointers.
466 mRenderTarget.image = &mSwapchainImages[mCurrentSwapchainImageIndex];
467 mRenderTarget.imageView = &mSwapchainImageViews[mCurrentSwapchainImageIndex];
468
469 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500470}
471
Jamie Madillfe548342017-06-19 11:13:24 -0400472egl::Error WindowSurfaceVk::postSubBuffer(const gl::Context *context,
473 EGLint x,
474 EGLint y,
475 EGLint width,
476 EGLint height)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500477{
478 // TODO(jmadill)
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500479 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500480}
481
482egl::Error WindowSurfaceVk::querySurfacePointerANGLE(EGLint attribute, void **value)
483{
484 UNREACHABLE();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500485 return egl::EglBadCurrentSurface();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500486}
487
488egl::Error WindowSurfaceVk::bindTexImage(gl::Texture *texture, EGLint buffer)
489{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500490 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500491}
492
493egl::Error WindowSurfaceVk::releaseTexImage(EGLint buffer)
494{
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500495 return egl::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500496}
497
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700498egl::Error WindowSurfaceVk::getSyncValues(EGLuint64KHR * /*ust*/,
499 EGLuint64KHR * /*msc*/,
500 EGLuint64KHR * /*sbc*/)
501{
502 UNIMPLEMENTED();
Yuly Novikovc4d18aa2017-03-09 18:45:02 -0500503 return egl::EglBadAccess();
Stanislav Chiknavaryanee218f22017-03-22 15:39:13 -0700504}
505
Jamie Madille09bd5d2016-11-29 16:20:35 -0500506void WindowSurfaceVk::setSwapInterval(EGLint interval)
507{
508}
509
510EGLint WindowSurfaceVk::getWidth() const
511{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500512 return static_cast<EGLint>(mRenderTarget.extents.width);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500513}
514
515EGLint WindowSurfaceVk::getHeight() const
516{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500517 return static_cast<EGLint>(mRenderTarget.extents.height);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500518}
519
520EGLint WindowSurfaceVk::isPostSubBufferSupported() const
521{
522 // TODO(jmadill)
523 return EGL_FALSE;
524}
525
526EGLint WindowSurfaceVk::getSwapBehavior() const
527{
528 // TODO(jmadill)
529 return EGL_BUFFER_DESTROYED;
530}
531
Jamie Madill4fd95d52017-04-05 11:22:18 -0400532gl::Error WindowSurfaceVk::getAttachmentRenderTarget(GLenum /*binding*/,
533 const gl::ImageIndex & /*target*/,
534 FramebufferAttachmentRenderTarget **rtOut)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500535{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500536 *rtOut = &mRenderTarget;
537 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400538}
539
Jamie Madillab9f9c32017-01-17 17:47:34 -0500540gl::ErrorOrResult<vk::Framebuffer *> WindowSurfaceVk::getCurrentFramebuffer(
541 VkDevice device,
542 const vk::RenderPass &compatibleRenderPass)
543{
544 if (!mSwapchainFramebuffers.empty())
545 {
546 // Validation layers should detect if the render pass is really compatible.
547 return &mSwapchainFramebuffers[mCurrentSwapchainImageIndex];
548 }
549
550 VkFramebufferCreateInfo framebufferInfo;
551
552 // TODO(jmadill): Depth/Stencil attachments.
553 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
554 framebufferInfo.pNext = nullptr;
555 framebufferInfo.flags = 0;
556 framebufferInfo.renderPass = compatibleRenderPass.getHandle();
557 framebufferInfo.attachmentCount = 1u;
558 framebufferInfo.pAttachments = nullptr;
559 framebufferInfo.width = static_cast<uint32_t>(mRenderTarget.extents.width);
560 framebufferInfo.height = static_cast<uint32_t>(mRenderTarget.extents.height);
561 framebufferInfo.layers = 1;
562
Jamie Madill5deea722017-02-16 10:44:46 -0500563 mSwapchainFramebuffers.resize(mSwapchainImageViews.size());
564 for (size_t imageIndex = 0; imageIndex < mSwapchainFramebuffers.size(); ++imageIndex)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500565 {
Jamie Madill5deea722017-02-16 10:44:46 -0500566 const auto &imageView = mSwapchainImageViews[imageIndex];
Jamie Madillab9f9c32017-01-17 17:47:34 -0500567 VkImageView imageViewHandle = imageView.getHandle();
568 framebufferInfo.pAttachments = &imageViewHandle;
569
Jamie Madill5deea722017-02-16 10:44:46 -0500570 vk::Framebuffer framebuffer;
571 ANGLE_TRY(framebuffer.init(device, framebufferInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500572
Jamie Madill5deea722017-02-16 10:44:46 -0500573 mSwapchainFramebuffers[imageIndex].retain(device, std::move(framebuffer));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500574 }
575
576 // We should only initialize framebuffers on the first swap.
577 ASSERT(mCurrentSwapchainImageIndex == 0u);
578 return &mSwapchainFramebuffers[mCurrentSwapchainImageIndex];
579}
580
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400581} // namespace rx