From d04c2ed5b4af0b1033ba6573d3790cdf568f4a66 Mon Sep 17 00:00:00 2001 From: "r.kuznetsov" Date: Fri, 11 Jan 2019 15:23:51 +0300 Subject: [PATCH] [Vulkan] Added memory management --- .../vulkan/android_vulkan_context_factory.cpp | 22 +++--- drape/CMakeLists.txt | 5 ++ drape/mesh_object.cpp | 10 ++- drape/mesh_object.hpp | 7 ++ drape/vulkan/vulkan_base_context.cpp | 4 +- drape/vulkan/vulkan_base_context.hpp | 9 ++- drape/vulkan/vulkan_device_holder.hpp | 20 ++++++ drape/vulkan/vulkan_memory_manager.cpp | 9 +++ drape/vulkan/vulkan_memory_manager.hpp | 21 ++++++ drape/vulkan/vulkan_mesh_object_impl.cpp | 69 +++++++++++++++++++ drape/vulkan/vulkan_mesh_object_impl.hpp | 41 +++++++++++ shaders/program_manager.cpp | 2 +- shaders/vulkan_program_pool.cpp | 17 +++-- shaders/vulkan_program_pool.hpp | 5 +- 14 files changed, 218 insertions(+), 23 deletions(-) create mode 100644 drape/vulkan/vulkan_device_holder.hpp create mode 100644 drape/vulkan/vulkan_memory_manager.cpp create mode 100644 drape/vulkan/vulkan_memory_manager.hpp create mode 100644 drape/vulkan/vulkan_mesh_object_impl.cpp create mode 100644 drape/vulkan/vulkan_mesh_object_impl.hpp diff --git a/android/jni/com/mapswithme/vulkan/android_vulkan_context_factory.cpp b/android/jni/com/mapswithme/vulkan/android_vulkan_context_factory.cpp index 56291fe28d..fe23dee413 100644 --- a/android/jni/com/mapswithme/vulkan/android_vulkan_context_factory.cpp +++ b/android/jni/com/mapswithme/vulkan/android_vulkan_context_factory.cpp @@ -20,8 +20,8 @@ class DrawVulkanContext : public dp::vulkan::VulkanBaseContext { public: DrawVulkanContext(VkInstance vulkanInstance, VkPhysicalDevice gpu, - VkDevice device) - : dp::vulkan::VulkanBaseContext(vulkanInstance, gpu, device) + VkDevice device, uint32_t renderingQueueFamilyIndex) + : dp::vulkan::VulkanBaseContext(vulkanInstance, gpu, device, renderingQueueFamilyIndex) {} }; @@ -29,8 +29,8 @@ class UploadVulkanContext : public dp::vulkan::VulkanBaseContext { public: UploadVulkanContext(VkInstance vulkanInstance, VkPhysicalDevice gpu, - VkDevice device) - : dp::vulkan::VulkanBaseContext(vulkanInstance, gpu, device) + VkDevice device, uint32_t renderingQueueFamilyIndex) + : dp::vulkan::VulkanBaseContext(vulkanInstance, gpu, device, renderingQueueFamilyIndex) {} void Present() override {} @@ -127,13 +127,13 @@ AndroidVulkanContextFactory::AndroidVulkanContextFactory() vkGetPhysicalDeviceQueueFamilyProperties(m_gpu, &queueFamilyCount, queueFamilyProperties.data()); - uint32_t queueFamilyIndex = 0; - for (; queueFamilyIndex < queueFamilyCount; ++queueFamilyIndex) + uint32_t renderingQueueFamilyIndex = 0; + for (; renderingQueueFamilyIndex < queueFamilyCount; ++renderingQueueFamilyIndex) { - if (queueFamilyProperties[queueFamilyIndex].queueFlags & VK_QUEUE_GRAPHICS_BIT) + if (queueFamilyProperties[renderingQueueFamilyIndex].queueFlags & VK_QUEUE_GRAPHICS_BIT) break; } - if (queueFamilyIndex == queueFamilyCount) + if (renderingQueueFamilyIndex == queueFamilyCount) { LOG_ERROR_VK("Any queue family with VK_QUEUE_GRAPHICS_BIT wasn't found."); return; @@ -169,8 +169,10 @@ AndroidVulkanContextFactory::AndroidVulkanContextFactory() return; } - m_drawContext = make_unique_dp(m_vulkanInstance, m_gpu, m_device); - m_uploadContext = make_unique_dp(m_vulkanInstance, m_gpu, m_device); + m_drawContext = make_unique_dp(m_vulkanInstance, m_gpu, m_device, + renderingQueueFamilyIndex); + m_uploadContext = make_unique_dp(m_vulkanInstance, m_gpu, m_device, + renderingQueueFamilyIndex); } AndroidVulkanContextFactory::~AndroidVulkanContextFactory() diff --git a/drape/CMakeLists.txt b/drape/CMakeLists.txt index 0569f9f63d..50375b7d10 100644 --- a/drape/CMakeLists.txt +++ b/drape/CMakeLists.txt @@ -152,9 +152,14 @@ append( SRC vulkan/vulkan_base_context.cpp vulkan/vulkan_base_context.hpp + vulkan/vulkan_device_holder.hpp vulkan/vulkan_gpu_program.hpp vulkan/vulkan_layers.cpp vulkan/vulkan_layers.hpp + vulkan/vulkan_memory_manager.cpp + vulkan/vulkan_memory_manager.hpp + vulkan/vulkan_mesh_object_impl.cpp + vulkan/vulkan_mesh_object_impl.hpp vulkan/vulkan_utils.cpp vulkan/vulkan_utils.hpp ) diff --git a/drape/mesh_object.cpp b/drape/mesh_object.cpp index 05bd8294a2..5dec31b5f5 100644 --- a/drape/mesh_object.cpp +++ b/drape/mesh_object.cpp @@ -7,6 +7,8 @@ #include "drape/glsl_types.hpp" #include "drape/texture_manager.hpp" +#include "drape/vulkan/vulkan_mesh_object_impl.hpp" + namespace { glConst GetGLDrawPrimitive(dp::MeshObject::DrawPrimitive drawPrimitive) @@ -159,8 +161,7 @@ MeshObject::MeshObject(ref_ptr context, DrawPrimitive drawP } else if (apiVersion == dp::ApiVersion::Vulkan) { - //TODO(@rokuz, @darina): Implement. - CHECK(false, ()); + InitForVulkan(); } CHECK(m_impl != nullptr, ()); } @@ -175,6 +176,11 @@ void MeshObject::InitForOpenGL() m_impl = make_unique_dp(make_ref(this)); } +void MeshObject::InitForVulkan() +{ + m_impl = make_unique_dp(make_ref(this)); +} + void MeshObject::SetBuffer(uint32_t bufferInd, std::vector && vertices, uint32_t stride) { CHECK_LESS_OR_EQUAL(bufferInd, GetNextBufferIndex(), ()); diff --git a/drape/mesh_object.hpp b/drape/mesh_object.hpp index 7d769e0f40..ff95917c0f 100644 --- a/drape/mesh_object.hpp +++ b/drape/mesh_object.hpp @@ -20,6 +20,11 @@ namespace metal class MetalMeshObjectImpl; } // namespace metal +namespace vulkan +{ +class VulkanMeshObjectImpl; +} // namespace vulkan + // This class implements a simple mesh object which does not use an index buffer. // Use this class only for simple geometry. class MeshObject @@ -27,6 +32,7 @@ class MeshObject friend class MeshObjectImpl; friend class GLMeshObjectImpl; friend class metal::MetalMeshObjectImpl; + friend class vulkan::VulkanMeshObjectImpl; public: enum class DrawPrimitive: uint8_t @@ -100,6 +106,7 @@ private: }; void InitForOpenGL(); + void InitForVulkan(); #if defined(OMIM_METAL_AVAILABLE) // Definition of this method is in a .mm-file. diff --git a/drape/vulkan/vulkan_base_context.cpp b/drape/vulkan/vulkan_base_context.cpp index b7dcb234c4..92dd67f5b5 100644 --- a/drape/vulkan/vulkan_base_context.cpp +++ b/drape/vulkan/vulkan_base_context.cpp @@ -11,11 +11,13 @@ namespace dp namespace vulkan { VulkanBaseContext::VulkanBaseContext(VkInstance vulkanInstance, VkPhysicalDevice gpu, - VkDevice device) + VkDevice device, uint32_t renderingQueueFamilyIndex) : m_vulkanInstance(vulkanInstance) , m_gpu(gpu) , m_device(device) + , m_renderingQueueFamilyIndex(renderingQueueFamilyIndex) { + m_deviceHolder = std::make_shared(m_device); vkGetPhysicalDeviceProperties(m_gpu, &m_gpuProperties); } diff --git a/drape/vulkan/vulkan_base_context.hpp b/drape/vulkan/vulkan_base_context.hpp index e2e5552ed1..d7f4af98e0 100644 --- a/drape/vulkan/vulkan_base_context.hpp +++ b/drape/vulkan/vulkan_base_context.hpp @@ -2,6 +2,7 @@ #include "drape/graphics_context.hpp" #include "drape/pointers.hpp" +#include "drape/vulkan/vulkan_device_holder.hpp" #include "geometry/point2d.hpp" @@ -20,7 +21,7 @@ class VulkanBaseContext : public dp::GraphicsContext { public: VulkanBaseContext(VkInstance vulkanInstance, VkPhysicalDevice gpu, - VkDevice device); + VkDevice device, uint32_t renderingQueueFamilyIndex); void Present() override {} void MakeCurrent() override {} @@ -54,15 +55,21 @@ public: void ResetSurface(); VkDevice GetDevice() const { return m_device; } + DeviceHolderPtr GetDeviceHolder() const { return m_deviceHolder; } + VkPhysicalDeviceProperties const & GetGpuProperties() const { return m_gpuProperties; } + uint32_t GetRenderingQueueFamilyIndex() { return m_renderingQueueFamilyIndex; } protected: VkInstance const m_vulkanInstance; VkPhysicalDevice const m_gpu; VkDevice const m_device; + uint32_t const m_renderingQueueFamilyIndex; VkPhysicalDeviceProperties m_gpuProperties; + std::shared_ptr m_deviceHolder; + boost::optional m_surface; uint32_t m_stencilReferenceValue = 1; diff --git a/drape/vulkan/vulkan_device_holder.hpp b/drape/vulkan/vulkan_device_holder.hpp new file mode 100644 index 0000000000..87a1da3483 --- /dev/null +++ b/drape/vulkan/vulkan_device_holder.hpp @@ -0,0 +1,20 @@ +#pragma once + +#include +#include + +#include + +namespace dp +{ +namespace vulkan +{ +struct DeviceHolder +{ + VkDevice const m_device; + explicit DeviceHolder(VkDevice const device) : m_device(device) {} +}; + +using DeviceHolderPtr = std::weak_ptr; +} // namespace vulkan +} // namespace dp diff --git a/drape/vulkan/vulkan_memory_manager.cpp b/drape/vulkan/vulkan_memory_manager.cpp new file mode 100644 index 0000000000..e9f98a3395 --- /dev/null +++ b/drape/vulkan/vulkan_memory_manager.cpp @@ -0,0 +1,9 @@ +#pragma "drape/vulkan/vulkan_memory_manager.cpp" + +namespace dp +{ +namespace vulkan +{ + +} // namespace vulkan +} // namespace dp diff --git a/drape/vulkan/vulkan_memory_manager.hpp b/drape/vulkan/vulkan_memory_manager.hpp new file mode 100644 index 0000000000..35f6031d2f --- /dev/null +++ b/drape/vulkan/vulkan_memory_manager.hpp @@ -0,0 +1,21 @@ +#pragma once + +#include +#include + +namespace dp +{ +namespace vulkan +{ +class VulkanMemoryManager +{ +public: + explicit VulkanMemoryManager(VkDevice device) : m_device(device) {} + + //VkDeviceMemory + +private: + VkDevice const m_device; +}; +} // namespace vulkan +} // namespace dp diff --git a/drape/vulkan/vulkan_mesh_object_impl.cpp b/drape/vulkan/vulkan_mesh_object_impl.cpp new file mode 100644 index 0000000000..e01abd7731 --- /dev/null +++ b/drape/vulkan/vulkan_mesh_object_impl.cpp @@ -0,0 +1,69 @@ +#include "drape/vulkan/vulkan_mesh_object_impl.hpp" + +#include "drape/vulkan/vulkan_base_context.hpp" +#include "drape/vulkan/vulkan_utils.hpp" + +namespace dp +{ +namespace vulkan +{ +void VulkanMeshObjectImpl::Build(ref_ptr context, ref_ptr program) +{ + ref_ptr vulkanContext = context; + uint32_t const queueFamilyIndex = vulkanContext->GetRenderingQueueFamilyIndex(); + m_deviceHolder = vulkanContext->GetDeviceHolder(); + + auto devicePtr = m_deviceHolder.lock(); + CHECK(devicePtr != nullptr, ()); + + m_geometryBuffers.resize(m_mesh->m_buffers.size()); + for (size_t i = 0; i < m_mesh->m_buffers.size(); i++) + { + if (m_mesh->m_buffers[i].m_data.empty()) + continue; + + VkBufferCreateInfo info = {}; + info.sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO; + info.pNext = nullptr; + info.flags = 0; + info.size = m_mesh->m_buffers[i].m_data.size() * sizeof(m_mesh->m_buffers[i].m_data[0]); + info.usage = VK_BUFFER_USAGE_VERTEX_BUFFER_BIT; + info.usage = VK_SHARING_MODE_CONCURRENT; + info.queueFamilyIndexCount = 1; + info.pQueueFamilyIndices = &queueFamilyIndex; + CHECK_VK_CALL(vkCreateBuffer(devicePtr->m_device, &info, nullptr, &m_geometryBuffers[i])); + + + } +} + +void VulkanMeshObjectImpl::Reset() +{ + auto devicePtr = m_deviceHolder.lock(); + CHECK(devicePtr != nullptr, ()); + + for (auto b : m_geometryBuffers) + vkDestroyBuffer(devicePtr->m_device, b, nullptr); + + m_geometryBuffers.clear(); +} + +void VulkanMeshObjectImpl::UpdateBuffer(uint32_t bufferInd) +{ + CHECK_LESS(bufferInd, static_cast(m_geometryBuffers.size()), ()); + + auto & buffer = m_mesh->m_buffers[bufferInd]; + CHECK(!buffer.m_data.empty(), ()); + +// uint8_t * bufferPointer = (uint8_t *)[m_geometryBuffers[bufferInd] contents]; +// auto const sizeInBytes = buffer.m_data.size() * sizeof(buffer.m_data[0]); +// memcpy(bufferPointer, buffer.m_data.data(), sizeInBytes); +} + +void VulkanMeshObjectImpl::DrawPrimitives(ref_ptr context, uint32_t verticesCount) +{ + //TODO (@rokuz, @darina): Implement. + CHECK(false, ()); +} +} // namespace vulkan +} // namespace dp diff --git a/drape/vulkan/vulkan_mesh_object_impl.hpp b/drape/vulkan/vulkan_mesh_object_impl.hpp new file mode 100644 index 0000000000..ac84f15007 --- /dev/null +++ b/drape/vulkan/vulkan_mesh_object_impl.hpp @@ -0,0 +1,41 @@ +#pragma once + +#include "drape/mesh_object.hpp" +#include "drape/pointers.hpp" +#include "drape/vulkan/vulkan_device_holder.hpp" + +#include "base/assert.hpp" + +#include +#include + +#include +#include + +namespace dp +{ +namespace vulkan +{ +class VulkanMeshObjectImpl : public MeshObjectImpl +{ +public: + VulkanMeshObjectImpl(ref_ptr mesh) + : m_mesh(std::move(mesh)) + {} + + void Build(ref_ptr context, ref_ptr program) override; + void Reset() override; + + void UpdateBuffer(uint32_t bufferInd) override; + void Bind(ref_ptr program) override {} + void Unbind() override {} + + void DrawPrimitives(ref_ptr context, uint32_t verticesCount) override; + +private: + ref_ptr m_mesh; + DeviceHolderPtr m_deviceHolder; + std::vector m_geometryBuffers; +}; +} // namespace vulkan +} // namespace dp diff --git a/shaders/program_manager.cpp b/shaders/program_manager.cpp index 8bd2764e91..6b4057f5c1 100644 --- a/shaders/program_manager.cpp +++ b/shaders/program_manager.cpp @@ -71,7 +71,7 @@ void ProgramManager::InitForVulkan(ref_ptr context) { ASSERT(dynamic_cast(context.get()) != nullptr, ()); ref_ptr vulkanContext = context; - m_pool = make_unique_dp(vulkanContext->GetDevice()); + m_pool = make_unique_dp(vulkanContext->GetDeviceHolder()); m_paramsSetter = make_unique_dp(); } diff --git a/shaders/vulkan_program_pool.cpp b/shaders/vulkan_program_pool.cpp index f96765f547..49e33d8722 100644 --- a/shaders/vulkan_program_pool.cpp +++ b/shaders/vulkan_program_pool.cpp @@ -115,9 +115,11 @@ VkShaderModule LoadShaderModule(VkDevice device, std::string const & filename) } } // namespace -VulkanProgramPool::VulkanProgramPool(VkDevice device) - : m_device(device) +VulkanProgramPool::VulkanProgramPool(dp::vulkan::DeviceHolderPtr deviceHolder) + : m_deviceHolder(deviceHolder) { + auto devicePtr = m_deviceHolder.lock(); + CHECK(devicePtr != nullptr, ()); auto reflection = ReadReflectionFile(base::JoinPath(kShadersDir, kShadersReflecton)); CHECK_EQUAL(reflection.size(), static_cast(Program::ProgramsCount), ()); for (size_t i = 0; i < static_cast(Program::ProgramsCount); ++i) @@ -126,8 +128,8 @@ VulkanProgramPool::VulkanProgramPool(VkDevice device) m_programs[i] = make_unique_dp( programName, std::move(reflection[i]), - LoadShaderModule(device, base::JoinPath(kShadersDir, programName + ".vert.spv")), - LoadShaderModule(device, base::JoinPath(kShadersDir, programName + ".frag.spv"))); + LoadShaderModule(devicePtr->m_device, base::JoinPath(kShadersDir, programName + ".vert.spv")), + LoadShaderModule(devicePtr->m_device, base::JoinPath(kShadersDir, programName + ".frag.spv"))); } ProgramParams::Init(); @@ -137,12 +139,15 @@ VulkanProgramPool::~VulkanProgramPool() { ProgramParams::Destroy(); + auto devicePtr = m_deviceHolder.lock(); + CHECK(devicePtr != nullptr, ()); + for (auto & p : m_programs) { if (p != nullptr) { - vkDestroyShaderModule(m_device, p->GetVertexShader(), nullptr); - vkDestroyShaderModule(m_device, p->GetFragmentShader(), nullptr); + vkDestroyShaderModule(devicePtr->m_device, p->GetVertexShader(), nullptr); + vkDestroyShaderModule(devicePtr->m_device, p->GetFragmentShader(), nullptr); } } } diff --git a/shaders/vulkan_program_pool.hpp b/shaders/vulkan_program_pool.hpp index 63a42f00a5..477404f52d 100644 --- a/shaders/vulkan_program_pool.hpp +++ b/shaders/vulkan_program_pool.hpp @@ -2,6 +2,7 @@ #include "shaders/program_pool.hpp" +#include "drape/vulkan/vulkan_device_holder.hpp" #include "drape/vulkan/vulkan_gpu_program.hpp" #include "drape/pointers.hpp" @@ -17,13 +18,13 @@ namespace vulkan class VulkanProgramPool : public ProgramPool { public: - explicit VulkanProgramPool(VkDevice device); + explicit VulkanProgramPool(dp::vulkan::DeviceHolderPtr deviceHolder); ~VulkanProgramPool() override; drape_ptr Get(Program program) override; private: - VkDevice const m_device; + dp::vulkan::DeviceHolderPtr m_deviceHolder; std::array, static_cast(Program::ProgramsCount)> m_programs; };