操作系统:Windows8.1

显卡:Nivida GTX965M

开发工具:Visual Studio 2017


Selecting a physical device

通过VkInstance初始化Vulkan后,我们需要在系统中查找并选择一个支持我们所需功能的显卡。实际上,我们可以选择任意数量的显卡并同时使用他们,但在本小节中,我们简单的设定选择规则,即将查找到的第一个图形卡作为我们适合的物理设备。

我们添加函数pickPhysicalDevice并在initVulkan函数中调用。

  1. void initVulkan() {
  2. createInstance();
  3. setupDebugCallback();
  4. pickPhysicalDevice();
  5. }
  6.  
  7. void pickPhysicalDevice() {
  8.  
  9. }

最终我们选择的图形显卡存储在类成员VkPhysicalDevice句柄中。当VkInstance销毁时,这个对象将会被隐式销毁,所以我们并不需要在cleanup函数中做任何操作。

  1. VkPhysicalDevice physicalDevice = VK_NULL_HANDLE;

关于获取图形卡列表的方式与获得扩展列表的方式类似。

  1. uint32_t deviceCount = ;
  2. vkEnumeratePhysicalDevices(instance, &deviceCount, nullptr);

如果Vulkan支持的设备数为0,那么没有任何意义进行下一步,我们选择抛出异常。

  1. if (deviceCount == ) {
  2. throw std::runtime_error("failed to find GPUs with Vulkan support!");
  3. }

否则我们分配数组存储所有VkPhysicalDevice的句柄。

  1. std::vector<VkPhysicalDevice> devices(deviceCount);
  2. vkEnumeratePhysicalDevices(instance, &deviceCount, devices.data());

现在我们需要对它们进行评估,检查它们是否适合我们要执行的操作,因为并不是所有的显卡功能一致。为此我们添加一个新的函数:

  1. bool isDeviceSuitable(VkPhysicalDevice device) {
  2. return true;
  3. }

我们将检查是否有任何物理设备符合我们的功能需求。

  1. for (const auto& device : devices) {
  2. if (isDeviceSuitable(device)) {
  3. physicalDevice = device;
  4. break;
  5. }
  6. }
  7.  
  8. if (physicalDevice == VK_NULL_HANDLE) {
  9. throw std::runtime_error("failed to find a suitable GPU!");
  10. }

下一节我们介绍isDeviceSuitable函数,并检查第一个需要满足的功能。在后续的小节中,我们将开始使用更多的Vulkan功能,我们会扩展此功能函数以满足更多的检查条件。

Base device suitability checks


评估合适的设备我们可以通过遍历一些细节来完成。基本的设备属性像name, type以及Vulkan版本都可以通过vkGetPhysicalDeviceProperties来遍历得到。

  1. VkPhysicalDeviceProperties deviceProperties;
  2. vkGetPhysicalDeviceProperties(device, &deviceProperties);

可以使用vkGetPhysicalDeviceFeatures查询对纹理压缩,64位浮点数和多视图渲染(VR非常有用)等可选功能的支持:

  1. VkPhysicalDeviceFeatures deviceFeatures;
  2. vkGetPhysicalDeviceFeatures(device, &deviceFeatures);

更多遍历物理设备细节的信息,诸如设备内存、队列簇我们将会在后续小节讨论。

例如,我们假设我们的应用程序仅适用于支持geometry shaders的专用显卡。那么isDeviceSuitable函数将如下所示:

  1. bool isDeviceSuitable(VkPhysicalDevice device) {
  2. VkPhysicalDeviceProperties deviceProperties;
  3. VkPhysicalDeviceFeatures deviceFeatures;
  4. vkGetPhysicalDeviceProperties(device, &deviceProperties);
  5. vkGetPhysicalDeviceFeatures(device, &deviceFeatures);
  6.  
  7. return deviceProperties.deviceType == VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU &&
  8. deviceFeatures.geometryShader;
  9. }

为了避免纯粹的单一的判断一个设备是否合适,尤其是当你发现多个设备都合适的条件下,你也可以给每一个设备做权值,选择最高的一个。这样,可以通过给予更高权值获取定制化的图形设备,但如果没有一个可用的设备,可以回滚到集成图形设备。你可以按照如下方式实现:

  1. #include <map>
  2.  
  3. ...
  4.  
  5. void pickPhysicalDevice() {
  6. ...
  7.  
  8. // Use an ordered map to automatically sort candidates by increasing score
  9. std::multimap<int, VkPhysicalDevice> candidates;
  10.  
  11. for (const auto& device : devices) {
  12. int score = rateDeviceSuitability(device);
  13. candidates.insert(std::make_pair(score, device));
  14. }
  15.  
  16. // Check if the best candidate is suitable at all
  17. if (candidates.rbegin()->first > ) {
  18. physicalDevice = candidates.rbegin()->second;
  19. } else {
  20. throw std::runtime_error("failed to find a suitable GPU!");
  21. }
  22. }
  23.  
  24. int rateDeviceSuitability(VkPhysicalDevice device) {
  25. ...
  26.  
  27. int score = ;
  28.  
  29. // Discrete GPUs have a significant performance advantage
  30. if (deviceProperties.deviceType == VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU) {
  31. score += ;
  32. }
  33.  
  34. // Maximum possible size of textures affects graphics quality
  35. score += deviceProperties.limits.maxImageDimension2D;
  36.  
  37. // Application can't function without geometry shaders
  38. if (!deviceFeatures.geometryShader) {
  39. return ;
  40. }
  41.  
  42. return score;
  43. }

我们不需要在小节内实现所有内容,但我们可以了解如何选择图形设备的过程。当然,我们也可以显示图形设备的名称列表,让用户选择。

因为我们刚刚开始,Vulkan的支持是我们唯一需要的,在这里假设任何GPU都可以:

  1. bool isDeviceSuitable(VkPhysicalDevice device) {
  2. return true;
  3. }

在下一小节中,我们将会讨论第一个真正需要检查的设备功能。

Queue families


之前已经简要的介绍过,几乎所有的Vulkan操作,从绘图到上传纹理,都需要将命令提交到队列中。有不同类型的队列来源于不同的队列簇,每个队列簇只允许部分commands。例如,可以有一个队列簇,只允许处理计算commands或者只允许内存传输commands:

我们需要检测设备中支持的队列簇,其中哪一个队列簇支持我们想要的commands。为此我们添加一个新的函数findQueueFamilies来查找我们需要的队列簇。现在我们只会寻找一个支持图形commands队列簇,但是我们可以在稍后的小节中扩展更多的内容。

此函数返回满足某个属性的队列簇索引。定义结构体,其中索引-1表示"未找到":

  1. struct QueueFamilyIndices {
  2. int graphicsFamily = -;
  3.  
  4. bool isComplete() {
  5. return graphicsFamily >= ;
  6. }
  7. };

现在我们实现findQueueFamilies函数:

  1. QueueFamilyIndices findQueueFamilies(VkPhysicalDevice device) {
  2. QueueFamilyIndices indices;
  3.  
  4. ...
  5.  
  6. return indices;
  7. }

获取队列簇的列表函数为vkGetPhysicalDeviceQueueFamilyProperties:

  1. uint32_t queueFamilyCount = ;
  2. vkGetPhysicalDeviceQueueFamilyProperties(device, &queueFamilyCount, nullptr);
  3.  
  4. std::vector<VkQueueFamilyProperties> queueFamilies(queueFamilyCount);
  5. vkGetPhysicalDeviceQueueFamilyProperties(device, &queueFamilyCount, queueFamilies.data());

有关队列簇,结构体VkQueueFamilyProperties包含了具体信息,包括支持的操作类型和基于当前队列簇可以创建的有效队列数。我们至少需要找到一个支持VK_QUEUE_GRAPHICS_BIT的队列簇。

  1. int i = ;
  2. for (const auto& queueFamily : queueFamilies) {
  3. if (queueFamily.queueCount > && queueFamily.queueFlags & VK_QUEUE_GRAPHICS_BIT) {
  4. indices.graphicsFamily = i;
  5. }
  6.  
  7. if (indices.isComplete()) {
  8. break;
  9. }
  10.  
  11. i++;
  12. }

现在我们有了比较理想的队列簇查询功能,我们可以在isDeviceSuitable函数中使用,确保物理设备可以处理我们需要的命令:

  1. bool isDeviceSuitable(VkPhysicalDevice device) {
  2. QueueFamilyIndices indices = findQueueFamilies(device);
  3.  
  4. return indices.isComplete();
  5. }

很好,我们已经找到了我们需要的物理设备,在下一个小节我们会讨论逻辑设备。

获取工程代码 GitHubcheckout

Vulkan Tutorial 05 物理设备与队列簇的更多相关文章

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

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

  2. [译]Vulkan教程(07)物理设备和队列家族

    [译]Vulkan教程(07)物理设备和队列家族 Selecting a physical device 选择一个物理设备 After initializing the Vulkan library ...

  3. Vulkan Tutorial 06 逻辑设备与队列

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

  4. Vulkan Tutorial 07 Window surface

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

  5. Vulkan Tutorial 08 交换链

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

  6. Vulkan Tutorial 16 Command buffers

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 诸如绘制和内存操作相关命令,在Vulkan中不是通过函数直接调用的.我们需要在命令缓 ...

  7. Vulkan Tutorial 20 Vertex buffer creation

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 Introduction 在Vulkan中,缓冲区是内存的一块区域,该区域用于向显卡 ...

  8. Vulkan Tutorial 21 Staging buffer

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 Introduction 顶点缓冲区现在已经可以正常工作,但相比于显卡内部读取数据, ...

  9. Vulkan Tutorial 25 Images

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 Introduction 到目前为止,几何图形使用每个顶点颜色进行着色处理,这是一个 ...

随机推荐

  1. OC 常见面试题以及答案

    一. #improt 和include区别?@class? 1.include 主要用用来引用C 文件  import 主要引用OC 头文件  import 确保一个文件只能被导入一次  避免了交叉编 ...

  2. PHP的laravel框架后台实现数据导出excel的功能

    要想在PHP后台实现excel导入导出功能,一种简单有效的方法就是使用phpexcel插件. 要使用phpexcel插件,首先需要下载composer,这个工具是专门用来管理项目中库之间的依赖关系的. ...

  3. QT链接数据库

    在介绍QT与数据的链接问题上,我在这里就不介绍关于QT环境与mysql.sqlite3环境的安装步骤了,以下的所有的操作都是建立在你已经安装了所有环境的基础上的.好的,那我们就具体来看一看QT环境中怎 ...

  4. Android中调用文件管理器并返回选中文件的路径

    实际项目中经常需要调用文件管理器,选择下载路径或者上传的本地文件路径.今天就给大家做个demo示范该功能的实现过程. 一.实现效果预览 以下为三星S6的样机测试效果,当然不同手机调用后的效果不一样. ...

  5. hadoop环境搭建之关于NAT模式静态IP的设置 ---VMware12+CentOs7

    很久没有更新了,主要是没有时间,今天挤出时间验证了一下,果然还是有些问题的,不过已经解决了,就发上来吧. PS:小豆腐看仔细了哦~ 关于hadoop环境搭建,从单机模式,到伪分布式,再到完全分布式,我 ...

  6. Magento中URL路径的获取

    //获得 media 带 http 的url 地址. Mage::getBaseUrl('media') //获得skin 和js 目录的地址: Mage::getBaseUrl('skin'); M ...

  7. C#总结(二)事件Event 介绍总结

    最近在总结一些基础的东西,主要是学起来很难懂,但是在日常又有可能会经常用到的东西.前面介绍了 C# 的 AutoResetEvent的使用介绍, 这次介绍事件(event). 事件(event),对于 ...

  8. Java项目中的一些注意事项

    一.关于包命名方式 例如:com.sun.spring.xxx.service.impl 第一级:公司域名的倒序com.sun 第二级:项目名称spring 第三级:模块信息xxx 第四级:功能顶层包 ...

  9. android在myeclipse上创建的项目各种报错

    这几天被android弄得头疼死了.差不多把电脑弄了个遍. 先是离线安装ADT,下载ADT,然后配置,但是因为ADT与MyEclipse冲突.所以直接不要再myeclipse下弄Android的环境了 ...

  10. 如何使用angular-ui的弹出框

    在开发项目时,我们经常性的会遇到弹出框的需求,例如登陆,注册,添加信息等等....面对这一需求,我们当然也可以使用自己的双手进行编写,如果你时间充足可以试试. 今天我们讲解一下如何在angular框架 ...