Fix deadlock in ThreadPool::Stop

This commit is contained in:
Gunnar Beutner 2014-11-13 10:00:49 +01:00
parent 02fef3f84a
commit f20d9010c0
3 changed files with 11 additions and 26 deletions

View File

@ -321,8 +321,6 @@ mainloop:
GetTP().Stop();
m_ShuttingDown = false;
GetTP().Join(true);
Timer::Uninitialize();
#endif /* _DEBUG */
}

View File

@ -41,7 +41,6 @@ ThreadPool::ThreadPool(size_t max_threads)
ThreadPool::~ThreadPool(void)
{
Stop();
Join(true);
}
void ThreadPool::Start(void)
@ -49,38 +48,26 @@ void ThreadPool::Start(void)
for (size_t i = 0; i < sizeof(m_Queues) / sizeof(m_Queues[0]); i++)
m_Queues[i].SpawnWorker(m_ThreadGroup);
m_ThreadGroup.create_thread(boost::bind(&ThreadPool::ManagerThreadProc, this));
m_MgmtThread = boost::move(boost::thread(boost::bind(&ThreadPool::ManagerThreadProc, this)));
}
void ThreadPool::Stop(void)
{
{
boost::mutex::scoped_lock lock(m_MgmtMutex);
m_Stopped = true;
m_MgmtCV.notify_all();
}
m_MgmtThread.join();
for (size_t i = 0; i < sizeof(m_Queues) / sizeof(m_Queues[0]); i++) {
boost::mutex::scoped_lock lock(m_Queues[i].Mutex);
m_Queues[i].Stopped = true;
m_Queues[i].CV.notify_all();
}
boost::mutex::scoped_lock lock(m_MgmtMutex);
m_Stopped = true;
m_MgmtCV.notify_all();
}
/**
* Waits for all worker threads to finish.
*/
void ThreadPool::Join(bool wait_for_stop)
{
if (wait_for_stop) {
m_ThreadGroup.join_all();
return;
}
for (size_t i = 0; i < sizeof(m_Queues) / sizeof(m_Queues[0]); i++) {
boost::mutex::scoped_lock lock(m_Queues[i].Mutex);
while (!m_Queues[i].Items.empty())
m_Queues[i].CVStarved.wait(lock);
}
m_ThreadGroup.join_all();
}
/**

View File

@ -53,7 +53,6 @@ public:
void Start(void);
void Stop(void);
void Join(bool wait_for_stop = false);
bool Post(const WorkFunction& callback, SchedulerPolicy policy = DefaultScheduler);
@ -122,6 +121,7 @@ private:
boost::thread_group m_ThreadGroup;
boost::thread m_MgmtThread;
boost::mutex m_MgmtMutex;
boost::condition_variable m_MgmtCV;
bool m_Stopped;