Commit 17813702 authored by Davis King's avatar Davis King

Disabled destruction of the global thread pool on program shutdown when using dlib

on MS Windows since this avoids a few program shutdown bugs when dlib us used as
part of a DLL.  For example, when creating python modules with threading the interpreter
sometimes hangs on program shutdown as a result of trying to destroy the global
thread pool.  Note that not destroying this resource on program shutdown is the
recommended approach according to Microsoft.
see http://blogs.msdn.com/b/oldnewthing/archive/2012/01/05/10253268.aspx
parent 1fd8afd5
...@@ -5,6 +5,7 @@ ...@@ -5,6 +5,7 @@
#include "threads_kernel_shared.h" #include "threads_kernel_shared.h"
#include "../assert.h" #include "../assert.h"
#include "../platform.h"
#include <iostream> #include <iostream>
...@@ -71,8 +72,27 @@ namespace dlib ...@@ -71,8 +72,27 @@ namespace dlib
data_ready(data_mutex), data_ready(data_mutex),
data_empty(data_mutex), data_empty(data_mutex),
destruct(false), destruct(false),
destructed(data_mutex) destructed(data_mutex),
{} do_not_ever_destruct(false)
{
#if WIN32
// Trying to destroy the global thread pool when we are part of a DLL and the
// DLL is being unloaded can sometimes lead to weird behavior. For example, in
// the python interpreter you will get the interpreter to hang. Or if we are
// part of a MATLAB mex file and the file is being unloaded there can also be
// similar weird issues. So when we are using dlib on windows we just disable
// the destruction of the global thread pool since it doesn't matter anyway.
// It's resources will just get freed by the OS. This is even the recommended
// thing to do by Microsoft (http://blogs.msdn.com/b/oldnewthing/archive/2012/01/05/10253268.aspx).
//
// As an aside, it's worth pointing out that the reason we try and free
// resources on program shutdown on other operating systems is so we can have
// clean reports from tools like valgrind which check for memory leaks. But
// trying to do this on windows is a lost cause so we give up in this case and
// follow the Microsoft recommendation.
do_not_ever_destruct = true;
#endif // WIN32
}
// ---------------------------------------------------------------------------------------- // ----------------------------------------------------------------------------------------
...@@ -98,6 +118,9 @@ namespace dlib ...@@ -98,6 +118,9 @@ namespace dlib
destruct_if_ready ( destruct_if_ready (
) )
{ {
if (do_not_ever_destruct)
return;
data_mutex.lock(); data_mutex.lock();
// if there aren't any active threads, just maybe some sitting around // if there aren't any active threads, just maybe some sitting around
......
...@@ -35,6 +35,7 @@ namespace dlib ...@@ -35,6 +35,7 @@ namespace dlib
- destruct == false - destruct == false
- total_count == 0 - total_count == 0
- function_pointer == 0 - function_pointer == 0
- do_not_ever_destruct == false
CONVENTION CONVENTION
- data_ready is associated with the mutex data_mutex - data_ready is associated with the mutex data_mutex
...@@ -73,7 +74,7 @@ namespace dlib ...@@ -73,7 +74,7 @@ namespace dlib
); );
/*! /*!
ensures ensures
- if (there are no threads currently running) then - if (there are no threads currently running and we haven't set do_not_ever_destruct) then
- calls delete this - calls delete this
- else - else
- does nothing - does nothing
...@@ -156,6 +157,7 @@ namespace dlib ...@@ -156,6 +157,7 @@ namespace dlib
signaler data_empty; // signaler to signal when the data is empty signaler data_empty; // signaler to signal when the data is empty
bool destruct; bool destruct;
signaler destructed; // signaler to signal when a thread has ended signaler destructed; // signaler to signal when a thread has ended
bool do_not_ever_destruct;
struct registry_type struct registry_type
{ {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment