diff --git a/android/jni/Android.mk b/android/jni/Android.mk index c06ec7c981..07f370c7fd 100644 --- a/android/jni/Android.mk +++ b/android/jni/Android.mk @@ -37,7 +37,6 @@ LOCAL_SRC_FILES := \ com/mapswithme/maps/DownloadUI.cpp \ com/mapswithme/maps/Framework.cpp \ com/mapswithme/maps/VideoTimer.cpp \ - com/mapswithme/maps/GesturesProcessor.cpp \ com/mapswithme/maps/MWMActivity.cpp \ com/mapswithme/maps/Lifecycle.cpp \ com/mapswithme/platform/Platform.cpp \ diff --git a/android/jni/com/mapswithme/maps/Framework.cpp b/android/jni/com/mapswithme/maps/Framework.cpp index dcb3607434..e80f4b3dd1 100644 --- a/android/jni/com/mapswithme/maps/Framework.cpp +++ b/android/jni/com/mapswithme/maps/Framework.cpp @@ -61,27 +61,23 @@ namespace android { struct make_all_invalid { - size_t m_threadCount; - - make_all_invalid(size_t threadCount) - : m_threadCount(threadCount) + make_all_invalid() {} void operator() (int, int, int, drule::BaseRule * p) { - for (size_t threadID = 0; threadID < m_threadCount; ++threadID) - { - p->MakeEmptyID(threadID); - p->MakeEmptyID2(threadID); - } + p->MakeEmptyID(); + p->MakeEmptyID2(); } }; } void Framework::DeleteRenderPolicy() { - drule::rules().ForEachRule(make_all_invalid(GetPlatform().CpuCores() + 1)); + LOG(LINFO, ("clearing current render policy.")); m_work.SetRenderPolicy(0); + LOG(LINFO, ("cleaning all cached ruleID values.")); + drule::rules().ForEachRule(make_all_invalid()); } void Framework::InitRenderPolicy() @@ -95,16 +91,13 @@ namespace android rmParams.m_videoMemoryLimit = 15 * 1024 * 1024; rmParams.m_rtFormat = yg::Rt8Bpp; - m_work.SetRenderPolicy(new PartialRenderPolicy(m_videoTimer, params, rmParams, make_shared_ptr(new android::RenderContext()))); + m_work.SetRenderPolicy(new PartialRenderPolicy(m_videoTimer, + params, + rmParams, + make_shared_ptr(new android::RenderContext()))); m_work.SetUpdatesEnabled(true); - DrawFrame(); - - LOG(LDEBUG, ("AF::InitRenderer 2")); - - m_work.ShowAll(); - LOG(LDEBUG, ("AF::InitRenderer 3")); } @@ -294,4 +287,22 @@ namespace android info.m_accuracy = accuracy; m_work.OnCompassUpdate(info); } + + void Framework::LoadState() + { + if (!m_work.LoadState()) + { + LOG(LINFO, ("no saved state, showing all world")); + m_work.ShowAll(); + } + else + { + LOG(LINFO, ("state loaded successfully")); + } + } + + void Framework::SaveState() + { + m_work.SaveState(); + } } diff --git a/android/jni/com/mapswithme/maps/Framework.hpp b/android/jni/com/mapswithme/maps/Framework.hpp index 1ce9f4a9bb..9c34a269b7 100644 --- a/android/jni/com/mapswithme/maps/Framework.hpp +++ b/android/jni/com/mapswithme/maps/Framework.hpp @@ -58,6 +58,9 @@ namespace android void Zoom(int mode, double x1, double y1, double x2, double y2); void Touch(int action, int mask, double x1, double y1, double x2, double y2); + void LoadState(); + void SaveState(); + void EnableLocation(bool enable); void UpdateLocation(uint64_t timestamp, double lat, double lon, float accuracy); void UpdateCompass(uint64_t timestamp, double magneticNorth, double trueNorth, float accuracy); diff --git a/android/jni/com/mapswithme/maps/GesturesProcessor.cpp b/android/jni/com/mapswithme/maps/GesturesProcessor.cpp deleted file mode 100644 index 24818c699f..0000000000 --- a/android/jni/com/mapswithme/maps/GesturesProcessor.cpp +++ /dev/null @@ -1,29 +0,0 @@ -/* - * GesturesProcessor.cpp - * - * Created on: Oct 13, 2011 - * Author: siarheirachytski - */ - -#include -#include "Framework.hpp" -#include "../../../../../base/assert.hpp" - -extern "C" -{ - JNIEXPORT void JNICALL - Java_com_mapswithme_maps_GesturesProcessor_nativeMove(JNIEnv * env, - jobject thiz, jint mode, jdouble x, jdouble y) - { - ASSERT ( g_framework, () ); - g_framework->Move(mode, x, y); - } - - JNIEXPORT void JNICALL - Java_com_mapswithme_maps_GesturesProcessor_nativeZoom(JNIEnv * env, - jobject thiz, jint mode, jdouble x1, jdouble y1, jdouble x2, jdouble y2) - { - ASSERT ( g_framework, () ); - g_framework->Zoom(mode, x1, y1, x2, y2); - } -} diff --git a/android/jni/com/mapswithme/maps/Lifecycle.cpp b/android/jni/com/mapswithme/maps/Lifecycle.cpp index bea83bbe3c..df45c4b426 100644 --- a/android/jni/com/mapswithme/maps/Lifecycle.cpp +++ b/android/jni/com/mapswithme/maps/Lifecycle.cpp @@ -18,6 +18,7 @@ #include "../../../nv_event/nv_event.hpp" #include "../../../nv_thread/nv_thread.hpp" #include "../../../../../base/logging.hpp" +#include "../../../../../yg/internal/opengl.hpp" #include "Framework.hpp" #include "../platform/Platform.hpp" @@ -31,12 +32,11 @@ static unsigned int s_swapCount = 0; static bool s_glesLoaded = false; static bool s_glesAutopaused = false; +static bool shouldLoadState = true; static bool renderGameplay() { g_framework->DrawFrame(); - //glClearColor(0.0f, 1.0f, 1.0f, 1.0f); - //glClear(GL_COLOR_BUFFER_BIT); return true; //return true to update screen } @@ -57,7 +57,6 @@ bool SetupGLESResources() NVDEBUG(reinterpret_cast(glGetString(GL_VERSION))); // NVDEBUG(reinterpret_cast(glGetString(GL_SHADING_LANGUAGE_VERSION))); - g_framework->DeleteRenderPolicy(); g_framework->InitRenderPolicy(); s_glesLoaded = true; @@ -76,6 +75,15 @@ bool ShutdownGLESResources() if (!NVEventStatusEGLIsBound()) { NVDEBUG("ShutdownGLESResources: GLES not bound, shutting down EGL to release"); + + yg::gl::g_doDeleteOnDestroy = false; + + g_framework->DeleteRenderPolicy(); + + yg::gl::g_doDeleteOnDestroy = true; + + NVDEBUG("Cleaning up EGL"); + if (NVEventCleanupEGL()) { s_glesLoaded = false; @@ -89,7 +97,7 @@ bool ShutdownGLESResources() NVDEBUG("ShutdownGLESResources event: GLES bound, manually deleting GLES resources"); - g_framework->DeleteRenderPolicy();/// delete RenderPolicy and ResourceManager here + g_framework->DeleteRenderPolicy(); s_glesLoaded = false; @@ -244,6 +252,11 @@ int32_t NVEventAppMain(int32_t argc, char** argv) s_winHeight = ev->m_data.m_size.m_h; g_framework->Resize(s_winWidth, s_winHeight); + /* if (shouldLoadState) + { + g_framework->LoadState(); + shouldLoadState = false; + }*/ NVDEBUG( "Surface create/resize event: %d x %d", s_winWidth, s_winHeight); @@ -259,26 +272,59 @@ int32_t NVEventAppMain(int32_t argc, char** argv) case NV_EVENT_FOCUS_LOST: NVDEBUG("Focus lost event"); // s_glesAutopaused = true; + /// need to investigate deeper, whether i could call it only once in a sequence of lifecycle events +/* if (g_framework) + { + NVDEBUG("saving state"); + g_framework->SaveState(); + }*/ + renderFrame(false); break; case NV_EVENT_PAUSE: NVDEBUG("Pause event"); + +/* /// need to investigate deeper, whether i could call it only once in a sequence of lifecycle events + if (g_framework) + { + NVDEBUG("saving state"); + g_framework->SaveState(); + }*/ + //s_glesAutopaused = true; renderFrame(false); break; case NV_EVENT_STOP: NVDEBUG("Stop event"); + +/* /// need to investigate deeper, whether i could call it only once in a sequence of lifecycle events + if (g_framework) + { + NVDEBUG("saving state"); + g_framework->SaveState(); + }*/ + // As per Google's recommendation, we release GLES resources here ShutdownGLESResources(); + break; + case NV_EVENT_QUIT: + NVDEBUG("Quit event"); + +/* /// need to investigate deeper, whether i could call it only once in a sequence of lifecycle events + if (g_framework) + { + NVDEBUG("saving state"); + g_framework->SaveState(); + }*/ + break; case NV_EVENT_ACCEL: case NV_EVENT_START: case NV_EVENT_RESTART: case NV_EVENT_RESUME: case NV_EVENT_FOCUS_GAINED: - case NV_EVENT_QUIT: NVDEBUG("%s event: no specific app action", NVEventGetEventStr(ev->m_type)); break; diff --git a/android/jni/com/mapswithme/maps/MWMActivity.cpp b/android/jni/com/mapswithme/maps/MWMActivity.cpp index bb2fc6d305..9395f47b0a 100644 --- a/android/jni/com/mapswithme/maps/MWMActivity.cpp +++ b/android/jni/com/mapswithme/maps/MWMActivity.cpp @@ -41,14 +41,11 @@ extern "C" JNIEXPORT void JNICALL Java_com_mapswithme_maps_MWMActivity_nativeInit(JNIEnv * env, jobject thiz, jstring apkPath, jstring storagePath) { - LOG(LDEBUG, ("Java_com_mapswithme_maps_MWMActivity_nativeInit 1")); if (!g_framework) { android::Platform::Instance().Initialize(env, apkPath, storagePath); g_framework = new android::Framework(g_jvm); } - - LOG(LDEBUG, ("Java_com_mapswithme_maps_MWMActivity_nativeInit 2")); } } // extern "C" diff --git a/android/jni/nv_event/nv_event.cpp b/android/jni/nv_event/nv_event.cpp index ba0f7f5409..3d082263ad 100644 --- a/android/jni/nv_event/nv_event.cpp +++ b/android/jni/nv_event/nv_event.cpp @@ -874,20 +874,6 @@ void InitNVEvent(JavaVM* vm) "(FFF)Z", (void *) NVEventAccelerometerEvent }, - - // TODO TBD - this should be done in NVTimeInit(), but we use a different - // class than most apps. Need to clean this up, as it is fragile w.r.t. - // changes in nv_time - { - "nvAcquireTimeExtension", - "()V", - (void *) nvAcquireTimeExtensionJNI - }, - { - "nvGetSystemTime", - "()J", - (void *) nvGetSystemTimeJNI - }, }; jclass k; diff --git a/android/jni/nv_event/nv_event.hpp b/android/jni/nv_event/nv_event.hpp index 20f38acd4e..1a582504bc 100644 --- a/android/jni/nv_event/nv_event.hpp +++ b/android/jni/nv_event/nv_event.hpp @@ -364,7 +364,7 @@ The application should always pair calls to this function that return non-NULL e to NVEventDoneWithEvent() @param waitMSecs The maximum time (in milisecs) to wait for an event before returning "no event". Pass NV_EVENT_WAIT_FOREVER to wait indefinitely for an event. Note that NV_EVENT_WAIT_FOREVER - does not gaurantee an event on return. The function can still return on error or if the + does not guarantee an event on return. The function can still return on error or if the app is exiting. Default is to return immediately, event or not. @return Non-NULL pointer to a constant event structure if an event was pending, NULL if no event was pending in the requested timeout period @@ -413,8 +413,8 @@ This function will be spawned in its own thread. extern int32_t NVEventAppMain(int32_t argc, char** argv); -/** Initializes EGL, queries a valid ES2 config and creates (but does not bind) - an ES2-compatible EGLContext +/** Initializes EGL, queries a valid ES1 config and creates (but does not bind) + an ES1-compatible EGLContext @return true on success, false on failure */ bool NVEventInitEGL(); diff --git a/indexer/drawing_rules.hpp b/indexer/drawing_rules.hpp index 310c10e535..1cba4b0bcd 100644 --- a/indexer/drawing_rules.hpp +++ b/indexer/drawing_rules.hpp @@ -59,6 +59,12 @@ namespace drule m_id1[threadID] = empty_id; } + void MakeEmptyID() + { + for (size_t i = 0; i < m_id1.size(); ++i) + MakeEmptyID(i); + } + uint32_t GetID2(size_t threadID) const { CheckSize(m_id2, threadID + 1); @@ -76,6 +82,12 @@ namespace drule CheckSize(m_id2, threadID + 1); m_id2[threadID] = empty_id; } + + void MakeEmptyID2() + { + for (size_t i = 0; i < m_id2.size(); ++i) + MakeEmptyID2(i); + } //@} void SetClassName(string const & cl) { m_class = cl; } diff --git a/map/framework.cpp b/map/framework.cpp index f528599b5a..b4caa2036c 100644 --- a/map/framework.cpp +++ b/map/framework.cpp @@ -607,9 +607,9 @@ void Framework::SetRenderPolicy(RenderPolicy * renderPolicy) Settings::Get("VisualLog", isVisualLogEnabled); m_informationDisplay.enableLog(isVisualLogEnabled, renderPolicy->GetWindowHandle().get()); m_informationDisplay.setVisualScale(GetPlatform().VisualScale()); - } - yg::gl::RenderContext::initParams(); + yg::gl::RenderContext::initParams(); + } m_renderPolicy.reset(); m_renderPolicy.reset(renderPolicy); diff --git a/map/partial_render_policy.cpp b/map/partial_render_policy.cpp index d1406401f8..23e3e20b11 100644 --- a/map/partial_render_policy.cpp +++ b/map/partial_render_policy.cpp @@ -139,8 +139,35 @@ void PartialRenderPolicy::SetRenderFn(TRenderFn renderFn) m_renderQueue->initializeGL(m_primaryRC, m_resourceManager); } +PartialRenderPolicy::~PartialRenderPolicy() +{ + LOG(LINFO, ("destroying PartialRenderPolicy")); + + { + threads::ConditionGuard guard(m_glCondition); + /// unlocking waiting renderThread + if (!m_glQueue.Empty()) + { + LOG(LINFO, ("clearing glQueue")); + m_glQueue.Clear(); + guard.Signal(); + } + } + + LOG(LINFO, ("shutting down renderQueue")); + + m_renderQueue.reset(); + + m_state.reset(); + m_curState.reset(); + + LOG(LINFO, ("PartialRenderPolicy destroyed")); +} + void PartialRenderPolicy::ProcessRenderQueue(list & renderQueue) { + threads::ConditionGuard g(m_glCondition); + if (renderQueue.empty()) { m_hasPacket = false; diff --git a/map/partial_render_policy.hpp b/map/partial_render_policy.hpp index 2385f5a8c4..f4a3ea902e 100644 --- a/map/partial_render_policy.hpp +++ b/map/partial_render_policy.hpp @@ -36,6 +36,8 @@ public: yg::ResourceManager::Params const & rmParams, shared_ptr const & primaryRC); + ~PartialRenderPolicy(); + void BeginFrame(shared_ptr const & paintEvent, ScreenBase const & screenBase); diff --git a/map/render_policy_st.cpp b/map/render_policy_st.cpp index a0d68f0204..04cf16aece 100644 --- a/map/render_policy_st.cpp +++ b/map/render_policy_st.cpp @@ -75,7 +75,7 @@ RenderPolicyST::RenderPolicyST(VideoTimer * videoTimer, "fonts_blacklist.txt", 2 * 1024 * 1024, 1, - GetPlatform().CpuCores()); + 0); rmp.m_useSingleThreadedOGL = false; diff --git a/map/render_queue_routine.cpp b/map/render_queue_routine.cpp index 6576257cef..d99cb89af2 100644 --- a/map/render_queue_routine.cpp +++ b/map/render_queue_routine.cpp @@ -54,11 +54,20 @@ RenderQueueRoutine::RenderQueueRoutine(shared_ptr const & r void RenderQueueRoutine::Cancel() { IRoutine::Cancel(); - /// Waking up the sleeping thread... - m_hasRenderCommands.Signal(); - /// ...Or cancelling the current rendering command in progress. - if (m_currentRenderCommand != 0) - m_currentRenderCommand->m_paintEvent->Cancel(); + + { + threads::ConditionGuard guard(m_hasRenderCommands); + + if (m_currentRenderCommand != 0) + { + LOG(LINFO, ("cancelling current renderCommand in progress")); + m_currentRenderCommand->m_paintEvent->Cancel(); + } + + LOG(LINFO, ("waking up the sleeping thread...")); + + m_hasRenderCommands.Signal(); + } } void RenderQueueRoutine::onSize(int w, int h) @@ -254,7 +263,10 @@ void RenderQueueRoutine::Do() waitForRenderCommand(m_renderCommands, guard); if (IsCancelled()) + { + LOG(LINFO, ("thread is cancelled while waiting for a render command")); break; + } m_currentRenderCommand = m_renderCommands.front(); m_renderCommands.erase(m_renderCommands.begin()); @@ -322,12 +334,6 @@ void RenderQueueRoutine::Do() m2::PointD(0, 0) * offsetM, redrawTextRect ); - -/* shared_ptr infoLayer(new yg::InfoLayer()); - infoLayer->merge(*m_threadDrawer->screen()->infoLayer().get(), - prevScreen.PtoGMatrix() * m_renderState->m_currentScreen.GtoPMatrix()); - - m_threadDrawer->screen()->setInfoLayer(infoLayer);*/ } else m_threadDrawer->screen()->infoLayer()->clear(); @@ -340,7 +346,11 @@ void RenderQueueRoutine::Do() /// main thread to add new rendering tasks and blit already /// rendered model while the current command is actually rendering. - if (m_currentRenderCommand != 0) + if (IsCancelled()) + { + LOG(LINFO, ("cancelled before processing currentRenderCommand")); + } + else if (m_currentRenderCommand != 0) { /// this fixes some strange issue with multisampled framebuffer. /// setRenderTarget should be made here. @@ -386,9 +396,15 @@ void RenderQueueRoutine::Do() glbRect, glbRect, scaleLevel); + + if (IsCancelled()) + break; } } + if (IsCancelled()) + break; + /// if something were actually drawn, or (exclusive or) we are repainting the whole rect if ((!m_renderState->m_isEmptyModelCurrent) || (fullRectRepaint)) m_renderState->m_isEmptyModelActual = m_renderState->m_isEmptyModelCurrent; @@ -420,26 +436,26 @@ void RenderQueueRoutine::Do() } invalidate(); - } - /// waiting for all collected commands to complete. - if (m_glQueue) - { + /// waiting for all collected commands to complete. + if (m_glQueue) { - threads::ConditionGuard guard(*m_glCondition); - if (!m_glQueue->Empty()) - guard.Wait(); + { + threads::ConditionGuard guard(*m_glCondition); + if (!m_glQueue->Empty()) + { + guard.Wait(); + } + } + } + + { + threads::MutexGuard guard(*m_renderState->m_mutex.get()); + /// refreshing shadow parameters from the primary parameters + m_renderState->m_shadowActualTarget = m_renderState->m_actualTarget; + m_renderState->m_shadowBackBuffer = m_renderState->m_backBuffer; } } - - { - threads::MutexGuard guard(*m_renderState->m_mutex.get()); - /// refreshing shadow parameters from the primary parameters - m_renderState->m_shadowActualTarget = m_renderState->m_actualTarget; - m_renderState->m_shadowBackBuffer = m_renderState->m_backBuffer; - } - - } // By VNG: We can't destroy render context in drawing thread. diff --git a/yg/base_texture.cpp b/yg/base_texture.cpp index 4826433d85..122c5aef4b 100644 --- a/yg/base_texture.cpp +++ b/yg/base_texture.cpp @@ -44,7 +44,7 @@ namespace yg BaseTexture::~BaseTexture() { - if (m_hasID) + if ((m_hasID) && (g_doDeleteOnDestroy)) OGLCHECK(glDeleteTextures(1, &m_id)); } diff --git a/yg/framebuffer.cpp b/yg/framebuffer.cpp index 65fced31af..58d548fd3a 100644 --- a/yg/framebuffer.cpp +++ b/yg/framebuffer.cpp @@ -41,7 +41,7 @@ namespace yg FrameBuffer::~FrameBuffer() { - if (m_id != 0) + if ((m_id != 0) && g_doDeleteOnDestroy) { #ifdef OMIM_GL_ES OGLCHECK(glDeleteFramebuffersOES(1, &m_id)); diff --git a/yg/indexbuffer.cpp b/yg/indexbuffer.cpp index addb298327..8bf31e04fd 100644 --- a/yg/indexbuffer.cpp +++ b/yg/indexbuffer.cpp @@ -59,7 +59,7 @@ namespace yg IndexBuffer::~IndexBuffer() { - if (!m_useVA) + if ((!m_useVA) && (g_doDeleteOnDestroy)) OGLCHECK(glDeleteBuffers(1, &m_id)); } diff --git a/yg/internal/opengl.cpp b/yg/internal/opengl.cpp index 0bdbfde510..ddb27460f5 100644 --- a/yg/internal/opengl.cpp +++ b/yg/internal/opengl.cpp @@ -16,6 +16,8 @@ namespace yg bool g_isRenderbufferSupported = true; bool g_isMultisamplingSupported = true; + bool g_doDeleteOnDestroy = true; + bool CheckExtensionSupport() { /// this functionality must be supported diff --git a/yg/internal/opengl.hpp b/yg/internal/opengl.hpp index 89bd367333..9fb7fe41cf 100644 --- a/yg/internal/opengl.hpp +++ b/yg/internal/opengl.hpp @@ -46,6 +46,13 @@ namespace yg extern bool g_isRenderbufferSupported; extern bool g_isMultisamplingSupported; + /// This flag controls, whether OpenGL resources should delete themselves upon destruction. + /// Sounds odd, but in EGL there are cases when the only function one should call to finish + /// its work with resources is eglTerminate, which by itself internally deletes all OpenGL resources. + /// In this case we should set this variable to true to correctly deletes all our classes. + + extern bool g_doDeleteOnDestroy; + /// return false to terminate program bool CheckExtensionSupport(); @@ -63,3 +70,4 @@ namespace yg #define OGLCHECKAFTER #define EGLCHECK #endif + diff --git a/yg/renderbuffer.cpp b/yg/renderbuffer.cpp index d65ae5b710..6e11e9d4eb 100644 --- a/yg/renderbuffer.cpp +++ b/yg/renderbuffer.cpp @@ -63,7 +63,7 @@ namespace yg RenderBuffer::~RenderBuffer() { - if (m_hasID) + if ((m_hasID) && (g_doDeleteOnDestroy)) { #ifdef OMIM_GL_ES OGLCHECK(glDeleteRenderbuffersOES(1, &m_id)); diff --git a/yg/vertexbuffer.cpp b/yg/vertexbuffer.cpp index 26f77a5fbc..373e12f81e 100644 --- a/yg/vertexbuffer.cpp +++ b/yg/vertexbuffer.cpp @@ -62,7 +62,8 @@ namespace yg if (m_useVA) delete [] (unsigned char*)m_gpuData; else - OGLCHECK(glDeleteBuffers(1, &m_id)); + if (g_doDeleteOnDestroy) + OGLCHECK(glDeleteBuffers(1, &m_id)); } void * VertexBuffer::data()