* commit 'afb488a3e91c7ac297b4f91e9985fd10c09d6f81': Fix possible leak in pthread_detach.
This commit is contained in:
@@ -27,29 +27,32 @@
|
||||
*/
|
||||
|
||||
#include <errno.h>
|
||||
#include <pthread.h>
|
||||
|
||||
#include "pthread_accessor.h"
|
||||
|
||||
int pthread_detach(pthread_t t) {
|
||||
pthread_accessor thread(t);
|
||||
if (thread.get() == NULL) {
|
||||
{
|
||||
pthread_accessor thread(t);
|
||||
if (thread.get() == NULL) {
|
||||
return ESRCH;
|
||||
}
|
||||
|
||||
if (thread->attr.flags & PTHREAD_ATTR_FLAG_DETACHED) {
|
||||
return EINVAL; // Already detached.
|
||||
}
|
||||
|
||||
if (thread->attr.flags & PTHREAD_ATTR_FLAG_JOINED) {
|
||||
return 0; // Already being joined; silently do nothing, like glibc.
|
||||
}
|
||||
|
||||
// If the thread has not exited, we can detach it safely.
|
||||
if ((thread->attr.flags & PTHREAD_ATTR_FLAG_ZOMBIE) == 0) {
|
||||
thread->attr.flags |= PTHREAD_ATTR_FLAG_DETACHED;
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
if (thread->attr.flags & PTHREAD_ATTR_FLAG_DETACHED) {
|
||||
return EINVAL; // Already detached.
|
||||
}
|
||||
|
||||
if (thread->attr.flags & PTHREAD_ATTR_FLAG_JOINED) {
|
||||
return 0; // Already being joined; silently do nothing, like glibc.
|
||||
}
|
||||
|
||||
if (thread->tid == 0) {
|
||||
// Already exited; clean up.
|
||||
_pthread_internal_remove_locked(thread.get(), true);
|
||||
return 0;
|
||||
}
|
||||
|
||||
thread->attr.flags |= PTHREAD_ATTR_FLAG_DETACHED;
|
||||
return 0;
|
||||
// The thread is in zombie state, use pthread_join to clean it up.
|
||||
return pthread_join(t, NULL);
|
||||
}
|
||||
|
@@ -99,6 +99,9 @@ void pthread_exit(void* return_value) {
|
||||
// pthread_internal_t is freed below with stack, not here.
|
||||
_pthread_internal_remove_locked(thread, false);
|
||||
free_mapped_space = true;
|
||||
} else {
|
||||
// Mark the thread as exiting without freeing pthread_internal_t.
|
||||
thread->attr.flags |= PTHREAD_ATTR_FLAG_ZOMBIE;
|
||||
}
|
||||
pthread_mutex_unlock(&g_thread_list_lock);
|
||||
|
||||
|
@@ -38,6 +38,9 @@
|
||||
/* Has the thread been joined by another thread? */
|
||||
#define PTHREAD_ATTR_FLAG_JOINED 0x00000002
|
||||
|
||||
/* Did the thread exit without freeing pthread_internal_t? */
|
||||
#define PTHREAD_ATTR_FLAG_ZOMBIE 0x00000004
|
||||
|
||||
/* Is this the main thread? */
|
||||
#define PTHREAD_ATTR_FLAG_MAIN_THREAD 0x80000000
|
||||
|
||||
|
Reference in New Issue
Block a user