blob: db5cc1ef4f08f0f18c56b64c39e4ab9791eb1a51 [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/GrVkUtil.h"
18#include "vk/GrVkTypes.h"
19
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;
64 if (!sk_gpu_test::CreateVkBackendContext(getProc, &backendContext, &extensions, &fDebugCallback,
Greg Daniel37329b32018-07-02 20:16:44 +000065 &fPresentQueueIndex, fCanPresentFn)) {
jvanverthb0d43522016-04-21 11:46:23 -070066 return;
67 }
68
Greg Daniel98bffae2018-08-01 13:25:41 -040069 if (!extensions.hasExtension(VK_KHR_SURFACE_EXTENSION_NAME) ||
70 !extensions.hasExtension(VK_KHR_SWAPCHAIN_EXTENSION_NAME)) {
Greg Danielf730c182018-07-02 20:15:37 +000071 return;
72 }
73
74 fInstance = backendContext.fInstance;
75 fPhysicalDevice = backendContext.fPhysicalDevice;
76 fDevice = backendContext.fDevice;
77 fGraphicsQueueIndex = backendContext.fGraphicsQueueIndex;
78 fGraphicsQueue = backendContext.fQueue;
Greg Danielc8cd45a2018-07-12 10:02:37 -040079 fInterface.reset(new GrVkInterface(backendContext.fGetProc, fInstance, fDevice,
Greg Daniel98bffae2018-08-01 13:25:41 -040080 &extensions));
Greg Danielf730c182018-07-02 20:15:37 +000081
82 GET_PROC(DestroyInstance);
jvanverthb0d43522016-04-21 11:46:23 -070083 GET_PROC(DestroySurfaceKHR);
84 GET_PROC(GetPhysicalDeviceSurfaceSupportKHR);
85 GET_PROC(GetPhysicalDeviceSurfaceCapabilitiesKHR);
86 GET_PROC(GetPhysicalDeviceSurfaceFormatsKHR);
87 GET_PROC(GetPhysicalDeviceSurfacePresentModesKHR);
Greg Danielf730c182018-07-02 20:15:37 +000088 GET_DEV_PROC(DeviceWaitIdle);
89 GET_DEV_PROC(QueueWaitIdle);
90 GET_DEV_PROC(DestroyDevice);
jvanverthb0d43522016-04-21 11:46:23 -070091 GET_DEV_PROC(CreateSwapchainKHR);
92 GET_DEV_PROC(DestroySwapchainKHR);
93 GET_DEV_PROC(GetSwapchainImagesKHR);
94 GET_DEV_PROC(AcquireNextImageKHR);
95 GET_DEV_PROC(QueuePresentKHR);
Greg Daniel35970ec2017-11-10 10:03:05 -050096 GET_DEV_PROC(GetDeviceQueue);
jvanverth9f372462016-04-06 06:08:59 -070097
Greg Danielf730c182018-07-02 20:15:37 +000098 fContext = GrContext::MakeVulkan(backendContext, fDisplayParams.fGrContextOptions);
jvanverth9f372462016-04-06 06:08:59 -070099
Greg Danielf730c182018-07-02 20:15:37 +0000100 fSurface = fCreateVkSurfaceFn(fInstance);
jvanverth9f372462016-04-06 06:08:59 -0700101 if (VK_NULL_HANDLE == fSurface) {
Greg Danielf730c182018-07-02 20:15:37 +0000102 this->destroyContext();
jvanverth9f372462016-04-06 06:08:59 -0700103 return;
104 }
105
jvanverth9f372462016-04-06 06:08:59 -0700106 VkBool32 supported;
Greg Danielf730c182018-07-02 20:15:37 +0000107 VkResult res = fGetPhysicalDeviceSurfaceSupportKHR(fPhysicalDevice, fPresentQueueIndex,
108 fSurface, &supported);
jvanverth9f372462016-04-06 06:08:59 -0700109 if (VK_SUCCESS != res) {
110 this->destroyContext();
111 return;
112 }
113
Jim Van Verthfbdc0802017-05-02 16:15:53 -0400114 if (!this->createSwapchain(-1, -1, fDisplayParams)) {
jvanverth9f372462016-04-06 06:08:59 -0700115 this->destroyContext();
116 return;
117 }
118
119 // create presentQueue
Greg Danielf730c182018-07-02 20:15:37 +0000120 fGetDeviceQueue(fDevice, fPresentQueueIndex, 0, &fPresentQueue);
jvanverth9f372462016-04-06 06:08:59 -0700121}
122
bsalomonccde4ab2016-07-27 08:50:12 -0700123bool VulkanWindowContext::createSwapchain(int width, int height,
brianosman05de2162016-05-06 13:28:57 -0700124 const DisplayParams& params) {
jvanverth9f372462016-04-06 06:08:59 -0700125 // check for capabilities
126 VkSurfaceCapabilitiesKHR caps;
Greg Danielf730c182018-07-02 20:15:37 +0000127 VkResult res = fGetPhysicalDeviceSurfaceCapabilitiesKHR(fPhysicalDevice, fSurface, &caps);
jvanverth9f372462016-04-06 06:08:59 -0700128 if (VK_SUCCESS != res) {
129 return false;
130 }
131
132 uint32_t surfaceFormatCount;
Greg Danielf730c182018-07-02 20:15:37 +0000133 res = fGetPhysicalDeviceSurfaceFormatsKHR(fPhysicalDevice, fSurface, &surfaceFormatCount,
134 nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700135 if (VK_SUCCESS != res) {
136 return false;
137 }
138
139 SkAutoMalloc surfaceFormatAlloc(surfaceFormatCount * sizeof(VkSurfaceFormatKHR));
140 VkSurfaceFormatKHR* surfaceFormats = (VkSurfaceFormatKHR*)surfaceFormatAlloc.get();
Greg Danielf730c182018-07-02 20:15:37 +0000141 res = fGetPhysicalDeviceSurfaceFormatsKHR(fPhysicalDevice, fSurface, &surfaceFormatCount,
142 surfaceFormats);
jvanverth9f372462016-04-06 06:08:59 -0700143 if (VK_SUCCESS != res) {
144 return false;
145 }
146
147 uint32_t presentModeCount;
Greg Danielf730c182018-07-02 20:15:37 +0000148 res = fGetPhysicalDeviceSurfacePresentModesKHR(fPhysicalDevice, fSurface, &presentModeCount,
149 nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700150 if (VK_SUCCESS != res) {
151 return false;
152 }
153
154 SkAutoMalloc presentModeAlloc(presentModeCount * sizeof(VkPresentModeKHR));
155 VkPresentModeKHR* presentModes = (VkPresentModeKHR*)presentModeAlloc.get();
Greg Danielf730c182018-07-02 20:15:37 +0000156 res = fGetPhysicalDeviceSurfacePresentModesKHR(fPhysicalDevice, fSurface, &presentModeCount,
157 presentModes);
jvanverth9f372462016-04-06 06:08:59 -0700158 if (VK_SUCCESS != res) {
159 return false;
160 }
161
162 VkExtent2D extent = caps.currentExtent;
163 // use the hints
164 if (extent.width == (uint32_t)-1) {
165 extent.width = width;
166 extent.height = height;
167 }
168
jvanverth9fab59d2016-04-06 12:08:51 -0700169 // clamp width; to protect us from broken hints
jvanverth9f372462016-04-06 06:08:59 -0700170 if (extent.width < caps.minImageExtent.width) {
171 extent.width = caps.minImageExtent.width;
172 } else if (extent.width > caps.maxImageExtent.width) {
173 extent.width = caps.maxImageExtent.width;
174 }
175 // clamp height
176 if (extent.height < caps.minImageExtent.height) {
177 extent.height = caps.minImageExtent.height;
178 } else if (extent.height > caps.maxImageExtent.height) {
179 extent.height = caps.maxImageExtent.height;
180 }
jvanverth1d155962016-05-23 13:13:36 -0700181
jvanverth9f372462016-04-06 06:08:59 -0700182 fWidth = (int)extent.width;
183 fHeight = (int)extent.height;
184
185 uint32_t imageCount = caps.minImageCount + 2;
186 if (caps.maxImageCount > 0 && imageCount > caps.maxImageCount) {
187 // Application must settle for fewer images than desired:
188 imageCount = caps.maxImageCount;
189 }
190
191 VkImageUsageFlags usageFlags = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
192 VK_IMAGE_USAGE_TRANSFER_SRC_BIT |
193 VK_IMAGE_USAGE_TRANSFER_DST_BIT;
194 SkASSERT((caps.supportedUsageFlags & usageFlags) == usageFlags);
195 SkASSERT(caps.supportedTransforms & caps.currentTransform);
196 SkASSERT(caps.supportedCompositeAlpha & (VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR |
197 VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR));
198 VkCompositeAlphaFlagBitsKHR composite_alpha =
199 (caps.supportedCompositeAlpha & VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR) ?
200 VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR :
201 VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
202
Brian Osman294fe292018-07-09 10:23:19 -0400203 // Pick our surface format.
brianosman05de2162016-05-06 13:28:57 -0700204 VkFormat surfaceFormat = VK_FORMAT_UNDEFINED;
205 VkColorSpaceKHR colorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
brianosman05de2162016-05-06 13:28:57 -0700206 for (uint32_t i = 0; i < surfaceFormatCount; ++i) {
Greg Daniel81b80592017-12-13 10:20:04 -0500207 VkFormat localFormat = surfaceFormats[i].format;
Brian Osman2b23c4b2018-06-01 12:25:08 -0400208 if (GrVkFormatIsSupported(localFormat)) {
Greg Daniel81b80592017-12-13 10:20:04 -0500209 surfaceFormat = localFormat;
brianosman05de2162016-05-06 13:28:57 -0700210 colorSpace = surfaceFormats[i].colorSpace;
211 break;
212 }
213 }
214 fDisplayParams = params;
csmartdalton578f0642017-02-24 16:04:47 -0700215 fSampleCount = params.fMSAASampleCount;
216 fStencilBits = 8;
brianosman05de2162016-05-06 13:28:57 -0700217
218 if (VK_FORMAT_UNDEFINED == surfaceFormat) {
219 return false;
220 }
jvanvertha4b0fed2016-04-27 11:42:21 -0700221
Greg Danielfaa095e2017-12-19 13:15:02 -0500222 SkColorType colorType;
223 switch (surfaceFormat) {
224 case VK_FORMAT_R8G8B8A8_UNORM: // fall through
225 case VK_FORMAT_R8G8B8A8_SRGB:
226 colorType = kRGBA_8888_SkColorType;
227 break;
228 case VK_FORMAT_B8G8R8A8_UNORM: // fall through
229 case VK_FORMAT_B8G8R8A8_SRGB:
230 colorType = kBGRA_8888_SkColorType;
231 break;
232 default:
233 return false;
234 }
235
jvanverth3d6ed3a2016-04-07 11:09:51 -0700236 // If mailbox mode is available, use it, as it is the lowest-latency non-
237 // tearing mode. If not, fall back to FIFO which is always available.
jvanverth9f372462016-04-06 06:08:59 -0700238 VkPresentModeKHR mode = VK_PRESENT_MODE_FIFO_KHR;
jvanverth9f372462016-04-06 06:08:59 -0700239 for (uint32_t i = 0; i < presentModeCount; ++i) {
jvanverth3d6ed3a2016-04-07 11:09:51 -0700240 // use mailbox
241 if (VK_PRESENT_MODE_MAILBOX_KHR == presentModes[i]) {
jvanverth9f372462016-04-06 06:08:59 -0700242 mode = presentModes[i];
jvanverth3d6ed3a2016-04-07 11:09:51 -0700243 break;
jvanverth9f372462016-04-06 06:08:59 -0700244 }
245 }
246
247 VkSwapchainCreateInfoKHR swapchainCreateInfo;
248 memset(&swapchainCreateInfo, 0, sizeof(VkSwapchainCreateInfoKHR));
249 swapchainCreateInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
250 swapchainCreateInfo.surface = fSurface;
251 swapchainCreateInfo.minImageCount = imageCount;
jvanvertha4b0fed2016-04-27 11:42:21 -0700252 swapchainCreateInfo.imageFormat = surfaceFormat;
253 swapchainCreateInfo.imageColorSpace = colorSpace;
jvanverth9f372462016-04-06 06:08:59 -0700254 swapchainCreateInfo.imageExtent = extent;
255 swapchainCreateInfo.imageArrayLayers = 1;
256 swapchainCreateInfo.imageUsage = usageFlags;
257
Greg Danielf730c182018-07-02 20:15:37 +0000258 uint32_t queueFamilies[] = { fGraphicsQueueIndex, fPresentQueueIndex };
259 if (fGraphicsQueueIndex != fPresentQueueIndex) {
jvanverth9f372462016-04-06 06:08:59 -0700260 swapchainCreateInfo.imageSharingMode = VK_SHARING_MODE_CONCURRENT;
261 swapchainCreateInfo.queueFamilyIndexCount = 2;
262 swapchainCreateInfo.pQueueFamilyIndices = queueFamilies;
263 } else {
264 swapchainCreateInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
265 swapchainCreateInfo.queueFamilyIndexCount = 0;
266 swapchainCreateInfo.pQueueFamilyIndices = nullptr;
267 }
268
Greg Daniele897b972016-10-06 13:57:57 -0400269 swapchainCreateInfo.preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
jvanverth9f372462016-04-06 06:08:59 -0700270 swapchainCreateInfo.compositeAlpha = composite_alpha;
271 swapchainCreateInfo.presentMode = mode;
272 swapchainCreateInfo.clipped = true;
273 swapchainCreateInfo.oldSwapchain = fSwapchain;
274
Greg Danielf730c182018-07-02 20:15:37 +0000275 res = fCreateSwapchainKHR(fDevice, &swapchainCreateInfo, nullptr, &fSwapchain);
jvanverth9f372462016-04-06 06:08:59 -0700276 if (VK_SUCCESS != res) {
277 return false;
278 }
279
280 // destroy the old swapchain
281 if (swapchainCreateInfo.oldSwapchain != VK_NULL_HANDLE) {
Greg Danielf730c182018-07-02 20:15:37 +0000282 fDeviceWaitIdle(fDevice);
jvanverth9f372462016-04-06 06:08:59 -0700283
284 this->destroyBuffers();
285
Greg Danielf730c182018-07-02 20:15:37 +0000286 fDestroySwapchainKHR(fDevice, swapchainCreateInfo.oldSwapchain, nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700287 }
288
Greg Danielfaa095e2017-12-19 13:15:02 -0500289 this->createBuffers(swapchainCreateInfo.imageFormat, colorType);
jvanverth9f372462016-04-06 06:08:59 -0700290
291 return true;
292}
293
Greg Danielfaa095e2017-12-19 13:15:02 -0500294void VulkanWindowContext::createBuffers(VkFormat format, SkColorType colorType) {
Greg Danielf730c182018-07-02 20:15:37 +0000295 fGetSwapchainImagesKHR(fDevice, fSwapchain, &fImageCount, nullptr);
jvanverth9f372462016-04-06 06:08:59 -0700296 SkASSERT(fImageCount);
297 fImages = new VkImage[fImageCount];
Greg Danielf730c182018-07-02 20:15:37 +0000298 fGetSwapchainImagesKHR(fDevice, fSwapchain, &fImageCount, fImages);
jvanverth9f372462016-04-06 06:08:59 -0700299
300 // set up initial image layouts and create surfaces
301 fImageLayouts = new VkImageLayout[fImageCount];
302 fSurfaces = new sk_sp<SkSurface>[fImageCount];
303 for (uint32_t i = 0; i < fImageCount; ++i) {
304 fImageLayouts[i] = VK_IMAGE_LAYOUT_UNDEFINED;
305
egdanielb2df0c22016-05-13 11:30:37 -0700306 GrVkImageInfo info;
jvanverth9f372462016-04-06 06:08:59 -0700307 info.fImage = fImages[i];
Greg Daniel8385a8a2018-02-26 13:29:37 -0500308 info.fAlloc = GrVkAlloc();
egdaniel580fa592016-08-31 11:03:50 -0700309 info.fImageLayout = VK_IMAGE_LAYOUT_UNDEFINED;
jvanverth9f372462016-04-06 06:08:59 -0700310 info.fImageTiling = VK_IMAGE_TILING_OPTIMAL;
egdaniel58a8d922016-04-21 08:03:10 -0700311 info.fFormat = format;
jvanverth3622a172016-05-10 06:42:18 -0700312 info.fLevelCount = 1;
jvanverthaf236b52016-05-20 06:01:06 -0700313
Brian Salomonafdc6b12018-03-09 12:02:32 -0500314 GrBackendRenderTarget backendRT(fWidth, fHeight, fSampleCount, info);
Greg Daniele79b4732017-04-20 14:07:46 -0400315
Brian Salomonafdc6b12018-03-09 12:02:32 -0500316 fSurfaces[i] = SkSurface::MakeFromBackendRenderTarget(fContext.get(),
317 backendRT,
318 kTopLeft_GrSurfaceOrigin,
319 colorType,
320 fDisplayParams.fColorSpace,
Ben Wagner37c54032018-04-13 14:30:23 -0400321 &fDisplayParams.fSurfaceProps);
jvanverth9f372462016-04-06 06:08:59 -0700322 }
323
324 // create the command pool for the command buffers
325 if (VK_NULL_HANDLE == fCommandPool) {
326 VkCommandPoolCreateInfo commandPoolInfo;
327 memset(&commandPoolInfo, 0, sizeof(VkCommandPoolCreateInfo));
328 commandPoolInfo.sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO;
329 // this needs to be on the render queue
Greg Danielf730c182018-07-02 20:15:37 +0000330 commandPoolInfo.queueFamilyIndex = fGraphicsQueueIndex;
jvanverth9f372462016-04-06 06:08:59 -0700331 commandPoolInfo.flags = VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT;
Greg Danielf730c182018-07-02 20:15:37 +0000332 GR_VK_CALL_ERRCHECK(fInterface,
333 CreateCommandPool(fDevice, &commandPoolInfo,
jvanverth9f372462016-04-06 06:08:59 -0700334 nullptr, &fCommandPool));
335 }
336
337 // set up the backbuffers
338 VkSemaphoreCreateInfo semaphoreInfo;
339 memset(&semaphoreInfo, 0, sizeof(VkSemaphoreCreateInfo));
340 semaphoreInfo.sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO;
341 semaphoreInfo.pNext = nullptr;
342 semaphoreInfo.flags = 0;
343 VkCommandBufferAllocateInfo commandBuffersInfo;
344 memset(&commandBuffersInfo, 0, sizeof(VkCommandBufferAllocateInfo));
345 commandBuffersInfo.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO;
346 commandBuffersInfo.pNext = nullptr;
347 commandBuffersInfo.commandPool = fCommandPool;
348 commandBuffersInfo.level = VK_COMMAND_BUFFER_LEVEL_PRIMARY;
349 commandBuffersInfo.commandBufferCount = 2;
350 VkFenceCreateInfo fenceInfo;
351 memset(&fenceInfo, 0, sizeof(VkFenceCreateInfo));
352 fenceInfo.sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO;
353 fenceInfo.pNext = nullptr;
354 fenceInfo.flags = VK_FENCE_CREATE_SIGNALED_BIT;
355
Greg Daniel1f05f442016-10-27 16:37:17 -0400356 // we create one additional backbuffer structure here, because we want to
jvanverth9f372462016-04-06 06:08:59 -0700357 // give the command buffers they contain a chance to finish before we cycle back
358 fBackbuffers = new BackbufferInfo[fImageCount + 1];
359 for (uint32_t i = 0; i < fImageCount + 1; ++i) {
360 fBackbuffers[i].fImageIndex = -1;
Greg Danielf730c182018-07-02 20:15:37 +0000361 GR_VK_CALL_ERRCHECK(fInterface,
362 CreateSemaphore(fDevice, &semaphoreInfo,
jvanverth9f372462016-04-06 06:08:59 -0700363 nullptr, &fBackbuffers[i].fAcquireSemaphore));
Greg Danielf730c182018-07-02 20:15:37 +0000364 GR_VK_CALL_ERRCHECK(fInterface,
365 CreateSemaphore(fDevice, &semaphoreInfo,
jvanverth9f372462016-04-06 06:08:59 -0700366 nullptr, &fBackbuffers[i].fRenderSemaphore));
Greg Danielf730c182018-07-02 20:15:37 +0000367 GR_VK_CALL_ERRCHECK(fInterface,
368 AllocateCommandBuffers(fDevice, &commandBuffersInfo,
jvanverth9f372462016-04-06 06:08:59 -0700369 fBackbuffers[i].fTransitionCmdBuffers));
Greg Danielf730c182018-07-02 20:15:37 +0000370 GR_VK_CALL_ERRCHECK(fInterface,
371 CreateFence(fDevice, &fenceInfo, nullptr,
jvanverth9f372462016-04-06 06:08:59 -0700372 &fBackbuffers[i].fUsageFences[0]));
Greg Danielf730c182018-07-02 20:15:37 +0000373 GR_VK_CALL_ERRCHECK(fInterface,
374 CreateFence(fDevice, &fenceInfo, nullptr,
jvanverth9f372462016-04-06 06:08:59 -0700375 &fBackbuffers[i].fUsageFences[1]));
376 }
377 fCurrentBackbufferIndex = fImageCount;
378}
379
jvanvertha8d0d6c2016-05-05 12:32:03 -0700380void VulkanWindowContext::destroyBuffers() {
jvanverth9f372462016-04-06 06:08:59 -0700381
382 if (fBackbuffers) {
383 for (uint32_t i = 0; i < fImageCount + 1; ++i) {
Greg Danielf730c182018-07-02 20:15:37 +0000384 GR_VK_CALL_ERRCHECK(fInterface,
385 WaitForFences(fDevice, 2,
jvanverth9f372462016-04-06 06:08:59 -0700386 fBackbuffers[i].fUsageFences,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700387 true, UINT64_MAX));
jvanverth9f372462016-04-06 06:08:59 -0700388 fBackbuffers[i].fImageIndex = -1;
Greg Danielf730c182018-07-02 20:15:37 +0000389 GR_VK_CALL(fInterface,
390 DestroySemaphore(fDevice,
jvanverth9f372462016-04-06 06:08:59 -0700391 fBackbuffers[i].fAcquireSemaphore,
392 nullptr));
Greg Danielf730c182018-07-02 20:15:37 +0000393 GR_VK_CALL(fInterface,
394 DestroySemaphore(fDevice,
jvanverth9f372462016-04-06 06:08:59 -0700395 fBackbuffers[i].fRenderSemaphore,
396 nullptr));
Greg Danielf730c182018-07-02 20:15:37 +0000397 GR_VK_CALL(fInterface,
398 FreeCommandBuffers(fDevice, fCommandPool, 2,
jvanverth9f372462016-04-06 06:08:59 -0700399 fBackbuffers[i].fTransitionCmdBuffers));
Greg Danielf730c182018-07-02 20:15:37 +0000400 GR_VK_CALL(fInterface,
401 DestroyFence(fDevice, fBackbuffers[i].fUsageFences[0], 0));
402 GR_VK_CALL(fInterface,
403 DestroyFence(fDevice, fBackbuffers[i].fUsageFences[1], 0));
jvanverth9f372462016-04-06 06:08:59 -0700404 }
405 }
406
407 delete[] fBackbuffers;
408 fBackbuffers = nullptr;
409
jvanverthaf236b52016-05-20 06:01:06 -0700410 // Does this actually free the surfaces?
jvanverth9f372462016-04-06 06:08:59 -0700411 delete[] fSurfaces;
412 fSurfaces = nullptr;
413 delete[] fImageLayouts;
414 fImageLayouts = nullptr;
415 delete[] fImages;
416 fImages = nullptr;
417}
418
jvanvertha8d0d6c2016-05-05 12:32:03 -0700419VulkanWindowContext::~VulkanWindowContext() {
jvanverth9f372462016-04-06 06:08:59 -0700420 this->destroyContext();
421}
422
jvanvertha8d0d6c2016-05-05 12:32:03 -0700423void VulkanWindowContext::destroyContext() {
Greg Daniel37329b32018-07-02 20:16:44 +0000424 if (this->isValid()) {
425 fQueueWaitIdle(fPresentQueue);
426 fDeviceWaitIdle(fDevice);
jvanverth9f372462016-04-06 06:08:59 -0700427
Greg Daniel37329b32018-07-02 20:16:44 +0000428 this->destroyBuffers();
jvanverth9f372462016-04-06 06:08:59 -0700429
Greg Daniel37329b32018-07-02 20:16:44 +0000430 if (VK_NULL_HANDLE != fCommandPool) {
431 GR_VK_CALL(fInterface, DestroyCommandPool(fDevice, fCommandPool, nullptr));
432 fCommandPool = VK_NULL_HANDLE;
433 }
jvanverth9f372462016-04-06 06:08:59 -0700434
Greg Daniel37329b32018-07-02 20:16:44 +0000435 if (VK_NULL_HANDLE != fSwapchain) {
436 fDestroySwapchainKHR(fDevice, fSwapchain, nullptr);
437 fSwapchain = VK_NULL_HANDLE;
438 }
jvanverth9f372462016-04-06 06:08:59 -0700439
Greg Daniel37329b32018-07-02 20:16:44 +0000440 if (VK_NULL_HANDLE != fSurface) {
441 fDestroySurfaceKHR(fInstance, fSurface, nullptr);
442 fSurface = VK_NULL_HANDLE;
443 }
jvanverth9f372462016-04-06 06:08:59 -0700444 }
445
Greg Daniel02611d92017-07-25 10:05:01 -0400446 fContext.reset();
Greg Danielf730c182018-07-02 20:15:37 +0000447 fInterface.reset();
jvanverth9f372462016-04-06 06:08:59 -0700448
Greg Danielf730c182018-07-02 20:15:37 +0000449 if (VK_NULL_HANDLE != fDevice) {
450 fDestroyDevice(fDevice, nullptr);
451 fDevice = VK_NULL_HANDLE;
452 }
Greg Daniel37329b32018-07-02 20:16:44 +0000453
454#ifdef SK_ENABLE_VK_LAYERS
455 if (fDebugCallback != VK_NULL_HANDLE) {
456 GR_VK_CALL(fInterface, DestroyDebugReportCallbackEXT(fInstance, fDebugCallback,
457 nullptr));
458 }
459#endif
460
Greg Danielf730c182018-07-02 20:15:37 +0000461 fPhysicalDevice = VK_NULL_HANDLE;
462
463 if (VK_NULL_HANDLE != fInstance) {
464 fDestroyInstance(fInstance, nullptr);
465 fInstance = VK_NULL_HANDLE;
466 }
jvanverth9f372462016-04-06 06:08:59 -0700467}
468
jvanvertha8d0d6c2016-05-05 12:32:03 -0700469VulkanWindowContext::BackbufferInfo* VulkanWindowContext::getAvailableBackbuffer() {
jvanverth9f372462016-04-06 06:08:59 -0700470 SkASSERT(fBackbuffers);
471
472 ++fCurrentBackbufferIndex;
egdaniel804af7d2016-09-26 12:30:46 -0700473 if (fCurrentBackbufferIndex > fImageCount) {
jvanverth9f372462016-04-06 06:08:59 -0700474 fCurrentBackbufferIndex = 0;
475 }
476
477 BackbufferInfo* backbuffer = fBackbuffers + fCurrentBackbufferIndex;
Greg Danielf730c182018-07-02 20:15:37 +0000478 GR_VK_CALL_ERRCHECK(fInterface,
479 WaitForFences(fDevice, 2, backbuffer->fUsageFences,
jvanverth9f372462016-04-06 06:08:59 -0700480 true, UINT64_MAX));
481 return backbuffer;
482}
483
jvanverthaf236b52016-05-20 06:01:06 -0700484sk_sp<SkSurface> VulkanWindowContext::getBackbufferSurface() {
jvanverth9f372462016-04-06 06:08:59 -0700485 BackbufferInfo* backbuffer = this->getAvailableBackbuffer();
486 SkASSERT(backbuffer);
487
488 // reset the fence
Greg Danielf730c182018-07-02 20:15:37 +0000489 GR_VK_CALL_ERRCHECK(fInterface,
490 ResetFences(fDevice, 2, backbuffer->fUsageFences));
jvanverth9f372462016-04-06 06:08:59 -0700491 // semaphores should be in unsignaled state
492
493 // acquire the image
Greg Danielf730c182018-07-02 20:15:37 +0000494 VkResult res = fAcquireNextImageKHR(fDevice, fSwapchain, UINT64_MAX,
jvanverthb0d43522016-04-21 11:46:23 -0700495 backbuffer->fAcquireSemaphore, VK_NULL_HANDLE,
496 &backbuffer->fImageIndex);
jvanverth9f372462016-04-06 06:08:59 -0700497 if (VK_ERROR_SURFACE_LOST_KHR == res) {
498 // need to figure out how to create a new vkSurface without the platformData*
jvanverth3d6ed3a2016-04-07 11:09:51 -0700499 // maybe use attach somehow? but need a Window
jvanverth9f372462016-04-06 06:08:59 -0700500 return nullptr;
501 }
jvanverth3d6ed3a2016-04-07 11:09:51 -0700502 if (VK_ERROR_OUT_OF_DATE_KHR == res) {
jvanverth9f372462016-04-06 06:08:59 -0700503 // tear swapchain down and try again
Greg Daniel1f05f442016-10-27 16:37:17 -0400504 if (!this->createSwapchain(-1, -1, fDisplayParams)) {
jvanverth9f372462016-04-06 06:08:59 -0700505 return nullptr;
506 }
Jim Van Verthd63c1022017-01-05 13:50:49 -0500507 backbuffer = this->getAvailableBackbuffer();
Greg Danielf730c182018-07-02 20:15:37 +0000508 GR_VK_CALL_ERRCHECK(fInterface,
509 ResetFences(fDevice, 2, backbuffer->fUsageFences));
jvanverth9f372462016-04-06 06:08:59 -0700510
511 // acquire the image
Greg Danielf730c182018-07-02 20:15:37 +0000512 res = fAcquireNextImageKHR(fDevice, fSwapchain, UINT64_MAX,
jvanverthb0d43522016-04-21 11:46:23 -0700513 backbuffer->fAcquireSemaphore, VK_NULL_HANDLE,
514 &backbuffer->fImageIndex);
jvanverth9f372462016-04-06 06:08:59 -0700515
516 if (VK_SUCCESS != res) {
517 return nullptr;
518 }
519 }
520
521 // set up layout transfer from initial to color attachment
522 VkImageLayout layout = fImageLayouts[backbuffer->fImageIndex];
egdaniel580fa592016-08-31 11:03:50 -0700523 SkASSERT(VK_IMAGE_LAYOUT_UNDEFINED == layout || VK_IMAGE_LAYOUT_PRESENT_SRC_KHR == layout);
jvanverth9f372462016-04-06 06:08:59 -0700524 VkPipelineStageFlags srcStageMask = (VK_IMAGE_LAYOUT_UNDEFINED == layout) ?
525 VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT :
526 VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
527 VkPipelineStageFlags dstStageMask = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
jvanvertha8d0d6c2016-05-05 12:32:03 -0700528 VkAccessFlags srcAccessMask = (VK_IMAGE_LAYOUT_UNDEFINED == layout) ?
jvanverth9f372462016-04-06 06:08:59 -0700529 0 : VK_ACCESS_MEMORY_READ_BIT;
530 VkAccessFlags dstAccessMask = VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT;
531
532 VkImageMemoryBarrier imageMemoryBarrier = {
533 VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER, // sType
534 NULL, // pNext
535 srcAccessMask, // outputMask
536 dstAccessMask, // inputMask
537 layout, // oldLayout
538 VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, // newLayout
539 fPresentQueueIndex, // srcQueueFamilyIndex
Greg Danielf730c182018-07-02 20:15:37 +0000540 fGraphicsQueueIndex, // dstQueueFamilyIndex
jvanverth9f372462016-04-06 06:08:59 -0700541 fImages[backbuffer->fImageIndex], // image
542 { VK_IMAGE_ASPECT_COLOR_BIT, 0, 1, 0, 1 } // subresourceRange
543 };
Greg Danielf730c182018-07-02 20:15:37 +0000544 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700545 ResetCommandBuffer(backbuffer->fTransitionCmdBuffers[0], 0));
546 VkCommandBufferBeginInfo info;
547 memset(&info, 0, sizeof(VkCommandBufferBeginInfo));
548 info.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO;
549 info.flags = 0;
Greg Danielf730c182018-07-02 20:15:37 +0000550 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700551 BeginCommandBuffer(backbuffer->fTransitionCmdBuffers[0], &info));
552
Greg Danielf730c182018-07-02 20:15:37 +0000553 GR_VK_CALL(fInterface,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700554 CmdPipelineBarrier(backbuffer->fTransitionCmdBuffers[0],
555 srcStageMask, dstStageMask, 0,
556 0, nullptr,
557 0, nullptr,
558 1, &imageMemoryBarrier));
jvanverth9f372462016-04-06 06:08:59 -0700559
Greg Danielf730c182018-07-02 20:15:37 +0000560 GR_VK_CALL_ERRCHECK(fInterface,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700561 EndCommandBuffer(backbuffer->fTransitionCmdBuffers[0]));
jvanverth9f372462016-04-06 06:08:59 -0700562
egdaniel58a8d922016-04-21 08:03:10 -0700563 VkPipelineStageFlags waitDstStageFlags = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
jvanverth9f372462016-04-06 06:08:59 -0700564 // insert the layout transfer into the queue and wait on the acquire
565 VkSubmitInfo submitInfo;
566 memset(&submitInfo, 0, sizeof(VkSubmitInfo));
567 submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO;
568 submitInfo.waitSemaphoreCount = 1;
569 submitInfo.pWaitSemaphores = &backbuffer->fAcquireSemaphore;
egdaniel58a8d922016-04-21 08:03:10 -0700570 submitInfo.pWaitDstStageMask = &waitDstStageFlags;
jvanverth9f372462016-04-06 06:08:59 -0700571 submitInfo.commandBufferCount = 1;
572 submitInfo.pCommandBuffers = &backbuffer->fTransitionCmdBuffers[0];
573 submitInfo.signalSemaphoreCount = 0;
jvanvertha8d0d6c2016-05-05 12:32:03 -0700574
Greg Danielf730c182018-07-02 20:15:37 +0000575 GR_VK_CALL_ERRCHECK(fInterface,
576 QueueSubmit(fGraphicsQueue, 1, &submitInfo,
jvanverth9f372462016-04-06 06:08:59 -0700577 backbuffer->fUsageFences[0]));
578
egdaniel580fa592016-08-31 11:03:50 -0700579 SkSurface* surface = fSurfaces[backbuffer->fImageIndex].get();
Robert Phillipsba375a82018-04-11 10:08:06 -0400580 GrBackendRenderTarget backendRT = surface->getBackendRenderTarget(
581 SkSurface::kFlushRead_BackendHandleAccess);
582 backendRT.setVkImageLayout(VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
583
egdaniel580fa592016-08-31 11:03:50 -0700584
585 return sk_ref_sp(surface);
jvanverth9f372462016-04-06 06:08:59 -0700586}
587
jvanvertha8d0d6c2016-05-05 12:32:03 -0700588void VulkanWindowContext::swapBuffers() {
jvanverth9f372462016-04-06 06:08:59 -0700589
590 BackbufferInfo* backbuffer = fBackbuffers + fCurrentBackbufferIndex;
egdaniel580fa592016-08-31 11:03:50 -0700591 SkSurface* surface = fSurfaces[backbuffer->fImageIndex].get();
jvanverth9f372462016-04-06 06:08:59 -0700592
Robert Phillipsba375a82018-04-11 10:08:06 -0400593 GrBackendRenderTarget backendRT = surface->getBackendRenderTarget(
594 SkSurface::kFlushRead_BackendHandleAccess);
595 GrVkImageInfo imageInfo;
596 SkAssertResult(backendRT.getVkImageInfo(&imageInfo));
597 // Check to make sure we never change the actually wrapped image
598 SkASSERT(imageInfo.fImage == fImages[backbuffer->fImageIndex]);
599
600 VkImageLayout layout = imageInfo.fImageLayout;
Greg Daniel6ddbafc2018-05-24 12:34:29 -0400601 VkPipelineStageFlags srcStageMask = GrVkImage::LayoutToPipelineStageFlags(layout);
jvanverth9f372462016-04-06 06:08:59 -0700602 VkPipelineStageFlags dstStageMask = VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT;
Greg Daniel6ddbafc2018-05-24 12:34:29 -0400603 VkAccessFlags srcAccessMask = GrVkImage::LayoutToSrcAccessMask(layout);
jvanverth9f372462016-04-06 06:08:59 -0700604 VkAccessFlags dstAccessMask = VK_ACCESS_MEMORY_READ_BIT;
605
606 VkImageMemoryBarrier imageMemoryBarrier = {
607 VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER, // sType
608 NULL, // pNext
609 srcAccessMask, // outputMask
610 dstAccessMask, // inputMask
611 layout, // oldLayout
612 VK_IMAGE_LAYOUT_PRESENT_SRC_KHR, // newLayout
Greg Danielf730c182018-07-02 20:15:37 +0000613 fGraphicsQueueIndex, // srcQueueFamilyIndex
jvanverth9f372462016-04-06 06:08:59 -0700614 fPresentQueueIndex, // dstQueueFamilyIndex
615 fImages[backbuffer->fImageIndex], // image
616 { VK_IMAGE_ASPECT_COLOR_BIT, 0, 1, 0, 1 } // subresourceRange
617 };
Greg Danielf730c182018-07-02 20:15:37 +0000618 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700619 ResetCommandBuffer(backbuffer->fTransitionCmdBuffers[1], 0));
620 VkCommandBufferBeginInfo info;
621 memset(&info, 0, sizeof(VkCommandBufferBeginInfo));
622 info.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO;
623 info.flags = 0;
Greg Danielf730c182018-07-02 20:15:37 +0000624 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700625 BeginCommandBuffer(backbuffer->fTransitionCmdBuffers[1], &info));
Greg Danielf730c182018-07-02 20:15:37 +0000626 GR_VK_CALL(fInterface,
jvanvertha8d0d6c2016-05-05 12:32:03 -0700627 CmdPipelineBarrier(backbuffer->fTransitionCmdBuffers[1],
628 srcStageMask, dstStageMask, 0,
629 0, nullptr,
630 0, nullptr,
631 1, &imageMemoryBarrier));
Greg Danielf730c182018-07-02 20:15:37 +0000632 GR_VK_CALL_ERRCHECK(fInterface,
jvanverth9f372462016-04-06 06:08:59 -0700633 EndCommandBuffer(backbuffer->fTransitionCmdBuffers[1]));
634
635 fImageLayouts[backbuffer->fImageIndex] = VK_IMAGE_LAYOUT_PRESENT_SRC_KHR;
636
637 // insert the layout transfer into the queue and wait on the acquire
638 VkSubmitInfo submitInfo;
639 memset(&submitInfo, 0, sizeof(VkSubmitInfo));
640 submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO;
641 submitInfo.waitSemaphoreCount = 0;
642 submitInfo.pWaitDstStageMask = 0;
643 submitInfo.commandBufferCount = 1;
644 submitInfo.pCommandBuffers = &backbuffer->fTransitionCmdBuffers[1];
645 submitInfo.signalSemaphoreCount = 1;
646 submitInfo.pSignalSemaphores = &backbuffer->fRenderSemaphore;
647
Greg Danielf730c182018-07-02 20:15:37 +0000648 GR_VK_CALL_ERRCHECK(fInterface,
649 QueueSubmit(fGraphicsQueue, 1, &submitInfo,
jvanverth9f372462016-04-06 06:08:59 -0700650 backbuffer->fUsageFences[1]));
651
652 // Submit present operation to present queue
653 const VkPresentInfoKHR presentInfo =
654 {
655 VK_STRUCTURE_TYPE_PRESENT_INFO_KHR, // sType
656 NULL, // pNext
657 1, // waitSemaphoreCount
658 &backbuffer->fRenderSemaphore, // pWaitSemaphores
659 1, // swapchainCount
660 &fSwapchain, // pSwapchains
661 &backbuffer->fImageIndex, // pImageIndices
662 NULL // pResults
663 };
664
jvanverthb0d43522016-04-21 11:46:23 -0700665 fQueuePresentKHR(fPresentQueue, &presentInfo);
jvanverth9f372462016-04-06 06:08:59 -0700666}
jvanvertha8d0d6c2016-05-05 12:32:03 -0700667
668} //namespace sk_app