From 0196e79dc5ded496a5a33caae876d7910dee9e3e Mon Sep 17 00:00:00 2001 From: "r.kuznetsov" Date: Thu, 10 Jan 2019 14:30:56 +0300 Subject: [PATCH] [Vulkan] Added checking max textures and lines width --- .../vulkan/android_vulkan_context_factory.cpp | 3 ++ drape/render_state.cpp | 6 ++-- drape/support_manager.cpp | 36 ++++++++++++------- drape/support_manager.hpp | 17 +++++++-- drape/texture.cpp | 6 ---- drape/texture.hpp | 1 - drape/texture_manager.cpp | 10 ++---- drape/vulkan/vulkan_base_context.cpp | 2 ++ drape/vulkan/vulkan_base_context.hpp | 2 +- drape_frontend/backend_renderer.cpp | 2 ++ drape_frontend/line_shape.cpp | 2 +- drape_frontend/traffic_renderer.cpp | 2 +- 12 files changed, 55 insertions(+), 34 deletions(-) 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 fe4dc25cd8..56291fe28d 100644 --- a/android/jni/com/mapswithme/vulkan/android_vulkan_context_factory.cpp +++ b/android/jni/com/mapswithme/vulkan/android_vulkan_context_factory.cpp @@ -178,7 +178,10 @@ AndroidVulkanContextFactory::~AndroidVulkanContextFactory() m_layers->Uninitialize(m_vulkanInstance); if (m_device != nullptr) + { + vkDeviceWaitIdle(m_device); vkDestroyDevice(m_device, nullptr); + } if (m_vulkanInstance != nullptr) vkDestroyInstance(m_vulkanInstance, nullptr); diff --git a/drape/render_state.cpp b/drape/render_state.cpp index 749ee03729..93ab82baa8 100644 --- a/drape/render_state.cpp +++ b/drape/render_state.cpp @@ -283,7 +283,9 @@ void ApplyState(ref_ptr context, ref_ptr program, R ASSERT_GREATER_OR_EQUAL(state.GetLineWidth(), 0, ()); GLFunctions::glLineWidth(static_cast(state.GetLineWidth())); } - - //TODO(@rokuz, @darina): Check if Vulkan support line width. + else if (apiVersion == dp::ApiVersion::Vulkan) + { + //TODO(@rokuz, @darina): Implement. + } } } // namespace dp diff --git a/drape/support_manager.cpp b/drape/support_manager.cpp index c12ccec6cf..3e8ab410c0 100644 --- a/drape/support_manager.cpp +++ b/drape/support_manager.cpp @@ -1,5 +1,6 @@ #include "drape/support_manager.hpp" #include "drape/gl_functions.hpp" +#include "drape/vulkan/vulkan_base_context.hpp" #include "platform/settings.hpp" @@ -19,17 +20,15 @@ char const * kSupportedAntialiasing = "Antialiasing"; void SupportManager::Init(ref_ptr context) { + std::lock_guard lock(m_mutex); + if (m_isInitialized) + return; + std::string const renderer = context->GetRendererName(); std::string const version = context->GetRendererVersion(); LOG(LINFO, ("Renderer =", renderer, "| Api =", context->GetApiVersion(), "| Version =", version)); - // On Android the engine may be recreated. Here we guarantee that GPU info is sent once per session. - static bool gpuInfoSent = false; - if (!gpuInfoSent) - { - alohalytics::Stats::Instance().LogEvent("GPU", renderer); - gpuInfoSent = true; - } + alohalytics::Stats::Instance().LogEvent("GPU", renderer); m_isSamsungGoogleNexus = (renderer == "PowerVR SGX 540" && version.find("GOOGLENEXUS.ED945322") != string::npos); if (m_isSamsungGoogleNexus) @@ -53,15 +52,26 @@ void SupportManager::Init(ref_ptr context) if (m_isTegra) LOG(LINFO, ("NVidia Tegra device detected.")); - // Metal does not support thick lines. auto const apiVersion = context->GetApiVersion(); if (apiVersion == dp::ApiVersion::OpenGLES2 || apiVersion == dp::ApiVersion::OpenGLES3) { - m_maxLineWidth = std::max(1, GLFunctions::glGetMaxLineWidth()); - LOG(LINFO, ("Max line width =", m_maxLineWidth)); + m_maxLineWidth = static_cast(std::max(1, GLFunctions::glGetMaxLineWidth())); + m_maxTextureSize = static_cast(GLFunctions::glGetInteger(gl_const::GLMaxTextureSize)); } - - //TODO(@rokuz, @darina): Check if Vulkan support line width. + else if (apiVersion == dp::ApiVersion::Metal) + { + // Metal does not support thick lines. + m_maxLineWidth = 1.0f; + m_maxTextureSize = 4096; + } + else if (apiVersion == dp::ApiVersion::Vulkan) + { + ref_ptr vulkanContext = context; + auto const & props = vulkanContext->GetGpuProperties(); + m_maxLineWidth = std::max(props.limits.lineWidthRange[0], props.limits.lineWidthRange[1]); + m_maxTextureSize = vulkanContext->GetGpuProperties().limits.maxImageDimension2D; + } + LOG(LINFO, ("Max line width =", m_maxLineWidth,"| Max texture size =", m_maxTextureSize)); // Set up default antialiasing value. // Turn off AA for a while by energy-saving issues. @@ -78,6 +88,8 @@ void SupportManager::Init(ref_ptr context) //#endif // settings::Set(kSupportedAntialiasing, m_isAntialiasingEnabledByDefault); // } + + m_isInitialized = true; } SupportManager & SupportManager::Instance() diff --git a/drape/support_manager.hpp b/drape/support_manager.hpp index 71b933a458..f12a997880 100644 --- a/drape/support_manager.hpp +++ b/drape/support_manager.hpp @@ -5,6 +5,9 @@ #include "base/macros.hpp" +#include +#include + namespace dp { extern char const * kSupportedAntialiasing; @@ -16,23 +19,33 @@ public: static SupportManager & Instance(); // Initialization must be called only when graphics context is created. + // Initialization happens once per application launch, so SupportManager + // must not contain any properties which can be changed in the case of contexts + // reinitialization. void Init(ref_ptr context); bool IsSamsungGoogleNexus() const { return m_isSamsungGoogleNexus; } bool IsAdreno200Device() const { return m_isAdreno200; } bool IsTegraDevice() const { return m_isTegra; } - int GetMaxLineWidth() const { return m_maxLineWidth; } bool IsAntialiasingEnabledByDefault() const { return m_isAntialiasingEnabledByDefault; } + float GetMaxLineWidth() const { return m_maxLineWidth; } + uint32_t GetMaxTextureSize() const { return m_maxTextureSize; } + private: SupportManager() = default; bool m_isSamsungGoogleNexus = false; bool m_isAdreno200 = false; bool m_isTegra = false; - int m_maxLineWidth = 1; bool m_isAntialiasingEnabledByDefault = false; + float m_maxLineWidth = 1; + uint32_t m_maxTextureSize = 1024; + + bool m_isInitialized = false; + std::mutex m_mutex; + DISALLOW_COPY_AND_MOVE(SupportManager); }; } // namespace dp diff --git a/drape/texture.cpp b/drape/texture.cpp index 5a3d43e6fa..a74c15b5fb 100644 --- a/drape/texture.cpp +++ b/drape/texture.cpp @@ -84,12 +84,6 @@ void Texture::SetFilter(TextureFilter filter) m_hwTexture->SetFilter(filter); } -// static -uint32_t Texture::GetMaxTextureSize() -{ - return static_cast(GLFunctions::glGetInteger(gl_const::GLMaxTextureSize)); -} - // static bool Texture::IsPowerOfTwo(uint32_t width, uint32_t height) { diff --git a/drape/texture.hpp b/drape/texture.hpp index e3f62610b8..aede9bbfcb 100644 --- a/drape/texture.hpp +++ b/drape/texture.hpp @@ -72,7 +72,6 @@ public: ref_ptr GetHardwareTexture() const; - static uint32_t GetMaxTextureSize(); static bool IsPowerOfTwo(uint32_t width, uint32_t height); protected: diff --git a/drape/texture_manager.cpp b/drape/texture_manager.cpp index a8fdd98369..2276b9e807 100644 --- a/drape/texture_manager.cpp +++ b/drape/texture_manager.cpp @@ -5,6 +5,7 @@ #include "drape/stipple_pen_resource.hpp" #include "drape/texture_of_colors.hpp" #include "drape/gl_functions.hpp" +#include "drape/support_manager.hpp" #include "drape/utils/glyph_usage_tracker.hpp" #include "platform/platform.hpp" @@ -477,17 +478,10 @@ void TextureManager::Init(ref_ptr context, Params const & p m_resPostfix = params.m_resPostfix; m_textureAllocator = CreateAllocator(context); + m_maxTextureSize = std::min(kMaxTextureSize, dp::SupportManager::Instance().GetMaxTextureSize()); auto const apiVersion = context->GetApiVersion(); if (apiVersion == dp::ApiVersion::OpenGLES2 || apiVersion == dp::ApiVersion::OpenGLES3) - { - m_maxTextureSize = std::min(kMaxTextureSize, - static_cast(GLFunctions::glGetInteger(gl_const::GLMaxTextureSize))); GLFunctions::glPixelStore(gl_const::GLUnpackAlignment, 1); - } - else - { - m_maxTextureSize = kMaxTextureSize; - } // Initialize symbols. for (size_t i = 0; i < ARRAY_SIZE(kSymbolTextures); ++i) diff --git a/drape/vulkan/vulkan_base_context.cpp b/drape/vulkan/vulkan_base_context.cpp index 2c00cc9e4f..b7dcb234c4 100644 --- a/drape/vulkan/vulkan_base_context.cpp +++ b/drape/vulkan/vulkan_base_context.cpp @@ -50,6 +50,8 @@ void VulkanBaseContext::SetSurface(VkSurfaceKHR surface, VkFormat surfaceFormat, void VulkanBaseContext::ResetSurface() { + vkDeviceWaitIdle(m_device); + //TODO: reset swapchains, image views and so on. m_surface.reset(); } diff --git a/drape/vulkan/vulkan_base_context.hpp b/drape/vulkan/vulkan_base_context.hpp index a41d09acf9..e2e5552ed1 100644 --- a/drape/vulkan/vulkan_base_context.hpp +++ b/drape/vulkan/vulkan_base_context.hpp @@ -54,6 +54,7 @@ public: void ResetSurface(); VkDevice GetDevice() const { return m_device; } + VkPhysicalDeviceProperties const & GetGpuProperties() const { return m_gpuProperties; } protected: VkInstance const m_vulkanInstance; @@ -63,7 +64,6 @@ protected: VkPhysicalDeviceProperties m_gpuProperties; boost::optional m_surface; - m2::PointU m_maxTextureSize; uint32_t m_stencilReferenceValue = 1; }; diff --git a/drape_frontend/backend_renderer.cpp b/drape_frontend/backend_renderer.cpp index 7f1313956c..16d1267204 100644 --- a/drape_frontend/backend_renderer.cpp +++ b/drape_frontend/backend_renderer.cpp @@ -13,6 +13,7 @@ #include "drape_frontend/user_mark_shapes.hpp" #include "drape_frontend/visual_params.hpp" +#include "drape/support_manager.hpp" #include "drape/texture_manager.hpp" #include "indexer/scales.hpp" @@ -616,6 +617,7 @@ void BackendRenderer::OnContextCreate() m_contextFactory->WaitForInitialization(m_context.get()); m_context->MakeCurrent(); m_context->Init(m_apiVersion); + dp::SupportManager::Instance().Init(m_context); //TODO(@rokuz, @darina): TEMPORARY return; diff --git a/drape_frontend/line_shape.cpp b/drape_frontend/line_shape.cpp index 3cfc3f09c0..772318b00c 100644 --- a/drape_frontend/line_shape.cpp +++ b/drape_frontend/line_shape.cpp @@ -472,7 +472,7 @@ bool LineShape::CanBeSimplified(int & lineWidth) const if (m_params.m_zoomLevel > 0 && m_params.m_zoomLevel <= scales::GetUpperCountryScale()) return false; - static float width = std::min(2.5f, static_cast(dp::SupportManager::Instance().GetMaxLineWidth())); + static float width = std::min(2.5f, dp::SupportManager::Instance().GetMaxLineWidth()); if (m_params.m_width <= width) { lineWidth = std::max(1, static_cast(m_params.m_width)); diff --git a/drape_frontend/traffic_renderer.cpp b/drape_frontend/traffic_renderer.cpp index d5acd85593..29ca8f2873 100644 --- a/drape_frontend/traffic_renderer.cpp +++ b/drape_frontend/traffic_renderer.cpp @@ -318,7 +318,7 @@ bool TrafficRenderer::CanBeRenderedAsLine(RoadClass const & roadClass, int zoomL if (it == lineDrawer->end()) return false; - width = max(1, base::rounds(TrafficRenderer::GetPixelWidthInternal(roadClass, zoomLevel))); + width = std::max(1, base::rounds(TrafficRenderer::GetPixelWidthInternal(roadClass, zoomLevel))); return width <= dp::SupportManager::Instance().GetMaxLineWidth(); } } // namespace df