Создание и инициализация произвольного буфера данных
This commit is contained in:
parent
167e46a506
commit
74c0872df9
12
include/vk.h
12
include/vk.h
|
@ -8,6 +8,8 @@
|
|||
#include "Surface.h"
|
||||
#include "Queue.h"
|
||||
|
||||
#include <map>
|
||||
|
||||
class Vulkan
|
||||
{
|
||||
public:
|
||||
|
@ -30,13 +32,13 @@ class Vulkan
|
|||
VkCommandPool commandPool; // Пул команд
|
||||
std::vector<VkCommandBuffer> commandBuffers; // Буферы команд
|
||||
VkBuffer vertexBuffer; // Буфер вершин
|
||||
VkDeviceMemory vertexBufferMemory; // Память буфера вершин
|
||||
VkBuffer indexBuffer; // Буфер индексов
|
||||
VkDeviceMemory indexBufferMemory; // Память буфера индексов
|
||||
std::vector<VkSemaphore> imageAvailableSemaphores; // семафор доступности изображения
|
||||
std::vector<VkSemaphore> renderFinishedSemaphores; // семафор окончания рендера
|
||||
std::vector<VkFence> inWorkFences; // барьер кадра в работе
|
||||
uint32_t currentFrame = 0; // Текущий кадр рендера
|
||||
|
||||
std::map<VkBuffer, std::pair<VkDevice, VkDeviceMemory>> databuffers; // Словарь для сопоставления дескриптору буфера дескрипторов лог. устройства и памяти
|
||||
|
||||
// Структура для хранения флагов
|
||||
struct
|
||||
|
@ -54,11 +56,11 @@ class Vulkan
|
|||
void createRenderpass(); // Создание проходов рендера
|
||||
VkShaderModule createShaderModule(const char * filename); // Создание шейдерного модуля
|
||||
void createGraphicPipeline(); // Создание графического конвеера
|
||||
void createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryPropertyFlags properties, VkBuffer& buffer, VkDeviceMemory& bufferMemory); // Создание произвольного буфера данных
|
||||
void createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryPropertyFlags properties, VkBuffer& buffer); // Создание произвольного буфера данных
|
||||
void destroyBuffer(VkBuffer buffer); // Уничтожение буфера и освобождение его памяти
|
||||
void createCommandPool(); // Создание пула команд
|
||||
void copyBuffer(VkBuffer srcBuffer, VkBuffer dstBuffer, VkDeviceSize size); // Копирование между буферами данных
|
||||
void createVertexBuffer(); // Создание буфера вершин
|
||||
void createIndexBuffer(); // Создание буфера индексов
|
||||
VkBuffer createDataBuffer(void* data, VkDeviceSize size, VkBufferUsageFlags usage); // Создание буфера данных
|
||||
void createSyncObjects(); // Создание объектов синхронизации
|
||||
void createFramebuffers(); // Создание буферов кадра
|
||||
};
|
||||
|
|
117
src/vk.cpp
117
src/vk.cpp
|
@ -21,8 +21,23 @@ void Vulkan::init(GLFWwindow* window)
|
|||
createFramebuffers(); // Создание буферов кадра
|
||||
createGraphicPipeline(); // Создание графического конвейера
|
||||
createCommandPool(); // Создание пула команд
|
||||
createVertexBuffer(); // Создание буфера вершин
|
||||
createIndexBuffer(); // Создание буфера индексов
|
||||
|
||||
//Вершины, записываемые в буфер
|
||||
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;
|
||||
vertexBuffer = createDataBuffer(vertices, bufferSize, VK_BUFFER_USAGE_VERTEX_BUFFER_BIT); // Создание буфера вершин
|
||||
// Индексы, записываемые в буфер
|
||||
uint16_t indices[] = {0, 1, 2, 2, 3, 0};
|
||||
// Размер буфера в байтах
|
||||
bufferSize = sizeof(uint16_t) * 6;
|
||||
indexBuffer = createDataBuffer(indices, bufferSize, VK_BUFFER_USAGE_INDEX_BUFFER_BIT); // Создание буфера индексов
|
||||
|
||||
createSyncObjects(); // Создание объектов синхронизации
|
||||
}
|
||||
|
||||
|
@ -31,11 +46,13 @@ void Vulkan::destroy()
|
|||
{
|
||||
vkDeviceWaitIdle(logicalDevice); // Ожидание окончания асинхронных задач
|
||||
|
||||
vkDestroyBuffer(logicalDevice, indexBuffer, nullptr); // Уничтожение буфера индексов
|
||||
vkFreeMemory(logicalDevice, indexBufferMemory, nullptr); // Освобождение памяти буфера индексов
|
||||
|
||||
vkDestroyBuffer(logicalDevice, vertexBuffer, nullptr); // Уничтожение буфера вершин
|
||||
vkFreeMemory(logicalDevice, vertexBufferMemory, nullptr); // Освобождение памяти буфера вершин
|
||||
// Освобождение буферов данных
|
||||
for (const auto& kv : databuffers)
|
||||
{
|
||||
vkDestroyBuffer(kv.second.first /*logicalDevice*/, kv.first /*VkBuffer*/, nullptr); // Уничтожение буфера
|
||||
vkFreeMemory(kv.second.first /*logicalDevice*/, kv.second.second /*VkDeviceMemory*/, nullptr); // Освобождение памяти буфера
|
||||
}
|
||||
databuffers.clear();
|
||||
|
||||
// Уничтожение объектов синхронизации
|
||||
for (int i = 0; i < surface.imageCount; i++)
|
||||
|
@ -748,7 +765,7 @@ void Vulkan::createGraphicPipeline()
|
|||
}
|
||||
|
||||
// Создание произвольного буфера данных
|
||||
void Vulkan::createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryPropertyFlags properties, VkBuffer& buffer, VkDeviceMemory& bufferMemory)
|
||||
void Vulkan::createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryPropertyFlags properties, VkBuffer& buffer)
|
||||
{
|
||||
// Информация о создаваемом буфере
|
||||
VkBufferCreateInfo bufferInfo{};
|
||||
|
@ -788,6 +805,7 @@ void Vulkan::createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryP
|
|||
allocInfo.allocationSize = memRequirements.size;
|
||||
allocInfo.memoryTypeIndex = index_memory;
|
||||
|
||||
VkDeviceMemory bufferMemory;
|
||||
// Выделение памяти
|
||||
if (vkAllocateMemory(logicalDevice, &allocInfo, nullptr, &bufferMemory) != VK_SUCCESS)
|
||||
{
|
||||
|
@ -796,6 +814,18 @@ void Vulkan::createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryP
|
|||
|
||||
// Привязка выделенной памяти к буферу
|
||||
vkBindBufferMemory(logicalDevice, buffer, bufferMemory, 0);
|
||||
|
||||
// Сохраним дескрипторы логического устройства и памяти буфера
|
||||
// по ключу дескриптора буфера
|
||||
databuffers[buffer] = {logicalDevice, bufferMemory};
|
||||
}
|
||||
|
||||
// Уничтожение буфера и освобождение его памяти
|
||||
void Vulkan::destroyBuffer(VkBuffer buffer)
|
||||
{
|
||||
vkDestroyBuffer(databuffers[buffer].first /*logicalDevice*/, buffer, nullptr); // Уничтожение буфера
|
||||
vkFreeMemory(databuffers[buffer].first /*logicalDevice*/, databuffers[buffer].second /*VkDeviceMemory*/, nullptr); // Освобождение памяти буфера
|
||||
databuffers.erase(buffer); // Удалим ключ из словаря
|
||||
}
|
||||
|
||||
// Создание пула команд
|
||||
|
@ -872,71 +902,32 @@ void Vulkan::copyBuffer(VkBuffer srcBuffer, VkBuffer dstBuffer, VkDeviceSize siz
|
|||
vkFreeCommandBuffers(logicalDevice, commandPool, 1, &commandBuffer);
|
||||
}
|
||||
|
||||
// Создание вершинного буфера
|
||||
void Vulkan::createVertexBuffer()
|
||||
// Создание и инициализация буфера данных
|
||||
VkBuffer Vulkan::createDataBuffer(void* data, VkDeviceSize size, VkBufferUsageFlags usage)
|
||||
{
|
||||
//Вершины, записываемые в буфер
|
||||
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 result;
|
||||
// Промежуточный буфер для переноса на устройство
|
||||
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);
|
||||
|
||||
createBuffer(size, VK_BUFFER_USAGE_TRANSFER_SRC_BIT, VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT, stagingBuffer);
|
||||
|
||||
// Отображение памяти буфера
|
||||
void* data;
|
||||
vkMapMemory(logicalDevice, stagingBufferMemory, 0, bufferSize, 0, &data);
|
||||
// Копирование вершин в промежуточный буфер
|
||||
memcpy(data, vertices, (size_t) bufferSize);
|
||||
void* stagingData;
|
||||
vkMapMemory(logicalDevice, databuffers[stagingBuffer].second /*VkDeviceMemory*/, 0, size, 0, &stagingData);
|
||||
// Копирование данных в промежуточный буфер
|
||||
memcpy(stagingData, data, (size_t) size);
|
||||
// Прекращение отображения памяти буфера
|
||||
vkUnmapMemory(logicalDevice, stagingBufferMemory);
|
||||
vkUnmapMemory(logicalDevice, databuffers[stagingBuffer].second /*VkDeviceMemory*/);
|
||||
|
||||
// Создание буфера вершин
|
||||
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);
|
||||
// Создание буфера данных
|
||||
createBuffer(size, VK_BUFFER_USAGE_TRANSFER_DST_BIT | usage, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT, result);
|
||||
// Копирование из промежуточного в буфер данных
|
||||
copyBuffer(stagingBuffer, result, size);
|
||||
|
||||
// Освобождение памяти и уничтожение буферов
|
||||
vkDestroyBuffer(logicalDevice, stagingBuffer, nullptr);
|
||||
vkFreeMemory(logicalDevice, stagingBufferMemory, nullptr);
|
||||
}
|
||||
destroyBuffer(stagingBuffer);
|
||||
|
||||
// Создание буфера индексов
|
||||
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);
|
||||
return result;
|
||||
}
|
||||
|
||||
// Создание объектов синхронизации
|
||||
|
|
Loading…
Reference in New Issue