diff options
author | Corinna Vinschen <corinna@vinschen.de> | 2012-05-23 13:24:57 +0000 |
---|---|---|
committer | Corinna Vinschen <corinna@vinschen.de> | 2012-05-23 13:24:57 +0000 |
commit | 750329ae90b72713963e4236c17e7f0bd167e4a3 (patch) | |
tree | 084535a59f9dc796bb7d5c0e418508539418c1d1 | |
parent | dacf4be3fa450920f16fe279e155e861f8cd258a (diff) | |
download | newlib-750329ae90b72713963e4236c17e7f0bd167e4a3.zip newlib-750329ae90b72713963e4236c17e7f0bd167e4a3.tar.gz newlib-750329ae90b72713963e4236c17e7f0bd167e4a3.tar.bz2 |
* thread.cc (pthread::precreate): Make sure mutex is recursive.
Explain why.
-rw-r--r-- | winsup/cygwin/ChangeLog | 5 | ||||
-rw-r--r-- | winsup/cygwin/thread.cc | 14 |
2 files changed, 17 insertions, 2 deletions
diff --git a/winsup/cygwin/ChangeLog b/winsup/cygwin/ChangeLog index ec7ff13..7b3d56d 100644 --- a/winsup/cygwin/ChangeLog +++ b/winsup/cygwin/ChangeLog @@ -1,5 +1,10 @@ 2012-05-23 Corinna Vinschen <corinna@vinschen.de> + * thread.cc (pthread::precreate): Make sure mutex is recursive. + Explain why. + +2012-05-23 Corinna Vinschen <corinna@vinschen.de> + * thread.cc (pthread::pop_cleanup_handler): Move setting the cancelstate to PTHREAD_CANCEL_DISABLE from here... (pthread::pop_all_cleanup_handlers): ...to here, otherwise any explicit diff --git a/winsup/cygwin/thread.cc b/winsup/cygwin/thread.cc index bb38ff3..7a914b2 100644 --- a/winsup/cygwin/thread.cc +++ b/winsup/cygwin/thread.cc @@ -431,8 +431,18 @@ pthread::precreate (pthread_attr *newattr) magic = 0; return; } - /* Change the mutex type to NORMAL to speed up mutex operations */ - mutex.set_type (PTHREAD_MUTEX_NORMAL); + /* This mutex MUST be recursive. Consider the following scenario: + - The thread installs a cleanup handler. + - The cleanup handler calls a function which itself installs a + cleanup handler. + - pthread_cancel is called for this thread. + - The thread's cleanup handler is called under mutex lock condition. + - The cleanup handler calls the subsequent function with cleanup handler. + - The function runs to completion, so it calls pthread_cleanup_pop. + - pthread_cleanup_pop calls pthread::pop_cleanup_handler which will again + try to lock the mutex. + - Deadlock. */ + mutex.set_type (PTHREAD_MUTEX_RECURSIVE); if (!create_cancel_event ()) magic = 0; } |