blob: e724b869ebdc641e591f71370127ae422b221150 [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
31} // namespace
32
Jamie Madille09bd5d2016-11-29 16:20:35 -050033OffscreenSurfaceVk::OffscreenSurfaceVk(const egl::SurfaceState &surfaceState,
34 EGLint width,
35 EGLint height)
36 : SurfaceImpl(surfaceState), mWidth(width), mHeight(height)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040037{
38}
39
Jamie Madille09bd5d2016-11-29 16:20:35 -050040OffscreenSurfaceVk::~OffscreenSurfaceVk()
Jamie Madill9e54b5a2016-05-25 12:57:39 -040041{
42}
43
Jamie Madille09bd5d2016-11-29 16:20:35 -050044egl::Error OffscreenSurfaceVk::initialize(const DisplayImpl *displayImpl)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040045{
Jamie Madille09bd5d2016-11-29 16:20:35 -050046 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040047}
48
Jamie Madille09bd5d2016-11-29 16:20:35 -050049FramebufferImpl *OffscreenSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040050{
Jamie Madill7b57b9d2017-01-13 09:33:38 -050051 // Use a user FBO for an offscreen RT.
52 return FramebufferVk::CreateUserFBO(state);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040053}
54
Jamie Madille09bd5d2016-11-29 16:20:35 -050055egl::Error OffscreenSurfaceVk::swap(const DisplayImpl *displayImpl)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040056{
Jamie Madille09bd5d2016-11-29 16:20:35 -050057 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040058}
59
Jamie Madille09bd5d2016-11-29 16:20:35 -050060egl::Error OffscreenSurfaceVk::postSubBuffer(EGLint /*x*/,
61 EGLint /*y*/,
62 EGLint /*width*/,
63 EGLint /*height*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040064{
Jamie Madille09bd5d2016-11-29 16:20:35 -050065 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040066}
67
Jamie Madille09bd5d2016-11-29 16:20:35 -050068egl::Error OffscreenSurfaceVk::querySurfacePointerANGLE(EGLint /*attribute*/, void ** /*value*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040069{
Jamie Madille09bd5d2016-11-29 16:20:35 -050070 UNREACHABLE();
71 return egl::Error(EGL_BAD_CURRENT_SURFACE);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040072}
73
Jamie Madille09bd5d2016-11-29 16:20:35 -050074egl::Error OffscreenSurfaceVk::bindTexImage(gl::Texture * /*texture*/, EGLint /*buffer*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040075{
Jamie Madille09bd5d2016-11-29 16:20:35 -050076 return egl::Error(EGL_SUCCESS);
Jamie Madill9e54b5a2016-05-25 12:57:39 -040077}
78
Jamie Madille09bd5d2016-11-29 16:20:35 -050079egl::Error OffscreenSurfaceVk::releaseTexImage(EGLint /*buffer*/)
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 -050084void OffscreenSurfaceVk::setSwapInterval(EGLint /*interval*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -040085{
Jamie Madill9e54b5a2016-05-25 12:57:39 -040086}
87
Jamie Madille09bd5d2016-11-29 16:20:35 -050088EGLint OffscreenSurfaceVk::getWidth() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -040089{
Jamie Madille09bd5d2016-11-29 16:20:35 -050090 return mWidth;
Jamie Madill9e54b5a2016-05-25 12:57:39 -040091}
92
Jamie Madille09bd5d2016-11-29 16:20:35 -050093EGLint OffscreenSurfaceVk::getHeight() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -040094{
Jamie Madille09bd5d2016-11-29 16:20:35 -050095 return mHeight;
Jamie Madill9e54b5a2016-05-25 12:57:39 -040096}
97
Jamie Madille09bd5d2016-11-29 16:20:35 -050098EGLint OffscreenSurfaceVk::isPostSubBufferSupported() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -040099{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500100 return EGL_FALSE;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400101}
102
Jamie Madille09bd5d2016-11-29 16:20:35 -0500103EGLint OffscreenSurfaceVk::getSwapBehavior() const
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400104{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500105 return EGL_BUFFER_PRESERVED;
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400106}
107
Jamie Madille09bd5d2016-11-29 16:20:35 -0500108gl::Error OffscreenSurfaceVk::getAttachmentRenderTarget(
109 const gl::FramebufferAttachment::Target & /*target*/,
110 FramebufferAttachmentRenderTarget ** /*rtOut*/)
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400111{
Jamie Madille09bd5d2016-11-29 16:20:35 -0500112 UNREACHABLE();
113 return gl::Error(GL_INVALID_OPERATION);
114}
115
Jamie Madill4d0bf552016-12-28 15:45:24 -0500116WindowSurfaceVk::WindowSurfaceVk(const egl::SurfaceState &surfaceState,
117 EGLNativeWindowType window,
118 EGLint width,
119 EGLint height)
120 : SurfaceImpl(surfaceState),
121 mNativeWindowType(window),
Jamie Madill4d0bf552016-12-28 15:45:24 -0500122 mSurface(VK_NULL_HANDLE),
123 mSwapchain(VK_NULL_HANDLE),
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500124 mRenderTarget(),
125 mCurrentSwapchainImageIndex(0)
Jamie Madille09bd5d2016-11-29 16:20:35 -0500126{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500127 mRenderTarget.extents.width = static_cast<GLint>(width);
128 mRenderTarget.extents.height = static_cast<GLint>(height);
129 mRenderTarget.extents.depth = 1;
Jamie Madille09bd5d2016-11-29 16:20:35 -0500130}
131
132WindowSurfaceVk::~WindowSurfaceVk()
133{
Jamie Madill70ee0f62017-02-06 16:04:20 -0500134 ASSERT(mSurface == VK_NULL_HANDLE);
135 ASSERT(mSwapchain == VK_NULL_HANDLE);
Jamie Madill70ee0f62017-02-06 16:04:20 -0500136}
137
138void WindowSurfaceVk::destroy(const DisplayImpl *displayImpl)
139{
140 const DisplayVk *displayVk = GetAs<DisplayVk>(displayImpl);
141 RendererVk *rendererVk = displayVk->getRenderer();
142 VkDevice device = rendererVk->getDevice();
143 VkInstance instance = rendererVk->getInstance();
144
Jamie Madill5deea722017-02-16 10:44:46 -0500145 mPresentCompleteSemaphore.destroy(device);
146
Jamie Madill70ee0f62017-02-06 16:04:20 -0500147 for (auto &imageView : mSwapchainImageViews)
148 {
149 imageView.destroy(device);
150 }
151
Jamie Madill70ee0f62017-02-06 16:04:20 -0500152 // Although we don't own the swapchain image handles, we need to keep our shutdown clean.
153 for (auto &image : mSwapchainImages)
154 {
155 image.reset();
156 }
157
Jamie Madill5deea722017-02-16 10:44:46 -0500158 for (auto &framebuffer : mSwapchainFramebuffers)
159 {
160 framebuffer.destroy(device);
161 }
Jamie Madill4d0bf552016-12-28 15:45:24 -0500162
163 if (mSwapchain)
164 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500165 vkDestroySwapchainKHR(device, mSwapchain, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500166 mSwapchain = VK_NULL_HANDLE;
167 }
168
169 if (mSurface)
170 {
Jamie Madill70ee0f62017-02-06 16:04:20 -0500171 vkDestroySurfaceKHR(instance, mSurface, nullptr);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500172 mSurface = VK_NULL_HANDLE;
173 }
Jamie Madille09bd5d2016-11-29 16:20:35 -0500174}
175
176egl::Error WindowSurfaceVk::initialize(const DisplayImpl *displayImpl)
177{
Jamie Madill4d0bf552016-12-28 15:45:24 -0500178 const DisplayVk *displayVk = GetAs<DisplayVk>(displayImpl);
179 return initializeImpl(displayVk->getRenderer()).toEGL(EGL_BAD_SURFACE);
180}
181
182vk::Error WindowSurfaceVk::initializeImpl(RendererVk *renderer)
183{
Jamie Madill4d0bf552016-12-28 15:45:24 -0500184 // TODO(jmadill): Make this platform-specific.
185 VkWin32SurfaceCreateInfoKHR createInfo;
186
187 createInfo.sType = VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR;
188 createInfo.pNext = nullptr;
189 createInfo.flags = 0;
190 createInfo.hinstance = GetModuleHandle(nullptr);
191 createInfo.hwnd = mNativeWindowType;
192 ANGLE_VK_TRY(vkCreateWin32SurfaceKHR(renderer->getInstance(), &createInfo, nullptr, &mSurface));
193
194 uint32_t presentQueue = 0;
195 ANGLE_TRY_RESULT(renderer->selectPresentQueueForSurface(mSurface), presentQueue);
196
197 const auto &physicalDevice = renderer->getPhysicalDevice();
198
199 VkSurfaceCapabilitiesKHR surfaceCaps;
200 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physicalDevice, mSurface, &surfaceCaps));
201
202 // Adjust width and height to the swapchain if necessary.
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500203 uint32_t width = surfaceCaps.currentExtent.width;
204 uint32_t height = surfaceCaps.currentExtent.height;
205
206 // TODO(jmadill): Support devices which don't support copy. We use this for ReadPixels.
207 ANGLE_VK_CHECK((surfaceCaps.supportedUsageFlags & VK_IMAGE_USAGE_TRANSFER_SRC_BIT) != 0,
208 VK_ERROR_INITIALIZATION_FAILED);
209
210 if (surfaceCaps.currentExtent.width == 0xFFFFFFFFu)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500211 {
212 ASSERT(surfaceCaps.currentExtent.height == 0xFFFFFFFFu);
213
214 RECT rect;
215 ANGLE_VK_CHECK(GetClientRect(mNativeWindowType, &rect) == TRUE,
216 VK_ERROR_INITIALIZATION_FAILED);
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500217 if (mRenderTarget.extents.width == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500218 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500219 width = static_cast<uint32_t>(rect.right - rect.left);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500220 }
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500221 if (mRenderTarget.extents.height == 0)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500222 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500223 height = static_cast<uint32_t>(rect.bottom - rect.top);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500224 }
225 }
226
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500227 mRenderTarget.extents.width = static_cast<int>(width);
228 mRenderTarget.extents.height = static_cast<int>(height);
229
Jamie Madill4d0bf552016-12-28 15:45:24 -0500230 uint32_t presentModeCount = 0;
231 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
232 &presentModeCount, nullptr));
233 ASSERT(presentModeCount > 0);
234
235 std::vector<VkPresentModeKHR> presentModes(presentModeCount);
236 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfacePresentModesKHR(physicalDevice, mSurface,
237 &presentModeCount, presentModes.data()));
238
239 // Use FIFO mode if available, since it throttles you to the display rate. Mailbox can lead
240 // to rendering frames which are never seen by the user, wasting power.
241 VkPresentModeKHR swapchainPresentMode = presentModes[0];
242 for (auto presentMode : presentModes)
243 {
244 if (presentMode == VK_PRESENT_MODE_FIFO_KHR)
245 {
246 swapchainPresentMode = VK_PRESENT_MODE_FIFO_KHR;
247 break;
248 }
249
250 // Fallback to immediate mode if FIFO is unavailable.
251 if (presentMode == VK_PRESENT_MODE_IMMEDIATE_KHR)
252 {
253 swapchainPresentMode = VK_PRESENT_MODE_IMMEDIATE_KHR;
254 }
255 }
256
257 // Determine number of swapchain images. Aim for one more than the minimum.
258 uint32_t minImageCount = surfaceCaps.minImageCount + 1;
259 if (surfaceCaps.maxImageCount > 0 && minImageCount > surfaceCaps.maxImageCount)
260 {
261 minImageCount = surfaceCaps.maxImageCount;
262 }
263
264 // Default to identity transform.
265 VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
266 if ((surfaceCaps.supportedTransforms & preTransform) == 0)
267 {
268 preTransform = surfaceCaps.currentTransform;
269 }
270
Jamie Madill4d0bf552016-12-28 15:45:24 -0500271 uint32_t surfaceFormatCount = 0;
272 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
273 nullptr));
274
275 std::vector<VkSurfaceFormatKHR> surfaceFormats(surfaceFormatCount);
276 ANGLE_VK_TRY(vkGetPhysicalDeviceSurfaceFormatsKHR(physicalDevice, mSurface, &surfaceFormatCount,
277 surfaceFormats.data()));
278
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500279 mRenderTarget.format = &GetVkFormatFromConfig(*mState.config);
280 auto nativeFormat = mRenderTarget.format->native;
281
Jamie Madill4d0bf552016-12-28 15:45:24 -0500282 if (surfaceFormatCount == 1u && surfaceFormats[0].format == VK_FORMAT_UNDEFINED)
283 {
284 // This is fine.
285 }
286 else
287 {
288 bool foundFormat = false;
289 for (const auto &surfaceFormat : surfaceFormats)
290 {
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500291 if (surfaceFormat.format == nativeFormat)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500292 {
293 foundFormat = true;
294 break;
295 }
296 }
297
298 ANGLE_VK_CHECK(foundFormat, VK_ERROR_INITIALIZATION_FAILED);
299 }
300
301 VkSwapchainCreateInfoKHR swapchainInfo;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500302 swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
303 swapchainInfo.pNext = nullptr;
304 swapchainInfo.flags = 0;
305 swapchainInfo.surface = mSurface;
306 swapchainInfo.minImageCount = minImageCount;
307 swapchainInfo.imageFormat = nativeFormat;
308 swapchainInfo.imageColorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
309 swapchainInfo.imageExtent.width = width;
310 swapchainInfo.imageExtent.height = height;
311 swapchainInfo.imageArrayLayers = 1;
312 swapchainInfo.imageUsage = (VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
313 VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500314 swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
315 swapchainInfo.queueFamilyIndexCount = 0;
316 swapchainInfo.pQueueFamilyIndices = nullptr;
317 swapchainInfo.preTransform = preTransform;
318 swapchainInfo.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
319 swapchainInfo.presentMode = swapchainPresentMode;
320 swapchainInfo.clipped = VK_TRUE;
321 swapchainInfo.oldSwapchain = VK_NULL_HANDLE;
322
323 const auto &device = renderer->getDevice();
324 ANGLE_VK_TRY(vkCreateSwapchainKHR(device, &swapchainInfo, nullptr, &mSwapchain));
325
326 // Intialize the swapchain image views.
327 uint32_t imageCount = 0;
328 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, nullptr));
329
330 std::vector<VkImage> swapchainImages(imageCount);
331 ANGLE_VK_TRY(vkGetSwapchainImagesKHR(device, mSwapchain, &imageCount, swapchainImages.data()));
332
333 // CommandBuffer is a singleton in the Renderer.
334 vk::CommandBuffer *commandBuffer = renderer->getCommandBuffer();
Jamie Madill5deea722017-02-16 10:44:46 -0500335 ANGLE_TRY(commandBuffer->begin(device));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500336
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500337 VkClearColorValue transparentBlack;
338 transparentBlack.float32[0] = 0.0f;
339 transparentBlack.float32[1] = 0.0f;
340 transparentBlack.float32[2] = 0.0f;
341 transparentBlack.float32[3] = 0.0f;
342
Jamie Madill5deea722017-02-16 10:44:46 -0500343 mSwapchainImages.resize(imageCount);
344 mSwapchainImageViews.resize(imageCount);
345
346 for (uint32_t imageIndex = 0; imageIndex < imageCount; ++imageIndex)
Jamie Madill4d0bf552016-12-28 15:45:24 -0500347 {
Jamie Madill5deea722017-02-16 10:44:46 -0500348 VkImage swapchainImage = swapchainImages[imageIndex];
349
Jamie Madill4d0bf552016-12-28 15:45:24 -0500350 VkImageViewCreateInfo imageViewInfo;
351 imageViewInfo.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
352 imageViewInfo.pNext = nullptr;
353 imageViewInfo.flags = 0;
354 imageViewInfo.image = swapchainImage;
355 imageViewInfo.viewType = VK_IMAGE_VIEW_TYPE_2D;
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500356 imageViewInfo.format = nativeFormat;
Jamie Madill4d0bf552016-12-28 15:45:24 -0500357 imageViewInfo.components.r = VK_COMPONENT_SWIZZLE_R;
358 imageViewInfo.components.g = VK_COMPONENT_SWIZZLE_G;
359 imageViewInfo.components.b = VK_COMPONENT_SWIZZLE_B;
360 imageViewInfo.components.a = VK_COMPONENT_SWIZZLE_A;
361 imageViewInfo.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
362 imageViewInfo.subresourceRange.baseMipLevel = 0;
363 imageViewInfo.subresourceRange.levelCount = 1;
364 imageViewInfo.subresourceRange.baseArrayLayer = 0;
365 imageViewInfo.subresourceRange.layerCount = 1;
366
367 vk::Image image(swapchainImage);
Jamie Madill5deea722017-02-16 10:44:46 -0500368 vk::ImageView imageView;
369 ANGLE_TRY(imageView.init(device, imageViewInfo));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500370
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500371 // Set transfer dest layout, and clear the image to black.
372 image.changeLayoutTop(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
373 commandBuffer);
374 commandBuffer->clearSingleColorImage(image, transparentBlack);
Jamie Madill4d0bf552016-12-28 15:45:24 -0500375
Jamie Madill5deea722017-02-16 10:44:46 -0500376 mSwapchainImages[imageIndex].retain(device, std::move(image));
377 mSwapchainImageViews[imageIndex].retain(device, std::move(imageView));
Jamie Madill4d0bf552016-12-28 15:45:24 -0500378 }
379
380 ANGLE_TRY(commandBuffer->end());
381 ANGLE_TRY(renderer->submitAndFinishCommandBuffer(*commandBuffer));
382
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500383 // Start by getting the next available swapchain image.
384 ANGLE_TRY(nextSwapchainImage(renderer));
385
Jamie Madill4d0bf552016-12-28 15:45:24 -0500386 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500387}
388
389FramebufferImpl *WindowSurfaceVk::createDefaultFramebuffer(const gl::FramebufferState &state)
390{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500391 return FramebufferVk::CreateDefaultFBO(state, this);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500392}
393
394egl::Error WindowSurfaceVk::swap(const DisplayImpl *displayImpl)
395{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500396 const DisplayVk *displayVk = GetAs<DisplayVk>(displayImpl);
397 return swapImpl(displayVk->getRenderer()).toEGL(EGL_BAD_ALLOC);
398}
399
400vk::Error WindowSurfaceVk::swapImpl(RendererVk *renderer)
401{
402 vk::CommandBuffer *currentCB = renderer->getCommandBuffer();
403
404 auto *image = &mSwapchainImages[mCurrentSwapchainImageIndex];
405
Jamie Madill5deea722017-02-16 10:44:46 -0500406 currentCB->begin(renderer->getDevice());
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500407 image->changeLayoutWithStages(VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
408 VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
409 VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, currentCB);
410 currentCB->end();
411
412 ANGLE_TRY(renderer->waitThenFinishCommandBuffer(*currentCB, mPresentCompleteSemaphore));
413
414 VkPresentInfoKHR presentInfo;
415 presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
416 presentInfo.pNext = nullptr;
417 presentInfo.waitSemaphoreCount = 0;
418 presentInfo.pWaitSemaphores = nullptr;
419 presentInfo.swapchainCount = 1;
420 presentInfo.pSwapchains = &mSwapchain;
421 presentInfo.pImageIndices = &mCurrentSwapchainImageIndex;
422 presentInfo.pResults = nullptr;
423
424 ANGLE_VK_TRY(vkQueuePresentKHR(renderer->getQueue(), &presentInfo));
425
426 // Get the next available swapchain iamge.
427 ANGLE_TRY(nextSwapchainImage(renderer));
428
429 return vk::NoError();
430}
431
432vk::Error WindowSurfaceVk::nextSwapchainImage(RendererVk *renderer)
433{
434 VkDevice device = renderer->getDevice();
435
Jamie Madill5deea722017-02-16 10:44:46 -0500436 vk::Semaphore presentComplete;
437 ANGLE_TRY(presentComplete.init(device));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500438
439 ANGLE_VK_TRY(vkAcquireNextImageKHR(device, mSwapchain, std::numeric_limits<uint64_t>::max(),
440 presentComplete.getHandle(), VK_NULL_HANDLE,
441 &mCurrentSwapchainImageIndex));
442
Jamie Madill5deea722017-02-16 10:44:46 -0500443 mPresentCompleteSemaphore.retain(device, std::move(presentComplete));
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500444
445 // Update RenderTarget pointers.
446 mRenderTarget.image = &mSwapchainImages[mCurrentSwapchainImageIndex];
447 mRenderTarget.imageView = &mSwapchainImageViews[mCurrentSwapchainImageIndex];
448
449 return vk::NoError();
Jamie Madille09bd5d2016-11-29 16:20:35 -0500450}
451
452egl::Error WindowSurfaceVk::postSubBuffer(EGLint x, EGLint y, EGLint width, EGLint height)
453{
454 // TODO(jmadill)
455 return egl::Error(EGL_SUCCESS);
456}
457
458egl::Error WindowSurfaceVk::querySurfacePointerANGLE(EGLint attribute, void **value)
459{
460 UNREACHABLE();
461 return egl::Error(EGL_BAD_CURRENT_SURFACE);
462}
463
464egl::Error WindowSurfaceVk::bindTexImage(gl::Texture *texture, EGLint buffer)
465{
466 return egl::Error(EGL_SUCCESS);
467}
468
469egl::Error WindowSurfaceVk::releaseTexImage(EGLint buffer)
470{
471 return egl::Error(EGL_SUCCESS);
472}
473
474void WindowSurfaceVk::setSwapInterval(EGLint interval)
475{
476}
477
478EGLint WindowSurfaceVk::getWidth() const
479{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500480 return static_cast<EGLint>(mRenderTarget.extents.width);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500481}
482
483EGLint WindowSurfaceVk::getHeight() const
484{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500485 return static_cast<EGLint>(mRenderTarget.extents.height);
Jamie Madille09bd5d2016-11-29 16:20:35 -0500486}
487
488EGLint WindowSurfaceVk::isPostSubBufferSupported() const
489{
490 // TODO(jmadill)
491 return EGL_FALSE;
492}
493
494EGLint WindowSurfaceVk::getSwapBehavior() const
495{
496 // TODO(jmadill)
497 return EGL_BUFFER_DESTROYED;
498}
499
500gl::Error WindowSurfaceVk::getAttachmentRenderTarget(
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500501 const gl::FramebufferAttachment::Target & /*target*/,
Jamie Madille09bd5d2016-11-29 16:20:35 -0500502 FramebufferAttachmentRenderTarget **rtOut)
503{
Jamie Madill7b57b9d2017-01-13 09:33:38 -0500504 *rtOut = &mRenderTarget;
505 return gl::NoError();
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400506}
507
Jamie Madillab9f9c32017-01-17 17:47:34 -0500508gl::ErrorOrResult<vk::Framebuffer *> WindowSurfaceVk::getCurrentFramebuffer(
509 VkDevice device,
510 const vk::RenderPass &compatibleRenderPass)
511{
512 if (!mSwapchainFramebuffers.empty())
513 {
514 // Validation layers should detect if the render pass is really compatible.
515 return &mSwapchainFramebuffers[mCurrentSwapchainImageIndex];
516 }
517
518 VkFramebufferCreateInfo framebufferInfo;
519
520 // TODO(jmadill): Depth/Stencil attachments.
521 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
522 framebufferInfo.pNext = nullptr;
523 framebufferInfo.flags = 0;
524 framebufferInfo.renderPass = compatibleRenderPass.getHandle();
525 framebufferInfo.attachmentCount = 1u;
526 framebufferInfo.pAttachments = nullptr;
527 framebufferInfo.width = static_cast<uint32_t>(mRenderTarget.extents.width);
528 framebufferInfo.height = static_cast<uint32_t>(mRenderTarget.extents.height);
529 framebufferInfo.layers = 1;
530
Jamie Madill5deea722017-02-16 10:44:46 -0500531 mSwapchainFramebuffers.resize(mSwapchainImageViews.size());
532 for (size_t imageIndex = 0; imageIndex < mSwapchainFramebuffers.size(); ++imageIndex)
Jamie Madillab9f9c32017-01-17 17:47:34 -0500533 {
Jamie Madill5deea722017-02-16 10:44:46 -0500534 const auto &imageView = mSwapchainImageViews[imageIndex];
Jamie Madillab9f9c32017-01-17 17:47:34 -0500535 VkImageView imageViewHandle = imageView.getHandle();
536 framebufferInfo.pAttachments = &imageViewHandle;
537
Jamie Madill5deea722017-02-16 10:44:46 -0500538 vk::Framebuffer framebuffer;
539 ANGLE_TRY(framebuffer.init(device, framebufferInfo));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500540
Jamie Madill5deea722017-02-16 10:44:46 -0500541 mSwapchainFramebuffers[imageIndex].retain(device, std::move(framebuffer));
Jamie Madillab9f9c32017-01-17 17:47:34 -0500542 }
543
544 // We should only initialize framebuffers on the first swap.
545 ASSERT(mCurrentSwapchainImageIndex == 0u);
546 return &mSwapchainFramebuffers[mCurrentSwapchainImageIndex];
547}
548
Jamie Madill9e54b5a2016-05-25 12:57:39 -0400549} // namespace rx