1 /*-------------------------------------------------------------------------
5 * Copyright (c) 2015 Google Inc.
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
11 * http://www.apache.org/licenses/LICENSE-2.0
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
21 * \brief Vulkan object builder utilities.
22 *//*--------------------------------------------------------------------*/
24 #include "vkBuilderUtil.hpp"
26 #include "vkRefUtil.hpp"
31 // DescriptorSetLayoutBuilder
33 DescriptorSetLayoutBuilder::DescriptorSetLayoutBuilder (void)
37 DescriptorSetLayoutBuilder& DescriptorSetLayoutBuilder::addBinding (VkDescriptorType descriptorType,
38 deUint32 descriptorCount,
39 VkShaderStageFlags stageFlags,
40 const VkSampler* pImmutableSamplers)
42 if (pImmutableSamplers)
44 const ImmutableSamplerInfo immutableSamplerInfo =
46 (deUint32)m_bindings.size(),
47 (deUint32)m_immutableSamplers.size()
50 m_immutableSamplerInfos.push_back(immutableSamplerInfo);
52 for (size_t descriptorNdx = 0; descriptorNdx < descriptorCount; descriptorNdx++)
53 m_immutableSamplers.push_back(pImmutableSamplers[descriptorNdx]);
56 // pImmutableSamplers will be updated at build time
57 const VkDescriptorSetLayoutBinding binding =
59 (deUint32)m_bindings.size(), // binding
60 descriptorType, // descriptorType
61 descriptorCount, // descriptorCount
62 stageFlags, // stageFlags
63 DE_NULL, // pImmutableSamplers
65 m_bindings.push_back(binding);
69 DescriptorSetLayoutBuilder& DescriptorSetLayoutBuilder::addIndexedBinding (VkDescriptorType descriptorType,
70 deUint32 descriptorCount,
71 VkShaderStageFlags stageFlags,
73 const VkSampler* pImmutableSamplers)
75 if (pImmutableSamplers)
77 const ImmutableSamplerInfo immutableSamplerInfo =
80 (deUint32)m_immutableSamplers.size()
83 m_immutableSamplerInfos.push_back(immutableSamplerInfo);
85 for (size_t descriptorNdx = 0; descriptorNdx < descriptorCount; descriptorNdx++)
86 m_immutableSamplers.push_back(pImmutableSamplers[descriptorNdx]);
89 // pImmutableSamplers will be updated at build time
90 const VkDescriptorSetLayoutBinding binding =
92 dstBinding, // binding
93 descriptorType, // descriptorType
94 descriptorCount, // descriptorCount
95 stageFlags, // stageFlags
96 DE_NULL, // pImmutableSamplers
98 m_bindings.push_back(binding);
102 Move<VkDescriptorSetLayout> DescriptorSetLayoutBuilder::build (const DeviceInterface& vk, VkDevice device, VkDescriptorSetLayoutCreateFlags extraFlags) const
104 // Create new layout bindings with pImmutableSamplers updated
105 std::vector<VkDescriptorSetLayoutBinding> bindings = m_bindings;
107 for (size_t samplerInfoNdx = 0; samplerInfoNdx < m_immutableSamplerInfos.size(); samplerInfoNdx++)
109 const ImmutableSamplerInfo& samplerInfo = m_immutableSamplerInfos[samplerInfoNdx];
110 deUint32 bindingNdx = 0;
112 while (bindings[bindingNdx].binding != samplerInfo.bindingIndex)
116 if (bindingNdx >= (deUint32)bindings.size())
117 DE_FATAL("Immutable sampler not found");
120 bindings[bindingNdx].pImmutableSamplers = &m_immutableSamplers[samplerInfo.samplerBaseIndex];
123 const VkDescriptorSetLayoutCreateInfo createInfo =
125 VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
127 (VkDescriptorSetLayoutCreateFlags)extraFlags, // flags
128 (deUint32)bindings.size(), // bindingCount
129 (bindings.empty()) ? (DE_NULL) : (&bindings.front()), // pBinding
132 return createDescriptorSetLayout(vk, device, &createInfo);
135 // DescriptorPoolBuilder
137 DescriptorPoolBuilder::DescriptorPoolBuilder (void)
141 DescriptorPoolBuilder& DescriptorPoolBuilder::addType (VkDescriptorType type, deUint32 numDescriptors)
143 if (numDescriptors == 0u)
150 for (size_t ndx = 0; ndx < m_counts.size(); ++ndx)
152 if (m_counts[ndx].type == type)
154 // augment existing requirement
155 m_counts[ndx].descriptorCount += numDescriptors;
162 const VkDescriptorPoolSize typeCount =
165 numDescriptors, // numDescriptors
168 m_counts.push_back(typeCount);
174 Move<VkDescriptorPool> DescriptorPoolBuilder::build (const DeviceInterface& vk, VkDevice device, VkDescriptorPoolCreateFlags flags, deUint32 maxSets, const void *pNext) const
176 const VkDescriptorPoolSize* const typeCountPtr = (m_counts.empty()) ? (DE_NULL) : (&m_counts[0]);
177 const VkDescriptorPoolCreateInfo createInfo =
179 VK_STRUCTURE_TYPE_DESCRIPTOR_POOL_CREATE_INFO,
183 (deUint32)m_counts.size(), // poolSizeCount
184 typeCountPtr, // pPoolSizes
187 return createDescriptorPool(vk, device, &createInfo);
190 // DescriptorSetUpdateBuilder
192 DescriptorSetUpdateBuilder::DescriptorSetUpdateBuilder (void)
196 DescriptorSetUpdateBuilder& DescriptorSetUpdateBuilder::write (VkDescriptorSet destSet,
197 deUint32 destBinding,
198 deUint32 destArrayElement,
200 VkDescriptorType descriptorType,
201 const VkDescriptorImageInfo* pImageInfo,
202 const VkDescriptorBufferInfo* pBufferInfo,
203 const VkBufferView* pTexelBufferView,
206 // pImageInfo, pBufferInfo and pTexelBufferView will be updated when calling update()
207 const VkWriteDescriptorSet writeParams =
209 VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
211 destSet, //!< destSet
212 destBinding, //!< destBinding
213 destArrayElement, //!< destArrayElement
215 descriptorType, //!< descriptorType
221 m_writes.push_back(writeParams);
223 // Store a copy of pImageInfo, pBufferInfo and pTexelBufferView
224 WriteDescriptorInfo writeInfo;
227 writeInfo.imageInfos.insert(writeInfo.imageInfos.end(), pImageInfo, pImageInfo + count);
230 writeInfo.bufferInfos.insert(writeInfo.bufferInfos.end(), pBufferInfo, pBufferInfo + count);
232 if (pTexelBufferView)
233 writeInfo.texelBufferViews.insert(writeInfo.texelBufferViews.end(), pTexelBufferView, pTexelBufferView + count);
235 m_writeDescriptorInfos.push_back(writeInfo);
240 DescriptorSetUpdateBuilder& DescriptorSetUpdateBuilder::copy (VkDescriptorSet srcSet,
242 deUint32 srcArrayElement,
243 VkDescriptorSet destSet,
244 deUint32 destBinding,
245 deUint32 destArrayElement,
248 const VkCopyDescriptorSet copyParams =
250 VK_STRUCTURE_TYPE_COPY_DESCRIPTOR_SET,
253 srcBinding, //!< srcBinding
254 srcArrayElement, //!< srcArrayElement
255 destSet, //!< destSet
256 destBinding, //!< destBinding
257 destArrayElement, //!< destArrayElement
260 m_copies.push_back(copyParams);
264 void DescriptorSetUpdateBuilder::update (const DeviceInterface& vk, VkDevice device) const
266 // Update VkWriteDescriptorSet structures with stored info
267 std::vector<VkWriteDescriptorSet> writes = m_writes;
269 for (size_t writeNdx = 0; writeNdx < m_writes.size(); writeNdx++)
271 const WriteDescriptorInfo& writeInfo = m_writeDescriptorInfos[writeNdx];
273 if (!writeInfo.imageInfos.empty())
274 writes[writeNdx].pImageInfo = &writeInfo.imageInfos[0];
276 if (!writeInfo.bufferInfos.empty())
277 writes[writeNdx].pBufferInfo = &writeInfo.bufferInfos[0];
279 if (!writeInfo.texelBufferViews.empty())
280 writes[writeNdx].pTexelBufferView = &writeInfo.texelBufferViews[0];
283 const VkWriteDescriptorSet* const writePtr = (m_writes.empty()) ? (DE_NULL) : (&writes[0]);
284 const VkCopyDescriptorSet* const copyPtr = (m_copies.empty()) ? (DE_NULL) : (&m_copies[0]);
286 vk.updateDescriptorSets(device, (deUint32)writes.size(), writePtr, (deUint32)m_copies.size(), copyPtr);
289 #ifndef CTS_USES_VULKANSC
291 void DescriptorSetUpdateBuilder::updateWithPush (const DeviceInterface& vk, VkCommandBuffer cmd, VkPipelineBindPoint bindPoint, VkPipelineLayout pipelineLayout, deUint32 setIdx, deUint32 descriptorIdx, deUint32 numDescriptors) const
293 // Write all descriptors or just a subset?
294 deUint32 count = (numDescriptors) ? numDescriptors : (deUint32)m_writes.size();
296 // Update VkWriteDescriptorSet structures with stored info
297 std::vector<VkWriteDescriptorSet> writes = m_writes;
299 for (size_t writeNdx = 0; writeNdx < m_writes.size(); writeNdx++)
301 const WriteDescriptorInfo& writeInfo = m_writeDescriptorInfos[writeNdx];
303 if (!writeInfo.imageInfos.empty())
304 writes[writeNdx].pImageInfo = &writeInfo.imageInfos[0];
306 if (!writeInfo.bufferInfos.empty())
307 writes[writeNdx].pBufferInfo = &writeInfo.bufferInfos[0];
309 if (!writeInfo.texelBufferViews.empty())
310 writes[writeNdx].pTexelBufferView = &writeInfo.texelBufferViews[0];
313 const VkWriteDescriptorSet* const writePtr = (m_writes.empty()) ? (DE_NULL) : (&writes[descriptorIdx]);
315 vk.cmdPushDescriptorSetKHR(cmd, bindPoint, pipelineLayout, setIdx, count, writePtr);
318 #endif // CTS_USES_VULKANSC
320 void DescriptorSetUpdateBuilder::clear(void)
322 m_writeDescriptorInfos.clear();