Allocate command buffers from pools
This commit is contained in:
@@ -15,22 +15,34 @@
|
||||
|
||||
namespace Nz
|
||||
{
|
||||
class VulkanCommandPool;
|
||||
|
||||
class NAZARA_VULKANRENDERER_API VulkanCommandBuffer final : public CommandBuffer
|
||||
{
|
||||
public:
|
||||
inline VulkanCommandBuffer(Vk::AutoCommandBuffer commandBuffer);
|
||||
inline VulkanCommandBuffer(std::vector<Vk::AutoCommandBuffer> commandBuffers);
|
||||
inline VulkanCommandBuffer(VulkanCommandPool& owner, std::size_t poolIndex, std::size_t bindingIndex, Vk::AutoCommandBuffer commandBuffer);
|
||||
inline VulkanCommandBuffer(VulkanCommandPool& owner, std::size_t poolIndex, std::size_t bindingIndex, std::vector<Vk::AutoCommandBuffer> commandBuffers);
|
||||
VulkanCommandBuffer(const VulkanCommandBuffer&) = delete;
|
||||
VulkanCommandBuffer(VulkanCommandBuffer&&) noexcept = default;
|
||||
~VulkanCommandBuffer() = default;
|
||||
|
||||
inline std::size_t GetBindingIndex() const;
|
||||
inline Vk::CommandBuffer& GetCommandBuffer(std::size_t imageIndex = 0);
|
||||
inline std::size_t GetPoolIndex() const;
|
||||
inline const VulkanCommandPool& GetOwner() const;
|
||||
|
||||
VulkanCommandBuffer& operator=(const VulkanCommandBuffer&) = delete;
|
||||
VulkanCommandBuffer& operator=(VulkanCommandBuffer&&) = delete;
|
||||
|
||||
private:
|
||||
inline VulkanCommandBuffer(VulkanCommandPool& owner, std::size_t poolIndex, std::size_t bindingIndex);
|
||||
|
||||
void Release() override;
|
||||
|
||||
std::size_t m_bindingIndex;
|
||||
std::size_t m_poolIndex;
|
||||
std::vector<Vk::AutoCommandBuffer> m_commandBuffers;
|
||||
VulkanCommandPool& m_owner;
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
@@ -7,20 +7,44 @@
|
||||
|
||||
namespace Nz
|
||||
{
|
||||
inline VulkanCommandBuffer::VulkanCommandBuffer(Vk::AutoCommandBuffer commandBuffer)
|
||||
inline VulkanCommandBuffer::VulkanCommandBuffer(VulkanCommandPool& owner, std::size_t poolIndex, std::size_t bindingIndex, Vk::AutoCommandBuffer commandBuffer) :
|
||||
VulkanCommandBuffer(owner, poolIndex, bindingIndex)
|
||||
{
|
||||
m_commandBuffers.push_back(std::move(commandBuffer));
|
||||
}
|
||||
|
||||
inline VulkanCommandBuffer::VulkanCommandBuffer(std::vector<Vk::AutoCommandBuffer> commandBuffers) :
|
||||
m_commandBuffers(std::move(commandBuffers))
|
||||
inline VulkanCommandBuffer::VulkanCommandBuffer(VulkanCommandPool& owner, std::size_t poolIndex, std::size_t bindingIndex, std::vector<Vk::AutoCommandBuffer> commandBuffers) :
|
||||
VulkanCommandBuffer(owner, poolIndex, bindingIndex)
|
||||
{
|
||||
m_commandBuffers = std::move(commandBuffers);
|
||||
}
|
||||
|
||||
inline VulkanCommandBuffer::VulkanCommandBuffer(VulkanCommandPool& owner, std::size_t poolIndex, std::size_t bindingIndex) :
|
||||
m_bindingIndex(bindingIndex),
|
||||
m_poolIndex(poolIndex),
|
||||
m_owner(owner)
|
||||
{
|
||||
}
|
||||
|
||||
inline std::size_t VulkanCommandBuffer::GetBindingIndex() const
|
||||
{
|
||||
return m_bindingIndex;
|
||||
}
|
||||
|
||||
inline Vk::CommandBuffer& VulkanCommandBuffer::GetCommandBuffer(std::size_t imageIndex)
|
||||
{
|
||||
return m_commandBuffers[imageIndex].Get();
|
||||
}
|
||||
|
||||
inline std::size_t VulkanCommandBuffer::GetPoolIndex() const
|
||||
{
|
||||
return m_poolIndex;
|
||||
}
|
||||
|
||||
inline const VulkanCommandPool& VulkanCommandBuffer::GetOwner() const
|
||||
{
|
||||
return m_owner;
|
||||
}
|
||||
}
|
||||
|
||||
#include <Nazara/VulkanRenderer/DebugOff.hpp>
|
||||
|
||||
@@ -8,14 +8,19 @@
|
||||
#define NAZARA_VULKANRENDERER_VULKANCOMMANDPOOL_HPP
|
||||
|
||||
#include <Nazara/Prerequisites.hpp>
|
||||
#include <Nazara/Core/Bitset.hpp>
|
||||
#include <Nazara/Renderer/CommandPool.hpp>
|
||||
#include <Nazara/VulkanRenderer/Config.hpp>
|
||||
#include <Nazara/VulkanRenderer/VulkanCommandBuffer.hpp>
|
||||
#include <Nazara/VulkanRenderer/Wrapper/CommandPool.hpp>
|
||||
#include <vector>
|
||||
|
||||
namespace Nz
|
||||
{
|
||||
class NAZARA_VULKANRENDERER_API VulkanCommandPool final : public CommandPool
|
||||
{
|
||||
friend VulkanCommandBuffer;
|
||||
|
||||
public:
|
||||
inline VulkanCommandPool(Vk::Device& device, QueueType queueType);
|
||||
inline VulkanCommandPool(Vk::Device& device, UInt32 queueFamilyIndex);
|
||||
@@ -23,12 +28,29 @@ namespace Nz
|
||||
VulkanCommandPool(VulkanCommandPool&&) noexcept = default;
|
||||
~VulkanCommandPool() = default;
|
||||
|
||||
std::unique_ptr<CommandBuffer> BuildCommandBuffer(const std::function<void(CommandBufferBuilder& builder)>& callback) override;
|
||||
CommandBufferPtr BuildCommandBuffer(const std::function<void(CommandBufferBuilder& builder)>& callback) override;
|
||||
|
||||
VulkanCommandPool& operator=(const VulkanCommandPool&) = delete;
|
||||
VulkanCommandPool& operator=(VulkanCommandPool&&) = delete;
|
||||
|
||||
private:
|
||||
struct CommandPool;
|
||||
|
||||
CommandPool& AllocatePool();
|
||||
template<typename... Args> CommandBufferPtr AllocateFromPool(std::size_t poolIndex, Args&&... args);
|
||||
void Release(CommandBuffer& commandBuffer);
|
||||
inline void TryToShrink();
|
||||
|
||||
struct CommandPool
|
||||
{
|
||||
using BindingStorage = std::aligned_storage_t<sizeof(VulkanCommandBuffer), alignof(VulkanCommandBuffer)>;
|
||||
|
||||
Bitset<UInt64> freeCommands;
|
||||
std::unique_ptr<BindingStorage[]> storage;
|
||||
};
|
||||
|
||||
MovablePtr<Vk::Device> m_device;
|
||||
std::vector<CommandPool> m_commandPools;
|
||||
Vk::CommandPool m_commandPool;
|
||||
};
|
||||
}
|
||||
|
||||
@@ -23,6 +23,38 @@ namespace Nz
|
||||
if (!m_commandPool.Create(device, queueFamilyIndex))
|
||||
throw std::runtime_error("Failed to create command pool: " + TranslateVulkanError(m_commandPool.GetLastErrorCode()));
|
||||
}
|
||||
|
||||
template<typename... Args>
|
||||
CommandBufferPtr VulkanCommandPool::AllocateFromPool(std::size_t poolIndex, Args&&... args)
|
||||
{
|
||||
auto& pool = m_commandPools[poolIndex];
|
||||
|
||||
std::size_t freeBindingId = pool.freeCommands.FindFirst();
|
||||
if (freeBindingId == pool.freeCommands.npos)
|
||||
return {}; //< No free binding in this pool
|
||||
|
||||
pool.freeCommands.Reset(freeBindingId);
|
||||
|
||||
VulkanCommandBuffer* freeBindingMemory = reinterpret_cast<VulkanCommandBuffer*>(&pool.storage[freeBindingId]);
|
||||
return CommandBufferPtr(PlacementNew(freeBindingMemory, *this, poolIndex, freeBindingId, std::forward<Args>(args)...));
|
||||
}
|
||||
|
||||
inline void VulkanCommandPool::TryToShrink()
|
||||
{
|
||||
std::size_t poolCount = m_commandPools.size();
|
||||
if (poolCount >= 2 && m_commandPools.back().freeCommands.TestAll())
|
||||
{
|
||||
for (std::size_t i = poolCount - 1; i > 0; --i)
|
||||
{
|
||||
if (!m_commandPools[i].freeCommands.TestAll())
|
||||
break;
|
||||
|
||||
poolCount--;
|
||||
}
|
||||
|
||||
m_commandPools.resize(poolCount);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#include <Nazara/VulkanRenderer/DebugOff.hpp>
|
||||
|
||||
Reference in New Issue
Block a user