/* Copyright (c) 2017-2018 Hans-Kristian Arntzen * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #pragma once #include "hash.hpp" #include "object_pool.hpp" #include "temporary_hashmap.hpp" #include "vulkan.hpp" #include "sampler.hpp" #include #include #include "cookie.hpp" namespace Vulkan { class Device; struct DescriptorSetLayout { uint32_t sampled_image_mask = 0; uint32_t storage_image_mask = 0; uint32_t uniform_buffer_mask = 0; uint32_t storage_buffer_mask = 0; uint32_t sampled_buffer_mask = 0; uint32_t input_attachment_mask = 0; uint32_t sampler_mask = 0; uint32_t separate_image_mask = 0; uint32_t fp_mask = 0; uint32_t immutable_sampler_mask = 0; uint64_t immutable_samplers = 0; }; // Avoid -Wclass-memaccess warnings since we hash DescriptorSetLayout. static inline bool has_immutable_sampler(const DescriptorSetLayout &layout, unsigned binding) { return (layout.immutable_sampler_mask & (1u << binding)) != 0; } static inline StockSampler get_immutable_sampler(const DescriptorSetLayout &layout, unsigned binding) { VK_ASSERT(has_immutable_sampler(layout, binding)); return static_cast((layout.immutable_samplers >> (4 * binding)) & 0xf); } static inline void set_immutable_sampler(DescriptorSetLayout &layout, unsigned binding, StockSampler sampler) { layout.immutable_samplers |= uint64_t(sampler) << (4 * binding); layout.immutable_sampler_mask |= 1u << binding; } static const unsigned VULKAN_NUM_SETS_PER_POOL = 16; static const unsigned VULKAN_DESCRIPTOR_RING_SIZE = 8; class DescriptorSetAllocator : public HashedObject { public: DescriptorSetAllocator(Util::Hash hash, Device *device, const DescriptorSetLayout &layout, const uint32_t *stages_for_bindings); ~DescriptorSetAllocator(); void operator=(const DescriptorSetAllocator &) = delete; DescriptorSetAllocator(const DescriptorSetAllocator &) = delete; void begin_frame(); std::pair find(unsigned thread_index, Util::Hash hash); VkDescriptorSetLayout get_layout() const { return set_layout; } void clear(); private: struct DescriptorSetNode : Util::TemporaryHashmapEnabled, Util::IntrusiveListEnabled { DescriptorSetNode(VkDescriptorSet set) : set(set) { } VkDescriptorSet set; }; Device *device; VkDescriptorSetLayout set_layout = VK_NULL_HANDLE; struct PerThread { Util::TemporaryHashmap set_nodes; std::vector pools; bool should_begin = true; }; std::vector> per_thread; std::vector pool_size; }; }