博客
关于我
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/

你可能感兴趣的文章
mysql 导入导出大文件
查看>>
mysql 将null转代为0
查看>>
mysql 常用
查看>>
MySQL 常用列类型
查看>>
mysql 常用命令
查看>>
Mysql 常见ALTER TABLE操作
查看>>
MySQL 常见的 9 种优化方法
查看>>
MySQL 常见的开放性问题
查看>>
Mysql 常见错误
查看>>
MYSQL 幻读(Phantom Problem)不可重复读
查看>>
mysql 往字段后面加字符串
查看>>
mysql 快速自增假数据, 新增假数据,mysql自增假数据
查看>>
Mysql 报错 Field 'id' doesn't have a default value
查看>>
MySQL 报错:Duplicate entry 'xxx' for key 'UNIQ_XXXX'
查看>>
Mysql 拼接多个字段作为查询条件查询方法
查看>>
mysql 排序id_mysql如何按特定id排序
查看>>
Mysql 提示:Communication link failure
查看>>
mysql 插入是否成功_PDO mysql:如何知道插入是否成功
查看>>
Mysql 数据库InnoDB存储引擎中主要组件的刷新清理条件:脏页、RedoLog重做日志、Insert Buffer或ChangeBuffer、Undo Log
查看>>
mysql 数据库备份及ibdata1的瘦身
查看>>