Wait for request_manager to be ready to be deleted (i.e. current request

have been properly aborted, and sign out done). Also shut down more of
STK after signaling the request manager to shut down, so that the user does
not notice the time for the sign out. Also added docs.
This commit is contained in:
hiker 2014-06-05 07:56:51 +10:00
parent c579267e9a
commit 8960b79181
5 changed files with 87 additions and 31 deletions

View File

@ -1,3 +1,5 @@
# Modify this file to change the last-modified date when you add/remove a file.
# This will then trigger a new cmake run automatically.
file(GLOB_RECURSE STK_SOURCES RELATIVE ${CMAKE_CURRENT_SOURCE_DIR} "src/*.cpp") file(GLOB_RECURSE STK_SOURCES RELATIVE ${CMAKE_CURRENT_SOURCE_DIR} "src/*.cpp")
file(GLOB_RECURSE STK_HEADERS RELATIVE ${CMAKE_CURRENT_SOURCE_DIR} "src/*.hpp") file(GLOB_RECURSE STK_HEADERS RELATIVE ${CMAKE_CURRENT_SOURCE_DIR} "src/*.hpp")
file(GLOB_RECURSE STK_SHADERS RELATIVE ${CMAKE_CURRENT_SOURCE_DIR} "data/shaders/*") file(GLOB_RECURSE STK_SHADERS RELATIVE ${CMAKE_CURRENT_SOURCE_DIR} "data/shaders/*")

View File

@ -1432,20 +1432,9 @@ static void cleanSuperTuxKart()
irr_driver->updateConfigIfRelevant(); irr_driver->updateConfigIfRelevant();
AchievementsManager::destroy(); AchievementsManager::destroy();
Referee::cleanup(); Referee::cleanup();
if(ReplayPlay::get()) ReplayPlay::destroy(); if(ReplayPlay::get()) ReplayPlay::destroy();
if(race_manager) delete race_manager; if(race_manager) delete race_manager;
//delete in reverse order of what they were created in.
//see InitTuxkart()
Online::RequestManager::deallocate();
Online::ServersManager::deallocate();
Online::ProfileManager::destroy();
GUIEngine::DialogQueue::deallocate();
if(addons_manager) delete addons_manager; if(addons_manager) delete addons_manager;
NetworkManager::kill();
if(grand_prix_manager) delete grand_prix_manager; if(grand_prix_manager) delete grand_prix_manager;
if(highscore_manager) delete highscore_manager; if(highscore_manager) delete highscore_manager;
if(attachment_manager) delete attachment_manager; if(attachment_manager) delete attachment_manager;
@ -1462,15 +1451,34 @@ static void cleanSuperTuxKart()
delete ParticleKindManager::get(); delete ParticleKindManager::get();
PlayerManager::destroy(); PlayerManager::destroy();
if(unlock_manager) delete unlock_manager; if(unlock_manager) delete unlock_manager;
Online::ProfileManager::destroy();
GUIEngine::DialogQueue::deallocate();
// Now finish shutting down objects which a separate thread. The
// RequestManager has been signaled to shut down as early as possible,
// the NewsManager thread should have finished quite early on anyway.
// But still give them some additional time to finish. It avoids a
// race condition where a thread might access the file manager after it
// was deleted (in cleanUserConfig below), but before STK finishes and
// the os takes all threads down.
// Wait (up to 2 seconds) for the news manager to be ready to be deleted,
// i.e. make sure it does not need the file_manager anymore (which will
// get deleted in cleanUserConfig).
if(!NewsManager::get()->waitForReadyToDeleted(2.0f)) if(!NewsManager::get()->waitForReadyToDeleted(2.0f))
{ {
Log::info("NewsManager", "News manager not stopping, exiting anyway."); Log::info("Thread", "News manager not stopping, exiting anyway.");
} }
NewsManager::deallocate(); NewsManager::deallocate();
if(!Online::RequestManager::get()->waitForReadyToDeleted(5.0f))
{
Log::info("Thread", "Request Manager not aborting in time, aborting.");
}
Online::RequestManager::deallocate();
// FIXME: do we need to wait for threads there, can they be
// moved further up?
Online::ServersManager::deallocate();
NetworkManager::kill();
cleanUserConfig(); cleanUserConfig();
StateManager::deallocate(); StateManager::deallocate();

View File

@ -1,7 +1,7 @@
// //
// SuperTuxKart - a fun racing game with go-kart // SuperTuxKart - a fun racing game with go-kart
// Copyright (C) 2010-2014 Lucas Baudin // Copyright (C) 2010-2014 Lucas Baudin
// 2011-201 Joerg Henrichs // 2011-2014 Joerg Henrichs
// 2013-2014 Glenn De Jonghe // 2013-2014 Glenn De Jonghe
// //
// This program is free software; you can redistribute it and/or // This program is free software; you can redistribute it and/or
@ -83,7 +83,7 @@ namespace Online
pthread_cond_init(&m_cond_request, NULL); pthread_cond_init(&m_cond_request, NULL);
m_abort.setAtomic(false); m_abort.setAtomic(false);
m_time_since_poll = MENU_POLLING_INTERVAL * 0.9; m_time_since_poll = MENU_POLLING_INTERVAL * 0.9;
} } // RequestManager
// ------------------------------------------------------------------------ // ------------------------------------------------------------------------
RequestManager::~RequestManager() RequestManager::~RequestManager()
@ -94,8 +94,7 @@ namespace Online
m_thread_id.unlock(); m_thread_id.unlock();
pthread_cond_destroy(&m_cond_request); pthread_cond_destroy(&m_cond_request);
curl_global_cleanup(); curl_global_cleanup();
} } // ~RequestManager
// ------------------------------------------------------------------------ // ------------------------------------------------------------------------
/** Start the actual network thread. This can not be done as part of /** Start the actual network thread. This can not be done as part of
@ -220,6 +219,11 @@ namespace Online
me->m_request_queue.lock(); me->m_request_queue.lock();
} // while } // while
// Signal that the request manager can now be deleted.
// We signal this even before cleaning up memory, since there's no
// need to keep the user waiting for STK to exit.
me->setCanBeDeleted();
// At this stage we have the lock for m_request_queue // At this stage we have the lock for m_request_queue
while(!me->m_request_queue.getData().empty()) while(!me->m_request_queue.getData().empty())
{ {

View File

@ -23,6 +23,7 @@
#include "io/xml_node.hpp" #include "io/xml_node.hpp"
#include "online/request.hpp" #include "online/request.hpp"
#include "utils/can_be_deleted.hpp"
#include "utils/string_utils.hpp" #include "utils/string_utils.hpp"
#include "utils/synchronised.hpp" #include "utils/synchronised.hpp"
@ -42,11 +43,41 @@
namespace Online namespace Online
{ {
/** /** A class to execute requests in a separate thread. Typically the
* \brief Class to connect with a server over HTTP(S) * requests involve a http(s) requests to be sent to the stk server, and
* \ingroup online * receive an answer (e.g. to sign in; or to download an addon). The
*/ * requests are sorted by priority (e.g. sign in and out have higher
class RequestManager * priority than downloading addon icons).
* A request is created and initialised from the main thread. When it
* is moved into the request queue, it must not be handled by the main
* thread anymore, only the RequestManager thread can handle it.
* Once the request is finished, it is put in a separate ready queue.
* The main thread regularly checks the ready queue for any ready
* request, and executes a callback. So there is no need to protect
* any functions or data members in requests, since they will either
* be handled by the main thread, or RequestManager thread, never by
* both.
* On exit, if necessary a high priority sign-out or client-quit request
* is put into the queue, and a flag is set which causes libcurl to
* abort any ongoing download. Then an additional 'quit' event with
* same priority as the sign-out is added to the queue (since it will
* be added later, the sign-out will be executed first, making sure that
* a logged in user is logged out (or its session saved). Once this is
* done, most of stk is deleted (except a few objects like the file
* manager which might be accessed if a download just finished before the
* abort). On executing the quit request, the request manager will set
* a flag that it is ready to be deleted (using the CanBeDeleted class).
* The main thread will wait for a certain amount of time for the
* RequestManager to be ready to be deleted (i.e. the sign-out and quit
* request have been processes), before deleting the RequestManager.
* Typically the RequestManager will finish while the rest of stk is
* shutting down, so the user will not experience any waiting time. Only
* on first start of stk (which will trigger downloading of all addon
* icons) is it possible that actually a download request is running,
* which might take a bit before it can be deleted.
* \ingroup online
*/
class RequestManager : public CanBeDeleted
{ {
public: public:
/** If stk has permission to access the internet (for news /** If stk has permission to access the internet (for news

View File

@ -23,8 +23,16 @@
#include "utils/synchronised.hpp" #include "utils/synchronised.hpp"
#include "utils/time.hpp" #include "utils/time.hpp"
/** A simple class that a /** A simple class that a adds a function to wait with a timeout for a
*/ * class to be ready to be deleted. It is used for objects with their
* own threads (e.g. RequestManager) to make sure they can be deleted.
* For example, the RequestManager might be executing a download request.
* So we have to signal libcurl to abort the download request, then
* potentially handle a high priority sign-out request before the thread
* can be deleted. With this object the main thread can wait for a given
* amount of time (in case that of a bad internet hickup) before deleting
* the RequestManager.
*/
class CanBeDeleted class CanBeDeleted
{ {
private: private:
@ -43,18 +51,21 @@ public:
{ {
if (m_can_be_deleted.getAtomic()) return true; if (m_can_be_deleted.getAtomic()) return true;
double start = StkTime::getRealTime(); double start = StkTime::getRealTime();
Log::verbose("Thread", "Start waiting %lf", start);
while(1) while(1)
{ {
if(m_can_be_deleted.getAtomic()) if(m_can_be_deleted.getAtomic())
{ {
Log::verbose("Thread", Log::verbose("Thread",
"Waited %lf for thread to become deleteable.", "Waited %lf seconds for thread to become deleteable.",
StkTime::getRealTime()-start); StkTime::getRealTime()-start);
Log::verbose("Thread", "Stop waiting %lf", StkTime::getRealTime());
return true; return true;
} }
StkTime::sleep(10); StkTime::sleep(10);
if(StkTime::getRealTime() - start > waiting_time) if(StkTime::getRealTime() - start > waiting_time)
{ {
Log::verbose("Thread", "Stop waiting %lf", StkTime::getRealTime());
Log::verbose("Thread", "Waited for more than %f seconds for " Log::verbose("Thread", "Waited for more than %f seconds for "
"thread to become deleteable", "thread to become deleteable",
waiting_time); waiting_time);