Commit e5f68408 authored by Piotr Caban's avatar Piotr Caban Committed by Alexandre Julliard

msvcrt: Avoid deadlock when _beginthread is executed inside loader critical section.

parent acc112ba
...@@ -25,7 +25,7 @@ WINE_DEFAULT_DEBUG_CHANNEL(msvcrt); ...@@ -25,7 +25,7 @@ WINE_DEFAULT_DEBUG_CHANNEL(msvcrt);
/********************************************************************/ /********************************************************************/
typedef struct { typedef struct {
HANDLE is_ready, thread; HANDLE 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;
...@@ -61,19 +61,12 @@ thread_data_t *msvcrt_get_thread_data(void) ...@@ -61,19 +61,12 @@ 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, *trampoline = arg; _beginthread_trampoline_t local_trampoline;
thread_data_t *data = msvcrt_get_thread_data(); thread_data_t *data = msvcrt_get_thread_data();
if(!DuplicateHandle(GetCurrentProcess(), GetCurrentThread(), GetCurrentProcess(), memcpy(&local_trampoline,arg,sizeof(local_trampoline));
&trampoline->thread, 0, TRUE, DUPLICATE_SAME_ACCESS)) {
trampoline->thread = NULL;
SetEvent(&trampoline->is_ready);
return 0;
}
memcpy(&local_trampoline, trampoline, sizeof(local_trampoline));
data->handle = local_trampoline.thread; data->handle = local_trampoline.thread;
SetEvent(trampoline->is_ready); MSVCRT_free(arg);
local_trampoline.start_address(local_trampoline.arglist); local_trampoline.start_address(local_trampoline.arglist);
return 0; return 0;
...@@ -87,35 +80,36 @@ MSVCRT_uintptr_t CDECL _beginthread( ...@@ -87,35 +80,36 @@ 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; HANDLE thread;
TRACE("(%p, %d, %p)\n", start_address, stack_size, arglist); TRACE("(%p, %d, %p)\n", start_address, stack_size, arglist);
trampoline.is_ready = CreateEventW(NULL, FALSE, FALSE, NULL); trampoline = MSVCRT_malloc(sizeof(*trampoline));
if(!trampoline.is_ready) { if(!trampoline) {
*MSVCRT__errno() = MSVCRT_EAGAIN; *MSVCRT__errno() = MSVCRT_EAGAIN;
return -1; return -1;
} }
trampoline.start_address = start_address;
trampoline.arglist = arglist;
thread = CreateThread(NULL, stack_size, _beginthread_trampoline, thread = CreateThread(NULL, stack_size, _beginthread_trampoline,
&trampoline, 0, NULL); trampoline, CREATE_SUSPENDED, NULL);
if(!thread) { if(!thread) {
MSVCRT_free(trampoline);
*MSVCRT__errno() = MSVCRT_EAGAIN; *MSVCRT__errno() = MSVCRT_EAGAIN;
return -1; return -1;
} }
CloseHandle(thread);
WaitForSingleObject(trampoline.is_ready, INFINITE); trampoline->thread = thread;
CloseHandle(trampoline.is_ready); trampoline->start_address = start_address;
trampoline->arglist = arglist;
if(!trampoline.thread) { if(ResumeThread(thread) == -1) {
MSVCRT_free(trampoline);
*MSVCRT__errno() = MSVCRT_EAGAIN; *MSVCRT__errno() = MSVCRT_EAGAIN;
return -1; return -1;
} }
return (MSVCRT_uintptr_t)trampoline.thread;
return (MSVCRT_uintptr_t)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