Compare commits

...

5 Commits
v0.2 ... master

4 changed files with 128 additions and 3 deletions

View File

@ -1,3 +1,3 @@
# 05 # 05
Пятая заметка по Vulkan API - вершинные буферы https://rekovalev.site/vulkan-api-5-vertex-buffers/ Пятая заметка по Vulkan API - буферы данных и командные буферы https://rekovalev.site/vulkan-api-5-vertex-buffers/

View File

@ -1,11 +1,12 @@
#ifndef VERTEX_H #ifndef VERTEX_H
#define VERTEX_H #define VERTEX_H
#include <GLM/vec3.hpp> #include <GLM/glm.hpp>
typedef struct _Vertex typedef struct _Vertex
{ {
glm::vec3 position, color; glm::vec2 position;
glm::vec3 color;
} Vertex; } Vertex;
#endif // VERTEX_H #endif // VERTEX_H

View File

@ -27,6 +27,10 @@ class Vulkan
VkPipeline graphicsPipeline; // Графический конвейер VkPipeline graphicsPipeline; // Графический конвейер
VkCommandPool commandPool; // Пул команд VkCommandPool commandPool; // Пул команд
std::vector<VkCommandBuffer> commandBuffers; // Буферы команд std::vector<VkCommandBuffer> commandBuffers; // Буферы команд
VkBuffer vertexBuffer; // Буфер вершин
VkDeviceMemory vertexBufferMemory; // Память буфера вершин
VkBuffer indexBuffer; // Буфер индексов
VkDeviceMemory indexBufferMemory; // Память буфера индексов
// Структура для хранения флагов // Структура для хранения флагов
struct struct
@ -46,6 +50,9 @@ class Vulkan
void createGraphicPipeline(); // Создание графического конвеера void createGraphicPipeline(); // Создание графического конвеера
void createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryPropertyFlags properties, VkBuffer& buffer, VkDeviceMemory& bufferMemory); // Создание произвольного буфера данных void createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryPropertyFlags properties, VkBuffer& buffer, VkDeviceMemory& bufferMemory); // Создание произвольного буфера данных
void createCommandPool(); // Создание пула команд void createCommandPool(); // Создание пула команд
void copyBuffer(VkBuffer srcBuffer, VkBuffer dstBuffer, VkDeviceSize size); // Копирование между буферами данных
void createVertexBuffer(); // Создание буфера вершин
void createIndexBuffer(); // Создание буфера индексов
}; };
#endif // VK_H #endif // VK_H

View File

@ -20,11 +20,19 @@ void Vulkan::init(GLFWwindow* window)
createRenderpass(); // Создание проходов рендера createRenderpass(); // Создание проходов рендера
createGraphicPipeline(); // Создание графического конвейера createGraphicPipeline(); // Создание графического конвейера
createCommandPool(); // Создание пула команд createCommandPool(); // Создание пула команд
createVertexBuffer(); // Создание буфера вершин
createIndexBuffer(); // Создание буфера индексов
} }
// завершение работы // завершение работы
void Vulkan::destroy() void Vulkan::destroy()
{ {
vkDestroyBuffer(logicalDevice, indexBuffer, nullptr); // Уничтожение буфера индексов
vkFreeMemory(logicalDevice, indexBufferMemory, nullptr); // Освобождение памяти буфера индексов
vkDestroyBuffer(logicalDevice, vertexBuffer, nullptr); // Уничтожение буфера вершин
vkFreeMemory(logicalDevice, vertexBufferMemory, nullptr); // Освобождение памяти буфера вершин
vkDestroyCommandPool(logicalDevice, commandPool, nullptr); // Уничтожение командного пула vkDestroyCommandPool(logicalDevice, commandPool, nullptr); // Уничтожение командного пула
vkDestroyPipeline(logicalDevice, graphicsPipeline, nullptr); // Уничтожение графического конвейера vkDestroyPipeline(logicalDevice, graphicsPipeline, nullptr); // Уничтожение графического конвейера
@ -803,3 +811,112 @@ void Vulkan::createCommandPool()
throw std::runtime_error("Unable to allocate command buffers"); throw std::runtime_error("Unable to allocate command buffers");
} }
} }
// Копирование между буферами данных
void Vulkan::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;
// Дескриптор и выделение буфера команд
VkCommandBuffer commandBuffer;
vkAllocateCommandBuffers(logicalDevice, &allocInfo, &commandBuffer);
// Начало записи команд
VkCommandBufferBeginInfo beginInfo{};
beginInfo.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO;
beginInfo.flags = VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT;
vkBeginCommandBuffer(commandBuffer, &beginInfo);
// Операция копирования
VkBufferCopy copyRegion{};
copyRegion.size = size;
vkCmdCopyBuffer(commandBuffer, srcBuffer, dstBuffer, 1, &copyRegion);
// Конец записи команд
vkEndCommandBuffer(commandBuffer);
// Информация о запускаемых буферах команд
VkSubmitInfo submitInfo{};
submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO;
submitInfo.commandBufferCount = 1;
submitInfo.pCommandBuffers = &commandBuffer;
// Запуск командного буфера копирования и ожидание завершения
vkQueueSubmit(queue.descriptor, 1, &submitInfo, VK_NULL_HANDLE);
vkQueueWaitIdle(queue.descriptor);
// Освобождение командного буфера копирования
vkFreeCommandBuffers(logicalDevice, commandPool, 1, &commandBuffer);
}
// Создание вершинного буфера
void Vulkan::createVertexBuffer()
{
//Вершины, записываемые в буфер
Vertex vertices[] = {
{ {-0.5f, -0.5f}, {1.0f, 0.0f, 0.0f} },
{ { 0.5f, -0.5f}, {0.0f, 1.0f, 0.0f} },
{ { 0.5f, 0.5f}, {0.0f, 0.0f, 1.0f} },
{ {-0.5f, 0.5f}, {1.0f, 1.0f, 1.0f} }
};
// Размер буфера в байтах
VkDeviceSize bufferSize = sizeof(Vertex) * 4;
// Промежуточный буфер для переноса на устройство
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(logicalDevice, stagingBufferMemory, 0, bufferSize, 0, &data);
// Копирование вершин в промежуточный буфер
memcpy(data, vertices, (size_t) bufferSize);
// Прекращение отображения памяти буфера
vkUnmapMemory(logicalDevice, stagingBufferMemory);
// Создание буфера вершин
createBuffer(bufferSize, VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT, vertexBuffer, vertexBufferMemory);
// Копирование из промежуточного в буфер вершин
copyBuffer(stagingBuffer, vertexBuffer, bufferSize);
// Освобождение памяти и уничтожение буферов
vkDestroyBuffer(logicalDevice, stagingBuffer, nullptr);
vkFreeMemory(logicalDevice, stagingBufferMemory, nullptr);
}
// Создание буфера индексов
void Vulkan::createIndexBuffer()
{
// Индексы, записываемые в буфер
uint16_t indices[] = {0, 1, 2, 2, 3, 0};
// Размер буфера в байтах
VkDeviceSize bufferSize = sizeof(uint16_t) * 6;
// Промежуточный буфер для переноса на устройство
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(logicalDevice, stagingBufferMemory, 0, bufferSize, 0, &data);
// Копирование вершин в промежуточный буфер
memcpy(data, indices, (size_t) bufferSize);
// Прекращение отображения памяти буфера
vkUnmapMemory(logicalDevice, stagingBufferMemory);
// Создание буфера вершин
createBuffer(bufferSize, VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_INDEX_BUFFER_BIT, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT, indexBuffer, indexBufferMemory);
// Копирование из промежуточного в буфер вершин
copyBuffer(stagingBuffer, indexBuffer, bufferSize);
// Освобождение памяти и уничтожение буферов
vkDestroyBuffer(logicalDevice, stagingBuffer, nullptr);
vkFreeMemory(logicalDevice, stagingBufferMemory, nullptr);
}