blob: 8abd160b09a96fea996c12a765dd32b1bb72a32f [file] [log] [blame]
Jesse Halld02edcb2015-09-08 07:44:48 -07001/*
2 * Copyright 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Jesse Hall4da3bd62016-01-16 22:14:40 -080017#include <algorithm>
18#include <array>
Jesse Hall73ab0ac2015-08-25 15:09:15 +010019#include <inttypes.h>
Jesse Hall04f4f472015-08-16 19:51:04 -070020#include <stdlib.h>
Jesse Hall1f91d392015-12-11 16:28:44 -080021#include <sstream>
Jesse Hall04f4f472015-08-16 19:51:04 -070022#include <vector>
23
24#define VK_PROTOTYPES
25#include <vulkan/vulkan.h>
Jesse Hall4da3bd62016-01-16 22:14:40 -080026#include <vulkan/vk_ext_debug_report.h>
Jesse Hall04f4f472015-08-16 19:51:04 -070027
28#define LOG_TAG "vkinfo"
29#include <log/log.h>
30
31namespace {
32
Jesse Hall09559b52016-01-07 21:50:19 -080033struct GpuInfo {
34 VkPhysicalDeviceProperties properties;
35 VkPhysicalDeviceMemoryProperties memory;
36 std::vector<VkQueueFamilyProperties> queue_families;
Jesse Hall6e4ab312016-01-07 22:26:20 -080037 std::vector<VkExtensionProperties> extensions;
38 std::vector<VkLayerProperties> layers;
39 std::vector<std::vector<VkExtensionProperties>> layer_extensions;
Jesse Hall09559b52016-01-07 21:50:19 -080040};
41struct VulkanInfo {
42 std::vector<VkExtensionProperties> extensions;
43 std::vector<VkLayerProperties> layers;
44 std::vector<std::vector<VkExtensionProperties>> layer_extensions;
45 std::vector<GpuInfo> gpus;
46};
47
48// ----------------------------------------------------------------------------
49
Jesse Hall04f4f472015-08-16 19:51:04 -070050[[noreturn]] void die(const char* proc, VkResult result) {
51 const char* result_str;
52 switch (result) {
53 // clang-format off
54 case VK_SUCCESS: result_str = "VK_SUCCESS"; break;
Jesse Hall04f4f472015-08-16 19:51:04 -070055 case VK_NOT_READY: result_str = "VK_NOT_READY"; break;
56 case VK_TIMEOUT: result_str = "VK_TIMEOUT"; break;
57 case VK_EVENT_SET: result_str = "VK_EVENT_SET"; break;
58 case VK_EVENT_RESET: result_str = "VK_EVENT_RESET"; break;
59 case VK_INCOMPLETE: result_str = "VK_INCOMPLETE"; break;
Jesse Hall04f4f472015-08-16 19:51:04 -070060 case VK_ERROR_OUT_OF_HOST_MEMORY: result_str = "VK_ERROR_OUT_OF_HOST_MEMORY"; break;
61 case VK_ERROR_OUT_OF_DEVICE_MEMORY: result_str = "VK_ERROR_OUT_OF_DEVICE_MEMORY"; break;
Jesse Hall5ae3abb2015-10-08 14:00:22 -070062 case VK_ERROR_INITIALIZATION_FAILED: result_str = "VK_ERROR_INITIALIZATION_FAILED"; break;
Jesse Hall04f4f472015-08-16 19:51:04 -070063 case VK_ERROR_DEVICE_LOST: result_str = "VK_ERROR_DEVICE_LOST"; break;
Jesse Hall04f4f472015-08-16 19:51:04 -070064 case VK_ERROR_MEMORY_MAP_FAILED: result_str = "VK_ERROR_MEMORY_MAP_FAILED"; break;
Jesse Hall5ae3abb2015-10-08 14:00:22 -070065 case VK_ERROR_LAYER_NOT_PRESENT: result_str = "VK_ERROR_LAYER_NOT_PRESENT"; break;
66 case VK_ERROR_EXTENSION_NOT_PRESENT: result_str = "VK_ERROR_EXTENSION_NOT_PRESENT"; break;
Jesse Hall04f4f472015-08-16 19:51:04 -070067 case VK_ERROR_INCOMPATIBLE_DRIVER: result_str = "VK_ERROR_INCOMPATIBLE_DRIVER"; break;
Jesse Hall04f4f472015-08-16 19:51:04 -070068 default: result_str = "<unknown VkResult>"; break;
69 // clang-format on
70 }
71 fprintf(stderr, "%s failed: %s (%d)\n", proc, result_str, result);
72 exit(1);
73}
74
Jesse Hall4da3bd62016-01-16 22:14:40 -080075bool HasExtension(const std::vector<VkExtensionProperties>& extensions,
76 const char* name) {
77 return std::find_if(extensions.cbegin(), extensions.cend(),
78 [=](const VkExtensionProperties& prop) {
79 return strcmp(prop.extensionName, name) == 0;
80 }) != extensions.end();
81}
82
Jesse Hall09559b52016-01-07 21:50:19 -080083void EnumerateInstanceExtensions(
84 const char* layer_name,
85 std::vector<VkExtensionProperties>* extensions) {
86 VkResult result;
87 uint32_t count;
88 result =
89 vkEnumerateInstanceExtensionProperties(layer_name, &count, nullptr);
90 if (result != VK_SUCCESS)
91 die("vkEnumerateInstanceExtensionProperties (count)", result);
92 do {
93 extensions->resize(count);
94 result = vkEnumerateInstanceExtensionProperties(layer_name, &count,
95 extensions->data());
96 } while (result == VK_INCOMPLETE);
97 if (result != VK_SUCCESS)
98 die("vkEnumerateInstanceExtensionProperties (data)", result);
99}
100
Jesse Hall6e4ab312016-01-07 22:26:20 -0800101void EnumerateDeviceExtensions(VkPhysicalDevice gpu,
102 const char* layer_name,
103 std::vector<VkExtensionProperties>* extensions) {
104 VkResult result;
105 uint32_t count;
106 result =
107 vkEnumerateDeviceExtensionProperties(gpu, layer_name, &count, nullptr);
108 if (result != VK_SUCCESS)
109 die("vkEnumerateDeviceExtensionProperties (count)", result);
110 do {
111 extensions->resize(count);
112 result = vkEnumerateDeviceExtensionProperties(gpu, layer_name, &count,
113 extensions->data());
114 } while (result == VK_INCOMPLETE);
115 if (result != VK_SUCCESS)
116 die("vkEnumerateDeviceExtensionProperties (data)", result);
117}
118
Jesse Hall09559b52016-01-07 21:50:19 -0800119void GatherInfo(VulkanInfo* info) {
120 VkResult result;
121 uint32_t count;
122
123 result = vkEnumerateInstanceLayerProperties(&count, nullptr);
124 if (result != VK_SUCCESS)
125 die("vkEnumerateInstanceLayerProperties (count)", result);
126 do {
127 info->layers.resize(count);
128 result =
129 vkEnumerateInstanceLayerProperties(&count, info->layers.data());
130 } while (result == VK_INCOMPLETE);
131 if (result != VK_SUCCESS)
132 die("vkEnumerateInstanceLayerProperties (data)", result);
133 info->layer_extensions.resize(info->layers.size());
134
135 EnumerateInstanceExtensions(nullptr, &info->extensions);
136 for (size_t i = 0; i < info->layers.size(); i++) {
137 EnumerateInstanceExtensions(info->layers[i].layerName,
138 &info->layer_extensions[i]);
139 }
140
Jesse Hall4da3bd62016-01-16 22:14:40 -0800141 const std::array<const char*, 1> kDesiredExtensions = {
142 {VK_EXT_DEBUG_REPORT_EXTENSION_NAME},
143 };
144 const char* extensions[kDesiredExtensions.size()];
145 uint32_t num_extensions = 0;
146 for (const auto& desired_ext : kDesiredExtensions) {
147 bool available = HasExtension(info->extensions, desired_ext);
148 for (size_t i = 0; !available && i < info->layer_extensions.size(); i++)
149 available = HasExtension(info->layer_extensions[i], desired_ext);
150 if (available)
151 extensions[num_extensions++] = desired_ext;
152 }
153
Jesse Hall09559b52016-01-07 21:50:19 -0800154 const VkInstanceCreateInfo create_info = {
155 .sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO,
Jesse Hall4da3bd62016-01-16 22:14:40 -0800156 .enabledExtensionCount = num_extensions,
157 .ppEnabledExtensionNames = extensions,
Jesse Hall09559b52016-01-07 21:50:19 -0800158 };
Jesse Hall4da3bd62016-01-16 22:14:40 -0800159 VkInstance instance;
Jesse Hall09559b52016-01-07 21:50:19 -0800160 result = vkCreateInstance(&create_info, nullptr, &instance);
161 if (result != VK_SUCCESS)
162 die("vkCreateInstance", result);
163
164 uint32_t num_gpus;
165 result = vkEnumeratePhysicalDevices(instance, &num_gpus, nullptr);
166 if (result != VK_SUCCESS)
167 die("vkEnumeratePhysicalDevices (count)", result);
168 std::vector<VkPhysicalDevice> gpus(num_gpus, VK_NULL_HANDLE);
169 do {
170 gpus.resize(num_gpus, VK_NULL_HANDLE);
171 result = vkEnumeratePhysicalDevices(instance, &num_gpus, gpus.data());
172 } while (result == VK_INCOMPLETE);
173 if (result != VK_SUCCESS)
174 die("vkEnumeratePhysicalDevices (data)", result);
175
176 info->gpus.resize(num_gpus);
177 for (size_t gpu_idx = 0; gpu_idx < gpus.size(); gpu_idx++) {
178 VkPhysicalDevice gpu = gpus[gpu_idx];
179 GpuInfo& gpu_info = info->gpus.at(gpu_idx);
180
181 vkGetPhysicalDeviceProperties(gpu, &gpu_info.properties);
182 vkGetPhysicalDeviceMemoryProperties(gpu, &gpu_info.memory);
183
184 vkGetPhysicalDeviceQueueFamilyProperties(gpu, &count, nullptr);
185 gpu_info.queue_families.resize(count);
186 vkGetPhysicalDeviceQueueFamilyProperties(
187 gpu, &count, gpu_info.queue_families.data());
Jesse Hall6e4ab312016-01-07 22:26:20 -0800188
189 result = vkEnumerateDeviceLayerProperties(gpu, &count, nullptr);
190 if (result != VK_SUCCESS)
191 die("vkEnumerateDeviceLayerProperties (count)", result);
192 do {
193 gpu_info.layers.resize(count);
194 result = vkEnumerateDeviceLayerProperties(gpu, &count,
195 gpu_info.layers.data());
196 } while (result == VK_INCOMPLETE);
197 if (result != VK_SUCCESS)
198 die("vkEnumerateDeviceLayerProperties (data)", result);
199 gpu_info.layer_extensions.resize(gpu_info.layers.size());
200
201 EnumerateDeviceExtensions(gpu, nullptr, &gpu_info.extensions);
202 for (size_t i = 0; i < gpu_info.layers.size(); i++) {
203 EnumerateDeviceExtensions(gpu, gpu_info.layers[i].layerName,
204 &gpu_info.layer_extensions[i]);
205 }
Jesse Hall09559b52016-01-07 21:50:19 -0800206 }
207
208 vkDestroyInstance(instance, nullptr);
209}
210
211// ----------------------------------------------------------------------------
212
Jesse Hallc1ab3032016-01-04 07:26:53 -0800213uint32_t ExtractMajorVersion(uint32_t version) {
214 return (version >> 22) & 0x3FF;
215}
216uint32_t ExtractMinorVersion(uint32_t version) {
217 return (version >> 12) & 0x3FF;
218}
219uint32_t ExtractPatchVersion(uint32_t version) {
220 return (version >> 0) & 0xFFF;
221}
222
Jesse Hall04f4f472015-08-16 19:51:04 -0700223const char* VkPhysicalDeviceTypeStr(VkPhysicalDeviceType type) {
224 switch (type) {
225 case VK_PHYSICAL_DEVICE_TYPE_OTHER:
226 return "OTHER";
227 case VK_PHYSICAL_DEVICE_TYPE_INTEGRATED_GPU:
228 return "INTEGRATED_GPU";
229 case VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU:
230 return "DISCRETE_GPU";
231 case VK_PHYSICAL_DEVICE_TYPE_VIRTUAL_GPU:
232 return "VIRTUAL_GPU";
233 case VK_PHYSICAL_DEVICE_TYPE_CPU:
234 return "CPU";
235 default:
236 return "<UNKNOWN>";
237 }
238}
239
Jesse Hall5ae3abb2015-10-08 14:00:22 -0700240const char* VkQueueFlagBitStr(VkQueueFlagBits bit) {
241 switch (bit) {
242 case VK_QUEUE_GRAPHICS_BIT:
243 return "GRAPHICS";
244 case VK_QUEUE_COMPUTE_BIT:
245 return "COMPUTE";
Jesse Hall65ab5522015-11-30 00:07:16 -0800246 case VK_QUEUE_TRANSFER_BIT:
247 return "TRANSFER";
Jesse Hallb00daad2015-11-29 19:46:20 -0800248 case VK_QUEUE_SPARSE_BINDING_BIT:
Jesse Hall5ae3abb2015-10-08 14:00:22 -0700249 return "SPARSE";
Jesse Hall5ae3abb2015-10-08 14:00:22 -0700250 }
251}
252
Jesse Hall09559b52016-01-07 21:50:19 -0800253void PrintExtensions(const std::vector<VkExtensionProperties>& extensions,
254 const char* prefix) {
255 for (const auto& e : extensions)
Jesse Hall8966bf42016-01-16 17:26:48 -0800256 printf("%s%s (v%u)\n", prefix, e.extensionName, e.specVersion);
Jesse Hall09559b52016-01-07 21:50:19 -0800257}
Jesse Hallc1ab3032016-01-04 07:26:53 -0800258
Jesse Hall09559b52016-01-07 21:50:19 -0800259void PrintGpuInfo(const GpuInfo& info) {
Jesse Hall04f4f472015-08-16 19:51:04 -0700260 VkResult result;
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100261 std::ostringstream strbuf;
Jesse Hall04f4f472015-08-16 19:51:04 -0700262
Jesse Hall8966bf42016-01-16 17:26:48 -0800263 printf(" \"%s\" (%s) %u.%u.%u/%#x [%04x:%04x]\n",
Jesse Hall09559b52016-01-07 21:50:19 -0800264 info.properties.deviceName,
265 VkPhysicalDeviceTypeStr(info.properties.deviceType),
266 ExtractMajorVersion(info.properties.apiVersion),
267 ExtractMinorVersion(info.properties.apiVersion),
268 ExtractPatchVersion(info.properties.apiVersion),
269 info.properties.driverVersion, info.properties.vendorID,
270 info.properties.deviceID);
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100271
Jesse Hall09559b52016-01-07 21:50:19 -0800272 for (uint32_t heap = 0; heap < info.memory.memoryHeapCount; heap++) {
273 if ((info.memory.memoryHeaps[heap].flags &
Jesse Halld1af8122015-11-29 23:50:38 -0800274 VK_MEMORY_HEAP_DEVICE_LOCAL_BIT) != 0)
275 strbuf << "DEVICE_LOCAL";
Jesse Hall8966bf42016-01-16 17:26:48 -0800276 printf(" Heap %u: %" PRIu64 " MiB (0x%" PRIx64 " B) %s\n", heap,
277 info.memory.memoryHeaps[heap].size / 0x1000000,
Jesse Hall09559b52016-01-07 21:50:19 -0800278 info.memory.memoryHeaps[heap].size, strbuf.str().c_str());
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100279 strbuf.str(std::string());
280
Jesse Hall09559b52016-01-07 21:50:19 -0800281 for (uint32_t type = 0; type < info.memory.memoryTypeCount; type++) {
282 if (info.memory.memoryTypes[type].heapIndex != heap)
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100283 continue;
284 VkMemoryPropertyFlags flags =
Jesse Hall09559b52016-01-07 21:50:19 -0800285 info.memory.memoryTypes[type].propertyFlags;
Jesse Halld1af8122015-11-29 23:50:38 -0800286 if ((flags & VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT) != 0)
Jesse Hall1f91d392015-12-11 16:28:44 -0800287 strbuf << " DEVICE_LOCAL";
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100288 if ((flags & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) != 0)
Jesse Hall1f91d392015-12-11 16:28:44 -0800289 strbuf << " HOST_VISIBLE";
Jesse Halld1af8122015-11-29 23:50:38 -0800290 if ((flags & VK_MEMORY_PROPERTY_HOST_COHERENT_BIT) != 0)
291 strbuf << " COHERENT";
292 if ((flags & VK_MEMORY_PROPERTY_HOST_CACHED_BIT) != 0)
293 strbuf << " CACHED";
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100294 if ((flags & VK_MEMORY_PROPERTY_LAZILY_ALLOCATED_BIT) != 0)
295 strbuf << " LAZILY_ALLOCATED";
Jesse Hall09559b52016-01-07 21:50:19 -0800296 printf(" Type %u:%s\n", type, strbuf.str().c_str());
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100297 strbuf.str(std::string());
298 }
299 }
Jesse Hall5ae3abb2015-10-08 14:00:22 -0700300
Jesse Hall09559b52016-01-07 21:50:19 -0800301 for (uint32_t family = 0; family < info.queue_families.size(); family++) {
302 const VkQueueFamilyProperties& qprops = info.queue_families[family];
Jesse Hall8966bf42016-01-16 17:26:48 -0800303 VkQueueFlags flags = qprops.queueFlags;
304 char flags_str[5];
305 flags_str[0] = (flags & VK_QUEUE_GRAPHICS_BIT) ? 'G' : '_';
306 flags_str[1] = (flags & VK_QUEUE_COMPUTE_BIT) ? 'C' : '_';
307 flags_str[2] = (flags & VK_QUEUE_TRANSFER_BIT) ? 'T' : '_';
308 flags_str[3] = (flags & VK_QUEUE_SPARSE_BINDING_BIT) ? 'S' : '_';
309 flags_str[4] = '\0';
310 printf(
311 " Queue Family %u: %ux %s\n"
312 " timestampValidBits: %ub\n"
313 " minImageTransferGranularity: (%u,%u,%u)\n",
314 family, qprops.queueCount, flags_str, qprops.timestampValidBits,
315 qprops.minImageTransferGranularity.width,
316 qprops.minImageTransferGranularity.height,
317 qprops.minImageTransferGranularity.depth);
Jesse Hall6e4ab312016-01-07 22:26:20 -0800318
319 if (!info.extensions.empty()) {
Jesse Hall30ac78b2016-01-11 21:29:40 -0800320 printf(" Extensions [%zu]:\n", info.extensions.size());
Jesse Hall6e4ab312016-01-07 22:26:20 -0800321 PrintExtensions(info.extensions, " ");
322 }
323 if (!info.layers.empty()) {
Jesse Hall30ac78b2016-01-11 21:29:40 -0800324 printf(" Layers [%zu]:\n", info.layers.size());
Jesse Hall6e4ab312016-01-07 22:26:20 -0800325 for (size_t i = 0; i < info.layers.size(); i++) {
326 const auto& layer = info.layers[i];
327 printf(" - %s %u.%u.%u/%u \"%s\"\n", layer.layerName,
328 ExtractMajorVersion(layer.specVersion),
329 ExtractMinorVersion(layer.specVersion),
330 ExtractPatchVersion(layer.specVersion),
331 layer.implementationVersion, layer.description);
332 if (!info.layer_extensions[i].empty()) {
333 printf(" Extensions [%zu]:\n",
334 info.layer_extensions.size());
335 PrintExtensions(info.layer_extensions[i], " ");
336 }
337 }
338 }
Jesse Hall5ae3abb2015-10-08 14:00:22 -0700339 }
Jesse Hall04f4f472015-08-16 19:51:04 -0700340}
341
Jesse Hallc1ab3032016-01-04 07:26:53 -0800342void PrintInfo(const VulkanInfo& info) {
Jesse Hall09559b52016-01-07 21:50:19 -0800343 std::ostringstream strbuf;
344
Jesse Hall30ac78b2016-01-11 21:29:40 -0800345 printf("Instance Extensions [%zu]:\n", info.extensions.size());
Jesse Hallc1ab3032016-01-04 07:26:53 -0800346 PrintExtensions(info.extensions, " ");
347 if (!info.layers.empty()) {
Jesse Hall30ac78b2016-01-11 21:29:40 -0800348 printf("Instance Layers [%zu]:\n", info.layers.size());
Jesse Hallc1ab3032016-01-04 07:26:53 -0800349 for (size_t i = 0; i < info.layers.size(); i++) {
350 const auto& layer = info.layers[i];
Jesse Hall8966bf42016-01-16 17:26:48 -0800351 printf(" %s %u.%u.%u/%u \"%s\"\n", layer.layerName,
Jesse Hall09559b52016-01-07 21:50:19 -0800352 ExtractMajorVersion(layer.specVersion),
353 ExtractMinorVersion(layer.specVersion),
354 ExtractPatchVersion(layer.specVersion),
355 layer.implementationVersion, layer.description);
Jesse Hallc1ab3032016-01-04 07:26:53 -0800356 if (!info.layer_extensions[i].empty()) {
Jesse Hall8966bf42016-01-16 17:26:48 -0800357 PrintExtensions(info.layer_extensions[i], " ");
Jesse Hallc1ab3032016-01-04 07:26:53 -0800358 }
359 }
360 }
Jesse Hall09559b52016-01-07 21:50:19 -0800361
362 printf("PhysicalDevices [%zu]:\n", info.gpus.size());
363 for (const auto& gpu : info.gpus)
364 PrintGpuInfo(gpu);
Jesse Hallc1ab3032016-01-04 07:26:53 -0800365}
366
Jesse Hall04f4f472015-08-16 19:51:04 -0700367} // namespace
368
Jesse Hall09559b52016-01-07 21:50:19 -0800369// ----------------------------------------------------------------------------
370
Jesse Hall04f4f472015-08-16 19:51:04 -0700371int main(int /*argc*/, char const* /*argv*/ []) {
Jesse Hallc1ab3032016-01-04 07:26:53 -0800372 VulkanInfo info;
373 GatherInfo(&info);
374 PrintInfo(info);
Jesse Hall04f4f472015-08-16 19:51:04 -0700375 return 0;
376}