vengine  0.0.1
3D graphics engine
Loading...
Searching...
No Matches
swapChain.cpp
Go to the documentation of this file.
1#include <iostream>
2#include <limits>
3#include <array>
4
6
8{
9 for (VkImageView_T *imageView : m_swapChainImageViews) {
10 vkDestroyImageView(m_device.device(), imageView, nullptr);
11 }
13
14 if (m_swapChain != nullptr) {
15 vkDestroySwapchainKHR(m_device.device(), m_swapChain, nullptr);
16 m_swapChain = nullptr;
17 }
18
19 for (size_t i = 0; i < m_depthImages.size(); i++) {
20 vkDestroyImageView(m_device.device(), m_depthImageViews[i], nullptr);
21 vkDestroyImage(m_device.device(), m_depthImages[i], nullptr);
22 vkFreeMemory(m_device.device(), m_depthImageMemory[i], nullptr);
23 }
24
25 for (VkFramebuffer_T *framebuffer : m_swapChainFrameBuffers) {
26 vkDestroyFramebuffer(m_device.device(), framebuffer, nullptr);
27 }
28
29 vkDestroyRenderPass(m_device.device(), m_renderPass, nullptr);
30
31 // cleanup synchronization objects
32 for (size_t i = 0; i < MAX_FRAMES_IN_FLIGHT; i++) {
33 vkDestroySemaphore(m_device.device(), m_renderFinishedSemaphores[i], nullptr);
34 vkDestroySemaphore(m_device.device(), m_imageAvailableSemaphores[i], nullptr);
35 vkDestroyFence(m_device.device(), m_inFlightFences[i], nullptr);
36 }
37}
38
40{
41 createSwapChain();
42 createImageViews();
43 createRenderPass();
44 createDepthResources();
45 createFrameBuffers();
46 createSyncObjects();
47}
48
49VkResult ven::SwapChain::acquireNextImage(uint32_t *imageIndex) const
50{
51 vkWaitForFences(m_device.device(), 1, &m_inFlightFences[m_currentFrame], VK_TRUE, std::numeric_limits<uint64_t>::max());
52
53 return vkAcquireNextImageKHR(m_device.device(), m_swapChain, std::numeric_limits<uint64_t>::max(), m_imageAvailableSemaphores[m_currentFrame], VK_NULL_HANDLE, imageIndex);;
54}
55
56VkResult ven::SwapChain::submitCommandBuffers(const VkCommandBuffer *buffers, const uint32_t *imageIndex)
57{
58 if (m_imagesInFlight[*imageIndex] != VK_NULL_HANDLE) {
59 vkWaitForFences(m_device.device(), 1, &m_imagesInFlight[*imageIndex], VK_TRUE, UINT64_MAX);
60 }
61 m_imagesInFlight[*imageIndex] = m_inFlightFences[m_currentFrame];
62
63 VkSubmitInfo submitInfo = {};
64 submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO;
65
66 const std::array<VkSemaphore, 1> waitSemaphores = {m_imageAvailableSemaphores[m_currentFrame]};
67 constexpr std::array<VkPipelineStageFlags, 1> waitStages = {VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT};
68 submitInfo.waitSemaphoreCount = 1;
69 submitInfo.pWaitSemaphores = waitSemaphores.data();
70 submitInfo.pWaitDstStageMask = waitStages.data();
71
72 submitInfo.commandBufferCount = 1;
73 submitInfo.pCommandBuffers = buffers;
74
75 const std::array<VkSemaphore, 1> signalSemaphores = {m_renderFinishedSemaphores[m_currentFrame]};
76 submitInfo.signalSemaphoreCount = 1;
77 submitInfo.pSignalSemaphores = signalSemaphores.data();
78
79 vkResetFences(m_device.device(), 1, &m_inFlightFences[m_currentFrame]);
80 if (vkQueueSubmit(m_device.graphicsQueue(), 1, &submitInfo, m_inFlightFences[m_currentFrame]) != VK_SUCCESS) {
81 throw std::runtime_error("failed to submit draw command buffer!");
82 }
83
84 VkPresentInfoKHR presentInfo = {};
85 presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
86
87 presentInfo.waitSemaphoreCount = 1;
88 presentInfo.pWaitSemaphores = signalSemaphores.data();
89
90 const std::array<VkSwapchainKHR, 1> swapChains = {m_swapChain};
91 presentInfo.swapchainCount = 1;
92 presentInfo.pSwapchains = swapChains.data();
93
94 presentInfo.pImageIndices = imageIndex;
95
96 const VkResult result = vkQueuePresentKHR(m_device.presentQueue(), &presentInfo);
97
98 m_currentFrame = (m_currentFrame + 1) % MAX_FRAMES_IN_FLIGHT;
99
100 return result;
101}
102
104{
105 const auto [capabilities, formats, presentModes] = m_device.getSwapChainSupport();
106
107 const auto [format, colorSpace] = chooseSwapSurfaceFormat(formats);
108 const VkPresentModeKHR presentMode = chooseSwapPresentMode(presentModes);
109 const VkExtent2D extent = chooseSwapExtent(capabilities);
110
111 uint32_t imageCount = capabilities.minImageCount + 1;
112 if (capabilities.maxImageCount > 0 && imageCount > capabilities.maxImageCount) {
113 imageCount = capabilities.maxImageCount;
114 }
115
116 VkSwapchainCreateInfoKHR createInfo = {};
117 createInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
118 createInfo.surface = m_device.surface();
119
120 createInfo.minImageCount = imageCount;
121 createInfo.imageFormat = format;
122 createInfo.imageColorSpace = colorSpace;
123 createInfo.imageExtent = extent;
124 createInfo.imageArrayLayers = 1;
125 createInfo.imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
126
127 const auto [graphicsFamily, presentFamily, graphicsFamilyHasValue, presentFamilyHasValue] = m_device.findPhysicalQueueFamilies();
128 const std::array<uint32_t, 2> queueFamilyIndices = {graphicsFamily, presentFamily};
129
130 if (graphicsFamily != presentFamily) {
131 createInfo.imageSharingMode = VK_SHARING_MODE_CONCURRENT;
132 createInfo.queueFamilyIndexCount = 2;
133 createInfo.pQueueFamilyIndices = queueFamilyIndices.data();
134 } else {
135 createInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
136 createInfo.queueFamilyIndexCount = 0; // Optional
137 createInfo.pQueueFamilyIndices = nullptr; // Optional
138 }
139
140 createInfo.preTransform = capabilities.currentTransform;
141 createInfo.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
142
143 createInfo.presentMode = presentMode;
144 createInfo.clipped = VK_TRUE;
145
146 createInfo.oldSwapchain = m_oldSwapChain == nullptr ? VK_NULL_HANDLE : m_oldSwapChain->m_swapChain;
147
148 if (vkCreateSwapchainKHR(m_device.device(), &createInfo, nullptr, &m_swapChain) != VK_SUCCESS) {
149 throw std::runtime_error("failed to create swap chain!");
150 }
151
152 vkGetSwapchainImagesKHR(m_device.device(), m_swapChain, &imageCount, nullptr);
153 m_swapChainImages.resize(imageCount);
154 vkGetSwapchainImagesKHR(m_device.device(), m_swapChain, &imageCount, m_swapChainImages.data());
155
156 m_swapChainImageFormat = format;
157 m_swapChainExtent = extent;
158}
159
161{
162 m_swapChainImageViews.resize(m_swapChainImages.size());
163 for (size_t i = 0; i < m_swapChainImages.size(); i++) {
164 VkImageViewCreateInfo viewInfo{};
165 viewInfo.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
166 viewInfo.image = m_swapChainImages[i];
167 viewInfo.viewType = VK_IMAGE_VIEW_TYPE_2D;
168 viewInfo.format = m_swapChainImageFormat;
169 viewInfo.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
170 viewInfo.subresourceRange.baseMipLevel = 0;
171 viewInfo.subresourceRange.levelCount = 1;
172 viewInfo.subresourceRange.baseArrayLayer = 0;
173 viewInfo.subresourceRange.layerCount = 1;
174
175 if (vkCreateImageView(m_device.device(), &viewInfo, nullptr, &m_swapChainImageViews[i]) != VK_SUCCESS) {
176 throw std::runtime_error("failed to create texture image view!");
177 }
178 }
179}
180
182{
183 VkAttachmentDescription depthAttachment{};
184 depthAttachment.format = findDepthFormat();
185 depthAttachment.samples = VK_SAMPLE_COUNT_1_BIT;
186 depthAttachment.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
187 depthAttachment.storeOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
188 depthAttachment.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
189 depthAttachment.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
190 depthAttachment.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
191 depthAttachment.finalLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
192
193 VkAttachmentReference depthAttachmentRef{};
194 depthAttachmentRef.attachment = 1;
195 depthAttachmentRef.layout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL;
196
197 VkAttachmentDescription colorAttachment = {};
198 colorAttachment.format = getSwapChainImageFormat();
199 colorAttachment.samples = VK_SAMPLE_COUNT_1_BIT;
200 colorAttachment.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
201 colorAttachment.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
202 colorAttachment.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
203 colorAttachment.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
204 colorAttachment.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
205 colorAttachment.finalLayout = VK_IMAGE_LAYOUT_PRESENT_SRC_KHR;
206
207 VkAttachmentReference colorAttachmentRef = {};
208 colorAttachmentRef.attachment = 0;
209 colorAttachmentRef.layout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;
210
211 VkSubpassDescription subpass = {};
212 subpass.pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS;
213 subpass.colorAttachmentCount = 1;
214 subpass.pColorAttachments = &colorAttachmentRef;
215 subpass.pDepthStencilAttachment = &depthAttachmentRef;
216
217 VkSubpassDependency dependency = {};
218 dependency.srcSubpass = VK_SUBPASS_EXTERNAL;
219 dependency.srcAccessMask = 0;
220 dependency.srcStageMask = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT | VK_PIPELINE_STAGE_EARLY_FRAGMENT_TESTS_BIT;
221 dependency.dstSubpass = 0;
222 dependency.dstStageMask = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT | VK_PIPELINE_STAGE_EARLY_FRAGMENT_TESTS_BIT;
223 dependency.dstAccessMask = VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT | VK_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE_BIT;
224
225 const std::array<VkAttachmentDescription, 2> attachments = {colorAttachment, depthAttachment};
226 VkRenderPassCreateInfo renderPassInfo = {};
227 renderPassInfo.sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO;
228 renderPassInfo.attachmentCount = static_cast<uint32_t>(attachments.size());
229 renderPassInfo.pAttachments = attachments.data();
230 renderPassInfo.subpassCount = 1;
231 renderPassInfo.pSubpasses = &subpass;
232 renderPassInfo.dependencyCount = 1;
233 renderPassInfo.pDependencies = &dependency;
234
235 if (vkCreateRenderPass(m_device.device(), &renderPassInfo, nullptr, &m_renderPass) != VK_SUCCESS) {
236 throw std::runtime_error("failed to create render pass!");
237 }
238}
239
241{
242 m_swapChainFrameBuffers.resize(imageCount());
243 for (size_t i = 0; i < imageCount(); i++) {
244 std::array<VkImageView, 2> attachments = {m_swapChainImageViews[i], m_depthImageViews[i]};
245
246 const auto [width, height] = getSwapChainExtent();
247 VkFramebufferCreateInfo framebufferInfo = {};
248 framebufferInfo.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
249 framebufferInfo.renderPass = m_renderPass;
250 framebufferInfo.attachmentCount = static_cast<uint32_t>(attachments.size());
251 framebufferInfo.pAttachments = attachments.data();
252 framebufferInfo.width = width;
253 framebufferInfo.height = height;
254 framebufferInfo.layers = 1;
255
256 if (vkCreateFramebuffer(m_device.device(), &framebufferInfo, nullptr, &m_swapChainFrameBuffers[i]) != VK_SUCCESS) {
257 throw std::runtime_error("failed to create framebuffer!");
258 }
259 }
260}
261
263{
264 const VkFormat depthFormat = findDepthFormat();
265 const auto [width, height] = getSwapChainExtent();
266
267 m_swapChainDepthFormat = depthFormat;
268 m_depthImages.resize(imageCount());
269 m_depthImageMemory.resize(imageCount());
270 m_depthImageViews.resize(imageCount());
271
272 for (size_t i = 0; i < m_depthImages.size(); i++) {
273 VkImageCreateInfo imageInfo{};
274 imageInfo.sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO;
275 imageInfo.imageType = VK_IMAGE_TYPE_2D;
276 imageInfo.extent.width = width;
277 imageInfo.extent.height = height;
278 imageInfo.extent.depth = 1;
279 imageInfo.mipLevels = 1;
280 imageInfo.arrayLayers = 1;
281 imageInfo.format = depthFormat;
282 imageInfo.tiling = VK_IMAGE_TILING_OPTIMAL;
283 imageInfo.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
284 imageInfo.usage = VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT;
285 imageInfo.samples = VK_SAMPLE_COUNT_1_BIT;
286 imageInfo.sharingMode = VK_SHARING_MODE_EXCLUSIVE;
287 imageInfo.flags = 0;
288
289 m_device.createImageWithInfo(imageInfo, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT, m_depthImages[i], m_depthImageMemory[i]);
290
291 VkImageViewCreateInfo viewInfo{};
292 viewInfo.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
293 viewInfo.image = m_depthImages[i];
294 viewInfo.viewType = VK_IMAGE_VIEW_TYPE_2D;
295 viewInfo.format = depthFormat;
296 viewInfo.subresourceRange.aspectMask = VK_IMAGE_ASPECT_DEPTH_BIT;
297 viewInfo.subresourceRange.baseMipLevel = 0;
298 viewInfo.subresourceRange.levelCount = 1;
299 viewInfo.subresourceRange.baseArrayLayer = 0;
300 viewInfo.subresourceRange.layerCount = 1;
301
302 if (vkCreateImageView(m_device.device(), &viewInfo, nullptr, &m_depthImageViews[i]) != VK_SUCCESS) {
303 throw std::runtime_error("failed to create texture image view!");
304 }
305 }
306}
307
309{
310 m_imageAvailableSemaphores.resize(MAX_FRAMES_IN_FLIGHT);
311 m_renderFinishedSemaphores.resize(MAX_FRAMES_IN_FLIGHT);
312 m_inFlightFences.resize(MAX_FRAMES_IN_FLIGHT);
313 m_imagesInFlight.resize(imageCount(), VK_NULL_HANDLE);
314
315 VkSemaphoreCreateInfo semaphoreInfo = {};
316 semaphoreInfo.sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO;
317
318 VkFenceCreateInfo fenceInfo = {};
319 fenceInfo.sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO;
320 fenceInfo.flags = VK_FENCE_CREATE_SIGNALED_BIT;
321
322 for (size_t i = 0; i < MAX_FRAMES_IN_FLIGHT; i++) {
323 if (vkCreateSemaphore(m_device.device(), &semaphoreInfo, nullptr, &m_imageAvailableSemaphores[i]) != VK_SUCCESS ||
324 vkCreateSemaphore(m_device.device(), &semaphoreInfo, nullptr, &m_renderFinishedSemaphores[i]) != VK_SUCCESS ||
325 vkCreateFence(m_device.device(), &fenceInfo, nullptr, &m_inFlightFences[i]) != VK_SUCCESS) {
326 throw std::runtime_error("failed to create synchronization objects for a frame!");
327 }
328 }
329}
330
331VkSurfaceFormatKHR ven::SwapChain::chooseSwapSurfaceFormat(const std::vector<VkSurfaceFormatKHR> &availableFormats)
332{
333 for (const auto &availableFormat : availableFormats) {
334 if (availableFormat.format == VK_FORMAT_B8G8R8A8_UNORM && availableFormat.colorSpace == VK_COLOR_SPACE_SRGB_NONLINEAR_KHR) {
335 return availableFormat;
336 }
337 }
338
339 return availableFormats[0];
340}
341
342VkPresentModeKHR ven::SwapChain::chooseSwapPresentMode(const std::vector<VkPresentModeKHR> &availablePresentModes)
343{
344 for (const auto &availablePresentMode : availablePresentModes) {
345 if (availablePresentMode == VK_PRESENT_MODE_MAILBOX_KHR) {
346 std::cout << "Present mode: Mailbox\n";
347 return availablePresentMode;
348 }
349 }
350
351 for (const auto &availablePresentMode : availablePresentModes) {
352 if (availablePresentMode == VK_PRESENT_MODE_IMMEDIATE_KHR) {
353 std::cout << "Present mode: Immediate" << '\n';
354 return availablePresentMode;
355 }
356 }
357
358 std::cout << "Present mode: V-Sync\n";
359 return VK_PRESENT_MODE_FIFO_KHR;
360}
361
362VkExtent2D ven::SwapChain::chooseSwapExtent(const VkSurfaceCapabilitiesKHR &capabilities) const
363{
364 if (capabilities.currentExtent.width != std::numeric_limits<uint32_t>::max()) {
365 return capabilities.currentExtent;
366 }
367 VkExtent2D actualExtent = m_windowExtent;
368 actualExtent.width = std::max(capabilities.minImageExtent.width, std::min(capabilities.maxImageExtent.width, actualExtent.width));
369 actualExtent.height = std::max(capabilities.minImageExtent.height, std::min(capabilities.maxImageExtent.height, actualExtent.height));
370
371 return actualExtent;
372}
373
375{
376 return m_device.findSupportedFormat(
377 {VK_FORMAT_D32_SFLOAT, VK_FORMAT_D32_SFLOAT_S8_UINT, VK_FORMAT_D24_UNORM_S8_UINT},
378 VK_IMAGE_TILING_OPTIMAL,
379 VK_FORMAT_FEATURE_DEPTH_STENCIL_ATTACHMENT_BIT);
380}
This file contains the Shader class.
VkDevice device() const
Definition Device.hpp:54
VkExtent2D chooseSwapExtent(const VkSurfaceCapabilitiesKHR &capabilities) const
std::vector< VkImageView > m_swapChainImageViews
Definition SwapChain.hpp:77
std::vector< VkFence > m_inFlightFences
Definition SwapChain.hpp:87
static VkPresentModeKHR chooseSwapPresentMode(const std::vector< VkPresentModeKHR > &availablePresentModes)
std::vector< VkSemaphore > m_imageAvailableSemaphores
Definition SwapChain.hpp:85
VkRenderPass m_renderPass
Definition SwapChain.hpp:71
VkResult acquireNextImage(uint32_t *imageIndex) const
Definition swapChain.cpp:49
void createSwapChain()
void createDepthResources()
std::vector< VkDeviceMemory > m_depthImageMemory
Definition SwapChain.hpp:74
void createSyncObjects()
void createImageViews()
std::vector< VkFramebuffer > m_swapChainFrameBuffers
Definition SwapChain.hpp:70
VkFormat findDepthFormat() const
Device & m_device
Definition SwapChain.hpp:79
std::vector< VkImage > m_depthImages
Definition SwapChain.hpp:73
std::vector< VkSemaphore > m_renderFinishedSemaphores
Definition SwapChain.hpp:86
VkResult submitCommandBuffers(const VkCommandBuffer *buffers, const uint32_t *imageIndex)
Definition swapChain.cpp:56
VkSwapchainKHR m_swapChain
Definition SwapChain.hpp:82
void createRenderPass()
void createFrameBuffers()
std::vector< VkImageView > m_depthImageViews
Definition SwapChain.hpp:75
static VkSurfaceFormatKHR chooseSwapSurfaceFormat(const std::vector< VkSurfaceFormatKHR > &availableFormats)
static constexpr int MAX_FRAMES_IN_FLIGHT
Definition SwapChain.hpp:15