1 #ifndef _VKQUERYUTIL_HPP
2 #define _VKQUERYUTIL_HPP
3 /*-------------------------------------------------------------------------
4 * Vulkan CTS Framework
5 * --------------------
6 *
7 * Copyright (c) 2015 Google Inc.
8 *
9 * Licensed under the Apache License, Version 2.0 (the "License");
10 * you may not use this file except in compliance with the License.
11 * You may obtain a copy of the License at
12 *
13 * http://www.apache.org/licenses/LICENSE-2.0
14 *
15 * Unless required by applicable law or agreed to in writing, software
16 * distributed under the License is distributed on an "AS IS" BASIS,
17 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18 * See the License for the specific language governing permissions and
19 * limitations under the License.
20 *
21 *//*!
22 * \file
23 * \brief Vulkan query utilities.
24 *//*--------------------------------------------------------------------*/
25
26 #include "vkDefs.hpp"
27 #include "tcuMaybe.hpp"
28 #include "deMemory.h"
29
30 #include <vector>
31 #include <string>
32
33 namespace vk
34 {
35
36 // API version introspection
37
38 void getCoreInstanceExtensions (deUint32 apiVersion, std::vector<const char*>& dst);
39 void getCoreDeviceExtensions (deUint32 apiVersion, std::vector<const char*>& dst);
40 bool isCoreInstanceExtension (const deUint32 apiVersion, const std::string& extension);
41 bool isCoreDeviceExtension (const deUint32 apiVersion, const std::string& extension);
42
43 // API queries
44
45 std::vector<VkPhysicalDevice> enumeratePhysicalDevices (const InstanceInterface& vk, VkInstance instance);
46 std::vector<VkPhysicalDeviceGroupProperties> enumeratePhysicalDeviceGroups (const InstanceInterface& vk, VkInstance instance);
47 std::vector<VkQueueFamilyProperties> getPhysicalDeviceQueueFamilyProperties (const InstanceInterface& vk, VkPhysicalDevice physicalDevice);
48 VkPhysicalDeviceFeatures getPhysicalDeviceFeatures (const InstanceInterface& vk, VkPhysicalDevice physicalDevice);
49 VkPhysicalDeviceFeatures2 getPhysicalDeviceFeatures2 (const InstanceInterface& vk, VkPhysicalDevice physicalDevice);
50 VkPhysicalDeviceVulkan11Features getPhysicalDeviceVulkan11Features (const InstanceInterface& vk, VkPhysicalDevice physicalDevice);
51 VkPhysicalDeviceVulkan12Features getPhysicalDeviceVulkan12Features (const InstanceInterface& vk, VkPhysicalDevice physicalDevice);
52 VkPhysicalDeviceVulkan11Properties getPhysicalDeviceVulkan11Properties (const InstanceInterface& vk, VkPhysicalDevice physicalDevice);
53 VkPhysicalDeviceVulkan12Properties getPhysicalDeviceVulkan12Properties (const InstanceInterface& vk, VkPhysicalDevice physicalDevice);
54 VkPhysicalDeviceProperties getPhysicalDeviceProperties (const InstanceInterface& vk, VkPhysicalDevice physicalDevice);
55 VkPhysicalDeviceMemoryProperties getPhysicalDeviceMemoryProperties (const InstanceInterface& vk, VkPhysicalDevice physicalDevice);
56 VkFormatProperties getPhysicalDeviceFormatProperties (const InstanceInterface& vk, VkPhysicalDevice physicalDevice, VkFormat format);
57 VkImageFormatProperties getPhysicalDeviceImageFormatProperties (const InstanceInterface& vk, VkPhysicalDevice physicalDevice, VkFormat format, VkImageType type, VkImageTiling tiling, VkImageUsageFlags usage, VkImageCreateFlags flags);
58 std::vector<VkSparseImageFormatProperties> getPhysicalDeviceSparseImageFormatProperties (const InstanceInterface& vk, VkPhysicalDevice physicalDevice, VkFormat format, VkImageType type, VkSampleCountFlagBits samples, VkImageUsageFlags usage, VkImageTiling tiling);
59
60 VkMemoryRequirements getBufferMemoryRequirements (const DeviceInterface& vk, VkDevice device, VkBuffer buffer);
61 VkMemoryRequirements getImageMemoryRequirements (const DeviceInterface& vk, VkDevice device, VkImage image);
62 VkMemoryRequirements getImagePlaneMemoryRequirements (const DeviceInterface& vk, VkDevice device, VkImage image, VkImageAspectFlagBits planeAspect);
63 std::vector<VkSparseImageMemoryRequirements> getImageSparseMemoryRequirements (const DeviceInterface& vk, VkDevice device, VkImage image);
64
65 std::vector<VkLayerProperties> enumerateInstanceLayerProperties (const PlatformInterface& vkp);
66 std::vector<VkExtensionProperties> enumerateInstanceExtensionProperties (const PlatformInterface& vkp, const char* layerName);
67 std::vector<VkLayerProperties> enumerateDeviceLayerProperties (const InstanceInterface& vki, VkPhysicalDevice physicalDevice);
68 std::vector<VkExtensionProperties> enumerateDeviceExtensionProperties (const InstanceInterface& vki, VkPhysicalDevice physicalDevice, const char* layerName);
69
70 VkQueue getDeviceQueue (const DeviceInterface& vkd, VkDevice device, deUint32 queueFamilyIndex, deUint32 queueIndex);
71 VkQueue getDeviceQueue2 (const DeviceInterface& vkd, VkDevice device, const VkDeviceQueueInfo2 *queueInfo);
72
73 // Feature / extension support
74
75 bool isShaderStageSupported (const VkPhysicalDeviceFeatures& deviceFeatures, VkShaderStageFlagBits stage);
76
77 struct RequiredExtension
78 {
79 std::string name;
80 tcu::Maybe<deUint32> minVersion;
81 tcu::Maybe<deUint32> maxVersion;
82
RequiredExtensionvk::RequiredExtension83 explicit RequiredExtension (const std::string& name_,
84 tcu::Maybe<deUint32> minVersion_ = tcu::Nothing,
85 tcu::Maybe<deUint32> maxVersion_ = tcu::Nothing)
86 : name (name_)
87 , minVersion (minVersion_)
88 , maxVersion (maxVersion_)
89 {}
90 };
91
92 struct RequiredLayer
93 {
94 std::string name;
95 tcu::Maybe<deUint32> minSpecVersion;
96 tcu::Maybe<deUint32> maxSpecVersion;
97 tcu::Maybe<deUint32> minImplVersion;
98 tcu::Maybe<deUint32> maxImplVersion;
99
RequiredLayervk::RequiredLayer100 explicit RequiredLayer (const std::string& name_,
101 tcu::Maybe<deUint32> minSpecVersion_ = tcu::Nothing,
102 tcu::Maybe<deUint32> maxSpecVersion_ = tcu::Nothing,
103 tcu::Maybe<deUint32> minImplVersion_ = tcu::Nothing,
104 tcu::Maybe<deUint32> maxImplVersion_ = tcu::Nothing)
105 : name (name_)
106 , minSpecVersion(minSpecVersion_)
107 , maxSpecVersion(maxSpecVersion_)
108 , minImplVersion(minImplVersion_)
109 , maxImplVersion(maxImplVersion_)
110 {}
111 };
112
113 bool isCompatible (const VkExtensionProperties& extensionProperties, const RequiredExtension& required);
114 bool isCompatible (const VkLayerProperties& layerProperties, const RequiredLayer& required);
115
116 template<typename ExtensionIterator>
117 bool isExtensionSupported (ExtensionIterator begin, ExtensionIterator end, const RequiredExtension& required);
118 bool isExtensionSupported (const std::vector<VkExtensionProperties>& extensions, const RequiredExtension& required);
119
120 bool isInstanceExtensionSupported (const deUint32 instanceVersion, const std::vector<std::string>& extensions, const std::string& required);
121
122 template<typename LayerIterator>
123 bool isLayerSupported (LayerIterator begin, LayerIterator end, const RequiredLayer& required);
124 bool isLayerSupported (const std::vector<VkLayerProperties>& layers, const RequiredLayer& required);
125
126 const void* findStructureInChain (const void* first, VkStructureType type);
127 void* findStructureInChain (void* first, VkStructureType type);
128
129 template<typename StructType>
130 VkStructureType getStructureType (void);
131
132 template<typename StructType>
findStructure(const void * first)133 const StructType* findStructure (const void* first)
134 {
135 return reinterpret_cast<const StructType*>(findStructureInChain(first, getStructureType<StructType>()));
136 }
137
138 template<typename StructType>
findStructure(void * first)139 StructType* findStructure (void* first)
140 {
141 return reinterpret_cast<StructType*>(findStructureInChain(first, getStructureType<StructType>()));
142 }
143
144 struct initVulkanStructure
145 {
initVulkanStructurevk::initVulkanStructure146 initVulkanStructure (void* pNext = DE_NULL) : m_next(pNext) {}
147
148 template<class StructType>
operator StructTypevk::initVulkanStructure149 operator StructType()
150 {
151 StructType result;
152
153 deMemset(&result, 0x00, sizeof(StructType));
154
155 result.sType = getStructureType<StructType>();
156 result.pNext = m_next;
157
158 return result;
159 }
160
161 private:
162 void* m_next;
163 };
164
165 template<class StructType>
addToChainVulkanStructure(void *** chainPNextPtr,StructType & structType)166 void addToChainVulkanStructure (void*** chainPNextPtr, StructType& structType)
167 {
168 DE_ASSERT(chainPNextPtr != DE_NULL);
169
170 (**chainPNextPtr) = &structType;
171
172 (*chainPNextPtr) = &structType.pNext;
173 }
174
175 struct initVulkanStructureConst
176 {
initVulkanStructureConstvk::initVulkanStructureConst177 initVulkanStructureConst (const void* pNext = DE_NULL) : m_next(pNext) {}
178
179 template<class StructType>
operator const StructTypevk::initVulkanStructureConst180 operator const StructType()
181 {
182 StructType result;
183
184 deMemset(&result, 0x00, sizeof(StructType));
185
186 result.sType = getStructureType<StructType>();
187 result.pNext = const_cast<void*>(m_next);
188
189 return result;
190 }
191
192 private:
193 const void* m_next;
194 };
195
196 struct getPhysicalDeviceExtensionProperties
197 {
getPhysicalDeviceExtensionPropertiesvk::getPhysicalDeviceExtensionProperties198 getPhysicalDeviceExtensionProperties (const InstanceInterface& vki, VkPhysicalDevice physicalDevice) : m_vki(vki), m_physicalDevice(physicalDevice) {}
199
200 template<class ExtensionProperties>
operator ExtensionPropertiesvk::getPhysicalDeviceExtensionProperties201 operator ExtensionProperties ()
202 {
203 VkPhysicalDeviceProperties2 properties2;
204 ExtensionProperties extensionProperties;
205
206 deMemset(&extensionProperties, 0x00, sizeof(ExtensionProperties));
207 extensionProperties.sType = getStructureType<ExtensionProperties>();
208
209 deMemset(&properties2, 0x00, sizeof(properties2));
210 properties2.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2;
211 properties2.pNext = &extensionProperties;
212
213 m_vki.getPhysicalDeviceProperties2(m_physicalDevice, &properties2);
214
215 return extensionProperties;
216 }
217
operator VkPhysicalDeviceProperties2vk::getPhysicalDeviceExtensionProperties218 operator VkPhysicalDeviceProperties2 ()
219 {
220 VkPhysicalDeviceProperties2 properties2;
221
222 deMemset(&properties2, 0x00, sizeof(properties2));
223 properties2.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2;
224
225 m_vki.getPhysicalDeviceProperties2(m_physicalDevice, &properties2);
226
227 return properties2;
228 }
229
230 private:
231 const InstanceInterface& m_vki;
232 const VkPhysicalDevice m_physicalDevice;
233 };
234
235 // Walks through chain to find empty pNext and assigns what to found pNext
236 void appendStructurePtrToVulkanChain (const void** chainHead, const void* structurePtr);
237
238 namespace ValidateQueryBits
239 {
240
241 typedef struct
242 {
243 size_t offset;
244 size_t size;
245 } QueryMemberTableEntry;
246
247 template <typename Context, typename Interface, typename Type>
248 //!< Return variable initialization validation
validateInitComplete(Context context,void (Interface::* Function)(Context,Type *)const,const Interface & interface,const QueryMemberTableEntry * queryMemberTableEntry)249 bool validateInitComplete(Context context, void (Interface::*Function)(Context, Type*)const, const Interface& interface, const QueryMemberTableEntry* queryMemberTableEntry)
250 {
251 const QueryMemberTableEntry *iterator;
252 Type vec[2];
253 deMemset(&vec[0], 0x00, sizeof(Type));
254 deMemset(&vec[1], 0xFF, sizeof(Type));
255
256 (interface.*Function)(context, &vec[0]);
257 (interface.*Function)(context, &vec[1]);
258
259 for (iterator = queryMemberTableEntry; iterator->size != 0; iterator++)
260 {
261 if (deMemCmp(((deUint8*)(&vec[0]))+iterator->offset, ((deUint8*)(&vec[1]))+iterator->offset, iterator->size) != 0)
262 return false;
263 }
264
265 return true;
266 }
267
268 template <typename Type>
269 //!< Return variable initialization validation
validateStructsWithGuard(const QueryMemberTableEntry * queryMemberTableEntry,Type * vec[2],const deUint8 guardValue,const deUint32 guardSize)270 bool validateStructsWithGuard (const QueryMemberTableEntry* queryMemberTableEntry, Type* vec[2], const deUint8 guardValue, const deUint32 guardSize)
271 {
272 const QueryMemberTableEntry *iterator;
273
274 for (iterator = queryMemberTableEntry; iterator->size != 0; iterator++)
275 {
276 if (deMemCmp(((deUint8*)(vec[0]))+iterator->offset, ((deUint8*)(vec[1]))+iterator->offset, iterator->size) != 0)
277 return false;
278 }
279
280 for (deUint32 vecNdx = 0; vecNdx < 2; ++vecNdx)
281 {
282 for (deUint32 ndx = 0; ndx < guardSize; ndx++)
283 {
284 if (((deUint8*)(vec[vecNdx]))[ndx + sizeof(Type)] != guardValue)
285 return false;
286 }
287 }
288
289 return true;
290 }
291
292 template<typename IterT>
293 //! Overwrite a range of objects with an 8-bit pattern.
fillBits(IterT beg,const IterT end,const deUint8 pattern=0xdeu)294 inline void fillBits (IterT beg, const IterT end, const deUint8 pattern = 0xdeu)
295 {
296 for (; beg < end; ++beg)
297 deMemset(&(*beg), static_cast<int>(pattern), sizeof(*beg));
298 }
299
300 template<typename IterT>
301 //! Verify that each byte of a range of objects is equal to an 8-bit pattern.
checkBits(IterT beg,const IterT end,const deUint8 pattern=0xdeu)302 bool checkBits (IterT beg, const IterT end, const deUint8 pattern = 0xdeu)
303 {
304 for (; beg < end; ++beg)
305 {
306 const deUint8* elementBytes = reinterpret_cast<const deUint8*>(&(*beg));
307 for (std::size_t i = 0u; i < sizeof(*beg); ++i)
308 {
309 if (elementBytes[i] != pattern)
310 return false;
311 }
312 }
313 return true;
314 }
315
316 } // ValidateQueryBits
317
318 // Template implementations
319
320 template<typename ExtensionIterator>
isExtensionSupported(ExtensionIterator begin,ExtensionIterator end,const RequiredExtension & required)321 bool isExtensionSupported (ExtensionIterator begin, ExtensionIterator end, const RequiredExtension& required)
322 {
323 for (ExtensionIterator cur = begin; cur != end; ++cur)
324 {
325 if (isCompatible(*cur, required))
326 return true;
327 }
328 return false;
329 }
330
331 template<typename LayerIterator>
isLayerSupported(LayerIterator begin,LayerIterator end,const RequiredLayer & required)332 bool isLayerSupported (LayerIterator begin, LayerIterator end, const RequiredLayer& required)
333 {
334 for (LayerIterator cur = begin; cur != end; ++cur)
335 {
336 if (isCompatible(*cur, required))
337 return true;
338 }
339 return false;
340 }
341
342 } // vk
343
344 #endif // _VKQUERYUTIL_HPP
345