博客
关于我
Vulkan临时缓冲区
阅读量:208 次
发布时间:2019-02-28

本文共 3925 字,大约阅读时间需要 13 分钟。

相比于显卡内部读取数据,单纯从CPU访问内存数据的方式性能并不理想。最佳的做法是使用VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT标志位,这种方式通常用于专用图形卡,因为这样可以确保CPU无法访问这些内存,从而提高性能。

创建临时缓冲区的最佳实现方式是通过辅助函数来完成。以下是一个常用的方法,涉及创建缓冲区和顶点缓冲区的实现:

void createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryPropertyFlags properties, VkBuffer& buffer, VkDeviceMemory& bufferMemory) {    VkBufferCreateInfo bufferInfo = {};    bufferInfo.sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO;    bufferInfo.size = size;    bufferInfo.usage = usage;    bufferInfo.sharingMode = VK_SHARING_MODE_EXCLUSIVE;    if (vkCreateBuffer(device, &bufferInfo, nullptr, &buffer) != VK_SUCCESS) {        throw std::runtime_error("failed to create buffer!");    }    VkMemoryRequirements memRequirements;    vkGetBufferMemoryRequirements(device, buffer, &memRequirements);    VkMemoryAllocateInfo allocInfo = {};    allocInfo.sType = VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO;    allocInfo.allocationSize = memRequirements.size;    allocInfo.memoryTypeIndex = findMemoryType(memRequirements.memoryTypeBits, properties);    if (vkAllocateMemory(device, &allocInfo, nullptr, &bufferMemory) != VK_SUCCESS) {        throw std::runtime_error("failed to allocate buffer memory!");    }    vkBindBufferMemory(device, buffer, bufferMemory, 0);}

这个函数需要传递缓冲区大小、内存属性和使用标志,以创建不同类型的缓冲区。最后两个参数用于存储输出的句柄。

以下是创建顶点缓冲区的实现:

void createVertexBuffer() {    VkDeviceSize bufferSize = sizeof(vertices[0]) * vertices.size();    createBuffer(bufferSize, VK_BUFFER_USAGE_VERTEX_BUFFER_BIT, VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT, vertexBuffer, vertexBufferMemory);}

为了实现临时缓冲区的使用,我们可以修改顶点缓冲区的创建方法:

void createVertexBuffer() {    VkDeviceSize bufferSize = sizeof(vertices[0]) * vertices.size();    VkBuffer stagingBuffer;    VkDeviceMemory stagingBufferMemory;    createBuffer(bufferSize, VK_BUFFER_USAGE_TRANSFER_SRC_BIT, VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT, stagingBuffer, stagingBufferMemory);    void* data;    vkMapMemory(device, stagingBufferMemory, 0, bufferSize, 0, &data);    memcpy(data, vertices.data(), bufferSize);    vkUnmapMemory(device, stagingBufferMemory);    createBuffer(bufferSize, VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT, vertexBuffer, vertexBufferMemory);}

在这个实现中,stagingBuffer用于临时存储顶点数据,然后通过copyBuffer函数将其复制到设备缓冲区中。

为了实现内存传输操作,我们需要使用命令缓冲区。以下是一个常用的方法:

void copyBuffer(VkBuffer srcBuffer, VkBuffer dstBuffer, VkDeviceSize size) {    VkCommandBufferAllocateInfo allocInfo = {};    allocInfo.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO;    allocInfo.level = VK_COMMAND_BUFFER_LEVEL_PRIMARY;    allocInfo.commandPool = commandPool;    allocInfo.commandBufferCount = 1;    if (vkAllocateCommandBuffers(device, &allocInfo, &commandBuffer) != VK_SUCCESS) {        throw std::runtime_error("failed to allocate command buffer!");    }}void copyBuffer() {    VkCommandBufferBeginInfo beginInfo = {};    beginInfo.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO;    beginInfo.flags = VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT;    if (vkBeginCommandBuffer(commandBuffer, &beginInfo) != VK_SUCCESS) {        throw std::runtime_error("failed to begin command buffer!");    }    VkBufferCopy copyRegion = {};    copyRegion.srcOffset = 0;    copyRegion.dstOffset = 0;    copyRegion.size = size;    if (vkCmdCopyBuffer(commandBuffer, srcBuffer, dstBuffer, 1, copyRegion) != VK_SUCCESS) {        throw std::runtime_error("failed to copy buffers!");    }    if (vkEndCommandBuffer(commandBuffer) != VK_SUCCESS) {        throw std::runtime_error("failed to end command buffer!");    }}

copyBuffer函数中,可以通过命令缓冲区执行内存传输操作。为了优化命令缓冲区的内存使用,可以为短期命令缓冲区分配内存,并使用VK_COMMAND_POOL_CREATE_TRANSIENT_BIT标志位。

最后,确保在完成内存传输后清理临时缓冲区:

void clearStagingBuffer() {    vkDestroyBuffer(device, stagingBuffer, nullptr);    vkFreeMemory(device, stagingBufferMemory, nullptr);}

通过上述方法,可以实现对顶点数据的高效内存传输和缓冲区管理。

转载地址:http://hpnp.baihongyu.com/

你可能感兴趣的文章
localhost:5000在MacOS V12(蒙特利)中不可用
查看>>
mac mysql 进程_Mac平台下启动MySQL到完全终止MySQL----终端八步走
查看>>
Mac OS 12.0.1 如何安装柯美287打印机驱动,刷卡打印
查看>>
MangoDB4.0版本的安装与配置
查看>>
Manjaro 24.1 “Xahea” 发布!具有 KDE Plasma 6.1.5、GNOME 46 和最新的内核增强功能
查看>>
mapping文件目录生成修改
查看>>
MapReduce程序依赖的jar包
查看>>
mariadb multi-source replication(mariadb多主复制)
查看>>
MariaDB的简单使用
查看>>
MaterialForm对tab页进行隐藏
查看>>
Member var and Static var.
查看>>
memcached高速缓存学习笔记001---memcached介绍和安装以及基本使用
查看>>
memcached高速缓存学习笔记003---利用JAVA程序操作memcached crud操作
查看>>
Memcached:Node.js 高性能缓存解决方案
查看>>
memcache、redis原理对比
查看>>
memset初始化高维数组为-1/0
查看>>
Metasploit CGI网关接口渗透测试实战
查看>>
Metasploit Web服务器渗透测试实战
查看>>
Moment.js常见用法总结
查看>>
MongoDB出现Error parsing command line: unrecognised option ‘--fork‘ 的解决方法
查看>>