blob: 8e0749ec1b37cd4e69279b6a7046190c9ec44cfc [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 Daniel6ddbafc2018-05-24 12:34:29 -040015#include "vk/GrVkImage.h"
jvanverth9f372462016-04-06 06:08:59 -070016#include "vk/GrVkUtil.h"
17#include "vk/GrVkTypes.h"
18
19#ifdef VK_USE_PLATFORM_WIN32_KHR
20// windows wants to define this as CreateSemaphoreA or CreateSemaphoreW
21#undef CreateSemaphore
22#endif
23
Greg Daniel93ae2332018-06-22 16:50:21 -040024#define GET_PROC(F) f ## F = (PFN_vk ## F) fGetInstanceProcAddr(fInstance, "vk" #F)
25#define GET_DEV_PROC(F) f ## F = (PFN_vk ## F) fGetDeviceProcAddr(fDevice, "vk" #F)
jvanverthb0d43522016-04-21 11:46:23 -070026
jvanvertha8d0d6c2016-05-05 12:32:03 -070027namespace sk_app {
28
bsalomond1bdd1f2016-07-26 12:02:50 -070029VulkanWindowContext::VulkanWindowContext(const DisplayParams& params,
30 CreateVkSurfaceFn createVkSurface,
Greg Daniel35970ec2017-11-10 10:03:05 -050031 CanPresentFn canPresent,
32 PFN_vkGetInstanceProcAddr instProc,
33 PFN_vkGetDeviceProcAddr devProc)
Jim Van Verthfbdc0802017-05-02 16:15:53 -040034 : WindowContext(params)
35 , fCreateVkSurfaceFn(createVkSurface)
36 , fCanPresentFn(canPresent)
jvanverthaf236b52016-05-20 06:01:06 -070037 , fSurface(VK_NULL_HANDLE)
jvanvertha8d0d6c2016-05-05 12:32:03 -070038 , fSwapchain(VK_NULL_HANDLE)
jvanverthaf236b52016-05-20 06:01:06 -070039 , fImages(nullptr)
40 , fImageLayouts(nullptr)
41 , fSurfaces(nullptr)
jvanvertha8d0d6c2016-05-05 12:32:03 -070042 , fCommandPool(VK_NULL_HANDLE)
43 , fBackbuffers(nullptr) {
Greg Daniel35970ec2017-11-10 10:03:05 -050044 fGetInstanceProcAddr = instProc;
45 fGetDeviceProcAddr = devProc;
Jim Van Verthfbdc0802017-05-02 16:15:53 -040046 this->initializeContext();
47}
jvanverth9f372462016-04-06 06:08:59 -070048
Jim Van Verthfbdc0802017-05-02 16:15:53 -040049void VulkanWindowContext::initializeContext() {
jvanverth9f372462016-04-06 06:08:59 -070050 // any config code here (particularly for msaa)?
jvanverth9f372462016-04-06 06:08:59 -070051
Greg Daniel93ae2332018-06-22 16:50:21 -040052 GrVkBackendContext backendContext;
53 if (!sk_gpu_test::CreateVkBackendContext(fGetInstanceProcAddr, fGetDeviceProcAddr,
Greg Danield4b2ade2018-06-25 17:16:44 -040054 &backendContext, &fDebugCallback,
55 &fPresentQueueIndex, fCanPresentFn)) {
jvanverthb0d43522016-04-21 11:46:23 -070056 return;
57 }
58
Greg Daniel93ae2332018-06-22 16:50:21 -040059 if (!(backendContext.fExtensions & kKHR_surface_GrVkExtensionFlag) ||
60 !(backendContext.fExtensions & kKHR_swapchain_GrVkExtensionFlag)) {
61 return;
62 }
63
64 fInstance = backendContext.fInstance;
65 fPhysicalDevice = backendContext.fPhysicalDevice;
66 fDevice = backendContext.fDevice;
67 fGraphicsQueueIndex = backendContext.fGraphicsQueueIndex;
68 fGraphicsQueue = backendContext.fQueue;
69 fInterface = backendContext.fInterface;
70
71 GET_PROC(DestroyInstance);
jvanverthb0d43522016-04-21 11:46:23 -070072 GET_PROC(DestroySurfaceKHR);
73 GET_PROC(GetPhysicalDeviceSurfaceSupportKHR);
74 GET_PROC(GetPhysicalDeviceSurfaceCapabilitiesKHR);
75 GET_PROC(GetPhysicalDeviceSurfaceFormatsKHR);
76 GET_PROC(GetPhysicalDeviceSurfacePresentModesKHR);
Greg Daniel93ae2332018-06-22 16:50:21 -040077 GET_DEV_PROC(DeviceWaitIdle);
78 GET_DEV_PROC(QueueWaitIdle);
79 GET_DEV_PROC(DestroyDevice);
jvanverthb0d43522016-04-21 11:46:23 -070080 GET_DEV_PROC(CreateSwapchainKHR);
81 GET_DEV_PROC(DestroySwapchainKHR);
82 GET_DEV_PROC(GetSwapchainImagesKHR);
83 GET_DEV_PROC(AcquireNextImageKHR);
84 GET_DEV_PROC(QueuePresentKHR);
Greg Daniel35970ec2017-11-10 10:03:05 -050085 GET_DEV_PROC(GetDeviceQueue);
jvanverth9f372462016-04-06 06:08:59 -070086
Greg Daniel93ae2332018-06-22 16:50:21 -040087 fContext = GrContext::MakeVulkan(backendContext, fDisplayParams.fGrContextOptions);
jvanverth9f372462016-04-06 06:08:59 -070088
Greg Daniel93ae2332018-06-22 16:50:21 -040089 fSurface = fCreateVkSurfaceFn(fInstance);
jvanverth9f372462016-04-06 06:08:59 -070090 if (VK_NULL_HANDLE == fSurface) {
Greg Daniel93ae2332018-06-22 16:50:21 -040091 this->destroyContext();
jvanverth9f372462016-04-06 06:08:59 -070092 return;
93 }
94
jvanverth9f372462016-04-06 06:08:59 -070095 VkBool32 supported;
Greg Daniel93ae2332018-06-22 16:50:21 -040096 VkResult res = fGetPhysicalDeviceSurfaceSupportKHR(fPhysicalDevice, fPresentQueueIndex,
97 fSurface, &supported);
jvanverth9f372462016-04-06 06:08:59 -070098 if (VK_SUCCESS != res) {
99 this->destroyContext();
100 return;
101 }
102
Jim Van Verthfbdc0802017-05-02 16:15:53 -0400103 if (!this->createSwapchain(-1, -1, fDisplayParams)) {
jvanverth9f372462016-04-06 06:08:59 -0700104 this->destroyContext();
105 return;
106 }
107
108 // create presentQueue
Greg Daniel93ae2332018-06-22 16:50:21 -0400109 fGetDeviceQueue(fDevice, fPresentQueueIndex, 0, &fPresentQueue);
jvanverth9f372462016-04-06 06:08:59 -0700110}
111
bsalomonccde4ab2016-07-27 08:50:12 -0700112bool VulkanWindowContext::createSwapchain(int width, int height,
brianosman05de2162016-05-06 13:28:57 -0700113 const DisplayParams& params) {
jvanverth9f372462016-04-06 06:08:59 -0700114 // check for capabilities
115 VkSurfaceCapabilitiesKHR caps;
Greg Daniel93ae2332018-06-22 16:50:21 -0400116 VkResult res = fGetPhysicalDeviceSurfaceCapabilitiesKHR(fPhysicalDevice, fSurface, &caps);
jvanverth9f372462016-04-06 06:08:59 -0700117 if (VK_SUCCESS != res) {
118 return false;
119 }
120
121 uint32_t surfaceFormatCount;
Greg Daniel93ae2332018-06-22 16:50:21 -0400122 res = fGetPhysicalDeviceSurfaceFormatsKHR(fPhysicalDevice, fSurface, &surfaceFormatCount,
123 nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700124 if (VK_SUCCESS != res) {
125 return false;
126 }
127
128 SkAutoMalloc surfaceFormatAlloc(surfaceFormatCount * sizeof(VkSurfaceFormatKHR));
129 VkSurfaceFormatKHR* surfaceFormats = (VkSurfaceFormatKHR*)surfaceFormatAlloc.get();
Greg Daniel93ae2332018-06-22 16:50:21 -0400130 res = fGetPhysicalDeviceSurfaceFormatsKHR(fPhysicalDevice, fSurface, &surfaceFormatCount,
131 surfaceFormats);
jvanverth9f372462016-04-06 06:08:59 -0700132 if (VK_SUCCESS != res) {
133 return false;
134 }
135
136 uint32_t presentModeCount;
Greg Daniel93ae2332018-06-22 16:50:21 -0400137 res = fGetPhysicalDeviceSurfacePresentModesKHR(fPhysicalDevice, fSurface, &presentModeCount,
138 nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700139 if (VK_SUCCESS != res) {
140 return false;
141 }
142
143 SkAutoMalloc presentModeAlloc(presentModeCount * sizeof(VkPresentModeKHR));
144 VkPresentModeKHR* presentModes = (VkPresentModeKHR*)presentModeAlloc.get();
Greg Daniel93ae2332018-06-22 16:50:21 -0400145 res = fGetPhysicalDeviceSurfacePresentModesKHR(fPhysicalDevice, fSurface, &presentModeCount,
146 presentModes);
jvanverth9f372462016-04-06 06:08:59 -0700147 if (VK_SUCCESS != res) {
148 return false;
149 }
150
151 VkExtent2D extent = caps.currentExtent;
152 // use the hints
153 if (extent.width == (uint32_t)-1) {
154 extent.width = width;
155 extent.height = height;
156 }
157
jvanverth9fab59d2016-04-06 12:08:51 -0700158 // clamp width; to protect us from broken hints
jvanverth9f372462016-04-06 06:08:59 -0700159 if (extent.width < caps.minImageExtent.width) {
160 extent.width = caps.minImageExtent.width;
161 } else if (extent.width > caps.maxImageExtent.width) {
162 extent.width = caps.maxImageExtent.width;
163 }
164 // clamp height
165 if (extent.height < caps.minImageExtent.height) {
166 extent.height = caps.minImageExtent.height;
167 } else if (extent.height > caps.maxImageExtent.height) {
168 extent.height = caps.maxImageExtent.height;
169 }
jvanverth1d155962016-05-23 13:13:36 -0700170
jvanverth9f372462016-04-06 06:08:59 -0700171 fWidth = (int)extent.width;
172 fHeight = (int)extent.height;
173
174 uint32_t imageCount = caps.minImageCount + 2;
175 if (caps.maxImageCount > 0 && imageCount > caps.maxImageCount) {
176 // Application must settle for fewer images than desired:
177 imageCount = caps.maxImageCount;
178 }
179
180 VkImageUsageFlags usageFlags = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
181 VK_IMAGE_USAGE_TRANSFER_SRC_BIT |
182 VK_IMAGE_USAGE_TRANSFER_DST_BIT;
183 SkASSERT((caps.supportedUsageFlags & usageFlags) == usageFlags);
184 SkASSERT(caps.supportedTransforms & caps.currentTransform);
185 SkASSERT(caps.supportedCompositeAlpha & (VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR |
186 VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR));
187 VkCompositeAlphaFlagBitsKHR composite_alpha =
188 (caps.supportedCompositeAlpha & VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR) ?
189 VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR :
190 VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
191
brianosman05de2162016-05-06 13:28:57 -0700192 // Pick our surface format. For now, just make sure it matches our sRGB request:
193 VkFormat surfaceFormat = VK_FORMAT_UNDEFINED;
194 VkColorSpaceKHR colorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
brianosman05de2162016-05-06 13:28:57 -0700195 for (uint32_t i = 0; i < surfaceFormatCount; ++i) {
Greg Daniel81b80592017-12-13 10:20:04 -0500196 VkFormat localFormat = surfaceFormats[i].format;
Brian Osman2b23c4b2018-06-01 12:25:08 -0400197 if (GrVkFormatIsSupported(localFormat)) {
Greg Daniel81b80592017-12-13 10:20:04 -0500198 surfaceFormat = localFormat;
brianosman05de2162016-05-06 13:28:57 -0700199 colorSpace = surfaceFormats[i].colorSpace;
200 break;
201 }
202 }
203 fDisplayParams = params;
csmartdalton578f0642017-02-24 16:04:47 -0700204 fSampleCount = params.fMSAASampleCount;
205 fStencilBits = 8;
brianosman05de2162016-05-06 13:28:57 -0700206
207 if (VK_FORMAT_UNDEFINED == surfaceFormat) {
208 return false;
209 }
jvanvertha4b0fed2016-04-27 11:42:21 -0700210
Greg Danielfaa095e2017-12-19 13:15:02 -0500211 SkColorType colorType;
212 switch (surfaceFormat) {
213 case VK_FORMAT_R8G8B8A8_UNORM: // fall through
214 case VK_FORMAT_R8G8B8A8_SRGB:
215 colorType = kRGBA_8888_SkColorType;
216 break;
217 case VK_FORMAT_B8G8R8A8_UNORM: // fall through
218 case VK_FORMAT_B8G8R8A8_SRGB:
219 colorType = kBGRA_8888_SkColorType;
220 break;
221 default:
222 return false;
223 }
224
jvanverth3d6ed3a2016-04-07 11:09:51 -0700225 // If mailbox mode is available, use it, as it is the lowest-latency non-
226 // tearing mode. If not, fall back to FIFO which is always available.
jvanverth9f372462016-04-06 06:08:59 -0700227 VkPresentModeKHR mode = VK_PRESENT_MODE_FIFO_KHR;
jvanverth9f372462016-04-06 06:08:59 -0700228 for (uint32_t i = 0; i < presentModeCount; ++i) {
jvanverth3d6ed3a2016-04-07 11:09:51 -0700229 // use mailbox
230 if (VK_PRESENT_MODE_MAILBOX_KHR == presentModes[i]) {
jvanverth9f372462016-04-06 06:08:59 -0700231 mode = presentModes[i];
jvanverth3d6ed3a2016-04-07 11:09:51 -0700232 break;
jvanverth9f372462016-04-06 06:08:59 -0700233 }
234 }
235
236 VkSwapchainCreateInfoKHR swapchainCreateInfo;
237 memset(&swapchainCreateInfo, 0, sizeof(VkSwapchainCreateInfoKHR));
238 swapchainCreateInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
239 swapchainCreateInfo.surface = fSurface;
240 swapchainCreateInfo.minImageCount = imageCount;
jvanvertha4b0fed2016-04-27 11:42:21 -0700241 swapchainCreateInfo.imageFormat = surfaceFormat;
242 swapchainCreateInfo.imageColorSpace = colorSpace;
jvanverth9f372462016-04-06 06:08:59 -0700243 swapchainCreateInfo.imageExtent = extent;
244 swapchainCreateInfo.imageArrayLayers = 1;
245 swapchainCreateInfo.imageUsage = usageFlags;
246
Greg Daniel93ae2332018-06-22 16:50:21 -0400247 uint32_t queueFamilies[] = { fGraphicsQueueIndex, fPresentQueueIndex };
248 if (fGraphicsQueueIndex != fPresentQueueIndex) {
jvanverth9f372462016-04-06 06:08:59 -0700249 swapchainCreateInfo.imageSharingMode = VK_SHARING_MODE_CONCURRENT;
250 swapchainCreateInfo.queueFamilyIndexCount = 2;
251 swapchainCreateInfo.pQueueFamilyIndices = queueFamilies;
252 } else {
253 swapchainCreateInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
254 swapchainCreateInfo.queueFamilyIndexCount = 0;
255 swapchainCreateInfo.pQueueFamilyIndices = nullptr;
256 }
257
Greg Daniele897b972016-10-06 13:57:57 -0400258 swapchainCreateInfo.preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
jvanverth9f372462016-04-06 06:08:59 -0700259 swapchainCreateInfo.compositeAlpha = composite_alpha;
260 swapchainCreateInfo.presentMode = mode;
261 swapchainCreateInfo.clipped = true;
262 swapchainCreateInfo.oldSwapchain = fSwapchain;
263
Greg Daniel93ae2332018-06-22 16:50:21 -0400264 res = fCreateSwapchainKHR(fDevice, &swapchainCreateInfo, nullptr, &fSwapchain);
jvanverth9f372462016-04-06 06:08:59 -0700265 if (VK_SUCCESS != res) {
266 return false;
267 }
268
269 // destroy the old swapchain
270 if (swapchainCreateInfo.oldSwapchain != VK_NULL_HANDLE) {
Greg Daniel93ae2332018-06-22 16:50:21 -0400271 fDeviceWaitIdle(fDevice);
jvanverth9f372462016-04-06 06:08:59 -0700272
273 this->destroyBuffers();
274
Greg Daniel93ae2332018-06-22 16:50:21 -0400275 fDestroySwapchainKHR(fDevice, swapchainCreateInfo.oldSwapchain, nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700276 }
277
Greg Danielfaa095e2017-12-19 13:15:02 -0500278 this->createBuffers(swapchainCreateInfo.imageFormat, colorType);
jvanverth9f372462016-04-06 06:08:59 -0700279
280 return true;
281}
282
Greg Danielfaa095e2017-12-19 13:15:02 -0500283void VulkanWindowContext::createBuffers(VkFormat format, SkColorType colorType) {
Greg Daniel93ae2332018-06-22 16:50:21 -0400284 fGetSwapchainImagesKHR(fDevice, fSwapchain, &fImageCount, nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700285 SkASSERT(fImageCount);
286 fImages = new VkImage[fImageCount];
Greg Daniel93ae2332018-06-22 16:50:21 -0400287 fGetSwapchainImagesKHR(fDevice, fSwapchain, &fImageCount, fImages);
jvanverth9f372462016-04-06 06:08:59 -0700288
289 // set up initial image layouts and create surfaces
290 fImageLayouts = new VkImageLayout[fImageCount];
291 fSurfaces = new sk_sp<SkSurface>[fImageCount];
292 for (uint32_t i = 0; i < fImageCount; ++i) {
293 fImageLayouts[i] = VK_IMAGE_LAYOUT_UNDEFINED;
294
egdanielb2df0c22016-05-13 11:30:37 -0700295 GrVkImageInfo info;
jvanverth9f372462016-04-06 06:08:59 -0700296 info.fImage = fImages[i];
Greg Daniel8385a8a2018-02-26 13:29:37 -0500297 info.fAlloc = GrVkAlloc();
egdaniel580fa592016-08-31 11:03:50 -0700298 info.fImageLayout = VK_IMAGE_LAYOUT_UNDEFINED;
jvanverth9f372462016-04-06 06:08:59 -0700299 info.fImageTiling = VK_IMAGE_TILING_OPTIMAL;
egdaniel58a8d922016-04-21 08:03:10 -0700300 info.fFormat = format;
jvanverth3622a172016-05-10 06:42:18 -0700301 info.fLevelCount = 1;
jvanverthaf236b52016-05-20 06:01:06 -0700302
Brian Salomonafdc6b12018-03-09 12:02:32 -0500303 GrBackendRenderTarget backendRT(fWidth, fHeight, fSampleCount, info);
Greg Daniele79b4732017-04-20 14:07:46 -0400304
Brian Salomonafdc6b12018-03-09 12:02:32 -0500305 fSurfaces[i] = SkSurface::MakeFromBackendRenderTarget(fContext.get(),
306 backendRT,
307 kTopLeft_GrSurfaceOrigin,
308 colorType,
309 fDisplayParams.fColorSpace,
Ben Wagner37c54032018-04-13 14:30:23 -0400310 &fDisplayParams.fSurfaceProps);
jvanverth9f372462016-04-06 06:08:59 -0700311 }
312
313 // create the command pool for the command buffers
314 if (VK_NULL_HANDLE == fCommandPool) {
315 VkCommandPoolCreateInfo commandPoolInfo;
316 memset(&commandPoolInfo, 0, sizeof(VkCommandPoolCreateInfo));
317 commandPoolInfo.sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO;
318 // this needs to be on the render queue
Greg Daniel93ae2332018-06-22 16:50:21 -0400319 commandPoolInfo.queueFamilyIndex = fGraphicsQueueIndex;
jvanverth9f372462016-04-06 06:08:59 -0700320 commandPoolInfo.flags = VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT;
Greg Daniel93ae2332018-06-22 16:50:21 -0400321 GR_VK_CALL_ERRCHECK(fInterface,
322 CreateCommandPool(fDevice, &commandPoolInfo,
jvanverth9f372462016-04-06 06:08:59 -0700323 nullptr, &fCommandPool));
324 }
325
326 // set up the backbuffers
327 VkSemaphoreCreateInfo semaphoreInfo;
328 memset(&semaphoreInfo, 0, sizeof(VkSemaphoreCreateInfo));
329 semaphoreInfo.sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO;
330 semaphoreInfo.pNext = nullptr;
331 semaphoreInfo.flags = 0;
332 VkCommandBufferAllocateInfo commandBuffersInfo;
333 memset(&commandBuffersInfo, 0, sizeof(VkCommandBufferAllocateInfo));
334 commandBuffersInfo.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO;
335 commandBuffersInfo.pNext = nullptr;
336 commandBuffersInfo.commandPool = fCommandPool;
337 commandBuffersInfo.level = VK_COMMAND_BUFFER_LEVEL_PRIMARY;
338 commandBuffersInfo.commandBufferCount = 2;
339 VkFenceCreateInfo fenceInfo;
340 memset(&fenceInfo, 0, sizeof(VkFenceCreateInfo));
341 fenceInfo.sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO;
342 fenceInfo.pNext = nullptr;
343 fenceInfo.flags = VK_FENCE_CREATE_SIGNALED_BIT;
344
Greg Daniel1f05f442016-10-27 16:37:17 -0400345 // we create one additional backbuffer structure here, because we want to
jvanverth9f372462016-04-06 06:08:59 -0700346 // give the command buffers they contain a chance to finish before we cycle back
347 fBackbuffers = new BackbufferInfo[fImageCount + 1];
348 for (uint32_t i = 0; i < fImageCount + 1; ++i) {
349 fBackbuffers[i].fImageIndex = -1;
Greg Daniel93ae2332018-06-22 16:50:21 -0400350 GR_VK_CALL_ERRCHECK(fInterface,
351 CreateSemaphore(fDevice, &semaphoreInfo,
jvanverth9f372462016-04-06 06:08:59 -0700352 nullptr, &fBackbuffers[i].fAcquireSemaphore));
Greg Daniel93ae2332018-06-22 16:50:21 -0400353 GR_VK_CALL_ERRCHECK(fInterface,
354 CreateSemaphore(fDevice, &semaphoreInfo,
jvanverth9f372462016-04-06 06:08:59 -0700355 nullptr, &fBackbuffers[i].fRenderSemaphore));
Greg Daniel93ae2332018-06-22 16:50:21 -0400356 GR_VK_CALL_ERRCHECK(fInterface,
357 AllocateCommandBuffers(fDevice, &commandBuffersInfo,
jvanverth9f372462016-04-06 06:08:59 -0700358 fBackbuffers[i].fTransitionCmdBuffers));
Greg Daniel93ae2332018-06-22 16:50:21 -0400359 GR_VK_CALL_ERRCHECK(fInterface,
360 CreateFence(fDevice, &fenceInfo, nullptr,
jvanverth9f372462016-04-06 06:08:59 -0700361 &fBackbuffers[i].fUsageFences[0]));
Greg Daniel93ae2332018-06-22 16:50:21 -0400362 GR_VK_CALL_ERRCHECK(fInterface,
363 CreateFence(fDevice, &fenceInfo, nullptr,
jvanverth9f372462016-04-06 06:08:59 -0700364 &fBackbuffers[i].fUsageFences[1]));
365 }
366 fCurrentBackbufferIndex = fImageCount;
367}
368
jvanvertha8d0d6c2016-05-05 12:32:03 -0700369void VulkanWindowContext::destroyBuffers() {
jvanverth9f372462016-04-06 06:08:59 -0700370
371 if (fBackbuffers) {
372 for (uint32_t i = 0; i < fImageCount + 1; ++i) {
Greg Daniel93ae2332018-06-22 16:50:21 -0400373 GR_VK_CALL_ERRCHECK(fInterface,
374 WaitForFences(fDevice, 2,
jvanverth9f372462016-04-06 06:08:59 -0700375 fBackbuffers[i].fUsageFences,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700376 true, UINT64_MAX));
jvanverth9f372462016-04-06 06:08:59 -0700377 fBackbuffers[i].fImageIndex = -1;
Greg Daniel93ae2332018-06-22 16:50:21 -0400378 GR_VK_CALL(fInterface,
379 DestroySemaphore(fDevice,
jvanverth9f372462016-04-06 06:08:59 -0700380 fBackbuffers[i].fAcquireSemaphore,
381 nullptr));
Greg Daniel93ae2332018-06-22 16:50:21 -0400382 GR_VK_CALL(fInterface,
383 DestroySemaphore(fDevice,
jvanverth9f372462016-04-06 06:08:59 -0700384 fBackbuffers[i].fRenderSemaphore,
385 nullptr));
Greg Daniel93ae2332018-06-22 16:50:21 -0400386 GR_VK_CALL(fInterface,
387 FreeCommandBuffers(fDevice, fCommandPool, 2,
jvanverth9f372462016-04-06 06:08:59 -0700388 fBackbuffers[i].fTransitionCmdBuffers));
Greg Daniel93ae2332018-06-22 16:50:21 -0400389 GR_VK_CALL(fInterface,
390 DestroyFence(fDevice, fBackbuffers[i].fUsageFences[0], 0));
391 GR_VK_CALL(fInterface,
392 DestroyFence(fDevice, fBackbuffers[i].fUsageFences[1], 0));
jvanverth9f372462016-04-06 06:08:59 -0700393 }
394 }
395
396 delete[] fBackbuffers;
397 fBackbuffers = nullptr;
398
jvanverthaf236b52016-05-20 06:01:06 -0700399 // Does this actually free the surfaces?
jvanverth9f372462016-04-06 06:08:59 -0700400 delete[] fSurfaces;
401 fSurfaces = nullptr;
402 delete[] fImageLayouts;
403 fImageLayouts = nullptr;
404 delete[] fImages;
405 fImages = nullptr;
406}
407
jvanvertha8d0d6c2016-05-05 12:32:03 -0700408VulkanWindowContext::~VulkanWindowContext() {
jvanverth9f372462016-04-06 06:08:59 -0700409 this->destroyContext();
410}
411
jvanvertha8d0d6c2016-05-05 12:32:03 -0700412void VulkanWindowContext::destroyContext() {
Greg Danield4b2ade2018-06-25 17:16:44 -0400413 if (this->isValid()) {
414 fQueueWaitIdle(fPresentQueue);
415 fDeviceWaitIdle(fDevice);
jvanverth9f372462016-04-06 06:08:59 -0700416
Greg Danield4b2ade2018-06-25 17:16:44 -0400417 this->destroyBuffers();
jvanverth9f372462016-04-06 06:08:59 -0700418
Greg Danield4b2ade2018-06-25 17:16:44 -0400419 if (VK_NULL_HANDLE != fCommandPool) {
420 GR_VK_CALL(fInterface, DestroyCommandPool(fDevice, fCommandPool, nullptr));
421 fCommandPool = VK_NULL_HANDLE;
422 }
jvanverth9f372462016-04-06 06:08:59 -0700423
Greg Danield4b2ade2018-06-25 17:16:44 -0400424 if (VK_NULL_HANDLE != fSwapchain) {
425 fDestroySwapchainKHR(fDevice, fSwapchain, nullptr);
426 fSwapchain = VK_NULL_HANDLE;
427 }
jvanverth9f372462016-04-06 06:08:59 -0700428
Greg Danield4b2ade2018-06-25 17:16:44 -0400429 if (VK_NULL_HANDLE != fSurface) {
430 fDestroySurfaceKHR(fInstance, fSurface, nullptr);
431 fSurface = VK_NULL_HANDLE;
432 }
jvanverth9f372462016-04-06 06:08:59 -0700433 }
434
Greg Daniel02611d92017-07-25 10:05:01 -0400435 fContext.reset();
Greg Daniel93ae2332018-06-22 16:50:21 -0400436 fInterface.reset();
jvanverth9f372462016-04-06 06:08:59 -0700437
Greg Daniel93ae2332018-06-22 16:50:21 -0400438 if (VK_NULL_HANDLE != fDevice) {
439 fDestroyDevice(fDevice, nullptr);
440 fDevice = VK_NULL_HANDLE;
441 }
Greg Danield4b2ade2018-06-25 17:16:44 -0400442
443#ifdef SK_ENABLE_VK_LAYERS
444 if (fDebugCallback != VK_NULL_HANDLE) {
445 GR_VK_CALL(fInterface, DestroyDebugReportCallbackEXT(fInstance, fDebugCallback,
446 nullptr));
447 }
448#endif
449
Greg Daniel93ae2332018-06-22 16:50:21 -0400450 fPhysicalDevice = VK_NULL_HANDLE;
451
452 if (VK_NULL_HANDLE != fInstance) {
453 fDestroyInstance(fInstance, nullptr);
454 fInstance = VK_NULL_HANDLE;
455 }
jvanverth9f372462016-04-06 06:08:59 -0700456}
457
jvanvertha8d0d6c2016-05-05 12:32:03 -0700458VulkanWindowContext::BackbufferInfo* VulkanWindowContext::getAvailableBackbuffer() {
jvanverth9f372462016-04-06 06:08:59 -0700459 SkASSERT(fBackbuffers);
460
461 ++fCurrentBackbufferIndex;
egdaniel804af7d2016-09-26 12:30:46 -0700462 if (fCurrentBackbufferIndex > fImageCount) {
jvanverth9f372462016-04-06 06:08:59 -0700463 fCurrentBackbufferIndex = 0;
464 }
465
466 BackbufferInfo* backbuffer = fBackbuffers + fCurrentBackbufferIndex;
Greg Daniel93ae2332018-06-22 16:50:21 -0400467 GR_VK_CALL_ERRCHECK(fInterface,
468 WaitForFences(fDevice, 2, backbuffer->fUsageFences,
jvanverth9f372462016-04-06 06:08:59 -0700469 true, UINT64_MAX));
470 return backbuffer;
471}
472
jvanverthaf236b52016-05-20 06:01:06 -0700473sk_sp<SkSurface> VulkanWindowContext::getBackbufferSurface() {
jvanverth9f372462016-04-06 06:08:59 -0700474 BackbufferInfo* backbuffer = this->getAvailableBackbuffer();
475 SkASSERT(backbuffer);
476
477 // reset the fence
Greg Daniel93ae2332018-06-22 16:50:21 -0400478 GR_VK_CALL_ERRCHECK(fInterface,
479 ResetFences(fDevice, 2, backbuffer->fUsageFences));
jvanverth9f372462016-04-06 06:08:59 -0700480 // semaphores should be in unsignaled state
481
482 // acquire the image
Greg Daniel93ae2332018-06-22 16:50:21 -0400483 VkResult res = fAcquireNextImageKHR(fDevice, fSwapchain, UINT64_MAX,
jvanverthb0d43522016-04-21 11:46:23 -0700484 backbuffer->fAcquireSemaphore, VK_NULL_HANDLE,
485 &backbuffer->fImageIndex);
jvanverth9f372462016-04-06 06:08:59 -0700486 if (VK_ERROR_SURFACE_LOST_KHR == res) {
487 // need to figure out how to create a new vkSurface without the platformData*
jvanverth3d6ed3a2016-04-07 11:09:51 -0700488 // maybe use attach somehow? but need a Window
jvanverth9f372462016-04-06 06:08:59 -0700489 return nullptr;
490 }
jvanverth3d6ed3a2016-04-07 11:09:51 -0700491 if (VK_ERROR_OUT_OF_DATE_KHR == res) {
jvanverth9f372462016-04-06 06:08:59 -0700492 // tear swapchain down and try again
Greg Daniel1f05f442016-10-27 16:37:17 -0400493 if (!this->createSwapchain(-1, -1, fDisplayParams)) {
jvanverth9f372462016-04-06 06:08:59 -0700494 return nullptr;
495 }
Jim Van Verthd63c1022017-01-05 13:50:49 -0500496 backbuffer = this->getAvailableBackbuffer();
Greg Daniel93ae2332018-06-22 16:50:21 -0400497 GR_VK_CALL_ERRCHECK(fInterface,
498 ResetFences(fDevice, 2, backbuffer->fUsageFences));
jvanverth9f372462016-04-06 06:08:59 -0700499
500 // acquire the image
Greg Daniel93ae2332018-06-22 16:50:21 -0400501 res = fAcquireNextImageKHR(fDevice, fSwapchain, UINT64_MAX,
jvanverthb0d43522016-04-21 11:46:23 -0700502 backbuffer->fAcquireSemaphore, VK_NULL_HANDLE,
503 &backbuffer->fImageIndex);
jvanverth9f372462016-04-06 06:08:59 -0700504
505 if (VK_SUCCESS != res) {
506 return nullptr;
507 }
508 }
509
510 // set up layout transfer from initial to color attachment
511 VkImageLayout layout = fImageLayouts[backbuffer->fImageIndex];
egdaniel580fa592016-08-31 11:03:50 -0700512 SkASSERT(VK_IMAGE_LAYOUT_UNDEFINED == layout || VK_IMAGE_LAYOUT_PRESENT_SRC_KHR == layout);
jvanverth9f372462016-04-06 06:08:59 -0700513 VkPipelineStageFlags srcStageMask = (VK_IMAGE_LAYOUT_UNDEFINED == layout) ?
514 VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT :
515 VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
516 VkPipelineStageFlags dstStageMask = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
jvanvertha8d0d6c2016-05-05 12:32:03 -0700517 VkAccessFlags srcAccessMask = (VK_IMAGE_LAYOUT_UNDEFINED == layout) ?
jvanverth9f372462016-04-06 06:08:59 -0700518 0 : VK_ACCESS_MEMORY_READ_BIT;
519 VkAccessFlags dstAccessMask = VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT;
520
521 VkImageMemoryBarrier imageMemoryBarrier = {
522 VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER, // sType
523 NULL, // pNext
524 srcAccessMask, // outputMask
525 dstAccessMask, // inputMask
526 layout, // oldLayout
527 VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, // newLayout
528 fPresentQueueIndex, // srcQueueFamilyIndex
Greg Daniel93ae2332018-06-22 16:50:21 -0400529 fGraphicsQueueIndex, // dstQueueFamilyIndex
jvanverth9f372462016-04-06 06:08:59 -0700530 fImages[backbuffer->fImageIndex], // image
531 { VK_IMAGE_ASPECT_COLOR_BIT, 0, 1, 0, 1 } // subresourceRange
532 };
Greg Daniel93ae2332018-06-22 16:50:21 -0400533 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700534 ResetCommandBuffer(backbuffer->fTransitionCmdBuffers[0], 0));
535 VkCommandBufferBeginInfo info;
536 memset(&info, 0, sizeof(VkCommandBufferBeginInfo));
537 info.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO;
538 info.flags = 0;
Greg Daniel93ae2332018-06-22 16:50:21 -0400539 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700540 BeginCommandBuffer(backbuffer->fTransitionCmdBuffers[0], &info));
541
Greg Daniel93ae2332018-06-22 16:50:21 -0400542 GR_VK_CALL(fInterface,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700543 CmdPipelineBarrier(backbuffer->fTransitionCmdBuffers[0],
544 srcStageMask, dstStageMask, 0,
545 0, nullptr,
546 0, nullptr,
547 1, &imageMemoryBarrier));
jvanverth9f372462016-04-06 06:08:59 -0700548
Greg Daniel93ae2332018-06-22 16:50:21 -0400549 GR_VK_CALL_ERRCHECK(fInterface,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700550 EndCommandBuffer(backbuffer->fTransitionCmdBuffers[0]));
jvanverth9f372462016-04-06 06:08:59 -0700551
egdaniel58a8d922016-04-21 08:03:10 -0700552 VkPipelineStageFlags waitDstStageFlags = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
jvanverth9f372462016-04-06 06:08:59 -0700553 // insert the layout transfer into the queue and wait on the acquire
554 VkSubmitInfo submitInfo;
555 memset(&submitInfo, 0, sizeof(VkSubmitInfo));
556 submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO;
557 submitInfo.waitSemaphoreCount = 1;
558 submitInfo.pWaitSemaphores = &backbuffer->fAcquireSemaphore;
egdaniel58a8d922016-04-21 08:03:10 -0700559 submitInfo.pWaitDstStageMask = &waitDstStageFlags;
jvanverth9f372462016-04-06 06:08:59 -0700560 submitInfo.commandBufferCount = 1;
561 submitInfo.pCommandBuffers = &backbuffer->fTransitionCmdBuffers[0];
562 submitInfo.signalSemaphoreCount = 0;
jvanvertha8d0d6c2016-05-05 12:32:03 -0700563
Greg Daniel93ae2332018-06-22 16:50:21 -0400564 GR_VK_CALL_ERRCHECK(fInterface,
565 QueueSubmit(fGraphicsQueue, 1, &submitInfo,
jvanverth9f372462016-04-06 06:08:59 -0700566 backbuffer->fUsageFences[0]));
567
egdaniel580fa592016-08-31 11:03:50 -0700568 SkSurface* surface = fSurfaces[backbuffer->fImageIndex].get();
Robert Phillipsba375a82018-04-11 10:08:06 -0400569 GrBackendRenderTarget backendRT = surface->getBackendRenderTarget(
570 SkSurface::kFlushRead_BackendHandleAccess);
571 backendRT.setVkImageLayout(VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
572
egdaniel580fa592016-08-31 11:03:50 -0700573
574 return sk_ref_sp(surface);
jvanverth9f372462016-04-06 06:08:59 -0700575}
576
jvanvertha8d0d6c2016-05-05 12:32:03 -0700577void VulkanWindowContext::swapBuffers() {
jvanverth9f372462016-04-06 06:08:59 -0700578
579 BackbufferInfo* backbuffer = fBackbuffers + fCurrentBackbufferIndex;
egdaniel580fa592016-08-31 11:03:50 -0700580 SkSurface* surface = fSurfaces[backbuffer->fImageIndex].get();
jvanverth9f372462016-04-06 06:08:59 -0700581
Robert Phillipsba375a82018-04-11 10:08:06 -0400582 GrBackendRenderTarget backendRT = surface->getBackendRenderTarget(
583 SkSurface::kFlushRead_BackendHandleAccess);
584 GrVkImageInfo imageInfo;
585 SkAssertResult(backendRT.getVkImageInfo(&imageInfo));
586 // Check to make sure we never change the actually wrapped image
587 SkASSERT(imageInfo.fImage == fImages[backbuffer->fImageIndex]);
588
589 VkImageLayout layout = imageInfo.fImageLayout;
Greg Daniel6ddbafc2018-05-24 12:34:29 -0400590 VkPipelineStageFlags srcStageMask = GrVkImage::LayoutToPipelineStageFlags(layout);
jvanverth9f372462016-04-06 06:08:59 -0700591 VkPipelineStageFlags dstStageMask = VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT;
Greg Daniel6ddbafc2018-05-24 12:34:29 -0400592 VkAccessFlags srcAccessMask = GrVkImage::LayoutToSrcAccessMask(layout);
jvanverth9f372462016-04-06 06:08:59 -0700593 VkAccessFlags dstAccessMask = VK_ACCESS_MEMORY_READ_BIT;
594
595 VkImageMemoryBarrier imageMemoryBarrier = {
596 VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER, // sType
597 NULL, // pNext
598 srcAccessMask, // outputMask
599 dstAccessMask, // inputMask
600 layout, // oldLayout
601 VK_IMAGE_LAYOUT_PRESENT_SRC_KHR, // newLayout
Greg Daniel93ae2332018-06-22 16:50:21 -0400602 fGraphicsQueueIndex, // srcQueueFamilyIndex
jvanverth9f372462016-04-06 06:08:59 -0700603 fPresentQueueIndex, // dstQueueFamilyIndex
604 fImages[backbuffer->fImageIndex], // image
605 { VK_IMAGE_ASPECT_COLOR_BIT, 0, 1, 0, 1 } // subresourceRange
606 };
Greg Daniel93ae2332018-06-22 16:50:21 -0400607 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700608 ResetCommandBuffer(backbuffer->fTransitionCmdBuffers[1], 0));
609 VkCommandBufferBeginInfo info;
610 memset(&info, 0, sizeof(VkCommandBufferBeginInfo));
611 info.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO;
612 info.flags = 0;
Greg Daniel93ae2332018-06-22 16:50:21 -0400613 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700614 BeginCommandBuffer(backbuffer->fTransitionCmdBuffers[1], &info));
Greg Daniel93ae2332018-06-22 16:50:21 -0400615 GR_VK_CALL(fInterface,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700616 CmdPipelineBarrier(backbuffer->fTransitionCmdBuffers[1],
617 srcStageMask, dstStageMask, 0,
618 0, nullptr,
619 0, nullptr,
620 1, &imageMemoryBarrier));
Greg Daniel93ae2332018-06-22 16:50:21 -0400621 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700622 EndCommandBuffer(backbuffer->fTransitionCmdBuffers[1]));
623
624 fImageLayouts[backbuffer->fImageIndex] = VK_IMAGE_LAYOUT_PRESENT_SRC_KHR;
625
626 // insert the layout transfer into the queue and wait on the acquire
627 VkSubmitInfo submitInfo;
628 memset(&submitInfo, 0, sizeof(VkSubmitInfo));
629 submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO;
630 submitInfo.waitSemaphoreCount = 0;
631 submitInfo.pWaitDstStageMask = 0;
632 submitInfo.commandBufferCount = 1;
633 submitInfo.pCommandBuffers = &backbuffer->fTransitionCmdBuffers[1];
634 submitInfo.signalSemaphoreCount = 1;
635 submitInfo.pSignalSemaphores = &backbuffer->fRenderSemaphore;
636
Greg Daniel93ae2332018-06-22 16:50:21 -0400637 GR_VK_CALL_ERRCHECK(fInterface,
638 QueueSubmit(fGraphicsQueue, 1, &submitInfo,
jvanverth9f372462016-04-06 06:08:59 -0700639 backbuffer->fUsageFences[1]));
640
641 // Submit present operation to present queue
642 const VkPresentInfoKHR presentInfo =
643 {
644 VK_STRUCTURE_TYPE_PRESENT_INFO_KHR, // sType
645 NULL, // pNext
646 1, // waitSemaphoreCount
647 &backbuffer->fRenderSemaphore, // pWaitSemaphores
648 1, // swapchainCount
649 &fSwapchain, // pSwapchains
650 &backbuffer->fImageIndex, // pImageIndices
651 NULL // pResults
652 };
653
jvanverthb0d43522016-04-21 11:46:23 -0700654 fQueuePresentKHR(fPresentQueue, &presentInfo);
jvanverth9f372462016-04-06 06:08:59 -0700655}
jvanvertha8d0d6c2016-05-05 12:32:03 -0700656
657} //namespace sk_app