操作系统:Windows8.1

显卡:Nivida GTX965M

开发工具:Visual Studio 2017


Introduction

现在我们已经成功的在屏幕上绘制出三角形,但是在某些情况下,它会出现异常情况。窗体surface会发生改变,使得交换链不在与其兼容。可能导致这种情况发生的原因之一是窗体的大小变化。我们必须在这个时机重新创建交换链。

Recreating the swap chain

添加新的函数recreateSwapChain并调用createSwapChain及依赖于交换链或者窗体大小的对象相关的所有创建函数。

  1. void recreateSwapChain() {
  2. vkDeviceWaitIdle(device);
  3.  
  4. createSwapChain();
  5. createImageViews();
  6. createRenderPass();
  7. createGraphicsPipeline();
  8. createFramebuffers();
  9. createCommandBuffers();
  10. }

我们首先调用vkDeviceIdle,就像前一个章节提到的,我们不能触碰正在使用中的资源。很明显,要做的第一件事情就是重新创建交换链本身。图像视图也需要重新创建,因为它们直接建立在交换链图像基础上。渲染通道需要重新创建,因为它依赖交换链图像的格式。在窗体调整大小的操作期间,交换链图像格式很少发生变化,但仍应该被处理。在创建图形管线期间指定Viewport和scissor 矩形大小,所以管线需要重新构建。可以使用动态状态改变viewports和scissor rectangles,避免重新创建。最后帧缓冲区和命令缓冲区也需要重新创建,因为它们也依赖交换链的图像。

为了确保重新创建相关的对象之前,老版本的对象被系统正确回收清理,我们需要移动一些cleanup代码到不同的函数中,这样可以在recreateSwapChain函数调用。该函数定义为cleanupSwapChain:

  1. void cleanupSwapChain() {
  2.  
  3. }
  4.  
  5. void recreateSwapChain() {
  6. vkDeviceWaitIdle(device);
  7.  
  8. cleanupSwapChain();
  9.  
  10. createSwapChain();
  11. createImageViews();
  12. createRenderPass();
  13. createGraphicsPipeline();
  14. createFramebuffers();
  15. createCommandBuffers();
  16. }

我们从cleanup中将需要被重新创建的对象所对应的清理代码移动到cleanupSwapChain中:

  1. void cleanupSwapChain() {
  2. for (size_t i = ; i < swapChainFramebuffers.size(); i++) {
  3. vkDestroyFramebuffer(device, swapChainFramebuffers[i], nullptr);
  4. }
  5.  
  6. vkFreeCommandBuffers(device, commandPool, static_cast<uint32_t>(commandBuffers.size()), commandBuffers.data());
  7.  
  8. vkDestroyPipeline(device, graphicsPipeline, nullptr);
  9. vkDestroyPipelineLayout(device, pipelineLayout, nullptr);
  10. vkDestroyRenderPass(device, renderPass, nullptr);
  11.  
  12. for (size_t i = ; i < swapChainImageViews.size(); i++) {
  13. vkDestroyImageView(device, swapChainImageViews[i], nullptr);
  14. }
  15.  
  16. vkDestroySwapchainKHR(device, swapChain, nullptr);
  17. }
  18.  
  19. void cleanup() {
  20. cleanupSwapChain();
  21.  
  22. vkDestroySemaphore(device, renderFinishedSemaphore, nullptr);
  23. vkDestroySemaphore(device, imageAvailableSemaphore, nullptr);
  24.  
  25. vkDestroyCommandPool(device, commandPool, nullptr);
  26.  
  27. vkDestroyDevice(device, nullptr);
  28. DestroyDebugReportCallbackEXT(instance, callback, nullptr);
  29. vkDestroySurfaceKHR(instance, surface, nullptr);
  30. vkDestroyInstance(instance, nullptr);
  31.  
  32. glfwDestroyWindow(window);
  33.  
  34. glfwTerminate();
  35. }

我们重头创建命令对象池command pool,但是比较浪费看起来。相反的,我们选择借助vkFreeCommandBuffers函数清理已经存在的命令缓冲区。这种方式可以重用对象池中已经分配的命令缓冲区。

以上部分就是重新创建交换链的工作!然而这样做的缺点就是在重新创建交换链完毕之前,会造成渲染停止。创建新交换链的同时允许在旧的交换链的图像上继续绘制命令。需要将之前的交换链传递到VkSwapchainCreateInfoKHR结构体中的oldSwapChain字段,并在使用之后立即销毁。

Window resizing


现在我们需要搞清楚哪些情况下重新创建交换链是必要的,并调用recreateSwapChain函数。一个通常的条件是窗体的大小变化。让我们调整窗体的大小,并观察捕捉到的事件。修改initWindow函数不再包含GLFW_RESIZABLE行,或者将其参数从GLFW_FALSE修改为GLFW_TRUE

  1. void initWindow() {
  2. glfwInit();
  3.  
  4. glfwWindowHint(GLFW_CLIENT_API, GLFW_NO_API);
  5.  
  6. window = glfwCreateWindow(WIDTH, HEIGHT, "Vulkan", nullptr, nullptr);
  7.  
  8. glfwSetWindowUserPointer(window, this);
  9. glfwSetWindowSizeCallback(window, HelloTriangleApplication::onWindowResized);
  10. }
  11.  
  12. ...
  13.  
  14. static void onWindowResized(GLFWwindow* window, int width, int height) {
  15. if (width == || height == ) return;
  16.  
  17. HelloTriangleApplication* app = reinterpret_cast<HelloTriangleApplication*>(glfwGetWindowUserPointer(window));
  18. app->recreateSwapChain();
  19. }

glfwSetWindowSizeCallback函数会在窗体发生大小变化的时候被事件回调。遗憾的是,它只能接受一个指针作为参数,所以我们不能直接使用成员函数。但幸运的是,GLFW允许我们使用glfwSetWindowUserPointer将任意指针存储在窗体对象中,因此可以指定静态类成员调用glfwGetWindowUserPointer返回原始的实例对象。然后我们可以继续调用recreateSwapChain,这种情况通常发生在,窗体最小化并且导致交换链创建失败时.

chooseSwapExtent函数应该增加更新逻辑,使用窗体最新的widthheight代替最初的WIDTHHEIGHT:

  1. int width, height;
  2. glfwGetWindowSize(window, &width, &height);
  3.  
  4. VkExtent2D actualExtent = {width, height};

Suboptimal or out-of-date swap chain


有些时候Vulkan可能告诉我们当前的交换链在presentation时不再兼容。vkAcquireNextImageKHRvkQueuePresentKHR函数可以返回具体的值明确。

  • VK_ERROR_OUT_DATE_KHR: 交换链与surface不再兼容,不可进行渲染
  • VK_SUBOPTIMAL_KHR: 交换链仍然可以向surface提交图像,但是surface的属性不再匹配准确。比如平台可能重新调整图像的尺寸适应窗体大小。
  1. VkResult result = vkAcquireNextImageKHR(device, swapChain, std::numeric_limits<uint64_t>::max(), imageAvailableSemaphore, VK_NULL_HANDLE, &imageIndex);
  2.  
  3. if (result == VK_ERROR_OUT_OF_DATE_KHR) {
  4. recreateSwapChain();
  5. return;
  6. } else if (result != VK_SUCCESS && result != VK_SUBOPTIMAL_KHR) {
  7. throw std::runtime_error("failed to acquire swap chain image!");
  8. }

如果交换链获取图像timeout,表明不再可用。所以我们需要立即重新创建交换链,并在下一次drawFrame调用中尝试获取。

你也可以选择在交换链不是最佳状态的时候,选择重新创建,比如刚才说的大小不匹配问题。在这里因为我们已经获得了一个图像,所以继续进行。VK_SUCCESSVK_SUBOPTIMAL_KHR都被认为是“成功”返回码。

  1. result = vkQueuePresentKHR(presentQueue, &presentInfo);
  2.  
  3. if (result == VK_ERROR_OUT_OF_DATE_KHR || result == VK_SUBOPTIMAL_KHR) {
  4. recreateSwapChain();
  5. } else if (result != VK_SUCCESS) {
  6. throw std::runtime_error("failed to present swap chain image!");
  7. }
  8.  
  9. vkQueueWaitIdle(presentQueue);

vkQueuePresentKHR函数返回同样的值。在我们的案例中我们,如果是非最佳状态,也选择重新创建交换链。因为我们需要最好的效果。尝试调整窗体的大小,帧缓冲区的大小变化与窗体匹配。

Congratulations,我们完结了第一个运行比较完整的Vulkan程序,在下面的章节中我们尝试摆脱之前的硬编码,使用顶点缓冲区代替vertex shader中写死顶点数据。

项目代码 GitHub地址。

Vulkan Tutorial 18 重构交换链的更多相关文章

  1. [译]Vulkan教程(20)重建交换链

    [译]Vulkan教程(20)重建交换链 Swap chain recreation 重建交换链 Introduction 入门 The application we have now success ...

  2. Vulkan Tutorial 08 交换链

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 在这一章节,我们了解一下将渲染图像提交到屏幕的基本机制.这种机制成为交换链,并且需要 ...

  3. [译]Vulkan教程(10)交换链

    [译]Vulkan教程(10)交换链 Vulkan does not have the concept of a "default framebuffer", hence it r ...

  4. [译]Vulkan教程(18)命令buffers

    [译]Vulkan教程(18)命令buffers Command buffers 命令buffer Commands in Vulkan, like drawing operations and me ...

  5. Vulkan Tutorial 05 逻辑设备与队列

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 Introduction 在选择要使用的物理设备之后,我们需要设置一个逻辑设备用于交 ...

  6. Vulkan Tutorial 07 Window surface

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 到目前为止,我们了解到Vulkan是一个与平台特性无关联的API集合.它不能直接与窗 ...

  7. Vulkan Tutorial 12 Fixed functions

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 早起的图形API在图形渲染管线的许多阶段提供了默认的状态.在Vulkan中,从vie ...

  8. Vulkan Tutorial 13 Render passes

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 Setup 在我们完成管线的创建工作,我们接下来需要告诉Vulkan渲染时候使用的f ...

  9. Vulkan Tutorial 14 Integration pipeline

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 我们现在整合前几章节的结构体和对象创建图形管线!以下是我们现在用到的对象类型,作为一 ...

随机推荐

  1. bzoj4784 [Zjoi2017]仙人掌

    Description 如果一个无自环无重边无向连通图的任意一条边最多属于一个简单环,我们就称之为仙人掌.所谓简单环即不经过重复的结点的环. 现在九条可怜手上有一张无自环无重边的无向连通图,但是她觉得 ...

  2. [KISSY5系列]淘宝全终端框架 KISSY 5--从零开始使用

    KISSY 是淘宝一个开源的 JavaScript 库,包含的组件有:日历.图片放大镜.卡片切换.弹出窗口.输入建议等 一.简介 KISSY 是一款跨终端.模块化.高性能.使用简单的 JavaScri ...

  3. 蓝桥杯-循环节长度-java

    /* (程序头部注释开始) * 程序的版权和版本声明部分 * Copyright (c) 2016, 广州科技贸易职业学院信息工程系学生 * All rights reserved. * 文件名称: ...

  4. Could not find a valid gem 'compass' (>= 0) in any repository compass安装失败解决方案

    安装完成ruby gem 之后,通过 gem install compass 安装compass~~ 出现如下报错 Could not find a valid gem 'compass' (> ...

  5. self 和 super 关键字

    self 相当于 java中的this self使用总结 1.self谁调用当前方法,self就代表谁 2.self在对象方法中,self代表当前对象 3.self在类方法中个,self代表类 [se ...

  6. 关于Yii框架的基础知识

    第一次写博文,也不知道怎么写,不太熟悉,带小伙伴学习一样我日常使用的Yii框架. PHP中的开发框架有很多,比如:ThinkPHP.Yii.CI.Laravel.Phalcon等.现在流行度最高的是L ...

  7. HttpClient--HttpGet使用

    最近公司在做一个爬虫工具,爬取公司现网的数据,留给方通项目使用 用到里阿帕奇的这两个类,在网上看到了一些资料结合自己的应用,这个贴出一个demo import com.alibaba.fastjson ...

  8. [玩耍]C++控制台扫雷

    其实是大一还不会GUI时闲着无聊写的.都是硬编码,也不支持自定义棋盘大小,现在看看这代码惨不忍睹.下载地址:http://download.csdn.net/download/xienaoban/98 ...

  9. JAVA IO中的设计模式

    在java语言 I/O库的设计中,使用了两个结构模式,即装饰模式和适配器模式.       在任何一种计算机语言中,输入/输出都是一个很重要的部分.与一般的计算机语言相比,java将输入/输出的功能和 ...

  10. JQuery控制下拉列表

    //遍历option和添加.移除option function changeShipMethod(shipping){ var len = $("select[@name=ISHIPTYPE ...