blob: 69dc2c568493161e69f2efcddb88579ad96a6aa1 [file] [log] [blame]
jvanverth9f372462016-04-06 06:08:59 -07001
2/*
3 * Copyright 2015 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
8
Greg Daniele79b4732017-04-20 14:07:46 -04009#include "GrBackendSurface.h"
jvanverth9f372462016-04-06 06:08:59 -070010#include "GrContext.h"
Hal Canary95e3c052017-01-11 12:44:43 -050011#include "SkAutoMalloc.h"
jvanverth9f372462016-04-06 06:08:59 -070012#include "SkSurface.h"
jvanvertha8d0d6c2016-05-05 12:32:03 -070013#include "VulkanWindowContext.h"
jvanverth9f372462016-04-06 06:08:59 -070014
Greg Daniel98bffae2018-08-01 13:25:41 -040015#include "vk/GrVkExtensions.h"
Greg Daniel6ddbafc2018-05-24 12:34:29 -040016#include "vk/GrVkImage.h"
jvanverth9f372462016-04-06 06:08:59 -070017#include "vk/GrVkTypes.h"
Greg Daniela31f4e52018-08-01 16:48:52 -040018#include "vk/GrVkUtil.h"
jvanverth9f372462016-04-06 06:08:59 -070019
20#ifdef VK_USE_PLATFORM_WIN32_KHR
21// windows wants to define this as CreateSemaphoreA or CreateSemaphoreW
22#undef CreateSemaphore
23#endif
24
Greg Danielf730c182018-07-02 20:15:37 +000025#define GET_PROC(F) f ## F = (PFN_vk ## F) fGetInstanceProcAddr(fInstance, "vk" #F)
26#define GET_DEV_PROC(F) f ## F = (PFN_vk ## F) fGetDeviceProcAddr(fDevice, "vk" #F)
jvanverthb0d43522016-04-21 11:46:23 -070027
jvanvertha8d0d6c2016-05-05 12:32:03 -070028namespace sk_app {
29
bsalomond1bdd1f2016-07-26 12:02:50 -070030VulkanWindowContext::VulkanWindowContext(const DisplayParams& params,
31 CreateVkSurfaceFn createVkSurface,
Greg Daniel35970ec2017-11-10 10:03:05 -050032 CanPresentFn canPresent,
33 PFN_vkGetInstanceProcAddr instProc,
34 PFN_vkGetDeviceProcAddr devProc)
Jim Van Verthfbdc0802017-05-02 16:15:53 -040035 : WindowContext(params)
36 , fCreateVkSurfaceFn(createVkSurface)
37 , fCanPresentFn(canPresent)
jvanverthaf236b52016-05-20 06:01:06 -070038 , fSurface(VK_NULL_HANDLE)
jvanvertha8d0d6c2016-05-05 12:32:03 -070039 , fSwapchain(VK_NULL_HANDLE)
jvanverthaf236b52016-05-20 06:01:06 -070040 , fImages(nullptr)
41 , fImageLayouts(nullptr)
42 , fSurfaces(nullptr)
jvanvertha8d0d6c2016-05-05 12:32:03 -070043 , fCommandPool(VK_NULL_HANDLE)
44 , fBackbuffers(nullptr) {
Greg Daniel35970ec2017-11-10 10:03:05 -050045 fGetInstanceProcAddr = instProc;
46 fGetDeviceProcAddr = devProc;
Jim Van Verthfbdc0802017-05-02 16:15:53 -040047 this->initializeContext();
48}
jvanverth9f372462016-04-06 06:08:59 -070049
Jim Van Verthfbdc0802017-05-02 16:15:53 -040050void VulkanWindowContext::initializeContext() {
jvanverth9f372462016-04-06 06:08:59 -070051 // any config code here (particularly for msaa)?
jvanverth9f372462016-04-06 06:08:59 -070052
Greg Danield3e65aa2018-08-01 09:19:45 -040053 PFN_vkGetInstanceProcAddr getInstanceProc = fGetInstanceProcAddr;
54 PFN_vkGetDeviceProcAddr getDeviceProc = fGetDeviceProcAddr;
55 auto getProc = [getInstanceProc, getDeviceProc](const char* proc_name,
56 VkInstance instance, VkDevice device) {
57 if (device != VK_NULL_HANDLE) {
58 return getDeviceProc(device, proc_name);
59 }
60 return getInstanceProc(instance, proc_name);
61 };
Greg Danielf730c182018-07-02 20:15:37 +000062 GrVkBackendContext backendContext;
Greg Daniel98bffae2018-08-01 13:25:41 -040063 GrVkExtensions extensions;
Greg Daniela0651ac2018-08-08 09:23:18 -040064 VkPhysicalDeviceFeatures2 features;
65 if (!sk_gpu_test::CreateVkBackendContext(getProc, &backendContext, &extensions, &features,
66 &fDebugCallback, &fPresentQueueIndex, fCanPresentFn)) {
67 sk_gpu_test::FreeVulkanFeaturesStructs(&features);
jvanverthb0d43522016-04-21 11:46:23 -070068 return;
69 }
70
Greg Daniela31f4e52018-08-01 16:48:52 -040071 if (!extensions.hasExtension(VK_KHR_SURFACE_EXTENSION_NAME, 25) ||
72 !extensions.hasExtension(VK_KHR_SWAPCHAIN_EXTENSION_NAME, 68)) {
Greg Daniela0651ac2018-08-08 09:23:18 -040073 sk_gpu_test::FreeVulkanFeaturesStructs(&features);
Greg Danielf730c182018-07-02 20:15:37 +000074 return;
75 }
76
77 fInstance = backendContext.fInstance;
78 fPhysicalDevice = backendContext.fPhysicalDevice;
79 fDevice = backendContext.fDevice;
80 fGraphicsQueueIndex = backendContext.fGraphicsQueueIndex;
81 fGraphicsQueue = backendContext.fQueue;
Greg Danielc0b03d82018-08-03 14:41:15 -040082
83 PFN_vkGetPhysicalDeviceProperties localGetPhysicalDeviceProperties =
84 reinterpret_cast<PFN_vkGetPhysicalDeviceProperties>(
85 backendContext.fGetProc("vkGetPhysicalDeviceProperties",
86 backendContext.fInstance,
87 VK_NULL_HANDLE));
88 if (!localGetPhysicalDeviceProperties) {
Greg Daniela0651ac2018-08-08 09:23:18 -040089 sk_gpu_test::FreeVulkanFeaturesStructs(&features);
Greg Danielc0b03d82018-08-03 14:41:15 -040090 return;
91 }
92 VkPhysicalDeviceProperties physDeviceProperties;
93 localGetPhysicalDeviceProperties(backendContext.fPhysicalDevice, &physDeviceProperties);
94 uint32_t physDevVersion = physDeviceProperties.apiVersion;
95
Greg Danielc8cd45a2018-07-12 10:02:37 -040096 fInterface.reset(new GrVkInterface(backendContext.fGetProc, fInstance, fDevice,
Greg Danielc0b03d82018-08-03 14:41:15 -040097 backendContext.fInstanceVersion, physDevVersion,
Greg Daniel98bffae2018-08-01 13:25:41 -040098 &extensions));
Greg Danielf730c182018-07-02 20:15:37 +000099
100 GET_PROC(DestroyInstance);
Greg Daniela31f4e52018-08-01 16:48:52 -0400101 if (fDebugCallback != VK_NULL_HANDLE) {
102 GET_PROC(DestroyDebugReportCallbackEXT);
103 }
jvanverthb0d43522016-04-21 11:46:23 -0700104 GET_PROC(DestroySurfaceKHR);
105 GET_PROC(GetPhysicalDeviceSurfaceSupportKHR);
106 GET_PROC(GetPhysicalDeviceSurfaceCapabilitiesKHR);
107 GET_PROC(GetPhysicalDeviceSurfaceFormatsKHR);
108 GET_PROC(GetPhysicalDeviceSurfacePresentModesKHR);
Greg Danielf730c182018-07-02 20:15:37 +0000109 GET_DEV_PROC(DeviceWaitIdle);
110 GET_DEV_PROC(QueueWaitIdle);
111 GET_DEV_PROC(DestroyDevice);
jvanverthb0d43522016-04-21 11:46:23 -0700112 GET_DEV_PROC(CreateSwapchainKHR);
113 GET_DEV_PROC(DestroySwapchainKHR);
114 GET_DEV_PROC(GetSwapchainImagesKHR);
115 GET_DEV_PROC(AcquireNextImageKHR);
116 GET_DEV_PROC(QueuePresentKHR);
Greg Daniel35970ec2017-11-10 10:03:05 -0500117 GET_DEV_PROC(GetDeviceQueue);
jvanverth9f372462016-04-06 06:08:59 -0700118
Greg Danielf730c182018-07-02 20:15:37 +0000119 fContext = GrContext::MakeVulkan(backendContext, fDisplayParams.fGrContextOptions);
jvanverth9f372462016-04-06 06:08:59 -0700120
Greg Danielf730c182018-07-02 20:15:37 +0000121 fSurface = fCreateVkSurfaceFn(fInstance);
jvanverth9f372462016-04-06 06:08:59 -0700122 if (VK_NULL_HANDLE == fSurface) {
Greg Danielf730c182018-07-02 20:15:37 +0000123 this->destroyContext();
Greg Daniela0651ac2018-08-08 09:23:18 -0400124 sk_gpu_test::FreeVulkanFeaturesStructs(&features);
jvanverth9f372462016-04-06 06:08:59 -0700125 return;
126 }
127
jvanverth9f372462016-04-06 06:08:59 -0700128 VkBool32 supported;
Greg Danielf730c182018-07-02 20:15:37 +0000129 VkResult res = fGetPhysicalDeviceSurfaceSupportKHR(fPhysicalDevice, fPresentQueueIndex,
130 fSurface, &supported);
jvanverth9f372462016-04-06 06:08:59 -0700131 if (VK_SUCCESS != res) {
132 this->destroyContext();
Greg Daniela0651ac2018-08-08 09:23:18 -0400133 sk_gpu_test::FreeVulkanFeaturesStructs(&features);
jvanverth9f372462016-04-06 06:08:59 -0700134 return;
135 }
136
Jim Van Verthfbdc0802017-05-02 16:15:53 -0400137 if (!this->createSwapchain(-1, -1, fDisplayParams)) {
jvanverth9f372462016-04-06 06:08:59 -0700138 this->destroyContext();
Greg Daniela0651ac2018-08-08 09:23:18 -0400139 sk_gpu_test::FreeVulkanFeaturesStructs(&features);
jvanverth9f372462016-04-06 06:08:59 -0700140 return;
141 }
142
143 // create presentQueue
Greg Danielf730c182018-07-02 20:15:37 +0000144 fGetDeviceQueue(fDevice, fPresentQueueIndex, 0, &fPresentQueue);
Greg Daniela0651ac2018-08-08 09:23:18 -0400145 sk_gpu_test::FreeVulkanFeaturesStructs(&features);
jvanverth9f372462016-04-06 06:08:59 -0700146}
147
bsalomonccde4ab2016-07-27 08:50:12 -0700148bool VulkanWindowContext::createSwapchain(int width, int height,
brianosman05de2162016-05-06 13:28:57 -0700149 const DisplayParams& params) {
jvanverth9f372462016-04-06 06:08:59 -0700150 // check for capabilities
151 VkSurfaceCapabilitiesKHR caps;
Greg Danielf730c182018-07-02 20:15:37 +0000152 VkResult res = fGetPhysicalDeviceSurfaceCapabilitiesKHR(fPhysicalDevice, fSurface, &caps);
jvanverth9f372462016-04-06 06:08:59 -0700153 if (VK_SUCCESS != res) {
154 return false;
155 }
156
157 uint32_t surfaceFormatCount;
Greg Danielf730c182018-07-02 20:15:37 +0000158 res = fGetPhysicalDeviceSurfaceFormatsKHR(fPhysicalDevice, fSurface, &surfaceFormatCount,
159 nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700160 if (VK_SUCCESS != res) {
161 return false;
162 }
163
164 SkAutoMalloc surfaceFormatAlloc(surfaceFormatCount * sizeof(VkSurfaceFormatKHR));
165 VkSurfaceFormatKHR* surfaceFormats = (VkSurfaceFormatKHR*)surfaceFormatAlloc.get();
Greg Danielf730c182018-07-02 20:15:37 +0000166 res = fGetPhysicalDeviceSurfaceFormatsKHR(fPhysicalDevice, fSurface, &surfaceFormatCount,
167 surfaceFormats);
jvanverth9f372462016-04-06 06:08:59 -0700168 if (VK_SUCCESS != res) {
169 return false;
170 }
171
172 uint32_t presentModeCount;
Greg Danielf730c182018-07-02 20:15:37 +0000173 res = fGetPhysicalDeviceSurfacePresentModesKHR(fPhysicalDevice, fSurface, &presentModeCount,
174 nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700175 if (VK_SUCCESS != res) {
176 return false;
177 }
178
179 SkAutoMalloc presentModeAlloc(presentModeCount * sizeof(VkPresentModeKHR));
180 VkPresentModeKHR* presentModes = (VkPresentModeKHR*)presentModeAlloc.get();
Greg Danielf730c182018-07-02 20:15:37 +0000181 res = fGetPhysicalDeviceSurfacePresentModesKHR(fPhysicalDevice, fSurface, &presentModeCount,
182 presentModes);
jvanverth9f372462016-04-06 06:08:59 -0700183 if (VK_SUCCESS != res) {
184 return false;
185 }
186
187 VkExtent2D extent = caps.currentExtent;
188 // use the hints
189 if (extent.width == (uint32_t)-1) {
190 extent.width = width;
191 extent.height = height;
192 }
193
jvanverth9fab59d2016-04-06 12:08:51 -0700194 // clamp width; to protect us from broken hints
jvanverth9f372462016-04-06 06:08:59 -0700195 if (extent.width < caps.minImageExtent.width) {
196 extent.width = caps.minImageExtent.width;
197 } else if (extent.width > caps.maxImageExtent.width) {
198 extent.width = caps.maxImageExtent.width;
199 }
200 // clamp height
201 if (extent.height < caps.minImageExtent.height) {
202 extent.height = caps.minImageExtent.height;
203 } else if (extent.height > caps.maxImageExtent.height) {
204 extent.height = caps.maxImageExtent.height;
205 }
jvanverth1d155962016-05-23 13:13:36 -0700206
jvanverth9f372462016-04-06 06:08:59 -0700207 fWidth = (int)extent.width;
208 fHeight = (int)extent.height;
209
210 uint32_t imageCount = caps.minImageCount + 2;
211 if (caps.maxImageCount > 0 && imageCount > caps.maxImageCount) {
212 // Application must settle for fewer images than desired:
213 imageCount = caps.maxImageCount;
214 }
215
216 VkImageUsageFlags usageFlags = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
217 VK_IMAGE_USAGE_TRANSFER_SRC_BIT |
218 VK_IMAGE_USAGE_TRANSFER_DST_BIT;
219 SkASSERT((caps.supportedUsageFlags & usageFlags) == usageFlags);
220 SkASSERT(caps.supportedTransforms & caps.currentTransform);
221 SkASSERT(caps.supportedCompositeAlpha & (VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR |
222 VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR));
223 VkCompositeAlphaFlagBitsKHR composite_alpha =
224 (caps.supportedCompositeAlpha & VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR) ?
225 VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR :
226 VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
227
Brian Osman294fe292018-07-09 10:23:19 -0400228 // Pick our surface format.
brianosman05de2162016-05-06 13:28:57 -0700229 VkFormat surfaceFormat = VK_FORMAT_UNDEFINED;
230 VkColorSpaceKHR colorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
brianosman05de2162016-05-06 13:28:57 -0700231 for (uint32_t i = 0; i < surfaceFormatCount; ++i) {
Greg Daniel81b80592017-12-13 10:20:04 -0500232 VkFormat localFormat = surfaceFormats[i].format;
Brian Osman2b23c4b2018-06-01 12:25:08 -0400233 if (GrVkFormatIsSupported(localFormat)) {
Greg Daniel81b80592017-12-13 10:20:04 -0500234 surfaceFormat = localFormat;
brianosman05de2162016-05-06 13:28:57 -0700235 colorSpace = surfaceFormats[i].colorSpace;
236 break;
237 }
238 }
239 fDisplayParams = params;
csmartdalton578f0642017-02-24 16:04:47 -0700240 fSampleCount = params.fMSAASampleCount;
241 fStencilBits = 8;
brianosman05de2162016-05-06 13:28:57 -0700242
243 if (VK_FORMAT_UNDEFINED == surfaceFormat) {
244 return false;
245 }
jvanvertha4b0fed2016-04-27 11:42:21 -0700246
Greg Danielfaa095e2017-12-19 13:15:02 -0500247 SkColorType colorType;
248 switch (surfaceFormat) {
249 case VK_FORMAT_R8G8B8A8_UNORM: // fall through
250 case VK_FORMAT_R8G8B8A8_SRGB:
251 colorType = kRGBA_8888_SkColorType;
252 break;
253 case VK_FORMAT_B8G8R8A8_UNORM: // fall through
254 case VK_FORMAT_B8G8R8A8_SRGB:
255 colorType = kBGRA_8888_SkColorType;
256 break;
257 default:
258 return false;
259 }
260
jvanverth3d6ed3a2016-04-07 11:09:51 -0700261 // If mailbox mode is available, use it, as it is the lowest-latency non-
262 // tearing mode. If not, fall back to FIFO which is always available.
jvanverth9f372462016-04-06 06:08:59 -0700263 VkPresentModeKHR mode = VK_PRESENT_MODE_FIFO_KHR;
jvanverth9f372462016-04-06 06:08:59 -0700264 for (uint32_t i = 0; i < presentModeCount; ++i) {
jvanverth3d6ed3a2016-04-07 11:09:51 -0700265 // use mailbox
266 if (VK_PRESENT_MODE_MAILBOX_KHR == presentModes[i]) {
jvanverth9f372462016-04-06 06:08:59 -0700267 mode = presentModes[i];
jvanverth3d6ed3a2016-04-07 11:09:51 -0700268 break;
jvanverth9f372462016-04-06 06:08:59 -0700269 }
270 }
271
272 VkSwapchainCreateInfoKHR swapchainCreateInfo;
273 memset(&swapchainCreateInfo, 0, sizeof(VkSwapchainCreateInfoKHR));
274 swapchainCreateInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
275 swapchainCreateInfo.surface = fSurface;
276 swapchainCreateInfo.minImageCount = imageCount;
jvanvertha4b0fed2016-04-27 11:42:21 -0700277 swapchainCreateInfo.imageFormat = surfaceFormat;
278 swapchainCreateInfo.imageColorSpace = colorSpace;
jvanverth9f372462016-04-06 06:08:59 -0700279 swapchainCreateInfo.imageExtent = extent;
280 swapchainCreateInfo.imageArrayLayers = 1;
281 swapchainCreateInfo.imageUsage = usageFlags;
282
Greg Danielf730c182018-07-02 20:15:37 +0000283 uint32_t queueFamilies[] = { fGraphicsQueueIndex, fPresentQueueIndex };
284 if (fGraphicsQueueIndex != fPresentQueueIndex) {
jvanverth9f372462016-04-06 06:08:59 -0700285 swapchainCreateInfo.imageSharingMode = VK_SHARING_MODE_CONCURRENT;
286 swapchainCreateInfo.queueFamilyIndexCount = 2;
287 swapchainCreateInfo.pQueueFamilyIndices = queueFamilies;
288 } else {
289 swapchainCreateInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
290 swapchainCreateInfo.queueFamilyIndexCount = 0;
291 swapchainCreateInfo.pQueueFamilyIndices = nullptr;
292 }
293
Greg Daniele897b972016-10-06 13:57:57 -0400294 swapchainCreateInfo.preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
jvanverth9f372462016-04-06 06:08:59 -0700295 swapchainCreateInfo.compositeAlpha = composite_alpha;
296 swapchainCreateInfo.presentMode = mode;
297 swapchainCreateInfo.clipped = true;
298 swapchainCreateInfo.oldSwapchain = fSwapchain;
299
Greg Danielf730c182018-07-02 20:15:37 +0000300 res = fCreateSwapchainKHR(fDevice, &swapchainCreateInfo, nullptr, &fSwapchain);
jvanverth9f372462016-04-06 06:08:59 -0700301 if (VK_SUCCESS != res) {
302 return false;
303 }
304
305 // destroy the old swapchain
306 if (swapchainCreateInfo.oldSwapchain != VK_NULL_HANDLE) {
Greg Danielf730c182018-07-02 20:15:37 +0000307 fDeviceWaitIdle(fDevice);
jvanverth9f372462016-04-06 06:08:59 -0700308
309 this->destroyBuffers();
310
Greg Danielf730c182018-07-02 20:15:37 +0000311 fDestroySwapchainKHR(fDevice, swapchainCreateInfo.oldSwapchain, nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700312 }
313
Greg Danielfaa095e2017-12-19 13:15:02 -0500314 this->createBuffers(swapchainCreateInfo.imageFormat, colorType);
jvanverth9f372462016-04-06 06:08:59 -0700315
316 return true;
317}
318
Greg Danielfaa095e2017-12-19 13:15:02 -0500319void VulkanWindowContext::createBuffers(VkFormat format, SkColorType colorType) {
Greg Danielf730c182018-07-02 20:15:37 +0000320 fGetSwapchainImagesKHR(fDevice, fSwapchain, &fImageCount, nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700321 SkASSERT(fImageCount);
322 fImages = new VkImage[fImageCount];
Greg Danielf730c182018-07-02 20:15:37 +0000323 fGetSwapchainImagesKHR(fDevice, fSwapchain, &fImageCount, fImages);
jvanverth9f372462016-04-06 06:08:59 -0700324
325 // set up initial image layouts and create surfaces
326 fImageLayouts = new VkImageLayout[fImageCount];
327 fSurfaces = new sk_sp<SkSurface>[fImageCount];
328 for (uint32_t i = 0; i < fImageCount; ++i) {
329 fImageLayouts[i] = VK_IMAGE_LAYOUT_UNDEFINED;
330
egdanielb2df0c22016-05-13 11:30:37 -0700331 GrVkImageInfo info;
jvanverth9f372462016-04-06 06:08:59 -0700332 info.fImage = fImages[i];
Greg Daniel8385a8a2018-02-26 13:29:37 -0500333 info.fAlloc = GrVkAlloc();
egdaniel580fa592016-08-31 11:03:50 -0700334 info.fImageLayout = VK_IMAGE_LAYOUT_UNDEFINED;
jvanverth9f372462016-04-06 06:08:59 -0700335 info.fImageTiling = VK_IMAGE_TILING_OPTIMAL;
egdaniel58a8d922016-04-21 08:03:10 -0700336 info.fFormat = format;
jvanverth3622a172016-05-10 06:42:18 -0700337 info.fLevelCount = 1;
jvanverthaf236b52016-05-20 06:01:06 -0700338
Brian Salomonafdc6b12018-03-09 12:02:32 -0500339 GrBackendRenderTarget backendRT(fWidth, fHeight, fSampleCount, info);
Greg Daniele79b4732017-04-20 14:07:46 -0400340
Brian Salomonafdc6b12018-03-09 12:02:32 -0500341 fSurfaces[i] = SkSurface::MakeFromBackendRenderTarget(fContext.get(),
342 backendRT,
343 kTopLeft_GrSurfaceOrigin,
344 colorType,
345 fDisplayParams.fColorSpace,
Ben Wagner37c54032018-04-13 14:30:23 -0400346 &fDisplayParams.fSurfaceProps);
jvanverth9f372462016-04-06 06:08:59 -0700347 }
348
349 // create the command pool for the command buffers
350 if (VK_NULL_HANDLE == fCommandPool) {
351 VkCommandPoolCreateInfo commandPoolInfo;
352 memset(&commandPoolInfo, 0, sizeof(VkCommandPoolCreateInfo));
353 commandPoolInfo.sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO;
354 // this needs to be on the render queue
Greg Danielf730c182018-07-02 20:15:37 +0000355 commandPoolInfo.queueFamilyIndex = fGraphicsQueueIndex;
jvanverth9f372462016-04-06 06:08:59 -0700356 commandPoolInfo.flags = VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT;
Greg Danielf730c182018-07-02 20:15:37 +0000357 GR_VK_CALL_ERRCHECK(fInterface,
358 CreateCommandPool(fDevice, &commandPoolInfo,
jvanverth9f372462016-04-06 06:08:59 -0700359 nullptr, &fCommandPool));
360 }
361
362 // set up the backbuffers
363 VkSemaphoreCreateInfo semaphoreInfo;
364 memset(&semaphoreInfo, 0, sizeof(VkSemaphoreCreateInfo));
365 semaphoreInfo.sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO;
366 semaphoreInfo.pNext = nullptr;
367 semaphoreInfo.flags = 0;
368 VkCommandBufferAllocateInfo commandBuffersInfo;
369 memset(&commandBuffersInfo, 0, sizeof(VkCommandBufferAllocateInfo));
370 commandBuffersInfo.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO;
371 commandBuffersInfo.pNext = nullptr;
372 commandBuffersInfo.commandPool = fCommandPool;
373 commandBuffersInfo.level = VK_COMMAND_BUFFER_LEVEL_PRIMARY;
374 commandBuffersInfo.commandBufferCount = 2;
375 VkFenceCreateInfo fenceInfo;
376 memset(&fenceInfo, 0, sizeof(VkFenceCreateInfo));
377 fenceInfo.sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO;
378 fenceInfo.pNext = nullptr;
379 fenceInfo.flags = VK_FENCE_CREATE_SIGNALED_BIT;
380
Greg Daniel1f05f442016-10-27 16:37:17 -0400381 // we create one additional backbuffer structure here, because we want to
jvanverth9f372462016-04-06 06:08:59 -0700382 // give the command buffers they contain a chance to finish before we cycle back
383 fBackbuffers = new BackbufferInfo[fImageCount + 1];
384 for (uint32_t i = 0; i < fImageCount + 1; ++i) {
385 fBackbuffers[i].fImageIndex = -1;
Greg Danielf730c182018-07-02 20:15:37 +0000386 GR_VK_CALL_ERRCHECK(fInterface,
387 CreateSemaphore(fDevice, &semaphoreInfo,
jvanverth9f372462016-04-06 06:08:59 -0700388 nullptr, &fBackbuffers[i].fAcquireSemaphore));
Greg Danielf730c182018-07-02 20:15:37 +0000389 GR_VK_CALL_ERRCHECK(fInterface,
390 CreateSemaphore(fDevice, &semaphoreInfo,
jvanverth9f372462016-04-06 06:08:59 -0700391 nullptr, &fBackbuffers[i].fRenderSemaphore));
Greg Danielf730c182018-07-02 20:15:37 +0000392 GR_VK_CALL_ERRCHECK(fInterface,
393 AllocateCommandBuffers(fDevice, &commandBuffersInfo,
jvanverth9f372462016-04-06 06:08:59 -0700394 fBackbuffers[i].fTransitionCmdBuffers));
Greg Danielf730c182018-07-02 20:15:37 +0000395 GR_VK_CALL_ERRCHECK(fInterface,
396 CreateFence(fDevice, &fenceInfo, nullptr,
jvanverth9f372462016-04-06 06:08:59 -0700397 &fBackbuffers[i].fUsageFences[0]));
Greg Danielf730c182018-07-02 20:15:37 +0000398 GR_VK_CALL_ERRCHECK(fInterface,
399 CreateFence(fDevice, &fenceInfo, nullptr,
jvanverth9f372462016-04-06 06:08:59 -0700400 &fBackbuffers[i].fUsageFences[1]));
401 }
402 fCurrentBackbufferIndex = fImageCount;
403}
404
jvanvertha8d0d6c2016-05-05 12:32:03 -0700405void VulkanWindowContext::destroyBuffers() {
jvanverth9f372462016-04-06 06:08:59 -0700406
407 if (fBackbuffers) {
408 for (uint32_t i = 0; i < fImageCount + 1; ++i) {
Greg Danielf730c182018-07-02 20:15:37 +0000409 GR_VK_CALL_ERRCHECK(fInterface,
410 WaitForFences(fDevice, 2,
jvanverth9f372462016-04-06 06:08:59 -0700411 fBackbuffers[i].fUsageFences,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700412 true, UINT64_MAX));
jvanverth9f372462016-04-06 06:08:59 -0700413 fBackbuffers[i].fImageIndex = -1;
Greg Danielf730c182018-07-02 20:15:37 +0000414 GR_VK_CALL(fInterface,
415 DestroySemaphore(fDevice,
jvanverth9f372462016-04-06 06:08:59 -0700416 fBackbuffers[i].fAcquireSemaphore,
417 nullptr));
Greg Danielf730c182018-07-02 20:15:37 +0000418 GR_VK_CALL(fInterface,
419 DestroySemaphore(fDevice,
jvanverth9f372462016-04-06 06:08:59 -0700420 fBackbuffers[i].fRenderSemaphore,
421 nullptr));
Greg Danielf730c182018-07-02 20:15:37 +0000422 GR_VK_CALL(fInterface,
423 FreeCommandBuffers(fDevice, fCommandPool, 2,
jvanverth9f372462016-04-06 06:08:59 -0700424 fBackbuffers[i].fTransitionCmdBuffers));
Greg Danielf730c182018-07-02 20:15:37 +0000425 GR_VK_CALL(fInterface,
426 DestroyFence(fDevice, fBackbuffers[i].fUsageFences[0], 0));
427 GR_VK_CALL(fInterface,
428 DestroyFence(fDevice, fBackbuffers[i].fUsageFences[1], 0));
jvanverth9f372462016-04-06 06:08:59 -0700429 }
430 }
431
432 delete[] fBackbuffers;
433 fBackbuffers = nullptr;
434
jvanverthaf236b52016-05-20 06:01:06 -0700435 // Does this actually free the surfaces?
jvanverth9f372462016-04-06 06:08:59 -0700436 delete[] fSurfaces;
437 fSurfaces = nullptr;
438 delete[] fImageLayouts;
439 fImageLayouts = nullptr;
440 delete[] fImages;
441 fImages = nullptr;
442}
443
jvanvertha8d0d6c2016-05-05 12:32:03 -0700444VulkanWindowContext::~VulkanWindowContext() {
jvanverth9f372462016-04-06 06:08:59 -0700445 this->destroyContext();
446}
447
jvanvertha8d0d6c2016-05-05 12:32:03 -0700448void VulkanWindowContext::destroyContext() {
Greg Daniel37329b32018-07-02 20:16:44 +0000449 if (this->isValid()) {
450 fQueueWaitIdle(fPresentQueue);
451 fDeviceWaitIdle(fDevice);
jvanverth9f372462016-04-06 06:08:59 -0700452
Greg Daniel37329b32018-07-02 20:16:44 +0000453 this->destroyBuffers();
jvanverth9f372462016-04-06 06:08:59 -0700454
Greg Daniel37329b32018-07-02 20:16:44 +0000455 if (VK_NULL_HANDLE != fCommandPool) {
456 GR_VK_CALL(fInterface, DestroyCommandPool(fDevice, fCommandPool, nullptr));
457 fCommandPool = VK_NULL_HANDLE;
458 }
jvanverth9f372462016-04-06 06:08:59 -0700459
Greg Daniel37329b32018-07-02 20:16:44 +0000460 if (VK_NULL_HANDLE != fSwapchain) {
461 fDestroySwapchainKHR(fDevice, fSwapchain, nullptr);
462 fSwapchain = VK_NULL_HANDLE;
463 }
jvanverth9f372462016-04-06 06:08:59 -0700464
Greg Daniel37329b32018-07-02 20:16:44 +0000465 if (VK_NULL_HANDLE != fSurface) {
466 fDestroySurfaceKHR(fInstance, fSurface, nullptr);
467 fSurface = VK_NULL_HANDLE;
468 }
jvanverth9f372462016-04-06 06:08:59 -0700469 }
470
Greg Daniel02611d92017-07-25 10:05:01 -0400471 fContext.reset();
Greg Danielf730c182018-07-02 20:15:37 +0000472 fInterface.reset();
jvanverth9f372462016-04-06 06:08:59 -0700473
Greg Danielf730c182018-07-02 20:15:37 +0000474 if (VK_NULL_HANDLE != fDevice) {
475 fDestroyDevice(fDevice, nullptr);
476 fDevice = VK_NULL_HANDLE;
477 }
Greg Daniel37329b32018-07-02 20:16:44 +0000478
479#ifdef SK_ENABLE_VK_LAYERS
480 if (fDebugCallback != VK_NULL_HANDLE) {
Greg Daniela31f4e52018-08-01 16:48:52 -0400481 fDestroyDebugReportCallbackEXT(fInstance, fDebugCallback, nullptr);
Greg Daniel37329b32018-07-02 20:16:44 +0000482 }
483#endif
484
Greg Danielf730c182018-07-02 20:15:37 +0000485 fPhysicalDevice = VK_NULL_HANDLE;
486
487 if (VK_NULL_HANDLE != fInstance) {
488 fDestroyInstance(fInstance, nullptr);
489 fInstance = VK_NULL_HANDLE;
490 }
jvanverth9f372462016-04-06 06:08:59 -0700491}
492
jvanvertha8d0d6c2016-05-05 12:32:03 -0700493VulkanWindowContext::BackbufferInfo* VulkanWindowContext::getAvailableBackbuffer() {
jvanverth9f372462016-04-06 06:08:59 -0700494 SkASSERT(fBackbuffers);
495
496 ++fCurrentBackbufferIndex;
egdaniel804af7d2016-09-26 12:30:46 -0700497 if (fCurrentBackbufferIndex > fImageCount) {
jvanverth9f372462016-04-06 06:08:59 -0700498 fCurrentBackbufferIndex = 0;
499 }
500
501 BackbufferInfo* backbuffer = fBackbuffers + fCurrentBackbufferIndex;
Greg Danielf730c182018-07-02 20:15:37 +0000502 GR_VK_CALL_ERRCHECK(fInterface,
503 WaitForFences(fDevice, 2, backbuffer->fUsageFences,
jvanverth9f372462016-04-06 06:08:59 -0700504 true, UINT64_MAX));
505 return backbuffer;
506}
507
jvanverthaf236b52016-05-20 06:01:06 -0700508sk_sp<SkSurface> VulkanWindowContext::getBackbufferSurface() {
jvanverth9f372462016-04-06 06:08:59 -0700509 BackbufferInfo* backbuffer = this->getAvailableBackbuffer();
510 SkASSERT(backbuffer);
511
512 // reset the fence
Greg Danielf730c182018-07-02 20:15:37 +0000513 GR_VK_CALL_ERRCHECK(fInterface,
514 ResetFences(fDevice, 2, backbuffer->fUsageFences));
jvanverth9f372462016-04-06 06:08:59 -0700515 // semaphores should be in unsignaled state
516
517 // acquire the image
Greg Danielf730c182018-07-02 20:15:37 +0000518 VkResult res = fAcquireNextImageKHR(fDevice, fSwapchain, UINT64_MAX,
jvanverthb0d43522016-04-21 11:46:23 -0700519 backbuffer->fAcquireSemaphore, VK_NULL_HANDLE,
520 &backbuffer->fImageIndex);
jvanverth9f372462016-04-06 06:08:59 -0700521 if (VK_ERROR_SURFACE_LOST_KHR == res) {
522 // need to figure out how to create a new vkSurface without the platformData*
jvanverth3d6ed3a2016-04-07 11:09:51 -0700523 // maybe use attach somehow? but need a Window
jvanverth9f372462016-04-06 06:08:59 -0700524 return nullptr;
525 }
jvanverth3d6ed3a2016-04-07 11:09:51 -0700526 if (VK_ERROR_OUT_OF_DATE_KHR == res) {
jvanverth9f372462016-04-06 06:08:59 -0700527 // tear swapchain down and try again
Greg Daniel1f05f442016-10-27 16:37:17 -0400528 if (!this->createSwapchain(-1, -1, fDisplayParams)) {
jvanverth9f372462016-04-06 06:08:59 -0700529 return nullptr;
530 }
Jim Van Verthd63c1022017-01-05 13:50:49 -0500531 backbuffer = this->getAvailableBackbuffer();
Greg Danielf730c182018-07-02 20:15:37 +0000532 GR_VK_CALL_ERRCHECK(fInterface,
533 ResetFences(fDevice, 2, backbuffer->fUsageFences));
jvanverth9f372462016-04-06 06:08:59 -0700534
535 // acquire the image
Greg Danielf730c182018-07-02 20:15:37 +0000536 res = fAcquireNextImageKHR(fDevice, fSwapchain, UINT64_MAX,
jvanverthb0d43522016-04-21 11:46:23 -0700537 backbuffer->fAcquireSemaphore, VK_NULL_HANDLE,
538 &backbuffer->fImageIndex);
jvanverth9f372462016-04-06 06:08:59 -0700539
540 if (VK_SUCCESS != res) {
541 return nullptr;
542 }
543 }
544
545 // set up layout transfer from initial to color attachment
546 VkImageLayout layout = fImageLayouts[backbuffer->fImageIndex];
egdaniel580fa592016-08-31 11:03:50 -0700547 SkASSERT(VK_IMAGE_LAYOUT_UNDEFINED == layout || VK_IMAGE_LAYOUT_PRESENT_SRC_KHR == layout);
jvanverth9f372462016-04-06 06:08:59 -0700548 VkPipelineStageFlags srcStageMask = (VK_IMAGE_LAYOUT_UNDEFINED == layout) ?
549 VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT :
550 VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
551 VkPipelineStageFlags dstStageMask = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
jvanvertha8d0d6c2016-05-05 12:32:03 -0700552 VkAccessFlags srcAccessMask = (VK_IMAGE_LAYOUT_UNDEFINED == layout) ?
jvanverth9f372462016-04-06 06:08:59 -0700553 0 : VK_ACCESS_MEMORY_READ_BIT;
554 VkAccessFlags dstAccessMask = VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT;
555
556 VkImageMemoryBarrier imageMemoryBarrier = {
557 VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER, // sType
558 NULL, // pNext
559 srcAccessMask, // outputMask
560 dstAccessMask, // inputMask
561 layout, // oldLayout
562 VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, // newLayout
563 fPresentQueueIndex, // srcQueueFamilyIndex
Greg Danielf730c182018-07-02 20:15:37 +0000564 fGraphicsQueueIndex, // dstQueueFamilyIndex
jvanverth9f372462016-04-06 06:08:59 -0700565 fImages[backbuffer->fImageIndex], // image
566 { VK_IMAGE_ASPECT_COLOR_BIT, 0, 1, 0, 1 } // subresourceRange
567 };
Greg Danielf730c182018-07-02 20:15:37 +0000568 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700569 ResetCommandBuffer(backbuffer->fTransitionCmdBuffers[0], 0));
570 VkCommandBufferBeginInfo info;
571 memset(&info, 0, sizeof(VkCommandBufferBeginInfo));
572 info.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO;
573 info.flags = 0;
Greg Danielf730c182018-07-02 20:15:37 +0000574 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700575 BeginCommandBuffer(backbuffer->fTransitionCmdBuffers[0], &info));
576
Greg Danielf730c182018-07-02 20:15:37 +0000577 GR_VK_CALL(fInterface,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700578 CmdPipelineBarrier(backbuffer->fTransitionCmdBuffers[0],
579 srcStageMask, dstStageMask, 0,
580 0, nullptr,
581 0, nullptr,
582 1, &imageMemoryBarrier));
jvanverth9f372462016-04-06 06:08:59 -0700583
Greg Danielf730c182018-07-02 20:15:37 +0000584 GR_VK_CALL_ERRCHECK(fInterface,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700585 EndCommandBuffer(backbuffer->fTransitionCmdBuffers[0]));
jvanverth9f372462016-04-06 06:08:59 -0700586
egdaniel58a8d922016-04-21 08:03:10 -0700587 VkPipelineStageFlags waitDstStageFlags = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
jvanverth9f372462016-04-06 06:08:59 -0700588 // insert the layout transfer into the queue and wait on the acquire
589 VkSubmitInfo submitInfo;
590 memset(&submitInfo, 0, sizeof(VkSubmitInfo));
591 submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO;
592 submitInfo.waitSemaphoreCount = 1;
593 submitInfo.pWaitSemaphores = &backbuffer->fAcquireSemaphore;
egdaniel58a8d922016-04-21 08:03:10 -0700594 submitInfo.pWaitDstStageMask = &waitDstStageFlags;
jvanverth9f372462016-04-06 06:08:59 -0700595 submitInfo.commandBufferCount = 1;
596 submitInfo.pCommandBuffers = &backbuffer->fTransitionCmdBuffers[0];
597 submitInfo.signalSemaphoreCount = 0;
jvanvertha8d0d6c2016-05-05 12:32:03 -0700598
Greg Danielf730c182018-07-02 20:15:37 +0000599 GR_VK_CALL_ERRCHECK(fInterface,
600 QueueSubmit(fGraphicsQueue, 1, &submitInfo,
jvanverth9f372462016-04-06 06:08:59 -0700601 backbuffer->fUsageFences[0]));
602
egdaniel580fa592016-08-31 11:03:50 -0700603 SkSurface* surface = fSurfaces[backbuffer->fImageIndex].get();
Robert Phillipsba375a82018-04-11 10:08:06 -0400604 GrBackendRenderTarget backendRT = surface->getBackendRenderTarget(
605 SkSurface::kFlushRead_BackendHandleAccess);
606 backendRT.setVkImageLayout(VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
607
egdaniel580fa592016-08-31 11:03:50 -0700608
609 return sk_ref_sp(surface);
jvanverth9f372462016-04-06 06:08:59 -0700610}
611
jvanvertha8d0d6c2016-05-05 12:32:03 -0700612void VulkanWindowContext::swapBuffers() {
jvanverth9f372462016-04-06 06:08:59 -0700613
614 BackbufferInfo* backbuffer = fBackbuffers + fCurrentBackbufferIndex;
egdaniel580fa592016-08-31 11:03:50 -0700615 SkSurface* surface = fSurfaces[backbuffer->fImageIndex].get();
jvanverth9f372462016-04-06 06:08:59 -0700616
Robert Phillipsba375a82018-04-11 10:08:06 -0400617 GrBackendRenderTarget backendRT = surface->getBackendRenderTarget(
618 SkSurface::kFlushRead_BackendHandleAccess);
619 GrVkImageInfo imageInfo;
620 SkAssertResult(backendRT.getVkImageInfo(&imageInfo));
621 // Check to make sure we never change the actually wrapped image
622 SkASSERT(imageInfo.fImage == fImages[backbuffer->fImageIndex]);
623
624 VkImageLayout layout = imageInfo.fImageLayout;
Greg Daniel6ddbafc2018-05-24 12:34:29 -0400625 VkPipelineStageFlags srcStageMask = GrVkImage::LayoutToPipelineStageFlags(layout);
jvanverth9f372462016-04-06 06:08:59 -0700626 VkPipelineStageFlags dstStageMask = VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT;
Greg Daniel6ddbafc2018-05-24 12:34:29 -0400627 VkAccessFlags srcAccessMask = GrVkImage::LayoutToSrcAccessMask(layout);
jvanverth9f372462016-04-06 06:08:59 -0700628 VkAccessFlags dstAccessMask = VK_ACCESS_MEMORY_READ_BIT;
629
630 VkImageMemoryBarrier imageMemoryBarrier = {
631 VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER, // sType
632 NULL, // pNext
633 srcAccessMask, // outputMask
634 dstAccessMask, // inputMask
635 layout, // oldLayout
636 VK_IMAGE_LAYOUT_PRESENT_SRC_KHR, // newLayout
Greg Danielf730c182018-07-02 20:15:37 +0000637 fGraphicsQueueIndex, // srcQueueFamilyIndex
jvanverth9f372462016-04-06 06:08:59 -0700638 fPresentQueueIndex, // dstQueueFamilyIndex
639 fImages[backbuffer->fImageIndex], // image
640 { VK_IMAGE_ASPECT_COLOR_BIT, 0, 1, 0, 1 } // subresourceRange
641 };
Greg Danielf730c182018-07-02 20:15:37 +0000642 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700643 ResetCommandBuffer(backbuffer->fTransitionCmdBuffers[1], 0));
644 VkCommandBufferBeginInfo info;
645 memset(&info, 0, sizeof(VkCommandBufferBeginInfo));
646 info.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO;
647 info.flags = 0;
Greg Danielf730c182018-07-02 20:15:37 +0000648 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700649 BeginCommandBuffer(backbuffer->fTransitionCmdBuffers[1], &info));
Greg Danielf730c182018-07-02 20:15:37 +0000650 GR_VK_CALL(fInterface,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700651 CmdPipelineBarrier(backbuffer->fTransitionCmdBuffers[1],
652 srcStageMask, dstStageMask, 0,
653 0, nullptr,
654 0, nullptr,
655 1, &imageMemoryBarrier));
Greg Danielf730c182018-07-02 20:15:37 +0000656 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700657 EndCommandBuffer(backbuffer->fTransitionCmdBuffers[1]));
658
659 fImageLayouts[backbuffer->fImageIndex] = VK_IMAGE_LAYOUT_PRESENT_SRC_KHR;
660
661 // insert the layout transfer into the queue and wait on the acquire
662 VkSubmitInfo submitInfo;
663 memset(&submitInfo, 0, sizeof(VkSubmitInfo));
664 submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO;
665 submitInfo.waitSemaphoreCount = 0;
666 submitInfo.pWaitDstStageMask = 0;
667 submitInfo.commandBufferCount = 1;
668 submitInfo.pCommandBuffers = &backbuffer->fTransitionCmdBuffers[1];
669 submitInfo.signalSemaphoreCount = 1;
670 submitInfo.pSignalSemaphores = &backbuffer->fRenderSemaphore;
671
Greg Danielf730c182018-07-02 20:15:37 +0000672 GR_VK_CALL_ERRCHECK(fInterface,
673 QueueSubmit(fGraphicsQueue, 1, &submitInfo,
jvanverth9f372462016-04-06 06:08:59 -0700674 backbuffer->fUsageFences[1]));
675
676 // Submit present operation to present queue
677 const VkPresentInfoKHR presentInfo =
678 {
679 VK_STRUCTURE_TYPE_PRESENT_INFO_KHR, // sType
680 NULL, // pNext
681 1, // waitSemaphoreCount
682 &backbuffer->fRenderSemaphore, // pWaitSemaphores
683 1, // swapchainCount
684 &fSwapchain, // pSwapchains
685 &backbuffer->fImageIndex, // pImageIndices
686 NULL // pResults
687 };
688
jvanverthb0d43522016-04-21 11:46:23 -0700689 fQueuePresentKHR(fPresentQueue, &presentInfo);
jvanverth9f372462016-04-06 06:08:59 -0700690}
jvanvertha8d0d6c2016-05-05 12:32:03 -0700691
692} //namespace sk_app