diff --git a/drape/render_state.cpp b/drape/render_state.cpp index 18985d0114..7e134e6e67 100644 --- a/drape/render_state.cpp +++ b/drape/render_state.cpp @@ -229,6 +229,7 @@ void TextureState::ApplyTextures(ref_ptr context, RenderState c else if (apiVersion == dp::ApiVersion::Vulkan) { ref_ptr vulkanContext = context; + vulkanContext->ClearParamDescriptors(); ref_ptr p = program; auto const & bindings = p->GetTextureBindings(); for (auto const & texture : state.GetTextures()) @@ -311,7 +312,7 @@ void ApplyState(ref_ptr context, ref_ptr program, R { ASSERT_GREATER_OR_EQUAL(state.GetLineWidth(), 0, ()); ref_ptr vulkanContext = context; - VkCommandBuffer commandBuffer = vulkanContext->GetCurrentCommandBuffer(); + VkCommandBuffer commandBuffer = vulkanContext->GetCurrentRenderingCommandBuffer(); CHECK(commandBuffer != nullptr, ()); vkCmdSetLineWidth(commandBuffer, static_cast(state.GetLineWidth())); } diff --git a/drape/vulkan/vulkan_base_context.cpp b/drape/vulkan/vulkan_base_context.cpp index 655e1a0e86..3f72fb995c 100644 --- a/drape/vulkan/vulkan_base_context.cpp +++ b/drape/vulkan/vulkan_base_context.cpp @@ -43,7 +43,7 @@ VulkanBaseContext::~VulkanBaseContext() DestroyDepthTexture(); DestroySwapchain(); DestroyRenderPass(); - DestroyCommandBuffer(); + DestroyCommandBuffers(); DestroyCommandPool(); } @@ -87,7 +87,7 @@ void VulkanBaseContext::SetFramebuffer(ref_ptr framebuffer) { if (m_renderPass != VK_NULL_HANDLE) { - vkCmdEndRenderPass(m_commandBuffer); + vkCmdEndRenderPass(m_renderingCommandBuffer); DestroyRenderPass(); } @@ -100,9 +100,11 @@ void VulkanBaseContext::SetFramebuffer(ref_ptr framebuffer) void VulkanBaseContext::ApplyFramebuffer(std::string const & framebufferLabel) { //TODO: set renderPass to m_pipelineKey - vkCmdSetStencilReference(m_commandBuffer, VK_STENCIL_FRONT_AND_BACK, m_stencilReferenceValue); + vkCmdSetStencilReference(m_renderingCommandBuffer, VK_STENCIL_FRONT_AND_BACK, + m_stencilReferenceValue); CreateRenderPass(); + m_pipelineKey.m_renderPass = m_renderPass; VkClearValue clearValues[2]; clearValues[0].color = {1.0f, 0.0f, 1.0f, 1.0f}; @@ -127,7 +129,7 @@ void VulkanBaseContext::ApplyFramebuffer(std::string const & framebufferLabel) CreateDefaultFramebuffer(); renderPassBeginInfo.framebuffer = m_defaultFramebuffers[m_imageIndex]; } - vkCmdBeginRenderPass(m_commandBuffer, &renderPassBeginInfo, VK_SUBPASS_CONTENTS_INLINE); + vkCmdBeginRenderPass(m_renderingCommandBuffer, &renderPassBeginInfo, VK_SUBPASS_CONTENTS_INLINE); } void VulkanBaseContext::Init(ApiVersion apiVersion) @@ -156,13 +158,13 @@ void VulkanBaseContext::SetViewport(uint32_t x, uint32_t y, uint32_t w, uint32_t viewport.height = h; viewport.minDepth = 0.0f; viewport.maxDepth = 1.0f; - vkCmdSetViewport(GetCurrentCommandBuffer(), 0, 1, &viewport); + vkCmdSetViewport(m_renderingCommandBuffer, 0, 1, &viewport); VkRect2D scissor = {}; scissor.extent = {w, h}; scissor.offset.x = x; scissor.offset.y = y; - vkCmdSetScissor(GetCurrentCommandBuffer(), 0, 1, &scissor); + vkCmdSetScissor(m_renderingCommandBuffer, 0, 1, &scissor); } void VulkanBaseContext::SetSurface(VkSurfaceKHR surface, VkSurfaceFormatKHR surfaceFormat, @@ -176,14 +178,14 @@ void VulkanBaseContext::SetSurface(VkSurfaceKHR surface, VkSurfaceFormatKHR surf if (m_surfaceFormat.is_initialized()) { DestroyRenderPass(); - DestroyCommandBuffer(); + DestroyCommandBuffers(); DestroyCommandPool(); DestroyDepthTexture(); } m_surfaceFormat = surfaceFormat; m_surfaceCapabilities = surfaceCapabilities; CreateCommandPool(); - CreateCommandBuffer(); + CreateCommandBuffers(); CreateDepthTexture(); } RecreateSwapchain(); @@ -209,7 +211,8 @@ void VulkanBaseContext::BeginRendering() VkCommandBufferBeginInfo commandBufferBeginInfo = {}; commandBufferBeginInfo.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO; - CHECK_VK_CALL(vkBeginCommandBuffer(m_commandBuffer, &commandBufferBeginInfo)); + CHECK_VK_CALL(vkBeginCommandBuffer(m_memoryCommandBuffer, &commandBufferBeginInfo)); + CHECK_VK_CALL(vkBeginCommandBuffer(m_renderingCommandBuffer, &commandBufferBeginInfo)); // Prepare frame. Acquire next image. // By setting timeout to UINT64_MAX we will always wait until the next image has been acquired or an actual @@ -241,24 +244,26 @@ void VulkanBaseContext::Present() if (m_renderPass != VK_NULL_HANDLE) { - vkCmdEndRenderPass(m_commandBuffer); + vkCmdEndRenderPass(m_renderingCommandBuffer); DestroyRenderPass(); } - CHECK_VK_CALL(vkEndCommandBuffer(m_commandBuffer)); + CHECK_VK_CALL(vkEndCommandBuffer(m_memoryCommandBuffer)); + CHECK_VK_CALL(vkEndCommandBuffer(m_renderingCommandBuffer)); // Pipeline stage at which the queue submission will wait (via pWaitSemaphores) const VkPipelineStageFlags waitStageMask = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT; VkSubmitInfo submitInfo = {}; + VkCommandBuffer commandBuffers[] = {m_memoryCommandBuffer, m_renderingCommandBuffer}; submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO; submitInfo.pWaitDstStageMask = &waitStageMask; submitInfo.pWaitSemaphores = &m_presentComplete; submitInfo.waitSemaphoreCount = 1; submitInfo.pSignalSemaphores = &m_renderComplete; submitInfo.signalSemaphoreCount = 1; - submitInfo.commandBufferCount = 1; - submitInfo.pCommandBuffers = &m_commandBuffer; + submitInfo.commandBufferCount = 2; + submitInfo.pCommandBuffers = commandBuffers; CHECK_VK_CALL(vkQueueSubmit(m_queue, 1, &submitInfo, m_fence)); //CHECK_VK_CALL(vkQueueWaitIdle(m_queue)); @@ -440,7 +445,7 @@ void VulkanBaseContext::DestroyCommandPool() vkDestroyCommandPool(m_device, m_commandPool, nullptr); } -void VulkanBaseContext::CreateCommandBuffer() +void VulkanBaseContext::CreateCommandBuffers() { // A fence is need to check for command buffer completion before we can recreate it VkFenceCreateInfo fenceCI = {}; @@ -461,15 +466,17 @@ void VulkanBaseContext::CreateCommandBuffer() cmdBufAllocateInfo.commandPool = m_commandPool; cmdBufAllocateInfo.commandBufferCount = 1; cmdBufAllocateInfo.level = VK_COMMAND_BUFFER_LEVEL_PRIMARY; - CHECK_VK_CALL(vkAllocateCommandBuffers(m_device, &cmdBufAllocateInfo, &m_commandBuffer)); + CHECK_VK_CALL(vkAllocateCommandBuffers(m_device, &cmdBufAllocateInfo, &m_memoryCommandBuffer)); + CHECK_VK_CALL(vkAllocateCommandBuffers(m_device, &cmdBufAllocateInfo, &m_renderingCommandBuffer)); } -void VulkanBaseContext::DestroyCommandBuffer() +void VulkanBaseContext::DestroyCommandBuffers() { vkDestroyFence(m_device, m_fence, nullptr); vkDestroySemaphore(m_device, m_presentComplete, nullptr); vkDestroySemaphore(m_device, m_renderComplete, nullptr); - vkFreeCommandBuffers(m_device, m_commandPool, 1, &m_commandBuffer); + vkFreeCommandBuffers(m_device, m_commandPool, 1, &m_memoryCommandBuffer); + vkFreeCommandBuffers(m_device, m_commandPool, 1, &m_renderingCommandBuffer); } void VulkanBaseContext::CreateDepthTexture() @@ -698,12 +705,12 @@ void VulkanBaseContext::SetStencilReferenceValue(uint32_t stencilReferenceValue) { m_stencilReferenceValue = stencilReferenceValue; } - + void VulkanBaseContext::SetPrimitiveTopology(VkPrimitiveTopology topology) { m_pipelineKey.m_primitiveTopology = topology; } - + void VulkanBaseContext::SetBindingInfo(std::vector const & bindingInfo) { m_pipelineKey.m_bindingInfo = bindingInfo; @@ -713,17 +720,28 @@ void VulkanBaseContext::SetProgram(ref_ptr program) { m_pipelineKey.m_program = program; } - + void VulkanBaseContext::SetBlendingEnabled(bool blendingEnabled) { m_pipelineKey.m_blendingEnabled = blendingEnabled; } - + void VulkanBaseContext::ApplyParamDescriptor(ParamDescriptor && descriptor) { + if (descriptor.m_type == ParamDescriptor::Type::DynamicUniformBuffer) + { + for (auto & param : m_paramDescriptors) + { + if (param.m_type == ParamDescriptor::Type::DynamicUniformBuffer) + { + param = std::move(descriptor); + return; + } + } + } m_paramDescriptors.push_back(std::move(descriptor)); } - + void VulkanBaseContext::ClearParamDescriptors() { m_paramDescriptors.clear(); diff --git a/drape/vulkan/vulkan_base_context.hpp b/drape/vulkan/vulkan_base_context.hpp index abfc12725f..bbeb1cc0be 100644 --- a/drape/vulkan/vulkan_base_context.hpp +++ b/drape/vulkan/vulkan_base_context.hpp @@ -85,7 +85,9 @@ public: ref_ptr GetObjectManager() const { return m_objectManager; } - VkCommandBuffer GetCurrentCommandBuffer() const { return m_commandBuffer; } + VkCommandBuffer GetCurrentMemoryCommandBuffer() const { return m_memoryCommandBuffer; } + VkCommandBuffer GetCurrentRenderingCommandBuffer() const { return m_renderingCommandBuffer; } + VkPipeline GetCurrentPipeline(); DescriptorSetGroup GetCurrentDescriptorSetGroup(); VkPipelineLayout GetCurrentPipelineLayout() const; @@ -110,8 +112,8 @@ protected: void CreateCommandPool(); void DestroyCommandPool(); - void CreateCommandBuffer(); - void DestroyCommandBuffer(); + void CreateCommandBuffers(); + void DestroyCommandBuffers(); void CreateDepthTexture(); void DestroyDepthTexture(); @@ -133,7 +135,8 @@ protected: VkQueue m_queue = {}; VkCommandPool m_commandPool = {}; - VkCommandBuffer m_commandBuffer = {}; + VkCommandBuffer m_renderingCommandBuffer = {}; + VkCommandBuffer m_memoryCommandBuffer = {}; VkRenderPass m_renderPass = {}; std::vector m_renderPassesToDestroy; @@ -163,6 +166,9 @@ protected: ref_ptr m_currentFramebuffer; + VkAttachmentDescription m_colorAttachment; + VkAttachmentDescription m_depthAttachment; + std::array>, static_cast(HandlerType::Count)> m_handlers; diff --git a/drape/vulkan/vulkan_gpu_buffer_impl.cpp b/drape/vulkan/vulkan_gpu_buffer_impl.cpp index a444f26980..23c520fac6 100644 --- a/drape/vulkan/vulkan_gpu_buffer_impl.cpp +++ b/drape/vulkan/vulkan_gpu_buffer_impl.cpp @@ -29,7 +29,7 @@ void * VulkanGPUBuffer::Map(ref_ptr context, uint32_t element uint32_t const mappingSizeInBytes = elementCount * elementSize; m_mappingByteOffset = elementOffset * elementSize; - VkCommandBuffer commandBuffer = context->GetCurrentCommandBuffer(); + VkCommandBuffer commandBuffer = context->GetCurrentMemoryCommandBuffer(); CHECK(commandBuffer != nullptr, ()); // Copy to default or temporary staging buffer. @@ -79,7 +79,7 @@ void VulkanGPUBuffer::UpdateData(void * gpuPtr, void const * data, void VulkanGPUBuffer::Unmap(ref_ptr context) { - VkCommandBuffer commandBuffer = context->GetCurrentCommandBuffer(); + VkCommandBuffer commandBuffer = context->GetCurrentMemoryCommandBuffer(); CHECK(commandBuffer != nullptr, ()); VkBuffer stagingBuffer = m_stagingBufferRef->GetReservationById(m_reservationId).m_stagingBuffer; @@ -117,17 +117,17 @@ void VulkanGPUBuffer::Resize(ref_ptr context, void const * da m_geometryBuffer = m_objectManager->CreateBuffer(VulkanMemoryManager::ResourceType::Geometry, sizeInBytes, 0 /* batcherHash */); + void * gpuPtr = m_objectManager->Map(m_geometryBuffer); if (data != nullptr) { - void * gpuPtr = m_objectManager->Map(m_geometryBuffer); memcpy(gpuPtr, data, sizeInBytes); m_objectManager->Flush(m_geometryBuffer); - m_objectManager->Unmap(m_geometryBuffer); } - CHECK_VK_CALL(vkBindBufferMemory(device, m_geometryBuffer.m_buffer, m_geometryBuffer.GetMemory(), m_geometryBuffer.GetAlignedOffset())); + m_objectManager->Unmap(m_geometryBuffer); + // If we have already set up data, we have to call SetDataSize. if (data != nullptr) SetDataSize(elementCount); diff --git a/drape/vulkan/vulkan_mesh_object_impl.cpp b/drape/vulkan/vulkan_mesh_object_impl.cpp index 6a1c6dbbb4..3dbd446c68 100644 --- a/drape/vulkan/vulkan_mesh_object_impl.cpp +++ b/drape/vulkan/vulkan_mesh_object_impl.cpp @@ -59,11 +59,11 @@ public: void * gpuPtr = m_objectManager->Map(m_geometryBuffers[i]); memcpy(gpuPtr, m_mesh->m_buffers[i].m_data.data(), sizeInBytes); m_objectManager->Flush(m_geometryBuffers[i]); - m_objectManager->Unmap(m_geometryBuffers[i]); CHECK_VK_CALL(vkBindBufferMemory(device, m_geometryBuffers[i].m_buffer, m_geometryBuffers[i].GetMemory(), m_geometryBuffers[i].GetAlignedOffset())); + m_objectManager->Unmap(m_geometryBuffers[i]); m_bindingInfo[i] = dp::BindingInfo(static_cast(m_mesh->m_buffers[i].m_attributes.size()), static_cast(i)); @@ -94,7 +94,7 @@ public: CHECK_LESS(bufferInd, static_cast(m_geometryBuffers.size()), ()); ref_ptr vulkanContext = context; - VkCommandBuffer commandBuffer = vulkanContext->GetCurrentCommandBuffer(); + VkCommandBuffer commandBuffer = vulkanContext->GetCurrentMemoryCommandBuffer(); CHECK(commandBuffer != nullptr, ()); auto & buffer = m_mesh->m_buffers[bufferInd]; @@ -154,7 +154,7 @@ public: void DrawPrimitives(ref_ptr context, uint32_t verticesCount) override { ref_ptr vulkanContext = context; - VkCommandBuffer commandBuffer = vulkanContext->GetCurrentCommandBuffer(); + VkCommandBuffer commandBuffer = vulkanContext->GetCurrentRenderingCommandBuffer(); CHECK(commandBuffer != nullptr, ()); vulkanContext->SetPrimitiveTopology(GetPrimitiveType(m_mesh->m_drawPrimitive)); @@ -176,8 +176,6 @@ public: vkCmdBindVertexBuffers(commandBuffer, i, 1, &m_geometryBuffers[i].m_buffer, offsets); vkCmdDraw(commandBuffer, verticesCount, 1, 0, 0); - - vulkanContext->ClearParamDescriptors(); } void Bind(ref_ptr program) override {} diff --git a/drape/vulkan/vulkan_object_manager.cpp b/drape/vulkan/vulkan_object_manager.cpp index b72683599c..0c0b0a702b 100644 --- a/drape/vulkan/vulkan_object_manager.cpp +++ b/drape/vulkan/vulkan_object_manager.cpp @@ -318,8 +318,8 @@ void VulkanObjectManager::CreateDescriptorPool() { std::vector poolSizes = { - {VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC, 1}, // Maximum uniform buffers count per draw call. - {VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER, 3}, // Maximum textures count per draw call. + {VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC, 500}, // Maximum uniform buffers count per draw call. + {VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER, 1000}, // Maximum textures count per draw call. }; VkDescriptorPoolCreateInfo descriptorPoolInfo = {}; @@ -327,7 +327,7 @@ void VulkanObjectManager::CreateDescriptorPool() descriptorPoolInfo.flags = VK_DESCRIPTOR_POOL_CREATE_FREE_DESCRIPTOR_SET_BIT; descriptorPoolInfo.poolSizeCount = static_cast(poolSizes.size()); descriptorPoolInfo.pPoolSizes = poolSizes.data(); - descriptorPoolInfo.maxSets = 500; // Approximately equal to doubled maximum VAO per frame. + descriptorPoolInfo.maxSets = 1500; // Approximately equal to doubled maximum VAO per frame. VkDescriptorPool descriptorPool; CHECK_VK_CALL(vkCreateDescriptorPool(m_device, &descriptorPoolInfo, nullptr, &descriptorPool)); diff --git a/drape/vulkan/vulkan_pipeline.cpp b/drape/vulkan/vulkan_pipeline.cpp index 7cc42d2f88..05db6d34cc 100644 --- a/drape/vulkan/vulkan_pipeline.cpp +++ b/drape/vulkan/vulkan_pipeline.cpp @@ -303,7 +303,7 @@ VkPipeline VulkanPipeline::GetPipeline(VkDevice device, PipelineKey const & key) size_t attribsCount = 0; for (size_t i = 0; i < key.m_bindingInfo.size(); ++i) { - bindingDescriptions[i].binding = key.m_bindingInfo[i].GetID(); + bindingDescriptions[i].binding = static_cast(i); bindingDescriptions[i].stride = key.m_bindingInfo[i].GetElementSize(); bindingDescriptions[i].inputRate = VK_VERTEX_INPUT_RATE_VERTEX; attribsCount += key.m_bindingInfo[i].GetCount(); @@ -316,8 +316,8 @@ VkPipeline VulkanPipeline::GetPipeline(VkDevice device, PipelineKey const & key) for (uint8_t j = 0; j < key.m_bindingInfo[i].GetCount(); ++j) { BindingDecl const & bindingDecl = key.m_bindingInfo[i].GetBindingDecl(j); - attributeDescriptions[bindingCounter].location = static_cast(i); - attributeDescriptions[bindingCounter].binding = bindingCounter; + attributeDescriptions[bindingCounter].location = bindingCounter; + attributeDescriptions[bindingCounter].binding = static_cast(i); attributeDescriptions[bindingCounter].format = GetAttributeFormat(bindingDecl.m_componentCount, bindingDecl.m_componentType); attributeDescriptions[bindingCounter].offset = bindingDecl.m_offset; diff --git a/drape/vulkan/vulkan_texture.cpp b/drape/vulkan/vulkan_texture.cpp index 5e2c6c4b51..30b754689d 100644 --- a/drape/vulkan/vulkan_texture.cpp +++ b/drape/vulkan/vulkan_texture.cpp @@ -174,7 +174,7 @@ void VulkanTexture::UploadData(ref_ptr context, uint32_t x, CHECK(data != nullptr, ()); ref_ptr vulkanContext = context; - VkCommandBuffer commandBuffer = vulkanContext->GetCurrentCommandBuffer(); + VkCommandBuffer commandBuffer = vulkanContext->GetCurrentMemoryCommandBuffer(); CHECK(commandBuffer != nullptr, ()); auto const sizeInBytes = GetBytesPerPixel(GetFormat()) * width * height; @@ -223,7 +223,7 @@ void VulkanTexture::UploadData(ref_ptr context, uint32_t x, void VulkanTexture::Bind(ref_ptr context) const { ref_ptr vulkanContext = context; - VkCommandBuffer commandBuffer = vulkanContext->GetCurrentCommandBuffer(); + VkCommandBuffer commandBuffer = vulkanContext->GetCurrentMemoryCommandBuffer(); CHECK(commandBuffer != nullptr, ()); // Fill texture on the first bind. diff --git a/drape/vulkan/vulkan_vertex_array_buffer_impl.cpp b/drape/vulkan/vulkan_vertex_array_buffer_impl.cpp index 786e4c2e30..139e9d96af 100644 --- a/drape/vulkan/vulkan_vertex_array_buffer_impl.cpp +++ b/drape/vulkan/vulkan_vertex_array_buffer_impl.cpp @@ -53,7 +53,7 @@ public: IndicesRange const & range) override { ref_ptr vulkanContext = context; - VkCommandBuffer commandBuffer = vulkanContext->GetCurrentCommandBuffer(); + VkCommandBuffer commandBuffer = vulkanContext->GetCurrentRenderingCommandBuffer(); CHECK(commandBuffer != nullptr, ()); vulkanContext->SetPrimitiveTopology(drawAsLine ? VK_PRIMITIVE_TOPOLOGY_LINE_LIST : @@ -94,8 +94,6 @@ public: vkCmdBindIndexBuffer(commandBuffer, vulkanIndexBuffer, 0, indexType); vkCmdDrawIndexed(commandBuffer, range.m_idxCount, 1, range.m_idxStart, 0, 0); - - vulkanContext->ClearParamDescriptors(); } private: