msvcrt: Close thread handle when _beginthread is used.
This commit is contained in:
parent
ee5a3216ed
commit
2bcc9d452a
|
@ -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 {
|
||||||
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);
|
||||||
/********************************************************************/
|
/********************************************************************/
|
||||||
|
|
||||||
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)
|
||||||
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)
|
||||||
*/
|
*/
|
||||||
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();
|
||||||
|
|
||||||
/* Maybe it's just being paranoid, but freeing arg right
|
if(!DuplicateHandle(GetCurrentProcess(), GetCurrentThread(), GetCurrentProcess(),
|
||||||
* away seems safer.
|
&trampoline->thread, 0, TRUE, DUPLICATE_SAME_ACCESS)) {
|
||||||
*/
|
trampoline->thread = NULL;
|
||||||
memcpy(&local_trampoline,arg,sizeof(local_trampoline));
|
SetEvent(&trampoline->is_ready);
|
||||||
MSVCRT_free(arg);
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(&local_trampoline, trampoline, sizeof(local_trampoline));
|
||||||
|
data->handle = local_trampoline.thread;
|
||||||
|
SetEvent(trampoline->is_ready);
|
||||||
|
|
||||||
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(
|
||||||
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
|
|
Loading…
Reference in New Issue