From 74c0872df93d8086114c6b7a8da7b597b5a3893b Mon Sep 17 00:00:00 2001 From: "re.kovalev" Date: Thu, 7 Apr 2022 09:01:43 +0300 Subject: [PATCH] =?UTF-8?q?=D0=A1=D0=BE=D0=B7=D0=B4=D0=B0=D0=BD=D0=B8?= =?UTF-8?q?=D0=B5=20=D0=B8=20=D0=B8=D0=BD=D0=B8=D1=86=D0=B8=D0=B0=D0=BB?= =?UTF-8?q?=D0=B8=D0=B7=D0=B0=D1=86=D0=B8=D1=8F=20=D0=BF=D1=80=D0=BE=D0=B8?= =?UTF-8?q?=D0=B7=D0=B2=D0=BE=D0=BB=D1=8C=D0=BD=D0=BE=D0=B3=D0=BE=20=D0=B1?= =?UTF-8?q?=D1=83=D1=84=D0=B5=D1=80=D0=B0=20=D0=B4=D0=B0=D0=BD=D0=BD=D1=8B?= =?UTF-8?q?=D1=85?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- include/vk.h | 12 +++--- src/vk.cpp | 117 ++++++++++++++++++++++++--------------------------- 2 files changed, 61 insertions(+), 68 deletions(-) diff --git a/include/vk.h b/include/vk.h index 88a5155..f642192 100644 --- a/include/vk.h +++ b/include/vk.h @@ -8,6 +8,8 @@ #include "Surface.h" #include "Queue.h" +#include + class Vulkan { public: @@ -30,13 +32,13 @@ class Vulkan VkCommandPool commandPool; // Пул команд std::vector commandBuffers; // Буферы команд VkBuffer vertexBuffer; // Буфер вершин - VkDeviceMemory vertexBufferMemory; // Память буфера вершин VkBuffer indexBuffer; // Буфер индексов - VkDeviceMemory indexBufferMemory; // Память буфера индексов std::vector imageAvailableSemaphores; // семафор доступности изображения std::vector renderFinishedSemaphores; // семафор окончания рендера std::vector inWorkFences; // барьер кадра в работе uint32_t currentFrame = 0; // Текущий кадр рендера + + std::map> 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(); // Создание буферов кадра }; diff --git a/src/vk.cpp b/src/vk.cpp index a47d13a..fd31ec8 100644 --- a/src/vk.cpp +++ b/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; } // Создание объектов синхронизации