Commit 2bcc9d45 authored by Piotr Caban's avatar Piotr Caban Committed by Alexandre Julliard

msvcrt: Close thread handle when _beginthread is used.

parent ee5a3216
...@@ -66,6 +66,7 @@ static inline void msvcrt_free_tls_mem(void) ...@@ -66,6 +66,7 @@ static inline void msvcrt_free_tls_mem(void)
thread_data_t *tls = TlsGetValue(msvcrt_tls_index); thread_data_t *tls = TlsGetValue(msvcrt_tls_index);
if (tls) if (tls)
{ {
CloseHandle(tls->handle);
HeapFree(GetProcessHeap(),0,tls->efcvt_buffer); HeapFree(GetProcessHeap(),0,tls->efcvt_buffer);
HeapFree(GetProcessHeap(),0,tls->asctime_buffer); HeapFree(GetProcessHeap(),0,tls->asctime_buffer);
HeapFree(GetProcessHeap(),0,tls->wasctime_buffer); HeapFree(GetProcessHeap(),0,tls->wasctime_buffer);
......
...@@ -107,6 +107,8 @@ struct MSVCRT_tm { ...@@ -107,6 +107,8 @@ struct MSVCRT_tm {
extern DWORD msvcrt_tls_index; extern DWORD msvcrt_tls_index;
struct __thread_data { struct __thread_data {
DWORD tid;
HANDLE handle;
int thread_errno; int thread_errno;
MSVCRT_ulong thread_doserrno; MSVCRT_ulong thread_doserrno;
unsigned int random_seed; /* seed for rand() */ unsigned int random_seed; /* seed for rand() */
......
...@@ -25,6 +25,7 @@ WINE_DEFAULT_DEBUG_CHANNEL(msvcrt); ...@@ -25,6 +25,7 @@ WINE_DEFAULT_DEBUG_CHANNEL(msvcrt);
/********************************************************************/ /********************************************************************/
typedef struct { typedef struct {
HANDLE is_ready, thread;
MSVCRT__beginthread_start_routine_t start_address; MSVCRT__beginthread_start_routine_t start_address;
void *arglist; void *arglist;
} _beginthread_trampoline_t; } _beginthread_trampoline_t;
...@@ -44,6 +45,8 @@ thread_data_t *msvcrt_get_thread_data(void) ...@@ -44,6 +45,8 @@ thread_data_t *msvcrt_get_thread_data(void)
if (!(ptr = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*ptr) ))) if (!(ptr = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*ptr) )))
_amsg_exit( _RT_THREAD ); _amsg_exit( _RT_THREAD );
if (!TlsSetValue( msvcrt_tls_index, ptr )) _amsg_exit( _RT_THREAD ); if (!TlsSetValue( msvcrt_tls_index, ptr )) _amsg_exit( _RT_THREAD );
ptr->tid = GetCurrentThreadId();
ptr->handle = INVALID_HANDLE_VALUE;
ptr->random_seed = 1; ptr->random_seed = 1;
} }
SetLastError( err ); SetLastError( err );
...@@ -56,13 +59,19 @@ thread_data_t *msvcrt_get_thread_data(void) ...@@ -56,13 +59,19 @@ thread_data_t *msvcrt_get_thread_data(void)
*/ */
static DWORD CALLBACK _beginthread_trampoline(LPVOID arg) static DWORD CALLBACK _beginthread_trampoline(LPVOID arg)
{ {
_beginthread_trampoline_t local_trampoline; _beginthread_trampoline_t local_trampoline, *trampoline = arg;
thread_data_t *data = msvcrt_get_thread_data();
if(!DuplicateHandle(GetCurrentProcess(), GetCurrentThread(), GetCurrentProcess(),
&trampoline->thread, 0, TRUE, DUPLICATE_SAME_ACCESS)) {
trampoline->thread = NULL;
SetEvent(&trampoline->is_ready);
return 0;
}
/* Maybe it's just being paranoid, but freeing arg right memcpy(&local_trampoline, trampoline, sizeof(local_trampoline));
* away seems safer. data->handle = local_trampoline.thread;
*/ SetEvent(trampoline->is_ready);
memcpy(&local_trampoline,arg,sizeof(local_trampoline));
MSVCRT_free(arg);
local_trampoline.start_address(local_trampoline.arglist); local_trampoline.start_address(local_trampoline.arglist);
return 0; return 0;
...@@ -76,21 +85,35 @@ MSVCRT_uintptr_t CDECL _beginthread( ...@@ -76,21 +85,35 @@ MSVCRT_uintptr_t CDECL _beginthread(
unsigned int stack_size, /* [in] Stack size for new thread or 0 */ unsigned int stack_size, /* [in] Stack size for new thread or 0 */
void *arglist) /* [in] Argument list to be passed to new thread or NULL */ void *arglist) /* [in] Argument list to be passed to new thread or NULL */
{ {
_beginthread_trampoline_t* trampoline; _beginthread_trampoline_t trampoline;
HANDLE thread;
TRACE("(%p, %d, %p)\n", start_address, stack_size, arglist); TRACE("(%p, %d, %p)\n", start_address, stack_size, arglist);
/* Allocate the trampoline here so that it is still valid when the thread trampoline.is_ready = CreateEventW(NULL, FALSE, FALSE, NULL);
* starts... typically after this function has returned. if(!trampoline.is_ready) {
* _beginthread_trampoline is responsible for freeing the trampoline *MSVCRT__errno() = MSVCRT_EAGAIN;
*/ return -1;
trampoline=MSVCRT_malloc(sizeof(*trampoline)); }
trampoline->start_address = start_address; trampoline.start_address = start_address;
trampoline->arglist = arglist; trampoline.arglist = arglist;
/* FIXME */ thread = CreateThread(NULL, stack_size, _beginthread_trampoline,
return (MSVCRT_uintptr_t)CreateThread(NULL, stack_size, _beginthread_trampoline, &trampoline, 0, NULL);
trampoline, 0, NULL); if(!thread) {
*MSVCRT__errno() = MSVCRT_EAGAIN;
return -1;
}
CloseHandle(thread);
WaitForSingleObject(trampoline.is_ready, INFINITE);
CloseHandle(trampoline.is_ready);
if(!trampoline.thread) {
*MSVCRT__errno() = MSVCRT_EAGAIN;
return -1;
}
return (MSVCRT_uintptr_t)trampoline.thread;
} }
/********************************************************************* /*********************************************************************
......
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