diff --git a/drape/graphics_context.hpp b/drape/graphics_context.hpp index b8aef1c294..804bfcffdd 100644 --- a/drape/graphics_context.hpp +++ b/drape/graphics_context.hpp @@ -52,6 +52,7 @@ class GraphicsContext public: virtual ~GraphicsContext() = default; virtual bool BeginRendering() { return true; } + virtual void EndRendering() {} virtual void Present() = 0; virtual void MakeCurrent() = 0; virtual void DoneCurrent() {} diff --git a/drape/utils/profiler.hpp b/drape/utils/profiler.hpp new file mode 100644 index 0000000000..57604ebe28 --- /dev/null +++ b/drape/utils/profiler.hpp @@ -0,0 +1,93 @@ +#pragma once + +#include "base/logging.hpp" + +#include +#include +#include +#include +#include +#include +#include + +namespace dp +{ +static size_t constexpr kLastMeasurementsCount = 30; + +class Profiler +{ +public: + static Profiler & Instance() + { + static Profiler profiler; + return profiler; + } + + void Measure(std::string const & name, uint32_t milliseconds) + { + auto & m = m_measurements[name]; + m.m_accumulator += milliseconds; + m.m_lastMeasurements[m.m_count % kLastMeasurementsCount] = milliseconds; + m.m_count++; + } + + void Print() + { + LOG(LINFO, ("--- DRAPE PROFILER ---")); + + for (auto const & m : m_measurements) + { + float const avg = static_cast(m.second.m_accumulator) / m.second.m_count; + + size_t const startIndex = m.second.m_count % kLastMeasurementsCount; + std::ostringstream ss; + ss << "["; + for (size_t i = startIndex; i < std::min(kLastMeasurementsCount, m.second.m_count); ++i) + ss << m.second.m_lastMeasurements[i] << ", "; + for (size_t i = 0; i < startIndex; ++i) + ss << m.second.m_lastMeasurements[i] << ", "; + ss << ']'; + + LOG(LINFO, (">", m.first, ": avg time =", avg, "count =", m.second.m_count, "last =", ss.str())); + } + LOG(LINFO, ("----------------------")); + } + +protected: + struct Measurement + { + size_t m_count = 0; + uint64_t m_accumulator = 0; + std::array m_lastMeasurements = {}; + }; + std::map m_measurements; +}; + +class ProfilerGuard +{ +public: + ProfilerGuard(std::string const & name) + : m_name(name) + { + m_start = std::chrono::steady_clock::now(); + } + + ~ProfilerGuard() + { + auto const dur = std::chrono::steady_clock::now() - m_start; + auto ms = static_cast(std::chrono::duration_cast(dur).count()); + if (!m_skipped) + Profiler::Instance().Measure(m_name, ms); + } + + void Skip() + { + m_skipped = true; + } + +private: + std::string m_name; + std::chrono::time_point m_start; + bool m_skipped = false; +}; +} // namespace dp diff --git a/drape/vulkan/vulkan_base_context.cpp b/drape/vulkan/vulkan_base_context.cpp index 5914aa2cd2..334b079013 100644 --- a/drape/vulkan/vulkan_base_context.cpp +++ b/drape/vulkan/vulkan_base_context.cpp @@ -196,6 +196,48 @@ bool VulkanBaseContext::BeginRendering() return true; } +void VulkanBaseContext::EndRendering() +{ + CHECK(m_presentAvailable, ()); + + // Flushing of the default staging buffer must be before submitting the queue. + // It guarantees the graphics data coherence. + m_defaultStagingBuffer->Flush(); + + for (auto const & h : m_handlers[static_cast(HandlerType::PrePresent)]) + h.second(make_ref(this)); + + CHECK(m_isActiveRenderPass, ()); + m_isActiveRenderPass = false; + vkCmdEndRenderPass(m_renderingCommandBuffer); + + CHECK_VK_CALL(vkEndCommandBuffer(m_memoryCommandBuffer)); + CHECK_VK_CALL(vkEndCommandBuffer(m_renderingCommandBuffer)); + + VkSubmitInfo submitInfo = {}; + VkPipelineStageFlags const waitStageMask = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT; + VkCommandBuffer commandBuffers[] = {m_memoryCommandBuffer, m_renderingCommandBuffer}; + submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO; + submitInfo.pWaitDstStageMask = &waitStageMask; + submitInfo.pWaitSemaphores = &m_acquireComplete; + submitInfo.waitSemaphoreCount = 1; + submitInfo.pSignalSemaphores = &m_renderComplete; + submitInfo.signalSemaphoreCount = 1; + submitInfo.commandBufferCount = 2; + submitInfo.pCommandBuffers = commandBuffers; + + auto const res = vkQueueSubmit(m_queue, 1, &submitInfo, m_fence); + if (res != VK_ERROR_DEVICE_LOST) + { + m_needPresent = true; + CHECK_RESULT_VK_CALL(vkQueueSubmit, res); + } + else + { + m_needPresent = false; + } +} + void VulkanBaseContext::SetFramebuffer(ref_ptr framebuffer) { if (m_isActiveRenderPass) @@ -364,39 +406,8 @@ void VulkanBaseContext::ApplyFramebuffer(std::string const & framebufferLabel) void VulkanBaseContext::Present() { - // Flushing of the default staging buffer must be before submitting the queue. - // It guarantees the graphics data coherence. - m_defaultStagingBuffer->Flush(); - - for (auto const & h : m_handlers[static_cast(HandlerType::PrePresent)]) - h.second(make_ref(this)); - - CHECK(m_isActiveRenderPass, ()); - m_isActiveRenderPass = false; - vkCmdEndRenderPass(m_renderingCommandBuffer); - - CHECK_VK_CALL(vkEndCommandBuffer(m_memoryCommandBuffer)); - CHECK_VK_CALL(vkEndCommandBuffer(m_renderingCommandBuffer)); - - VkSubmitInfo submitInfo = {}; - VkPipelineStageFlags const waitStageMask = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT; - VkCommandBuffer commandBuffers[] = {m_memoryCommandBuffer, m_renderingCommandBuffer}; - submitInfo.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO; - submitInfo.pWaitDstStageMask = &waitStageMask; - submitInfo.pWaitSemaphores = &m_acquireComplete; - submitInfo.waitSemaphoreCount = 1; - submitInfo.pSignalSemaphores = &m_renderComplete; - submitInfo.signalSemaphoreCount = 1; - submitInfo.commandBufferCount = 2; - submitInfo.pCommandBuffers = commandBuffers; - - // For commands that wait indefinitely for device execution - // a return value of VK_ERROR_DEVICE_LOST is equivalent to VK_SUCCESS. - VkResult res = vkQueueSubmit(m_queue, 1, &submitInfo, m_fence); - if (res != VK_ERROR_DEVICE_LOST) + if (m_needPresent && m_presentAvailable) { - CHECK_RESULT_VK_CALL(vkQueueSubmit, res); - VkPresentInfoKHR presentInfo = {}; presentInfo.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR; presentInfo.pNext = nullptr; @@ -406,12 +417,12 @@ void VulkanBaseContext::Present() presentInfo.pWaitSemaphores = &m_renderComplete; presentInfo.waitSemaphoreCount = 1; - res = vkQueuePresentKHR(m_queue, &presentInfo); + auto const res = vkQueuePresentKHR(m_queue, &presentInfo); if (res != VK_SUCCESS && res != VK_SUBOPTIMAL_KHR && res != VK_ERROR_OUT_OF_DATE_KHR) CHECK_RESULT_VK_CALL(vkQueuePresentKHR, res); } - res = vkWaitForFences(m_device, 1, &m_fence, VK_TRUE, UINT64_MAX); + auto const res = vkWaitForFences(m_device, 1, &m_fence, VK_TRUE, UINT64_MAX); if (res != VK_SUCCESS && res != VK_ERROR_DEVICE_LOST) CHECK_RESULT_VK_CALL(vkWaitForFences, res); diff --git a/drape/vulkan/vulkan_base_context.hpp b/drape/vulkan/vulkan_base_context.hpp index cbd9449f3b..62dc2048db 100644 --- a/drape/vulkan/vulkan_base_context.hpp +++ b/drape/vulkan/vulkan_base_context.hpp @@ -40,6 +40,7 @@ public: using ContextHandler = std::function)>; bool BeginRendering() override; + void EndRendering() override; void Present() override; void CollectMemory() override; void DoneCurrent() override {} @@ -212,6 +213,7 @@ protected: drape_ptr m_defaultStagingBuffer; std::atomic m_presentAvailable; uint32_t m_frameCounter = 0; + bool m_needPresent = true; }; } // namespace vulkan } // namespace dp diff --git a/drape_frontend/frontend_renderer.cpp b/drape_frontend/frontend_renderer.cpp index a4e6cc509a..b6a8450c31 100755 --- a/drape_frontend/frontend_renderer.cpp +++ b/drape_frontend/frontend_renderer.cpp @@ -1603,6 +1603,7 @@ void FrontendRenderer::RenderEmptyFrame() m_context->ApplyFramebuffer("Empty frame"); m_viewport.Apply(m_context); + m_context->EndRendering(); m_context->Present(); } @@ -1655,6 +1656,8 @@ void FrontendRenderer::RenderFrame() RenderScene(modelView, isActiveFrameForScene); + m_context->EndRendering(); + auto const hasForceUpdate = m_forceUpdateScene || m_forceUpdateUserMarks; isActiveFrame |= hasForceUpdate; #if defined(SCENARIO_ENABLE)