Use vma in GEVulkanDynamicBuffer

This commit is contained in:
Benau 2022-07-13 11:23:08 +08:00
parent b0f91ff33d
commit cf8bafaab3
2 changed files with 59 additions and 97 deletions

View File

@ -9,7 +9,7 @@
namespace GE namespace GE
{ {
VkMemoryPropertyFlags GEVulkanDynamicBuffer::m_host_flag = (VkMemoryPropertyFlags)-1; int GEVulkanDynamicBuffer::m_supports_host_transfer = -1;
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
GEVulkanDynamicBuffer::GEVulkanDynamicBuffer(GEVulkanDynamicBufferType t, GEVulkanDynamicBuffer::GEVulkanDynamicBuffer(GEVulkanDynamicBufferType t,
VkBufferUsageFlags usage, VkBufferUsageFlags usage,
@ -20,12 +20,12 @@ GEVulkanDynamicBuffer::GEVulkanDynamicBuffer(GEVulkanDynamicBufferType t,
m_size = m_real_size = initial_size; m_size = m_real_size = initial_size;
m_buffer = new VkBuffer[GEVulkanDriver::getMaxFrameInFlight()]; m_buffer = new VkBuffer[GEVulkanDriver::getMaxFrameInFlight()];
m_memory = new VkDeviceMemory[GEVulkanDriver::getMaxFrameInFlight()]; m_memory = new VmaAllocation[GEVulkanDriver::getMaxFrameInFlight()];
m_mapped_addr = new void*[GEVulkanDriver::getMaxFrameInFlight()]; m_mapped_addr = new void*[GEVulkanDriver::getMaxFrameInFlight()];
if (t == GVDBT_GPU_RAM) if (t == GVDBT_GPU_RAM)
{ {
m_staging_buffer = new VkBuffer[GEVulkanDriver::getMaxFrameInFlight()]; m_staging_buffer = new VkBuffer[GEVulkanDriver::getMaxFrameInFlight()];
m_staging_memory = new VkDeviceMemory[GEVulkanDriver::getMaxFrameInFlight()]; m_staging_memory = new VmaAllocation[GEVulkanDriver::getMaxFrameInFlight()];
} }
else else
{ {
@ -51,6 +51,7 @@ GEVulkanDynamicBuffer::~GEVulkanDynamicBuffer()
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
void GEVulkanDynamicBuffer::initPerFrame(unsigned frame) void GEVulkanDynamicBuffer::initPerFrame(unsigned frame)
{ {
GEVulkanDriver* vk = getVKDriver();
m_buffer[frame] = VK_NULL_HANDLE; m_buffer[frame] = VK_NULL_HANDLE;
m_memory[frame] = VK_NULL_HANDLE; m_memory[frame] = VK_NULL_HANDLE;
m_mapped_addr[frame] = NULL; m_mapped_addr[frame] = NULL;
@ -60,128 +61,85 @@ void GEVulkanDynamicBuffer::initPerFrame(unsigned frame)
m_staging_memory[frame] = VK_NULL_HANDLE; m_staging_memory[frame] = VK_NULL_HANDLE;
} }
start:
VkMemoryPropertyFlags prop = {};
VkBuffer host_buffer = VK_NULL_HANDLE; VkBuffer host_buffer = VK_NULL_HANDLE;
VkDeviceMemory host_memory = VK_NULL_HANDLE; VmaAllocation host_memory = VK_NULL_HANDLE;
VkMemoryPropertyFlags host_flag = 0; VmaAllocationCreateInfo host_info = {};
if (m_host_flag == (VkFlags)-1) host_info.usage = VMA_MEMORY_USAGE_AUTO;
host_info.flags = VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT |
VMA_ALLOCATION_CREATE_MAPPED_BIT;
host_info.preferredFlags = VK_MEMORY_PROPERTY_HOST_COHERENT_BIT;
if ((m_type == GVDBT_SYSTEM_RAM && m_supports_host_transfer == 1) ||
(m_type == GVDBT_SYSTEM_RAM && m_supports_host_transfer == -1))
{ {
// https://zeux.io/2020/02/27/writing-an-efficient-vulkan-renderer/ host_info.flags |=
m_host_flag = VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT | VMA_ALLOCATION_CREATE_HOST_ACCESS_ALLOW_TRANSFER_INSTEAD_BIT;
VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT |
VK_MEMORY_PROPERTY_HOST_COHERENT_BIT;
bool better_flag_exist = false;
if (getVKDriver()->createBuffer(m_size,
m_usage | VK_BUFFER_USAGE_TRANSFER_SRC_BIT, m_host_flag,
host_buffer, host_memory))
{
better_flag_exist = true;
if (m_type == GVDBT_SYSTEM_RAM)
goto succeed;
} }
if (host_buffer != VK_NULL_HANDLE) if (!vk->createBuffer(m_size, m_usage | VK_BUFFER_USAGE_TRANSFER_SRC_BIT,
vkDestroyBuffer(getVKDriver()->getDevice(), host_buffer, NULL); host_info, host_buffer, host_memory))
host_buffer = VK_NULL_HANDLE;
if (host_memory != VK_NULL_HANDLE)
vkFreeMemory(getVKDriver()->getDevice(), host_memory, NULL);
host_memory = VK_NULL_HANDLE;
if (!better_flag_exist)
{ {
m_host_flag = VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | vmaDestroyBuffer(vk->getVmaAllocator(), host_buffer, host_memory);
VK_MEMORY_PROPERTY_HOST_COHERENT_BIT;
}
}
host_flag = m_host_flag;
// From the above website:
// This flag should be used to store staging buffers that are used to
// populate static resources allocated with
// VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT with data.
if (m_type == GVDBT_GPU_RAM)
{
host_flag = VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT |
VK_MEMORY_PROPERTY_HOST_COHERENT_BIT;
}
if (!getVKDriver()->createBuffer(m_size,
m_usage | VK_BUFFER_USAGE_TRANSFER_SRC_BIT, host_flag,
host_buffer, host_memory))
{
if (host_buffer != VK_NULL_HANDLE)
vkDestroyBuffer(getVKDriver()->getDevice(), host_buffer, NULL);
if (host_memory != VK_NULL_HANDLE)
vkFreeMemory(getVKDriver()->getDevice(), host_memory, NULL);
return; return;
} }
succeed: if (m_type == GVDBT_SYSTEM_RAM && m_supports_host_transfer == -1)
{
vmaGetAllocationMemoryProperties(vk->getVmaAllocator(), host_memory,
&prop);
if ((prop & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) == 0)
{
m_supports_host_transfer = 0;
vmaDestroyBuffer(vk->getVmaAllocator(), host_buffer, host_memory);
goto start;
}
else
m_supports_host_transfer = 1;
}
VmaAllocationInfo info = {};
vmaGetAllocationInfo(vk->getVmaAllocator(), host_memory, &info);
if (m_type == GVDBT_SYSTEM_RAM) if (m_type == GVDBT_SYSTEM_RAM)
{ {
m_buffer[frame] = host_buffer; m_buffer[frame] = host_buffer;
m_memory[frame] = host_memory; m_memory[frame] = host_memory;
m_mapped_addr[frame] = info.pMappedData;
if (vkMapMemory(getVKDriver()->getDevice(), m_memory[frame], 0, m_size,
0, &m_mapped_addr[frame]) != VK_SUCCESS)
{
destroyPerFrame(frame);
}
return; return;
} }
VkBuffer local_buffer = VK_NULL_HANDLE; VkBuffer local_buffer = VK_NULL_HANDLE;
VkDeviceMemory local_memory = VK_NULL_HANDLE; VmaAllocation local_memory = VK_NULL_HANDLE;
if (!getVKDriver()->createBuffer(m_size, VmaAllocationCreateInfo local_info = {};
m_usage | VK_BUFFER_USAGE_TRANSFER_DST_BIT, local_info.usage = VMA_MEMORY_USAGE_AUTO;
VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT, local_buffer, local_memory))
if (!vk->createBuffer(m_size, m_usage | VK_BUFFER_USAGE_TRANSFER_DST_BIT,
local_info, local_buffer, local_memory))
{ {
if (host_buffer != VK_NULL_HANDLE) vmaDestroyBuffer(vk->getVmaAllocator(), host_buffer, host_memory);
vkDestroyBuffer(getVKDriver()->getDevice(), host_buffer, NULL); vmaDestroyBuffer(vk->getVmaAllocator(), local_buffer, local_memory);
if (host_memory != VK_NULL_HANDLE)
vkFreeMemory(getVKDriver()->getDevice(), host_memory, NULL);
if (local_buffer != VK_NULL_HANDLE)
vkDestroyBuffer(getVKDriver()->getDevice(), local_buffer, NULL);
if (local_memory != VK_NULL_HANDLE)
vkFreeMemory(getVKDriver()->getDevice(), local_memory, NULL);
return; return;
} }
m_buffer[frame] = local_buffer; m_buffer[frame] = local_buffer;
m_memory[frame] = local_memory; m_memory[frame] = local_memory;
m_staging_buffer[frame] = host_buffer; m_staging_buffer[frame] = host_buffer;
m_staging_memory[frame] = host_memory; m_staging_memory[frame] = host_memory;
m_mapped_addr[frame] = info.pMappedData;
if (vkMapMemory(getVKDriver()->getDevice(), m_staging_memory[frame], 0,
m_size, 0, &m_mapped_addr[frame]) != VK_SUCCESS)
{
destroyPerFrame(frame);
}
} // initPerFrame } // initPerFrame
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
void GEVulkanDynamicBuffer::destroyPerFrame(unsigned frame) void GEVulkanDynamicBuffer::destroyPerFrame(unsigned frame)
{ {
if ((m_staging_memory && m_staging_memory[frame] != VK_NULL_HANDLE) || GEVulkanDriver* vk = getVKDriver();
m_memory[frame] != VK_NULL_HANDLE) vmaDestroyBuffer(vk->getVmaAllocator(), m_buffer[frame], m_memory[frame]);
{
vkUnmapMemory(getVKDriver()->getDevice(), m_type == GVDBT_GPU_RAM ?
m_staging_memory[frame] : m_memory[frame]);
}
if (m_memory[frame] != VK_NULL_HANDLE)
vkFreeMemory(getVKDriver()->getDevice(), m_memory[frame], NULL);
if (m_buffer[frame] != VK_NULL_HANDLE)
vkDestroyBuffer(getVKDriver()->getDevice(), m_buffer[frame], NULL);
m_buffer[frame] = VK_NULL_HANDLE; m_buffer[frame] = VK_NULL_HANDLE;
m_memory[frame] = VK_NULL_HANDLE; m_memory[frame] = VK_NULL_HANDLE;
m_mapped_addr[frame] = NULL; m_mapped_addr[frame] = NULL;
if (m_type == GVDBT_GPU_RAM) if (m_type == GVDBT_GPU_RAM)
{ {
if (m_staging_buffer[frame] != VK_NULL_HANDLE) vmaDestroyBuffer(vk->getVmaAllocator(), m_staging_buffer[frame],
vkDestroyBuffer(getVKDriver()->getDevice(), m_staging_buffer[frame], NULL); m_staging_memory[frame]);
if (m_staging_memory[frame] != VK_NULL_HANDLE)
vkFreeMemory(getVKDriver()->getDevice(), m_staging_memory[frame], NULL);
m_staging_buffer[frame] = VK_NULL_HANDLE; m_staging_buffer[frame] = VK_NULL_HANDLE;
m_staging_memory[frame] = VK_NULL_HANDLE; m_staging_memory[frame] = VK_NULL_HANDLE;
} }
@ -198,7 +156,8 @@ void GEVulkanDynamicBuffer::destroy()
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
void GEVulkanDynamicBuffer::setCurrentData(void* data, size_t size) void GEVulkanDynamicBuffer::setCurrentData(void* data, size_t size)
{ {
const unsigned cur_frame = getVKDriver()->getCurrentFrame(); GEVulkanDriver* vk = getVKDriver();
const unsigned cur_frame = vk->getCurrentFrame();
if (size > m_size) if (size > m_size)
{ {
destroy(); destroy();
@ -210,11 +169,14 @@ void GEVulkanDynamicBuffer::setCurrentData(void* data, size_t size)
if (size == 0 || m_mapped_addr[cur_frame] == NULL) if (size == 0 || m_mapped_addr[cur_frame] == NULL)
return; return;
memcpy(m_mapped_addr[cur_frame], data, size); memcpy(m_mapped_addr[cur_frame], data, size);
vmaFlushAllocation(vk->getVmaAllocator(), m_staging_memory != NULL ?
m_staging_memory[cur_frame] : m_memory[cur_frame], 0, size);
if (m_type == GVDBT_GPU_RAM) if (m_type == GVDBT_GPU_RAM)
{ {
VkBufferCopy copy_region = {}; VkBufferCopy copy_region = {};
copy_region.size = size; copy_region.size = size;
vkCmdCopyBuffer(getVKDriver()->getCurrentCommandBuffer(), vkCmdCopyBuffer(vk->getCurrentCommandBuffer(),
m_staging_buffer[cur_frame], m_buffer[cur_frame], 1, &copy_region); m_staging_buffer[cur_frame], m_buffer[cur_frame], 1, &copy_region);
} }
} // setCurrentData } // setCurrentData

View File

@ -2,6 +2,7 @@
#define HEADER_GE_VULKAN_DYNAMIC_BUFFER_HPP #define HEADER_GE_VULKAN_DYNAMIC_BUFFER_HPP
#include "vulkan_wrapper.h" #include "vulkan_wrapper.h"
#include "ge_vma.hpp"
namespace GE namespace GE
{ {
@ -17,11 +18,11 @@ class GEVulkanDynamicBuffer
private: private:
VkBuffer* m_buffer; VkBuffer* m_buffer;
VkDeviceMemory* m_memory; VmaAllocation* m_memory;
VkBuffer* m_staging_buffer; VkBuffer* m_staging_buffer;
VkDeviceMemory* m_staging_memory; VmaAllocation* m_staging_memory;
void** m_mapped_addr; void** m_mapped_addr;
@ -29,10 +30,9 @@ private:
GEVulkanDynamicBufferType m_type; GEVulkanDynamicBufferType m_type;
static VkMemoryPropertyFlags m_host_flag;
size_t m_size, m_real_size; size_t m_size, m_real_size;
static int m_supports_host_transfer;
// ------------------------------------------------------------------------ // ------------------------------------------------------------------------
void initPerFrame(unsigned frame); void initPerFrame(unsigned frame);
// ------------------------------------------------------------------------ // ------------------------------------------------------------------------