OpenGL: Implement RenderPipelineLayout
This commit is contained in:
parent
0b05feb7e3
commit
32157503e8
|
|
@ -12,11 +12,7 @@
|
||||||
#include <Nazara/Renderer/RenderPipelineLayout.hpp>
|
#include <Nazara/Renderer/RenderPipelineLayout.hpp>
|
||||||
#include <Nazara/OpenGLRenderer/Config.hpp>
|
#include <Nazara/OpenGLRenderer/Config.hpp>
|
||||||
#include <Nazara/OpenGLRenderer/OpenGLShaderBinding.hpp>
|
#include <Nazara/OpenGLRenderer/OpenGLShaderBinding.hpp>
|
||||||
#include <Nazara/OpenGLRenderer/Wrapper/Device.hpp>
|
#include <Nazara/OpenGLRenderer/Wrapper/CoreFunctions.hpp>
|
||||||
#include <Nazara/OpenGLRenderer/Wrapper/DescriptorPool.hpp>
|
|
||||||
#include <Nazara/OpenGLRenderer/Wrapper/DescriptorSet.hpp>
|
|
||||||
#include <Nazara/OpenGLRenderer/Wrapper/DescriptorSetLayout.hpp>
|
|
||||||
#include <Nazara/OpenGLRenderer/Wrapper/PipelineLayout.hpp>
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <type_traits>
|
#include <type_traits>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
@ -28,39 +24,59 @@ namespace Nz
|
||||||
friend OpenGLShaderBinding;
|
friend OpenGLShaderBinding;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
OpenGLRenderPipelineLayout() = default;
|
OpenGLRenderPipelineLayout(RenderPipelineLayoutInfo layoutInfo);
|
||||||
|
OpenGLRenderPipelineLayout(const OpenGLRenderPipelineLayout&) = delete;
|
||||||
|
OpenGLRenderPipelineLayout(OpenGLRenderPipelineLayout&&) = delete;
|
||||||
~OpenGLRenderPipelineLayout();
|
~OpenGLRenderPipelineLayout();
|
||||||
|
|
||||||
ShaderBindingPtr AllocateShaderBinding() override;
|
ShaderBindingPtr AllocateShaderBinding() override;
|
||||||
|
|
||||||
bool Create(Vk::Device& device, RenderPipelineLayoutInfo layoutInfo);
|
inline const RenderPipelineLayoutInfo& GetLayoutInfo() const;
|
||||||
|
|
||||||
inline Vk::Device* GetDevice() const;
|
inline std::size_t GetTextureDescriptorCount() const;
|
||||||
|
inline std::size_t GetUniformBufferDescriptorCount() const;
|
||||||
|
|
||||||
inline const Vk::DescriptorSetLayout& GetDescriptorSetLayout() const;
|
OpenGLRenderPipelineLayout& operator=(const OpenGLRenderPipelineLayout&) = delete;
|
||||||
inline const Vk::PipelineLayout& GetPipelineLayout() const;
|
OpenGLRenderPipelineLayout& operator=(OpenGLRenderPipelineLayout&&) = delete;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
struct DescriptorPool;
|
struct DescriptorPool;
|
||||||
|
struct TextureDescriptor;
|
||||||
|
struct UniformBufferDescriptor;
|
||||||
|
|
||||||
DescriptorPool& AllocatePool();
|
DescriptorPool& AllocatePool();
|
||||||
ShaderBindingPtr AllocateFromPool(std::size_t poolIndex);
|
ShaderBindingPtr AllocateFromPool(std::size_t poolIndex);
|
||||||
|
TextureDescriptor& GetTextureDescriptor(std::size_t poolIndex, std::size_t bindingIndex, std::size_t textureIndex);
|
||||||
|
UniformBufferDescriptor& GetUniformBufferDescriptor(std::size_t poolIndex, std::size_t bindingIndex, std::size_t uniformBufferIndex);
|
||||||
void Release(ShaderBinding& binding);
|
void Release(ShaderBinding& binding);
|
||||||
inline void TryToShrink();
|
inline void TryToShrink();
|
||||||
|
|
||||||
|
struct TextureDescriptor
|
||||||
|
{
|
||||||
|
GLuint texture;
|
||||||
|
GLuint sampler;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct UniformBufferDescriptor
|
||||||
|
{
|
||||||
|
GLuint buffer;
|
||||||
|
GLintptr offset;
|
||||||
|
GLsizeiptr size;
|
||||||
|
};
|
||||||
|
|
||||||
struct DescriptorPool
|
struct DescriptorPool
|
||||||
{
|
{
|
||||||
using BindingStorage = std::aligned_storage_t<sizeof(OpenGLShaderBinding), alignof(OpenGLShaderBinding)>;
|
using BindingStorage = std::aligned_storage_t<sizeof(OpenGLShaderBinding), alignof(OpenGLShaderBinding)>;
|
||||||
|
|
||||||
Bitset<UInt64> freeBindings;
|
Bitset<UInt64> freeBindings;
|
||||||
Vk::DescriptorPool descriptorPool;
|
std::vector<TextureDescriptor> textureDescriptor;
|
||||||
|
std::vector<UniformBufferDescriptor> uniformBufferDescriptor;
|
||||||
std::unique_ptr<BindingStorage[]> storage;
|
std::unique_ptr<BindingStorage[]> storage;
|
||||||
};
|
};
|
||||||
|
|
||||||
MovablePtr<Vk::Device> m_device;
|
std::size_t m_textureDescriptorCount;
|
||||||
|
std::size_t m_uniformBufferDescriptorCount;
|
||||||
std::vector<DescriptorPool> m_descriptorPools;
|
std::vector<DescriptorPool> m_descriptorPools;
|
||||||
Vk::DescriptorSetLayout m_descriptorSetLayout;
|
|
||||||
Vk::PipelineLayout m_pipelineLayout;
|
|
||||||
RenderPipelineLayoutInfo m_layoutInfo;
|
RenderPipelineLayoutInfo m_layoutInfo;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -7,19 +7,19 @@
|
||||||
|
|
||||||
namespace Nz
|
namespace Nz
|
||||||
{
|
{
|
||||||
inline Vk::Device* OpenGLRenderPipelineLayout::GetDevice() const
|
inline const RenderPipelineLayoutInfo& OpenGLRenderPipelineLayout::GetLayoutInfo() const
|
||||||
{
|
{
|
||||||
return m_device.Get();
|
return m_layoutInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline const Vk::DescriptorSetLayout& OpenGLRenderPipelineLayout::GetDescriptorSetLayout() const
|
inline std::size_t OpenGLRenderPipelineLayout::GetTextureDescriptorCount() const
|
||||||
{
|
{
|
||||||
return m_descriptorSetLayout;
|
return m_textureDescriptorCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline const Vk::PipelineLayout& OpenGLRenderPipelineLayout::GetPipelineLayout() const
|
inline std::size_t OpenGLRenderPipelineLayout::GetUniformBufferDescriptorCount() const
|
||||||
{
|
{
|
||||||
return m_pipelineLayout;
|
return m_uniformBufferDescriptorCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void OpenGLRenderPipelineLayout::TryToShrink()
|
inline void OpenGLRenderPipelineLayout::TryToShrink()
|
||||||
|
|
|
||||||
|
|
@ -8,8 +8,8 @@
|
||||||
#define NAZARA_OPENGLRENDERER_OPENGLSHADERBINDING_HPP
|
#define NAZARA_OPENGLRENDERER_OPENGLSHADERBINDING_HPP
|
||||||
|
|
||||||
#include <Nazara/Prerequisites.hpp>
|
#include <Nazara/Prerequisites.hpp>
|
||||||
|
#include <Nazara/OpenGLRenderer/Config.hpp>
|
||||||
#include <Nazara/Renderer/ShaderBinding.hpp>
|
#include <Nazara/Renderer/ShaderBinding.hpp>
|
||||||
#include <Nazara/OpenGLRenderer/Wrapper/DescriptorSet.hpp>
|
|
||||||
|
|
||||||
namespace Nz
|
namespace Nz
|
||||||
{
|
{
|
||||||
|
|
@ -18,13 +18,12 @@ namespace Nz
|
||||||
class NAZARA_OPENGLRENDERER_API OpenGLShaderBinding : public ShaderBinding
|
class NAZARA_OPENGLRENDERER_API OpenGLShaderBinding : public ShaderBinding
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
inline OpenGLShaderBinding(OpenGLRenderPipelineLayout& owner, std::size_t poolIndex, std::size_t bindingIndex, Vk::DescriptorSet descriptorSet);
|
inline OpenGLShaderBinding(OpenGLRenderPipelineLayout& owner, std::size_t poolIndex, std::size_t bindingIndex);
|
||||||
OpenGLShaderBinding(const OpenGLShaderBinding&) = default;
|
OpenGLShaderBinding(const OpenGLShaderBinding&) = default;
|
||||||
OpenGLShaderBinding(OpenGLShaderBinding&&) noexcept = default;
|
OpenGLShaderBinding(OpenGLShaderBinding&&) noexcept = default;
|
||||||
~OpenGLShaderBinding() = default;
|
~OpenGLShaderBinding() = default;
|
||||||
|
|
||||||
inline std::size_t GetBindingIndex() const;
|
inline std::size_t GetBindingIndex() const;
|
||||||
inline const Vk::DescriptorSet& GetDescriptorSet() const;
|
|
||||||
inline std::size_t GetPoolIndex() const;
|
inline std::size_t GetPoolIndex() const;
|
||||||
inline const OpenGLRenderPipelineLayout& GetOwner() const;
|
inline const OpenGLRenderPipelineLayout& GetOwner() const;
|
||||||
|
|
||||||
|
|
@ -36,7 +35,6 @@ namespace Nz
|
||||||
private:
|
private:
|
||||||
void Release() override;
|
void Release() override;
|
||||||
|
|
||||||
Vk::AutoDescriptorSet m_descriptorSet;
|
|
||||||
OpenGLRenderPipelineLayout& m_owner;
|
OpenGLRenderPipelineLayout& m_owner;
|
||||||
std::size_t m_bindingIndex;
|
std::size_t m_bindingIndex;
|
||||||
std::size_t m_poolIndex;
|
std::size_t m_poolIndex;
|
||||||
|
|
|
||||||
|
|
@ -7,8 +7,7 @@
|
||||||
|
|
||||||
namespace Nz
|
namespace Nz
|
||||||
{
|
{
|
||||||
inline OpenGLShaderBinding::OpenGLShaderBinding(OpenGLRenderPipelineLayout& owner, std::size_t poolIndex, std::size_t bindingIndex, Vk::DescriptorSet descriptorSet) :
|
inline OpenGLShaderBinding::OpenGLShaderBinding(OpenGLRenderPipelineLayout& owner, std::size_t poolIndex, std::size_t bindingIndex) :
|
||||||
m_descriptorSet(std::move(descriptorSet)),
|
|
||||||
m_owner(owner),
|
m_owner(owner),
|
||||||
m_bindingIndex(bindingIndex),
|
m_bindingIndex(bindingIndex),
|
||||||
m_poolIndex(poolIndex)
|
m_poolIndex(poolIndex)
|
||||||
|
|
@ -25,11 +24,6 @@ namespace Nz
|
||||||
return m_poolIndex;
|
return m_poolIndex;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline const Vk::DescriptorSet& OpenGLShaderBinding::GetDescriptorSet() const
|
|
||||||
{
|
|
||||||
return m_descriptorSet;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline const OpenGLRenderPipelineLayout& OpenGLShaderBinding::GetOwner() const
|
inline const OpenGLRenderPipelineLayout& OpenGLShaderBinding::GetOwner() const
|
||||||
{
|
{
|
||||||
return m_owner;
|
return m_owner;
|
||||||
|
|
|
||||||
|
|
@ -26,6 +26,7 @@ namespace Nz
|
||||||
PixelFormat GetFormat() const override;
|
PixelFormat GetFormat() const override;
|
||||||
UInt8 GetLevelCount() const override;
|
UInt8 GetLevelCount() const override;
|
||||||
Vector3ui GetSize(UInt8 level = 0) const override;
|
Vector3ui GetSize(UInt8 level = 0) const override;
|
||||||
|
inline const GL::Texture& GetTexture() const;
|
||||||
ImageType GetType() const override;
|
ImageType GetType() const override;
|
||||||
|
|
||||||
bool Update(const void* ptr) override;
|
bool Update(const void* ptr) override;
|
||||||
|
|
|
||||||
|
|
@ -7,6 +7,10 @@
|
||||||
|
|
||||||
namespace Nz
|
namespace Nz
|
||||||
{
|
{
|
||||||
|
inline const GL::Texture& OpenGLTexture::GetTexture() const
|
||||||
|
{
|
||||||
|
return m_texture;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#include <Nazara/OpenGLRenderer/DebugOff.hpp>
|
#include <Nazara/OpenGLRenderer/DebugOff.hpp>
|
||||||
|
|
|
||||||
|
|
@ -5,6 +5,7 @@
|
||||||
#include <Nazara/OpenGLRenderer/OpenGLDevice.hpp>
|
#include <Nazara/OpenGLRenderer/OpenGLDevice.hpp>
|
||||||
#include <Nazara/Renderer/CommandPool.hpp>
|
#include <Nazara/Renderer/CommandPool.hpp>
|
||||||
#include <Nazara/OpenGLRenderer/OpenGLBuffer.hpp>
|
#include <Nazara/OpenGLRenderer/OpenGLBuffer.hpp>
|
||||||
|
#include <Nazara/OpenGLRenderer/OpenGLRenderPipelineLayout.hpp>
|
||||||
#include <Nazara/OpenGLRenderer/OpenGLShaderStage.hpp>
|
#include <Nazara/OpenGLRenderer/OpenGLShaderStage.hpp>
|
||||||
#include <Nazara/OpenGLRenderer/OpenGLTexture.hpp>
|
#include <Nazara/OpenGLRenderer/OpenGLTexture.hpp>
|
||||||
#include <Nazara/OpenGLRenderer/Wrapper/Loader.hpp>
|
#include <Nazara/OpenGLRenderer/Wrapper/Loader.hpp>
|
||||||
|
|
@ -58,7 +59,7 @@ namespace Nz
|
||||||
|
|
||||||
std::shared_ptr<RenderPipelineLayout> OpenGLDevice::InstantiateRenderPipelineLayout(RenderPipelineLayoutInfo pipelineLayoutInfo)
|
std::shared_ptr<RenderPipelineLayout> OpenGLDevice::InstantiateRenderPipelineLayout(RenderPipelineLayoutInfo pipelineLayoutInfo)
|
||||||
{
|
{
|
||||||
return {};
|
return std::make_shared<OpenGLRenderPipelineLayout>(std::move(pipelineLayoutInfo));
|
||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<ShaderStageImpl> OpenGLDevice::InstantiateShaderStage(ShaderStageType type, ShaderLanguage lang, const void* source, std::size_t sourceSize)
|
std::shared_ptr<ShaderStageImpl> OpenGLDevice::InstantiateShaderStage(ShaderStageType type, ShaderLanguage lang, const void* source, std::size_t sourceSize)
|
||||||
|
|
|
||||||
|
|
@ -2,8 +2,6 @@
|
||||||
// This file is part of the "Nazara Engine - OpenGL Renderer"
|
// This file is part of the "Nazara Engine - OpenGL Renderer"
|
||||||
// For conditions of distribution and use, see copyright notice in Config.hpp
|
// For conditions of distribution and use, see copyright notice in Config.hpp
|
||||||
|
|
||||||
#if 0
|
|
||||||
|
|
||||||
#include <Nazara/OpenGLRenderer/OpenGLRenderPipelineLayout.hpp>
|
#include <Nazara/OpenGLRenderer/OpenGLRenderPipelineLayout.hpp>
|
||||||
#include <Nazara/Core/ErrorFlags.hpp>
|
#include <Nazara/Core/ErrorFlags.hpp>
|
||||||
#include <Nazara/Core/MemoryHelper.hpp>
|
#include <Nazara/Core/MemoryHelper.hpp>
|
||||||
|
|
@ -15,6 +13,29 @@
|
||||||
|
|
||||||
namespace Nz
|
namespace Nz
|
||||||
{
|
{
|
||||||
|
OpenGLRenderPipelineLayout::OpenGLRenderPipelineLayout(RenderPipelineLayoutInfo layoutInfo) :
|
||||||
|
m_textureDescriptorCount(0),
|
||||||
|
m_uniformBufferDescriptorCount(0),
|
||||||
|
m_layoutInfo(std::move(layoutInfo))
|
||||||
|
{
|
||||||
|
for (const auto& bindingInfo : m_layoutInfo.bindings)
|
||||||
|
{
|
||||||
|
switch (bindingInfo.type)
|
||||||
|
{
|
||||||
|
case ShaderBindingType::Texture:
|
||||||
|
m_textureDescriptorCount++;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case ShaderBindingType::UniformBuffer:
|
||||||
|
m_uniformBufferDescriptorCount++;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw std::runtime_error(("unknown binding type 0x" + String::Number(UnderlyingCast(bindingInfo.type), 16)).ToStdString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
OpenGLRenderPipelineLayout::~OpenGLRenderPipelineLayout()
|
OpenGLRenderPipelineLayout::~OpenGLRenderPipelineLayout()
|
||||||
{
|
{
|
||||||
for (auto& pool : m_descriptorPools)
|
for (auto& pool : m_descriptorPools)
|
||||||
|
|
@ -46,50 +67,15 @@ namespace Nz
|
||||||
return bindingPtr;
|
return bindingPtr;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OpenGLRenderPipelineLayout::Create(Vk::Device& device, RenderPipelineLayoutInfo layoutInfo)
|
|
||||||
{
|
|
||||||
m_device = &device;
|
|
||||||
m_layoutInfo = std::move(layoutInfo);
|
|
||||||
|
|
||||||
StackVector<VkDescriptorSetLayoutBinding> layoutBindings = NazaraStackVector(VkDescriptorSetLayoutBinding, m_layoutInfo.bindings.size());
|
|
||||||
|
|
||||||
for (const auto& bindingInfo : m_layoutInfo.bindings)
|
|
||||||
{
|
|
||||||
VkDescriptorSetLayoutBinding& layoutBinding = layoutBindings.emplace_back();
|
|
||||||
layoutBinding.binding = bindingInfo.index;
|
|
||||||
layoutBinding.descriptorCount = 1U;
|
|
||||||
layoutBinding.descriptorType = ToOpenGL(bindingInfo.type);
|
|
||||||
layoutBinding.stageFlags = ToOpenGL(bindingInfo.shaderStageFlags);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!m_descriptorSetLayout.Create(*m_device, UInt32(layoutBindings.size()), layoutBindings.data()))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (!m_pipelineLayout.Create(*m_device, m_descriptorSetLayout))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto OpenGLRenderPipelineLayout::AllocatePool() -> DescriptorPool&
|
auto OpenGLRenderPipelineLayout::AllocatePool() -> DescriptorPool&
|
||||||
{
|
{
|
||||||
StackVector<VkDescriptorPoolSize> poolSizes = NazaraStackVector(VkDescriptorPoolSize, m_layoutInfo.bindings.size());
|
|
||||||
|
|
||||||
constexpr UInt32 MaxSet = 128;
|
constexpr UInt32 MaxSet = 128;
|
||||||
|
|
||||||
for (const auto& bindingInfo : m_layoutInfo.bindings)
|
|
||||||
{
|
|
||||||
VkDescriptorPoolSize& poolSize = poolSizes.emplace_back();
|
|
||||||
poolSize.descriptorCount = MaxSet;
|
|
||||||
poolSize.type = ToOpenGL(bindingInfo.type);
|
|
||||||
}
|
|
||||||
|
|
||||||
DescriptorPool pool;
|
DescriptorPool pool;
|
||||||
if (!pool.descriptorPool.Create(*m_device, MaxSet, UInt32(poolSizes.size()), poolSizes.data(), VK_DESCRIPTOR_POOL_CREATE_FREE_DESCRIPTOR_SET_BIT))
|
|
||||||
throw std::runtime_error("Failed to allocate new descriptor pool: " + TranslateOpenGLError(pool.descriptorPool.GetLastErrorCode()));
|
|
||||||
|
|
||||||
pool.freeBindings.Resize(MaxSet, true);
|
pool.freeBindings.Resize(MaxSet, true);
|
||||||
pool.storage = std::make_unique<DescriptorPool::BindingStorage[]>(MaxSet);
|
pool.storage = std::make_unique<DescriptorPool::BindingStorage[]>(MaxSet);
|
||||||
|
pool.textureDescriptor.resize(m_textureDescriptorCount * MaxSet);
|
||||||
|
pool.uniformBufferDescriptor.resize(m_uniformBufferDescriptorCount * MaxSet);
|
||||||
|
|
||||||
return m_descriptorPools.emplace_back(std::move(pool));
|
return m_descriptorPools.emplace_back(std::move(pool));
|
||||||
}
|
}
|
||||||
|
|
@ -102,17 +88,30 @@ namespace Nz
|
||||||
if (freeBindingId == pool.freeBindings.npos)
|
if (freeBindingId == pool.freeBindings.npos)
|
||||||
return {}; //< No free binding in this pool
|
return {}; //< No free binding in this pool
|
||||||
|
|
||||||
Vk::DescriptorSet descriptorSet = pool.descriptorPool.AllocateDescriptorSet(m_descriptorSetLayout);
|
|
||||||
if (!descriptorSet)
|
|
||||||
{
|
|
||||||
NazaraWarning("Failed to allocate descriptor set: " + TranslateOpenGLError(pool.descriptorPool.GetLastErrorCode()));
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
pool.freeBindings.Reset(freeBindingId);
|
pool.freeBindings.Reset(freeBindingId);
|
||||||
|
|
||||||
OpenGLShaderBinding* freeBindingMemory = reinterpret_cast<OpenGLShaderBinding*>(&pool.storage[freeBindingId]);
|
OpenGLShaderBinding* freeBindingMemory = reinterpret_cast<OpenGLShaderBinding*>(&pool.storage[freeBindingId]);
|
||||||
return ShaderBindingPtr(PlacementNew(freeBindingMemory, *this, poolIndex, freeBindingId, std::move(descriptorSet)));
|
return ShaderBindingPtr(PlacementNew(freeBindingMemory, *this, poolIndex, freeBindingId));
|
||||||
|
}
|
||||||
|
|
||||||
|
auto OpenGLRenderPipelineLayout::GetTextureDescriptor(std::size_t poolIndex, std::size_t bindingIndex, std::size_t textureIndex) -> TextureDescriptor&
|
||||||
|
{
|
||||||
|
assert(poolIndex < m_descriptorPools.size());
|
||||||
|
auto& pool = m_descriptorPools[poolIndex];
|
||||||
|
assert(!pool.freeBindings.Test(bindingIndex));
|
||||||
|
assert(textureIndex < m_textureDescriptorCount);
|
||||||
|
|
||||||
|
return pool.textureDescriptor[bindingIndex * m_textureDescriptorCount + textureIndex];
|
||||||
|
}
|
||||||
|
|
||||||
|
auto OpenGLRenderPipelineLayout::GetUniformBufferDescriptor(std::size_t poolIndex, std::size_t bindingIndex, std::size_t uniformBufferIndex) -> UniformBufferDescriptor&
|
||||||
|
{
|
||||||
|
assert(poolIndex < m_descriptorPools.size());
|
||||||
|
auto& pool = m_descriptorPools[poolIndex];
|
||||||
|
assert(!pool.freeBindings.Test(bindingIndex));
|
||||||
|
assert(uniformBufferIndex < m_uniformBufferDescriptorCount);
|
||||||
|
|
||||||
|
return pool.uniformBufferDescriptor[bindingIndex * m_uniformBufferDescriptorCount + uniformBufferIndex];
|
||||||
}
|
}
|
||||||
|
|
||||||
void OpenGLRenderPipelineLayout::Release(ShaderBinding& binding)
|
void OpenGLRenderPipelineLayout::Release(ShaderBinding& binding)
|
||||||
|
|
@ -136,5 +135,3 @@ namespace Nz
|
||||||
TryToShrink();
|
TryToShrink();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
|
||||||
|
|
@ -2,8 +2,6 @@
|
||||||
// This file is part of the "Nazara Engine - OpenGL Renderer"
|
// This file is part of the "Nazara Engine - OpenGL Renderer"
|
||||||
// For conditions of distribution and use, see copyright notice in Config.hpp
|
// For conditions of distribution and use, see copyright notice in Config.hpp
|
||||||
|
|
||||||
#if 0
|
|
||||||
|
|
||||||
#include <Nazara/OpenGLRenderer/OpenGLShaderBinding.hpp>
|
#include <Nazara/OpenGLRenderer/OpenGLShaderBinding.hpp>
|
||||||
#include <Nazara/Core/Algorithm.hpp>
|
#include <Nazara/Core/Algorithm.hpp>
|
||||||
#include <Nazara/Core/StackVector.hpp>
|
#include <Nazara/Core/StackVector.hpp>
|
||||||
|
|
@ -17,57 +15,58 @@ namespace Nz
|
||||||
{
|
{
|
||||||
void OpenGLShaderBinding::Update(std::initializer_list<Binding> bindings)
|
void OpenGLShaderBinding::Update(std::initializer_list<Binding> bindings)
|
||||||
{
|
{
|
||||||
StackVector<VkDescriptorBufferInfo> bufferBinding = NazaraStackVector(VkDescriptorBufferInfo, bindings.size());
|
const auto& layoutInfo = m_owner.GetLayoutInfo();
|
||||||
StackVector<VkDescriptorImageInfo> imageBinding = NazaraStackVector(VkDescriptorImageInfo, bindings.size());
|
|
||||||
StackVector<VkWriteDescriptorSet> writeOps = NazaraStackVector(VkWriteDescriptorSet, bindings.size());
|
|
||||||
|
|
||||||
for (const Binding& binding : bindings)
|
for (const Binding& binding : bindings)
|
||||||
{
|
{
|
||||||
VkWriteDescriptorSet& writeOp = writeOps.emplace_back();
|
assert(binding.bindingIndex < layoutInfo.bindings.size());
|
||||||
writeOp.sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
|
const auto& bindingDesc = layoutInfo.bindings[binding.bindingIndex];
|
||||||
writeOp.dstSet = m_descriptorSet;
|
|
||||||
writeOp.dstBinding = UInt32(binding.bindingIndex);
|
|
||||||
|
|
||||||
std::visit([&](auto&& arg)
|
std::size_t resourceIndex = 0;
|
||||||
|
for (std::size_t i = binding.bindingIndex; i > 0; --i)
|
||||||
{
|
{
|
||||||
using T = std::decay_t<decltype(arg)>;
|
// Use i-1 to prevent underflow in for loop
|
||||||
|
if (layoutInfo.bindings[i - 1].type == bindingDesc.type)
|
||||||
|
resourceIndex++;
|
||||||
|
}
|
||||||
|
|
||||||
if constexpr (std::is_same_v<T, TextureBinding>)
|
switch (bindingDesc.type)
|
||||||
|
{
|
||||||
|
case ShaderBindingType::Texture:
|
||||||
{
|
{
|
||||||
OpenGLTexture& vkTexture = *static_cast<OpenGLTexture*>(arg.texture);
|
if (!std::holds_alternative<TextureBinding>(binding.content))
|
||||||
OpenGLTextureSampler& vkSampler = *static_cast<OpenGLTextureSampler*>(arg.sampler);
|
throw std::runtime_error("binding #" + std::to_string(binding.bindingIndex) + " is a texture but another binding type has been supplied");
|
||||||
|
|
||||||
VkDescriptorImageInfo& imageInfo = imageBinding.emplace_back();
|
const TextureBinding& texBinding = std::get<TextureBinding>(binding.content);
|
||||||
imageInfo.imageLayout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL;
|
|
||||||
imageInfo.imageView = vkTexture.GetImageView();
|
|
||||||
imageInfo.sampler = vkSampler.GetSampler();
|
|
||||||
|
|
||||||
writeOp.descriptorCount = 1;
|
OpenGLTexture& glTexture = *static_cast<OpenGLTexture*>(texBinding.texture);
|
||||||
writeOp.descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER;
|
OpenGLTextureSampler& glSampler = *static_cast<OpenGLTextureSampler*>(texBinding.sampler);
|
||||||
|
|
||||||
writeOp.pImageInfo = &imageInfo;
|
auto& textureDescriptor = m_owner.GetTextureDescriptor(m_poolIndex, m_bindingIndex, resourceIndex);
|
||||||
|
textureDescriptor.sampler = glSampler.GetSampler().GetObjectId();
|
||||||
|
textureDescriptor.texture = glTexture.GetTexture().GetObjectId();
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
else if constexpr (std::is_same_v<T, UniformBufferBinding>)
|
|
||||||
|
case ShaderBindingType::UniformBuffer:
|
||||||
{
|
{
|
||||||
OpenGLBuffer& vkBuffer = *static_cast<OpenGLBuffer*>(arg.buffer);
|
if (!std::holds_alternative<UniformBufferBinding>(binding.content))
|
||||||
|
throw std::runtime_error("binding #" + std::to_string(binding.bindingIndex) + " is an uniform buffer but another binding type has been supplied");
|
||||||
|
|
||||||
VkDescriptorBufferInfo& bufferInfo = bufferBinding.emplace_back();
|
const UniformBufferBinding& uboBinding = std::get<UniformBufferBinding>(binding.content);
|
||||||
bufferInfo.buffer = vkBuffer.GetBuffer();
|
|
||||||
bufferInfo.offset = arg.offset;
|
|
||||||
bufferInfo.range = arg.range;
|
|
||||||
|
|
||||||
writeOp.descriptorCount = 1;
|
OpenGLBuffer& glBuffer = *static_cast<OpenGLBuffer*>(uboBinding.buffer);
|
||||||
writeOp.descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER;
|
if (glBuffer.GetType() != BufferType_Uniform)
|
||||||
|
throw std::runtime_error("expected uniform buffer");
|
||||||
|
|
||||||
writeOp.pBufferInfo = &bufferInfo;
|
auto& uboDescriptor = m_owner.GetUniformBufferDescriptor(m_poolIndex, m_bindingIndex, resourceIndex);
|
||||||
|
uboDescriptor.buffer = glBuffer.GetBuffer().GetObjectId();
|
||||||
|
uboDescriptor.offset = uboBinding.offset;
|
||||||
|
uboDescriptor.size = uboBinding.range;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
else
|
}
|
||||||
static_assert(AlwaysFalse<T>::value, "non-exhaustive visitor");
|
|
||||||
|
|
||||||
}, binding.content);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
m_owner.GetDevice()->vkUpdateDescriptorSets(*m_owner.GetDevice(), UInt32(writeOps.size()), writeOps.data(), 0U, nullptr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OpenGLShaderBinding::Release()
|
void OpenGLShaderBinding::Release()
|
||||||
|
|
@ -75,5 +74,3 @@ namespace Nz
|
||||||
m_owner.Release(*this);
|
m_owner.Release(*this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue