// Copyright (C) 2022 Jérôme "Lynix" Leclercq (lynix680@gmail.com) // This file is part of the "Nazara Engine - Vulkan renderer" // For conditions of distribution and use, see copyright notice in Config.hpp #include #include #include #include #include #include #include #include namespace Nz { void VulkanShaderBinding::Update(const Binding* bindings, std::size_t bindingCount) { std::size_t bufferBindingCount = 0; std::size_t imageBindingCount = 0; for (std::size_t i = 0; i < bindingCount; ++i) { const Binding& binding = bindings[i]; std::visit([&](auto&& arg) { using T = std::decay_t; if constexpr (std::is_same_v || std::is_same_v) bufferBindingCount++; else if constexpr (std::is_same_v || std::is_same_v) imageBindingCount++; else if constexpr (std::is_same_v) imageBindingCount += arg.arraySize; else static_assert(AlwaysFalse(), "non-exhaustive visitor"); }, binding.content); } NazaraAssert(bufferBindingCount < 128, "too many concurrent buffer update"); NazaraAssert(imageBindingCount < 128, "too many concurrent image binding update"); NazaraAssert(bindingCount < 128, "too many binding update"); StackVector bufferBinding = NazaraStackVector(VkDescriptorBufferInfo, bufferBindingCount); StackVector imageBinding = NazaraStackVector(VkDescriptorImageInfo, imageBindingCount); StackVector writeOps = NazaraStackVector(VkWriteDescriptorSet, bindingCount); for (std::size_t i = 0; i < bindingCount; ++i) { const Binding& binding = bindings[i]; VkWriteDescriptorSet& writeOp = writeOps.emplace_back(); writeOp.sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET; writeOp.dstSet = m_descriptorSet; writeOp.dstBinding = UInt32(binding.bindingIndex); std::visit([&](auto&& arg) { using T = std::decay_t; if constexpr (std::is_same_v) { const VulkanTexture* vkTexture = SafeCast(arg.texture); const VulkanTextureSampler* vkSampler = SafeCast(arg.sampler); VkDescriptorImageInfo& imageInfo = imageBinding.emplace_back(); imageInfo.imageLayout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL; imageInfo.imageView = (vkTexture) ? vkTexture->GetImageView() : VK_NULL_HANDLE; imageInfo.sampler = (vkSampler) ? vkSampler->GetSampler() : VK_NULL_HANDLE; writeOp.descriptorCount = 1; writeOp.descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER; writeOp.pImageInfo = &imageInfo; } else if constexpr (std::is_same_v) { for (UInt32 i = 0; i < arg.arraySize; ++i) { const VulkanTexture* vkTexture = SafeCast(arg.textureBindings[i].texture); const VulkanTextureSampler* vkSampler = SafeCast(arg.textureBindings[i].sampler); VkDescriptorImageInfo& imageInfo = imageBinding.emplace_back(); imageInfo.imageLayout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL; imageInfo.imageView = (vkTexture) ? vkTexture->GetImageView() : VK_NULL_HANDLE; imageInfo.sampler = (vkSampler) ? vkSampler->GetSampler() : VK_NULL_HANDLE; } writeOp.descriptorCount = arg.arraySize; writeOp.descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER; writeOp.pImageInfo = &imageBinding[imageBinding.size() - arg.arraySize]; } else if constexpr (std::is_same_v) { VulkanBuffer* vkBuffer = SafeCast(arg.buffer); VkDescriptorBufferInfo& bufferInfo = bufferBinding.emplace_back(); bufferInfo.buffer = (vkBuffer) ? vkBuffer->GetBuffer() : VK_NULL_HANDLE; bufferInfo.offset = arg.offset; bufferInfo.range = arg.range; writeOp.descriptorCount = 1; writeOp.descriptorType = VK_DESCRIPTOR_TYPE_STORAGE_BUFFER; writeOp.pBufferInfo = &bufferInfo; } else if constexpr (std::is_same_v) { const VulkanTexture* vkTexture = SafeCast(arg.texture); VkDescriptorImageInfo& imageInfo = imageBinding.emplace_back(); imageInfo.imageLayout = VK_IMAGE_LAYOUT_GENERAL; imageInfo.imageView = (vkTexture) ? vkTexture->GetImageView() : VK_NULL_HANDLE; imageInfo.sampler = VK_NULL_HANDLE; writeOp.descriptorCount = 1; writeOp.descriptorType = VK_DESCRIPTOR_TYPE_STORAGE_IMAGE; writeOp.pImageInfo = &imageInfo; } else if constexpr (std::is_same_v) { VulkanBuffer* vkBuffer = static_cast(arg.buffer); VkDescriptorBufferInfo& bufferInfo = bufferBinding.emplace_back(); bufferInfo.buffer = (vkBuffer) ? vkBuffer->GetBuffer() : VK_NULL_HANDLE; bufferInfo.offset = arg.offset; bufferInfo.range = arg.range; writeOp.descriptorCount = 1; writeOp.descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER; writeOp.pBufferInfo = &bufferInfo; } else static_assert(AlwaysFalse(), "non-exhaustive visitor"); }, binding.content); } m_owner.GetDevice()->vkUpdateDescriptorSets(*m_owner.GetDevice(), UInt32(writeOps.size()), writeOps.data(), 0U, nullptr); } void VulkanShaderBinding::UpdateDebugName(std::string_view name) { return m_owner.m_device->SetDebugName(VK_OBJECT_TYPE_DESCRIPTOR_SET, static_cast(reinterpret_cast(static_cast(m_descriptorSet))), name); } void VulkanShaderBinding::Release() { m_owner.Release(*this); } } #if defined(NAZARA_PLATFORM_WINDOWS) #include #endif