ULRE/src/SceneGraph/Vulkan/VKDeviceBuffer.cpp

137 lines
4.5 KiB
C++
Raw Normal View History

#include<hgl/graph/VKDevice.h>
#include<hgl/graph/VKIndexBuffer.h>
#include<hgl/graph/VKVertexAttribBuffer.h>
#include<hgl/graph/VKPhysicalDevice.h>
2023-07-28 20:17:46 +08:00
#include<iostream>
VK_NAMESPACE_BEGIN
const VkDeviceSize VulkanDevice::GetUBOAlign (){return attr->physical_device->GetUBOAlign();}
const VkDeviceSize VulkanDevice::GetSSBOAlign (){return attr->physical_device->GetSSBOAlign();}
const VkDeviceSize VulkanDevice::GetUBORange (){return attr->physical_device->GetUBORange();}
const VkDeviceSize VulkanDevice::GetSSBORange (){return attr->physical_device->GetSSBORange();}
2023-03-24 22:14:05 +08:00
bool VulkanDevice::CreateBuffer(DeviceBufferData *buf,VkBufferUsageFlags buf_usage,VkDeviceSize range,VkDeviceSize size,const void *data,SharingMode sharing_mode)
{
2023-09-07 18:09:31 +08:00
if(size<=0)return(false);
2020-09-27 20:58:25 +08:00
BufferCreateInfo buf_info;
2020-09-02 18:36:24 +08:00
buf_info.usage = buf_usage;
buf_info.size = size;
buf_info.queueFamilyIndexCount = 0;
buf_info.pQueueFamilyIndices = nullptr;
buf_info.sharingMode = VkSharingMode(sharing_mode);
2019-11-25 22:05:05 +08:00
if(vkCreateBuffer(attr->device,&buf_info,nullptr,&buf->buffer)!=VK_SUCCESS)
return(false);
2019-11-25 22:05:05 +08:00
VkMemoryRequirements mem_reqs;
2019-11-25 22:05:05 +08:00
vkGetBufferMemoryRequirements(attr->device,buf->buffer,&mem_reqs);
DeviceMemory *dm=CreateMemory(mem_reqs, VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT //CPU端可以Map
|VK_MEMORY_PROPERTY_HOST_COHERENT_BIT); //CPU端无需Flush,即可被GPU访问
//注:这个模式并非最佳效能,但是在开发时最为方便
//Device Local模式仅支持GPU访问是性能最佳考虑在一些极端情况下使用
2019-11-25 22:05:05 +08:00
if(dm&&dm->BindBuffer(buf->buffer))
{
buf->info.buffer =buf->buffer;
buf->info.offset =0;
buf->info.range =range;
2019-06-26 18:38:35 +08:00
2019-11-25 22:05:05 +08:00
buf->memory =dm;
2019-11-25 22:05:05 +08:00
if(!data)
return(true);
dm->Write(data,0,size);
2019-11-25 22:05:05 +08:00
return(true);
}
2019-11-25 22:05:05 +08:00
delete dm;
vkDestroyBuffer(attr->device,buf->buffer,nullptr);
return(false);
}
VAB *VulkanDevice::CreateVAB(VkFormat format,uint32_t count,const void *data,SharingMode sharing_mode)
{
2023-09-07 18:09:31 +08:00
if(count==0)return(nullptr);
const uint32_t stride=GetStrideByFormat(format);
if(stride==0)
{
std::cerr<<"format["<<format<<"] stride length is 0,please use \"VulkanDevice::CreateBuffer(VkBufferUsageFlags,VkDeviceSize,VkSharingMode)\" function.";
return(nullptr);
}
const VkDeviceSize size=stride*count;
DeviceBufferData buf;
if(!CreateBuffer(&buf,VK_BUFFER_USAGE_VERTEX_BUFFER_BIT,size,size,data,sharing_mode))
return(nullptr);
return(new VertexAttribBuffer(attr->device,buf,format,stride,count));
}
const IndexType VulkanDevice::ChooseIndexType(const VkDeviceSize &vertex_count)const
2024-04-19 00:40:51 +08:00
{
if(vertex_count<=0)return(IndexType::ERR);
2024-05-05 15:12:14 +08:00
if(attr->uint8_index_type&& vertex_count<=0xFF )return IndexType::U8; else
if( vertex_count<=0xFFFF)return IndexType::U16; else
if(attr->uint32_index_type )return IndexType::U32; else
2024-04-19 00:40:51 +08:00
return IndexType::ERR;
}
const bool VulkanDevice::CheckIndexType(const IndexType it,const VkDeviceSize &vertex_count)const
2024-04-19 00:40:51 +08:00
{
if(vertex_count<=0)return(false);
if(it==IndexType::U16&&vertex_count<=0xFFFF)return(true);
2024-05-05 15:12:14 +08:00
if(it==IndexType::U32&& attr->uint32_index_type)return(true);
2024-04-19 00:40:51 +08:00
2024-05-05 15:12:14 +08:00
if(it==IndexType::U8 &&vertex_count<=0xFF&& attr->uint8_index_type)return(true);
2024-04-19 00:40:51 +08:00
return(false);
}
IndexBuffer *VulkanDevice::CreateIBO(IndexType index_type,uint32_t count,const void *data,SharingMode sharing_mode)
{
2023-09-07 18:09:31 +08:00
if(count==0)return(nullptr);
uint32_t stride;
2023-05-11 21:07:26 +08:00
if(index_type==IndexType::U8 )stride=1;else
if(index_type==IndexType::U16)stride=2;else
if(index_type==IndexType::U32)stride=4;else
2019-05-06 12:00:03 +08:00
return(nullptr);
const VkDeviceSize size=stride*count;
DeviceBufferData buf;
2019-11-25 22:05:05 +08:00
if(!CreateBuffer(&buf,VK_BUFFER_USAGE_INDEX_BUFFER_BIT,size,data,sharing_mode))
return(nullptr);
2019-11-25 22:05:05 +08:00
return(new IndexBuffer(attr->device,buf,index_type,count));
}
DeviceBuffer *VulkanDevice::CreateBuffer(VkBufferUsageFlags buf_usage,VkDeviceSize range,VkDeviceSize size,const void *data,SharingMode sharing_mode)
{
2023-09-07 18:09:31 +08:00
if(size<=0)return(nullptr);
DeviceBufferData buf;
if(!CreateBuffer(&buf,buf_usage,range,size,data,sharing_mode))
return(nullptr);
return(new DeviceBuffer(attr->device,buf));
}
VK_NAMESPACE_END