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