blob: 255a6ed084fbca5fd8f83b112c6a59c5744723f8 [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
Jesse Hall04f4f472015-08-16 19:51:04 -070024#include <vulkan/vulkan.h>
Jesse Hall4da3bd62016-01-16 22:14:40 -080025#include <vulkan/vk_ext_debug_report.h>
Jesse Hall04f4f472015-08-16 19:51:04 -070026
27#define LOG_TAG "vkinfo"
28#include <log/log.h>
29
30namespace {
31
Jesse Hall09559b52016-01-07 21:50:19 -080032struct GpuInfo {
33 VkPhysicalDeviceProperties properties;
34 VkPhysicalDeviceMemoryProperties memory;
Jesse Hallb1471272016-01-17 21:36:58 -080035 VkPhysicalDeviceFeatures features;
Jesse Hall09559b52016-01-07 21:50:19 -080036 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 Hallb1471272016-01-17 21:36:58 -0800119void GatherGpuInfo(VkPhysicalDevice gpu, GpuInfo& info) {
120 VkResult result;
121 uint32_t count;
122
123 vkGetPhysicalDeviceProperties(gpu, &info.properties);
124 vkGetPhysicalDeviceMemoryProperties(gpu, &info.memory);
125 vkGetPhysicalDeviceFeatures(gpu, &info.features);
126
127 vkGetPhysicalDeviceQueueFamilyProperties(gpu, &count, nullptr);
128 info.queue_families.resize(count);
129 vkGetPhysicalDeviceQueueFamilyProperties(gpu, &count,
130 info.queue_families.data());
131
132 result = vkEnumerateDeviceLayerProperties(gpu, &count, nullptr);
133 if (result != VK_SUCCESS)
134 die("vkEnumerateDeviceLayerProperties (count)", result);
135 do {
136 info.layers.resize(count);
137 result =
138 vkEnumerateDeviceLayerProperties(gpu, &count, info.layers.data());
139 } while (result == VK_INCOMPLETE);
140 if (result != VK_SUCCESS)
141 die("vkEnumerateDeviceLayerProperties (data)", result);
142 info.layer_extensions.resize(info.layers.size());
143
144 EnumerateDeviceExtensions(gpu, nullptr, &info.extensions);
145 for (size_t i = 0; i < info.layers.size(); i++) {
146 EnumerateDeviceExtensions(gpu, info.layers[i].layerName,
147 &info.layer_extensions[i]);
148 }
149
150 const std::array<const char*, 1> kDesiredExtensions = {
151 {VK_KHR_SWAPCHAIN_EXTENSION_NAME},
152 };
153 const char* extensions[kDesiredExtensions.size()];
154 uint32_t num_extensions = 0;
155 for (const auto& desired_ext : kDesiredExtensions) {
156 bool available = HasExtension(info.extensions, desired_ext);
157 for (size_t i = 0; !available && i < info.layer_extensions.size(); i++)
158 available = HasExtension(info.layer_extensions[i], desired_ext);
159 if (available)
160 extensions[num_extensions++] = desired_ext;
161 }
162
163 VkDevice device;
164 const VkDeviceQueueCreateInfo queue_create_info = {
165 .sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO,
166 .queueFamilyIndex = 0,
167 .queueCount = 1,
168 };
169 const VkDeviceCreateInfo create_info = {
170 .sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO,
171 .queueCreateInfoCount = 1,
172 .pQueueCreateInfos = &queue_create_info,
173 .enabledExtensionCount = num_extensions,
174 .ppEnabledExtensionNames = extensions,
175 .pEnabledFeatures = &info.features,
176 };
177 result = vkCreateDevice(gpu, &create_info, nullptr, &device);
178 if (result != VK_SUCCESS)
179 die("vkCreateDevice", result);
180 vkDestroyDevice(device, nullptr);
181}
182
Jesse Hall09559b52016-01-07 21:50:19 -0800183void GatherInfo(VulkanInfo* info) {
184 VkResult result;
185 uint32_t count;
186
187 result = vkEnumerateInstanceLayerProperties(&count, nullptr);
188 if (result != VK_SUCCESS)
189 die("vkEnumerateInstanceLayerProperties (count)", result);
190 do {
191 info->layers.resize(count);
192 result =
193 vkEnumerateInstanceLayerProperties(&count, info->layers.data());
194 } while (result == VK_INCOMPLETE);
195 if (result != VK_SUCCESS)
196 die("vkEnumerateInstanceLayerProperties (data)", result);
197 info->layer_extensions.resize(info->layers.size());
198
199 EnumerateInstanceExtensions(nullptr, &info->extensions);
200 for (size_t i = 0; i < info->layers.size(); i++) {
201 EnumerateInstanceExtensions(info->layers[i].layerName,
202 &info->layer_extensions[i]);
203 }
204
Jesse Hallae3b70d2016-01-17 22:05:29 -0800205 const char* kDesiredExtensions[] = {
206 VK_EXT_DEBUG_REPORT_EXTENSION_NAME,
Jesse Hall4da3bd62016-01-16 22:14:40 -0800207 };
Jesse Hallae3b70d2016-01-17 22:05:29 -0800208 const char*
209 extensions[sizeof(kDesiredExtensions) / sizeof(kDesiredExtensions[0])];
Jesse Hall4da3bd62016-01-16 22:14:40 -0800210 uint32_t num_extensions = 0;
211 for (const auto& desired_ext : kDesiredExtensions) {
212 bool available = HasExtension(info->extensions, desired_ext);
213 for (size_t i = 0; !available && i < info->layer_extensions.size(); i++)
214 available = HasExtension(info->layer_extensions[i], desired_ext);
215 if (available)
216 extensions[num_extensions++] = desired_ext;
217 }
218
Jesse Hall09559b52016-01-07 21:50:19 -0800219 const VkInstanceCreateInfo create_info = {
220 .sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO,
Jesse Hall4da3bd62016-01-16 22:14:40 -0800221 .enabledExtensionCount = num_extensions,
222 .ppEnabledExtensionNames = extensions,
Jesse Hall09559b52016-01-07 21:50:19 -0800223 };
Jesse Hall4da3bd62016-01-16 22:14:40 -0800224 VkInstance instance;
Jesse Hall09559b52016-01-07 21:50:19 -0800225 result = vkCreateInstance(&create_info, nullptr, &instance);
226 if (result != VK_SUCCESS)
227 die("vkCreateInstance", result);
228
229 uint32_t num_gpus;
230 result = vkEnumeratePhysicalDevices(instance, &num_gpus, nullptr);
231 if (result != VK_SUCCESS)
232 die("vkEnumeratePhysicalDevices (count)", result);
233 std::vector<VkPhysicalDevice> gpus(num_gpus, VK_NULL_HANDLE);
234 do {
235 gpus.resize(num_gpus, VK_NULL_HANDLE);
236 result = vkEnumeratePhysicalDevices(instance, &num_gpus, gpus.data());
237 } while (result == VK_INCOMPLETE);
238 if (result != VK_SUCCESS)
239 die("vkEnumeratePhysicalDevices (data)", result);
240
241 info->gpus.resize(num_gpus);
Jesse Hallb1471272016-01-17 21:36:58 -0800242 for (size_t i = 0; i < gpus.size(); i++)
243 GatherGpuInfo(gpus[i], info->gpus.at(i));
Jesse Hall09559b52016-01-07 21:50:19 -0800244
245 vkDestroyInstance(instance, nullptr);
246}
247
248// ----------------------------------------------------------------------------
249
Jesse Hall38cb8402016-01-18 03:41:35 -0800250struct Options {
251 bool layer_description;
252 bool layer_extensions;
253};
254
255const size_t kMaxIndent = 8;
256const size_t kIndentSize = 3;
257std::array<char, kMaxIndent * kIndentSize + 1> kIndent;
258const char* Indent(size_t n) {
259 static bool initialized = false;
260 if (!initialized) {
261 kIndent.fill(' ');
262 kIndent.back() = '\0';
263 initialized = true;
264 }
265 return kIndent.data() +
266 (kIndent.size() - (kIndentSize * std::min(n, kMaxIndent) + 1));
267}
268
Jesse Hallc1ab3032016-01-04 07:26:53 -0800269uint32_t ExtractMajorVersion(uint32_t version) {
270 return (version >> 22) & 0x3FF;
271}
272uint32_t ExtractMinorVersion(uint32_t version) {
273 return (version >> 12) & 0x3FF;
274}
275uint32_t ExtractPatchVersion(uint32_t version) {
276 return (version >> 0) & 0xFFF;
277}
278
Jesse Hall04f4f472015-08-16 19:51:04 -0700279const char* VkPhysicalDeviceTypeStr(VkPhysicalDeviceType type) {
280 switch (type) {
281 case VK_PHYSICAL_DEVICE_TYPE_OTHER:
282 return "OTHER";
283 case VK_PHYSICAL_DEVICE_TYPE_INTEGRATED_GPU:
284 return "INTEGRATED_GPU";
285 case VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU:
286 return "DISCRETE_GPU";
287 case VK_PHYSICAL_DEVICE_TYPE_VIRTUAL_GPU:
288 return "VIRTUAL_GPU";
289 case VK_PHYSICAL_DEVICE_TYPE_CPU:
290 return "CPU";
291 default:
292 return "<UNKNOWN>";
293 }
294}
295
Jesse Hall09559b52016-01-07 21:50:19 -0800296void PrintExtensions(const std::vector<VkExtensionProperties>& extensions,
Jesse Hall38cb8402016-01-18 03:41:35 -0800297 const Options& /*options*/,
298 size_t indent) {
Jesse Hall09559b52016-01-07 21:50:19 -0800299 for (const auto& e : extensions)
Jesse Hall38cb8402016-01-18 03:41:35 -0800300 printf("%s%s (v%u)\n", Indent(indent), e.extensionName, e.specVersion);
Jesse Hall09559b52016-01-07 21:50:19 -0800301}
Jesse Hallc1ab3032016-01-04 07:26:53 -0800302
Jesse Hallaa410942016-01-17 13:07:10 -0800303void PrintLayers(
304 const std::vector<VkLayerProperties>& layers,
305 const std::vector<std::vector<VkExtensionProperties>> extensions,
Jesse Hall38cb8402016-01-18 03:41:35 -0800306 const Options& options,
307 size_t indent) {
Jesse Hallaa410942016-01-17 13:07:10 -0800308 for (size_t i = 0; i < layers.size(); i++) {
Jesse Hall38cb8402016-01-18 03:41:35 -0800309 printf("%s%s %u.%u.%u/%u\n", Indent(indent), layers[i].layerName,
310 ExtractMajorVersion(layers[i].specVersion),
311 ExtractMinorVersion(layers[i].specVersion),
312 ExtractPatchVersion(layers[i].specVersion),
313 layers[i].implementationVersion);
314 if (options.layer_description)
315 printf("%s%s\n", Indent(indent + 1), layers[i].description);
316 if (options.layer_extensions && !extensions[i].empty()) {
317 if (!extensions[i].empty()) {
318 printf("%sExtensions [%zu]:\n", Indent(indent + 1),
319 extensions[i].size());
320 PrintExtensions(extensions[i], options, indent + 2);
321 }
322 }
Jesse Hallaa410942016-01-17 13:07:10 -0800323 }
324}
325
Jesse Hall38cb8402016-01-18 03:41:35 -0800326void PrintGpuInfo(const GpuInfo& info, const Options& options, size_t indent) {
Jesse Hall04f4f472015-08-16 19:51:04 -0700327 VkResult result;
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100328 std::ostringstream strbuf;
Jesse Hall04f4f472015-08-16 19:51:04 -0700329
Jesse Hall38cb8402016-01-18 03:41:35 -0800330 printf("%s\"%s\" (%s) %u.%u.%u/%#x [%04x:%04x]\n", Indent(indent),
Jesse Hall09559b52016-01-07 21:50:19 -0800331 info.properties.deviceName,
332 VkPhysicalDeviceTypeStr(info.properties.deviceType),
333 ExtractMajorVersion(info.properties.apiVersion),
334 ExtractMinorVersion(info.properties.apiVersion),
335 ExtractPatchVersion(info.properties.apiVersion),
336 info.properties.driverVersion, info.properties.vendorID,
337 info.properties.deviceID);
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100338
Jesse Hall09559b52016-01-07 21:50:19 -0800339 for (uint32_t heap = 0; heap < info.memory.memoryHeapCount; heap++) {
340 if ((info.memory.memoryHeaps[heap].flags &
Jesse Halld1af8122015-11-29 23:50:38 -0800341 VK_MEMORY_HEAP_DEVICE_LOCAL_BIT) != 0)
342 strbuf << "DEVICE_LOCAL";
Jesse Hall38cb8402016-01-18 03:41:35 -0800343 printf("%sHeap %u: %" PRIu64 " MiB (0x%" PRIx64 " B) %s\n",
344 Indent(indent + 1), heap,
Jesse Hall8966bf42016-01-16 17:26:48 -0800345 info.memory.memoryHeaps[heap].size / 0x1000000,
Jesse Hall09559b52016-01-07 21:50:19 -0800346 info.memory.memoryHeaps[heap].size, strbuf.str().c_str());
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100347 strbuf.str(std::string());
348
Jesse Hall09559b52016-01-07 21:50:19 -0800349 for (uint32_t type = 0; type < info.memory.memoryTypeCount; type++) {
350 if (info.memory.memoryTypes[type].heapIndex != heap)
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100351 continue;
352 VkMemoryPropertyFlags flags =
Jesse Hall09559b52016-01-07 21:50:19 -0800353 info.memory.memoryTypes[type].propertyFlags;
Jesse Halld1af8122015-11-29 23:50:38 -0800354 if ((flags & VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT) != 0)
Jesse Hall1f91d392015-12-11 16:28:44 -0800355 strbuf << " DEVICE_LOCAL";
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100356 if ((flags & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) != 0)
Jesse Hall1f91d392015-12-11 16:28:44 -0800357 strbuf << " HOST_VISIBLE";
Jesse Halld1af8122015-11-29 23:50:38 -0800358 if ((flags & VK_MEMORY_PROPERTY_HOST_COHERENT_BIT) != 0)
359 strbuf << " COHERENT";
360 if ((flags & VK_MEMORY_PROPERTY_HOST_CACHED_BIT) != 0)
361 strbuf << " CACHED";
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100362 if ((flags & VK_MEMORY_PROPERTY_LAZILY_ALLOCATED_BIT) != 0)
363 strbuf << " LAZILY_ALLOCATED";
Jesse Hall38cb8402016-01-18 03:41:35 -0800364 printf("%sType %u:%s\n", Indent(indent + 2), type,
365 strbuf.str().c_str());
Jesse Hall73ab0ac2015-08-25 15:09:15 +0100366 strbuf.str(std::string());
367 }
368 }
Jesse Hall5ae3abb2015-10-08 14:00:22 -0700369
Jesse Hall09559b52016-01-07 21:50:19 -0800370 for (uint32_t family = 0; family < info.queue_families.size(); family++) {
371 const VkQueueFamilyProperties& qprops = info.queue_families[family];
Jesse Hall8966bf42016-01-16 17:26:48 -0800372 VkQueueFlags flags = qprops.queueFlags;
373 char flags_str[5];
374 flags_str[0] = (flags & VK_QUEUE_GRAPHICS_BIT) ? 'G' : '_';
375 flags_str[1] = (flags & VK_QUEUE_COMPUTE_BIT) ? 'C' : '_';
376 flags_str[2] = (flags & VK_QUEUE_TRANSFER_BIT) ? 'T' : '_';
377 flags_str[3] = (flags & VK_QUEUE_SPARSE_BINDING_BIT) ? 'S' : '_';
378 flags_str[4] = '\0';
379 printf(
Jesse Hall38cb8402016-01-18 03:41:35 -0800380 "%sQueue Family %u: %ux %s\n"
381 "%stimestampValidBits: %ub\n"
382 "%sminImageTransferGranularity: (%u,%u,%u)\n",
383 Indent(indent + 1), family, qprops.queueCount, flags_str,
384 Indent(indent + 2), qprops.timestampValidBits, Indent(indent + 2),
Jesse Hall8966bf42016-01-16 17:26:48 -0800385 qprops.minImageTransferGranularity.width,
386 qprops.minImageTransferGranularity.height,
387 qprops.minImageTransferGranularity.depth);
Jesse Hallb1471272016-01-17 21:36:58 -0800388 }
Jesse Hall6e4ab312016-01-07 22:26:20 -0800389
Jesse Hall38cb8402016-01-18 03:41:35 -0800390 printf("%sExtensions [%zu]:\n", Indent(indent + 1), info.extensions.size());
391 if (!info.extensions.empty())
392 PrintExtensions(info.extensions, options, indent + 2);
393 printf("%sLayers [%zu]:\n", Indent(indent + 1), info.layers.size());
394 if (!info.layers.empty())
395 PrintLayers(info.layers, info.layer_extensions, options, indent + 2);
Jesse Hall04f4f472015-08-16 19:51:04 -0700396}
397
Jesse Hall38cb8402016-01-18 03:41:35 -0800398void PrintInfo(const VulkanInfo& info, const Options& options) {
Jesse Hall09559b52016-01-07 21:50:19 -0800399 std::ostringstream strbuf;
Jesse Hall38cb8402016-01-18 03:41:35 -0800400 size_t indent = 0;
Jesse Hall09559b52016-01-07 21:50:19 -0800401
Jesse Hall38cb8402016-01-18 03:41:35 -0800402 printf("%sInstance Extensions [%zu]:\n", Indent(indent),
403 info.extensions.size());
404 PrintExtensions(info.extensions, options, indent + 1);
405 printf("%sInstance Layers [%zu]:\n", Indent(indent), info.layers.size());
406 if (!info.layers.empty())
407 PrintLayers(info.layers, info.layer_extensions, options, indent + 1);
Jesse Hall09559b52016-01-07 21:50:19 -0800408
Jesse Hall38cb8402016-01-18 03:41:35 -0800409 printf("%sPhysicalDevices [%zu]:\n", Indent(indent), info.gpus.size());
Jesse Hall09559b52016-01-07 21:50:19 -0800410 for (const auto& gpu : info.gpus)
Jesse Hall38cb8402016-01-18 03:41:35 -0800411 PrintGpuInfo(gpu, options, indent + 1);
Jesse Hallc1ab3032016-01-04 07:26:53 -0800412}
413
Jesse Hall04f4f472015-08-16 19:51:04 -0700414} // namespace
415
Jesse Hall09559b52016-01-07 21:50:19 -0800416// ----------------------------------------------------------------------------
417
Jesse Hall38cb8402016-01-18 03:41:35 -0800418int main(int argc, char const* argv[]) {
419 Options options = {
420 .layer_description = false, .layer_extensions = false,
421 };
422 for (int argi = 1; argi < argc; argi++) {
423 if (strcmp(argv[argi], "-v") == 0) {
424 options.layer_description = true;
425 options.layer_extensions = true;
426 } else if (strcmp(argv[argi], "-layer_description") == 0) {
427 options.layer_description = true;
428 } else if (strcmp(argv[argi], "-layer_extensions") == 0) {
429 options.layer_extensions = true;
430 }
431 }
432
Jesse Hallc1ab3032016-01-04 07:26:53 -0800433 VulkanInfo info;
434 GatherInfo(&info);
Jesse Hall38cb8402016-01-18 03:41:35 -0800435 PrintInfo(info, options);
Jesse Hall04f4f472015-08-16 19:51:04 -0700436 return 0;
437}