操作系统:Windows8.1

显卡:Nivida GTX965M

开发工具:Visual Studio 2017

到目前为止,我们了解到Vulkan是一个与平台特性无关联的API集合。它不能直接与窗口系统进行交互。为了将渲染结果呈现到屏幕,需要建立Vulkan与窗体系统之间的连接,我们需要使用WSI(窗体系统集成)扩展。在本小节中,我们将讨论第一个,即VK_KHR_surface。它暴露了VkSurfaceKHR,它代表surface的一个抽象类型,用以呈现渲染图像使用。我们程序中将要使用到的surface是由我们已经引入的GLFW扩展及其打开的相关窗体支持的。简单来说surface就是Vulkan与窗体系统的连接桥梁。

VK_KHR_surface扩展是一个instance级扩展,我们目前为止已经启用过它,它包含在glfwGetrequiredInstanceExtensions返回的列表中。该列表还包括将在接下来几小节中使用的一些其他WSI扩展。

需要在instance创建之后立即创建窗体surface,因为它会影响物理设备的选择。之所以在本小节将surface创建逻辑纳入讨论范围,是因为窗体surface对于渲染、呈现方式是一个比较大的课题,如果过早的在创建物理设备加入这部分内容,会混淆基本的物理设备设置工作。另外窗体surface本身对于Vulkan也是非强制的。Vulkan允许这样做,不需要同OpenGL一样必须要创建窗体surface。

Window surface creation

现在开始着手创建窗体surface,在类成员debugCallback下加入成员变量surface

VkSurfaceKHRsurface;

虽然VkSurfaceKHR对象及其用法与平台无关联,但创建过程需要依赖具体的窗体系统的细节。比如,在Windows平台中,它需要WIndows上的HWNDHMODULE句柄。因此针对特定平台提供相应的扩展,在Windows上为VK_KHR_win32_surface,它自动包含在glfwGetrequiredInstanceExtensions列表中。

我们将会演示如何使用特定平台的扩展来创建Windows上的surface桥,但是不会在教程中实际使用它。使用GLFW这样的库避免了编写没有任何意义的跨平台相关代码。GLFW实际上通过glfwCreateWindowSurface很好的处理了平台差异性。当然了,比较理想是在依赖它们帮助我们完成具体工作之前,了解一下背后的实现是有帮助的。

因为一个窗体surface是一个Vulkan对象,它需要填充VkWin32SurfaceCreateInfoKHR结构体,这里有两个比较重要的参数:hwndhinstance。如果熟悉windows下开发应该知道,这些是窗口和进程的句柄。

VkWin32SurfaceCreateInfoKHRcreateInfo;
createInfo.sType=VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR;
createInfo.hwnd=glfwGetWin32Window(window);
createInfo.hinstance=GetModuleHandle(nullptr);

glfwGetWin32Window函数用于从GLFW窗体对象获取原始的HWNDGetModuleHandle函数返回当前进程的HINSTANCE句柄。

填充完结构体之后,可以利用vkCreateWin32SurfaceKHR创建surface桥,和之前获取创建、销毁DebugReportCallEXT一样,这里同样需要通过instance获取创建surface用到的函数。这里涉及到的参数分别为instance,surface创建的信息,自定义分配器和最终保存surface的句柄变量。

autocreateWin32SurfaceKHR=(PFN_vkCreateWin32SurfaceKHR)vkGetInstanceProcAddr(instance,"vkCreateWin32SurfaceKHR");if(!CreateWin32SurfaceKHR||CreateWin32SurfaceKHR(instance,&createInfo,nullptr,&surface)!=VK_SUCCESS){throwstd::runtime_error("Failedtocreatewindowsurface!");
}

该过程与其他平台类似,比如Linux,使用X11界面窗体系统,可以通过vkCreateXcbSurfaceKHR函数建立连接。

glfwCreateWindowSurface函数根据不同平台的差异性,在实现细节上会有所不同。我们现在将其整合到我们的程序中。从initVulkan中添加一个函数createSurface,安排在createInstnacesetupDebugCallback函数之后。

voidinitVulkan(){
createInstance();
setupDebugCallback();
createSurface();
pickPhysicalDevice();
createLogicalDevice();
}voidcreateSurface(){

}

GLFW没有使用结构体,而是选择非常直接的参数传递来调用函数。

voidcreateSurface(){if(glfwCreateWindowSurface(instance,window,&surface)!=VK_SUCCESS){throwstd::runtime_error("Failedtocreatewindowsurface!");
}
}

参数是VkInstance,GLFW窗体的指针,自定义分配器和用于存储VkSurfaceKHR变量的指针。对于不同平台统一返回VkResult。GLFW没有提供专用的函数销毁surface,但是可以简单的通过Vulkan原始的API完成:

voidcleanup(){
...
vkDestroySurfaceKHR(instance,surface,nullptr);
vkDestroyInstance(instance,nullptr);
...
}

最后请确保surface的清理是在instance销毁之前完成。

Querying for presentation support

虽然Vulkan的实现支持窗体集成功能,但是并不意味着系统中的每一个物理设备都支持它。因此,我们需要扩展isDeviceSuitable函数,确保设备可以将图像呈现到我们创建的surface。由于presentation是一个队列的特×××,因此解决问题的方法就是找到支持presentation的队列簇,最终获取队列满足surface创建的需要。

实际情况是,支持graphics命令的的队列簇和支持presentation命令的队列簇可能不是同一个簇。因此,我们需要修改QueueFamilyIndices结构体,以支持差异化的存储。

structQueueFamilyIndices{intgraphicsFamily=-1;intpresentFamily=-1;boolisComplete(){returngraphicsFamily>=0&&presentFamily>=0;
}
};

接下来,我们修改findQueueFamilies函数来查找具备presentation功能的队列簇。函数中用于检查的核心代码是vkGetPhysicalDeviceSurfaceSupportKHR,它将物理设备、队列簇索引和surface作为参数。在VK_QUEUE_GRAPHICS_BIT相同的循环体中添加函数的调用:

VkBool32presentSupport=false;
vkGetPhysicalDeviceSurfaceSupportKHR(device,i,&presentSupport);

然后之需要检查布尔值并存储presentation队列簇的索引:

if(queueFamily.queueCount>0&&presentSupport){
indices.presentFamily=i;
}

需要注意的是,为了支持graphics和presentation功能,我们实际环境中得到的可能是同一个队列簇,也可能不同,为此在我们的程序数据结构及选择逻辑中,将按照均来自不同的队列簇分别处理,这样便可以统一处理以上两种情况。除此之外,出于性能的考虑,我们也可以通过添加逻辑明确的指定物理设备所使用的graphics和presentation功能来自同一个队列簇。

Creating the presentation queue

剩下的事情是修改逻辑设备创建过程,在于创建presentation队列并获取VkQueue的句柄。添加保存队列句柄的成员变量:

VkQueuepresentQueue;

接下来,我们需要多个VkDeviceQueueCreateInfo结构来创建不同功能的队列。一个优雅的方式是针对不同功能的队列簇创建一个set集合确保队列簇的唯一性:

#include<set>...

QueueFamilyIndicesindices=findQueueFamilies(physicalDevice);

std::vector<VkDeviceQueueCreateInfo>queueCreateInfos;
std::set<int>uniqueQueueFamilies={indices.graphicsFamily,indices.presentFamily};floatqueuePriority=1.0f;for(intqueueFamily:uniqueQueueFamilies){
VkDeviceQueueCreateInfoqueueCreateInfo={};
queueCreateInfo.sType=VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO;
queueCreateInfo.queueFamilyIndex=queueFamily;
queueCreateInfo.queueCount=1;
queueCreateInfo.pQueuePriorities=&queuePriority;
queueCreateInfos.push_back(queueCreateInfo);
}

同时还要修改VkDeviceCreateInfo指向队列集合:

createInfo.queueCreateInfoCount=static_cast<uint32_t>(queueCreateInfos.size());
createInfo.pQueueCreateInfos=queueCreateInfos.data();

如果队列簇相同,那么我们之需要传递一次索引。最后,添加一个调用检索队列句柄:

vkGetDeviceQueue(device,indices.presentFamily,&presentQueue);

在这个例子中,队列簇是相同的,两个句柄可能会有相同的值。在下一个章节中我们会看看交换链,以及它们如何使我们能够将图像呈现给surface。

获取工程代码GitHubcheckout

Vulkan Tutorial 07 Window surface的更多相关文章

  1. 如何直接从Android Surface访问EGL图像以用于MediaCodec视频解码器?

    )而无需进行格式转换,但目前还没有类似的机制.我不知道比你描述的更好的方法.更新:我的办公室伙伴提出了一个建议:使用着色器将缓冲区渲染为两个纹理,一个用于Y,另一个用于CbCr.这样可以保持GLES中的所有操作,而不会扩展到完整的RGB.在内部,它会将MediaCodec输出转换为RGB并通过它进行两次研磨,但这可能比将其复制到用户空间并在cpu上自行完成更便宜.

  2. android – Camera2 ImageReader冻结重复捕获请求

    我正在尝试使用camera2API从相机捕获图像数据.我主要使用的是从androidCapture2RAW示例中获取的代码.在完全停止之前,只有少数图像通过.我尝试使用不同大小的RAW_SENSOR和JPEG格式捕获相同的结果.我究竟做错了什么?解决方法固定它.ImageReader生成的图像需要关闭,否则会很快填满内存.

  3. android – 在Surface View中动画和旋转图像

    我想在SurfaceView上制作动画动画.理想情况下,我想在动画结束后收到通知.例如:我可能有一辆朝北的汽车.如果我想动画它以使其面向南方持续500ms,我该怎么办呢?如果这最终成为你的问题,你无法解决问题,让我知道,我会发布我是如何做到的.(我不知道这是否是最有效的方法,但只要位图小于300×300左右,它就能顺利运行.也许如果有人知道更好的方法,他们可以告诉我们!

  4. android – 如何获得具有有效Surface的SurfaceHolder(EGL.eglCreateWindowSurface需要)?

    我尝试用EGL初始化GLES(因为我想从主线程中绘制)而不是使用渲染器和从onDrawFrame内部绘图).我收到错误:“确保SurfaceView或相关的SurfaceHolder具有有效的Surface”.显然mEgl.eglCreateWindowSurface失败,因为surfaceHolder没有有效的表面.如何获得具有有效Surface的SurfaceHolder?我的Activity

  5. android – 缓冲SurfaceCodec的Surface输入

    >来自Android的任何人都可以评论相机和MediaCodec之间是否协调ByteBuffer格式是否在路线图上?

  6. Android Camera2 API显示已处理的预览图像

    新的Camera2API与旧的相比非常不同.显示操纵的相机帧到管道的用户部分让我感到困惑.我知道CamerapreviewimagedataprocessingwithAndroidLandCamera2API有很好的解释,但显示帧仍然不清楚.我的问题是,在一些处理过程中,在ImageReaders回调函数中显示帧的方式是什么,同时保持Camera2api管道的效率和速度?示例流程:camera.

  7. android – 使用MediaCodec录制曲面

    我将并行录制音频,使用FFMPEG合并该视频和音频,并将视频应用于视频.解决方法您可以在Grafika中看到一个完整的例子.特别地,“显示拍摄相机”活动将相机输出记录到.mp4.它也是demonstrates在GL着色器中应用一些简单的图像处理技术.它使用GLSurfaceView和复杂的舞蹈来保持记录的方向变化.也可能感兴趣的是,“使用FBO记录GL应用程序”活动记录OpenGLES呈现了几种不同的方式.它使用简单的SurfaceView,更直接.

  8. jQueryMobile之窗体长内容的缺陷与解决方法实例分析

    这篇文章主要介绍了jQueryMobile之窗体长内容的缺陷与解决方法,结合具体实例形式分析了jQueryMobile底部悬浮层遮挡情况下的解决方法,非常简单实用,需要的朋友可以参考下

  9. 在同一窗体中使用PHP来处理多个提交任务

    由于一个窗体只能处理一个唯一的任务,但是相同的PHP脚本可以根据被点击的按钮和执行合适的代码段来处理以上四种任务。>当一个窗体被提交给PHP脚本时,根据使用的提交方法,PHP自动建立一个特定的$_POST或者$_GET数组。特别值得注意的是,如何在以上脚本中处理提交任务的按扭。

  10. Java编写实现窗体程序显示日历

    这篇文章主要为大家详细介绍了Java编写实现窗体程序显示日历,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

随机推荐

  1. static – 在页面之间共享数据的最佳实践

    我想知道在UWP的页面之间发送像’selectedItem’等变量的最佳做法是什么?创建一个每个页面都知道的静态全局变量类是一个好主意吗?

  2. .net – 为Windows窗体控件提供百分比宽度/高度

    WindowsForm开发的新手,但在Web开发方面经验丰富.有没有办法为Windows窗体控件指定百分比宽度/高度,以便在用户调整窗口大小时扩展/缩小?当窗口调整大小时,可以编写代码来改变控件的宽度/高度,但我希望有更好的方法,比如在HTML/CSS中.在那儿?

  3. 使用Windows Azure查询表存储数据

    我需要使用特定帐户吗?>将应用程序部署到Azure服务后,如何查询数据?GoogleAppEngine有一个数据查看器/查询工具,Azure有类似的东西吗?>您可以看到的sqlExpressintance仅在开发结构中,并且一旦您表示没有等效,所以请小心使用它.>您可以尝试使用Linqpad查询表格.看看JamieThomson的thispost.

  4. windows – SetupDiGetClassDevs是否与文档中的设备实例ID一起使用?

    有没有更好的方法可以使用DBT_DEVICEARRIVAL事件中的数据获取设备的更多信息?您似乎必须指定DIGCF_ALLCLASSES标志以查找与给定设备实例ID匹配的所有类,或者指定ClassGuid并使用DIGCF_DEFAULT标志.这对我有用:带输出:

  5. Windows Live ID是OpenID提供商吗?

    不,WindowsLiveID不是OpenID提供商.他们使用专有协议.自从他们的“测试版”期结束以来,他们从未宣布计划继续它.

  6. 如果我在代码中进行了更改,是否需要重新安装Windows服务?

    我写了一个Windows服务并安装它.现在我对代码进行了一些更改并重新构建了解决方案.我还应该重新安装服务吗?不,只需停止它,替换文件,然后重新启动它.

  7. 带有双引号的字符串回显使用Windows批处理输出文件

    我正在尝试使用Windows批处理文件重写配置文件.我循环遍历文件的行并查找我想要用指定的新行替换的行.我有一个’函数’将行写入文件问题是%Text%是一个嵌入双引号的字符串.然后失败了.可能还有其他角色也会导致失败.如何才能使用配置文件中的所有文本?尝试将所有“在文本中替换为^”.^是转义字符,因此“将被视为常规字符你可以尝试以下方法:其他可能导致错误的字符是:

  8. .net – 将控制台应用程序转换为服务?

    我正在寻找不同的优势/劣势,将我们长期使用的控制台应用程序转换为Windows服务.我们为ActiveMQ使用了一个叫做java服务包装器的东西,我相信人们告诉我你可以用它包装任何东西.这并不是说你应该用它包装任何东西;我们遇到了这个问题.控制台应用程序是一个.NET控制台应用程序,默认情况下会将大量信息记录到控制台,尽管这是可配置的.任何推荐?我们应该在VisualStudio中将其重建为服务吗?我使用“-install”/“-uninstall”开关执行此操作.例如,seehere.

  9. windows – 捕获外部程序的STDOUT和STDERR *同时*它正在执行(Ruby)

    哦,我在Windows上:-(实际上,它比我想象的要简单,这看起来很完美:…是的,它适用于Windows!

  10. windows – 当我试图批量打印变量时,为什么我得到“Echo is on”

    我想要执行一个简单的批处理文件脚本:当我在XP中运行时,它给了我预期的输出,但是当我在Vista或Windows7中运行它时,我在尝试打印值时得到“EchoisOn”.以下是程序的输出:摆脱集合表达式中的空格.等号(=)的两侧可以并且应该没有空格BTW:我通常在@echo关闭的情况下启动所有批处理文件,并以@echo结束它们,所以我可以避免将代码与批处理文件的输出混合.它只是使您的批处理文件输出更好,更清洁.

返回
顶部