2012-09-18 03:17:19 +04:00
|
|
|
/**
|
|
|
|
* WinPR: Windows Portable Runtime
|
|
|
|
* Process Thread Functions
|
|
|
|
*
|
|
|
|
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
#include <assert.h>
|
|
|
|
|
2012-09-18 20:57:19 +04:00
|
|
|
#include <winpr/handle.h>
|
|
|
|
|
2012-09-18 03:17:19 +04:00
|
|
|
#include <winpr/thread.h>
|
|
|
|
|
|
|
|
/**
|
|
|
|
* api-ms-win-core-processthreads-l1-1-1.dll
|
2014-07-15 13:11:59 +04:00
|
|
|
*
|
2012-09-18 03:17:19 +04:00
|
|
|
* CreateRemoteThread
|
|
|
|
* CreateRemoteThreadEx
|
|
|
|
* CreateThread
|
|
|
|
* DeleteProcThreadAttributeList
|
|
|
|
* ExitThread
|
|
|
|
* FlushInstructionCache
|
|
|
|
* FlushProcessWriteBuffers
|
|
|
|
* GetCurrentThread
|
|
|
|
* GetCurrentThreadId
|
|
|
|
* GetCurrentThreadStackLimits
|
|
|
|
* GetExitCodeThread
|
|
|
|
* GetPriorityClass
|
|
|
|
* GetStartupInfoW
|
|
|
|
* GetThreadContext
|
|
|
|
* GetThreadId
|
|
|
|
* GetThreadIdealProcessorEx
|
|
|
|
* GetThreadPriority
|
|
|
|
* GetThreadPriorityBoost
|
|
|
|
* GetThreadTimes
|
|
|
|
* InitializeProcThreadAttributeList
|
|
|
|
* OpenThread
|
|
|
|
* OpenThreadToken
|
|
|
|
* QueryProcessAffinityUpdateMode
|
|
|
|
* QueueUserAPC
|
|
|
|
* ResumeThread
|
|
|
|
* SetPriorityClass
|
|
|
|
* SetThreadContext
|
|
|
|
* SetThreadPriority
|
|
|
|
* SetThreadPriorityBoost
|
|
|
|
* SetThreadStackGuarantee
|
|
|
|
* SetThreadToken
|
|
|
|
* SuspendThread
|
|
|
|
* SwitchToThread
|
|
|
|
* TerminateThread
|
|
|
|
* UpdateProcThreadAttribute
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _WIN32
|
|
|
|
|
2012-09-23 21:54:14 +04:00
|
|
|
#include <winpr/crt.h>
|
2014-05-11 13:23:30 +04:00
|
|
|
#include <winpr/platform.h>
|
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
#ifdef HAVE_UNISTD_H
|
2014-05-11 13:23:30 +04:00
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
2012-09-23 21:54:14 +04:00
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
#ifdef HAVE_EVENTFD_H
|
|
|
|
#include <sys/eventfd.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAVE_EXECINFO_H
|
|
|
|
#include <execinfo.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include <winpr/collections.h>
|
|
|
|
|
2012-12-13 07:03:40 +04:00
|
|
|
#include "thread.h"
|
2012-09-18 03:17:19 +04:00
|
|
|
|
2013-05-17 01:32:58 +04:00
|
|
|
#include "../handle/handle.h"
|
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
static pthread_once_t thread_initialized = PTHREAD_ONCE_INIT;
|
2014-07-15 13:11:59 +04:00
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
static HANDLE_CLOSE_CB _ThreadHandleCloseCb;
|
|
|
|
static wListDictionary *thread_list = NULL;
|
|
|
|
|
|
|
|
static BOOL ThreadCloseHandle(HANDLE handle);
|
2014-07-15 13:11:59 +04:00
|
|
|
static void cleanup_handle(void *obj);
|
2014-07-14 21:36:31 +04:00
|
|
|
|
|
|
|
static BOOL ThreadIsHandled(HANDLE handle)
|
|
|
|
{
|
|
|
|
WINPR_THREAD *pThread = (WINPR_THREAD *)handle;
|
|
|
|
|
|
|
|
if (!pThread || pThread->Type != HANDLE_TYPE_THREAD)
|
|
|
|
{
|
|
|
|
SetLastError(ERROR_INVALID_HANDLE);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void ThreadInitialize(void)
|
|
|
|
{
|
|
|
|
_ThreadHandleCloseCb.IsHandled = ThreadIsHandled;
|
|
|
|
_ThreadHandleCloseCb.CloseHandle = ThreadCloseHandle;
|
|
|
|
RegisterHandleCloseCb(&_ThreadHandleCloseCb);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dump_thread(WINPR_THREAD *thread)
|
|
|
|
{
|
|
|
|
#if defined(WITH_DEBUG_THREADS) && defined(HAVE_EXECINFO_H)
|
|
|
|
void *stack[20];
|
2014-07-16 13:56:25 +04:00
|
|
|
fprintf(stderr, "[%s]: Called from:\n", __FUNCTION__);
|
2014-07-14 21:36:31 +04:00
|
|
|
backtrace_symbols_fd(stack, 20, STDERR_FILENO);
|
2014-07-16 13:56:25 +04:00
|
|
|
fprintf(stderr, "[%s]: Thread handle created still not closed!\n", __FUNCTION__);
|
2014-07-14 21:36:31 +04:00
|
|
|
backtrace_symbols_fd(thread->create_stack, 20, STDERR_FILENO);
|
|
|
|
|
|
|
|
if (thread->started)
|
2014-07-16 13:56:25 +04:00
|
|
|
fprintf(stderr, "[%s]: Thread still running!\n", __FUNCTION__);
|
2014-07-14 21:36:31 +04:00
|
|
|
else if (!thread->exit_stack)
|
2014-07-16 13:56:25 +04:00
|
|
|
fprintf(stderr, "[%s]: Thread suspended.\n", __FUNCTION__);
|
2014-07-14 21:36:31 +04:00
|
|
|
else
|
|
|
|
{
|
2014-07-16 13:56:25 +04:00
|
|
|
fprintf(stderr, "[%s]: Thread exited at:\n", __FUNCTION__);
|
2014-07-14 21:36:31 +04:00
|
|
|
backtrace_symbols_fd(thread->exit_stack, 20, STDERR_FILENO);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2012-09-20 05:42:46 +04:00
|
|
|
/**
|
|
|
|
* TODO: implement thread suspend/resume using pthreads
|
|
|
|
* http://stackoverflow.com/questions/3140867/suspend-pthreads-without-using-condition
|
|
|
|
*/
|
2014-07-14 21:36:31 +04:00
|
|
|
static BOOL set_event(WINPR_THREAD *thread)
|
|
|
|
{
|
|
|
|
int length;
|
|
|
|
BOOL status = FALSE;
|
|
|
|
#ifdef HAVE_EVENTFD_H
|
|
|
|
eventfd_t val = 1;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
length = eventfd_write(thread->pipe_fd[0], val);
|
|
|
|
}
|
|
|
|
while ((length < 0) && (errno == EINTR));
|
|
|
|
|
|
|
|
status = (length == 0) ? TRUE : FALSE;
|
|
|
|
#else
|
|
|
|
|
|
|
|
if (WaitForSingleObject(thread, 0) != WAIT_OBJECT_0)
|
|
|
|
{
|
|
|
|
length = write(thread->pipe_fd[1], "-", 1);
|
|
|
|
|
|
|
|
if (length == 1)
|
|
|
|
status = TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
status = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL reset_event(WINPR_THREAD *thread)
|
|
|
|
{
|
|
|
|
int length;
|
|
|
|
BOOL status = FALSE;
|
|
|
|
#ifdef HAVE_EVENTFD_H
|
2014-07-16 11:56:30 +04:00
|
|
|
eventfd_t value;
|
2014-07-14 21:36:31 +04:00
|
|
|
|
2014-07-16 11:56:30 +04:00
|
|
|
do
|
|
|
|
{
|
|
|
|
length = eventfd_read(thread->pipe_fd[0], &value);
|
|
|
|
}
|
|
|
|
while ((length < 0) && (errno == EINTR));
|
2014-07-14 21:36:31 +04:00
|
|
|
|
2014-07-16 11:56:30 +04:00
|
|
|
if ((length > 0) && (!status))
|
|
|
|
status = TRUE;
|
2014-07-14 21:36:31 +04:00
|
|
|
|
|
|
|
#else
|
2014-07-16 11:56:30 +04:00
|
|
|
length = read(thread->pipe_fd[0], &length, 1);
|
2014-07-14 21:36:31 +04:00
|
|
|
|
2014-07-16 11:56:30 +04:00
|
|
|
if ((length == 1) && (!status))
|
|
|
|
status = TRUE;
|
2014-07-14 21:36:31 +04:00
|
|
|
|
|
|
|
#endif
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int thread_compare(void *a, void *b)
|
|
|
|
{
|
|
|
|
pthread_t *p1 = a;
|
|
|
|
pthread_t *p2 = b;
|
|
|
|
int rc = pthread_equal(*p1, *p2);
|
|
|
|
return rc;
|
|
|
|
}
|
2012-09-20 05:42:46 +04:00
|
|
|
|
2014-07-15 13:11:59 +04:00
|
|
|
/* Thread launcher function responsible for registering
|
|
|
|
* cleanup handlers and calling pthread_exit, if not done
|
|
|
|
* in thread function. */
|
|
|
|
static void *thread_launcher(void *arg)
|
2012-09-23 21:54:14 +04:00
|
|
|
{
|
2014-07-15 13:11:59 +04:00
|
|
|
void *rc = NULL;
|
|
|
|
WINPR_THREAD *thread = (WINPR_THREAD *)arg;
|
|
|
|
|
|
|
|
if (!thread)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: Called with invalid argument %p\n", __FUNCTION__, arg);
|
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
void *(*fkt)(void *) = (void *)thread->lpStartAddress;
|
|
|
|
|
|
|
|
if (!fkt)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: Thread function argument is %p\n", fkt);
|
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = fkt(thread->lpParameter);
|
|
|
|
}
|
|
|
|
|
|
|
|
exit:
|
2014-07-16 11:40:59 +04:00
|
|
|
|
2014-07-15 14:42:53 +04:00
|
|
|
if (!thread->exited)
|
|
|
|
thread->dwExitCode = (DWORD)(size_t)rc;
|
2012-09-23 21:54:14 +04:00
|
|
|
|
2014-07-16 11:40:59 +04:00
|
|
|
set_event(thread);
|
|
|
|
|
2014-07-16 13:56:25 +04:00
|
|
|
if (thread->detached || !thread->started)
|
2014-07-15 13:11:59 +04:00
|
|
|
cleanup_handle(thread);
|
|
|
|
|
2014-07-16 13:56:25 +04:00
|
|
|
pthread_exit(thread->dwExitCode);
|
2014-07-15 13:11:59 +04:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void winpr_StartThread(WINPR_THREAD *thread)
|
|
|
|
{
|
|
|
|
pthread_attr_t attr;
|
2012-09-23 21:54:14 +04:00
|
|
|
pthread_attr_init(&attr);
|
2013-01-23 01:19:32 +04:00
|
|
|
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
|
2012-09-23 21:54:14 +04:00
|
|
|
|
|
|
|
if (thread->dwStackSize > 0)
|
|
|
|
pthread_attr_setstacksize(&attr, (size_t) thread->dwStackSize);
|
|
|
|
|
2014-07-15 13:11:59 +04:00
|
|
|
pthread_create(&thread->thread, &attr, thread_launcher, thread);
|
2012-09-23 21:54:14 +04:00
|
|
|
pthread_attr_destroy(&attr);
|
2014-07-14 21:36:31 +04:00
|
|
|
reset_event(thread);
|
|
|
|
ListDictionary_Add(thread_list, &thread->thread, thread);
|
|
|
|
dump_thread(thread);
|
2014-07-16 13:56:25 +04:00
|
|
|
thread->started = TRUE;
|
2012-09-18 08:10:14 +04:00
|
|
|
}
|
|
|
|
|
2012-09-18 03:17:19 +04:00
|
|
|
HANDLE CreateThread(LPSECURITY_ATTRIBUTES lpThreadAttributes, SIZE_T dwStackSize,
|
2014-07-15 13:11:59 +04:00
|
|
|
LPTHREAD_START_ROUTINE lpStartAddress, LPVOID lpParameter, DWORD dwCreationFlags, LPDWORD lpThreadId)
|
2012-09-18 03:17:19 +04:00
|
|
|
{
|
2012-09-19 03:24:03 +04:00
|
|
|
HANDLE handle;
|
2014-07-15 13:11:59 +04:00
|
|
|
WINPR_THREAD *thread;
|
|
|
|
thread = (WINPR_THREAD *) calloc(1, sizeof(WINPR_THREAD));
|
2014-05-11 13:23:30 +04:00
|
|
|
|
2014-04-09 18:07:06 +04:00
|
|
|
if (!thread)
|
|
|
|
return NULL;
|
2012-09-20 05:42:46 +04:00
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
pthread_once(&thread_initialized, ThreadInitialize);
|
2012-09-23 21:54:14 +04:00
|
|
|
thread->dwStackSize = dwStackSize;
|
|
|
|
thread->lpParameter = lpParameter;
|
|
|
|
thread->lpStartAddress = lpStartAddress;
|
|
|
|
thread->lpThreadAttributes = lpThreadAttributes;
|
2014-07-14 21:36:31 +04:00
|
|
|
#if defined(WITH_DEBUG_THREADS) && defined(HAVE_EXECINFO_H)
|
|
|
|
backtrace(thread->create_stack, 20);
|
|
|
|
dump_thread(thread);
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_EVENTFD_H
|
|
|
|
thread->pipe_fd[0] = eventfd(0, EFD_NONBLOCK);
|
|
|
|
|
|
|
|
if (thread->pipe_fd[0] < 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: failed to create thread\n", __FUNCTION__);
|
|
|
|
free(thread);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
2012-09-20 05:42:46 +04:00
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
if (pipe(thread->pipe_fd) < 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: failed to create thread\n", __FUNCTION__);
|
|
|
|
free(thread);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2012-09-23 21:54:14 +04:00
|
|
|
pthread_mutex_init(&thread->mutex, 0);
|
2013-05-17 02:27:26 +04:00
|
|
|
WINPR_HANDLE_SET_TYPE(thread, HANDLE_TYPE_THREAD);
|
|
|
|
handle = (HANDLE) thread;
|
2012-09-19 03:24:03 +04:00
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
if (NULL == thread_list)
|
|
|
|
{
|
|
|
|
thread_list = ListDictionary_New(TRUE);
|
|
|
|
thread_list->objectKey.fnObjectEquals = thread_compare;
|
|
|
|
}
|
|
|
|
|
2012-09-23 21:54:14 +04:00
|
|
|
if (!(dwCreationFlags & CREATE_SUSPENDED))
|
|
|
|
winpr_StartThread(thread);
|
2014-07-14 21:36:31 +04:00
|
|
|
else
|
|
|
|
set_event(thread);
|
2012-09-20 05:42:46 +04:00
|
|
|
|
2012-09-19 03:24:03 +04:00
|
|
|
return handle;
|
2012-09-18 03:17:19 +04:00
|
|
|
}
|
|
|
|
|
2014-07-15 13:11:59 +04:00
|
|
|
void cleanup_handle(void *obj)
|
2014-07-14 21:36:31 +04:00
|
|
|
{
|
2014-07-15 13:11:59 +04:00
|
|
|
WINPR_THREAD *thread = (WINPR_THREAD *)obj;
|
2014-07-14 21:36:31 +04:00
|
|
|
int rc = pthread_mutex_destroy(&thread->mutex);
|
|
|
|
|
|
|
|
if (rc)
|
2014-07-15 13:11:59 +04:00
|
|
|
fprintf(stderr, "[%s]: failed to destroy mutex [%d] %s (%d)\n",
|
|
|
|
__FUNCTION__, rc, strerror(errno), errno);
|
2014-07-14 21:36:31 +04:00
|
|
|
|
|
|
|
if (thread->pipe_fd[0])
|
|
|
|
close(thread->pipe_fd[0]);
|
|
|
|
|
|
|
|
if (thread->pipe_fd[1])
|
|
|
|
close(thread->pipe_fd[1]);
|
|
|
|
|
2014-07-15 13:11:59 +04:00
|
|
|
if (thread_list && ListDictionary_Contains(thread_list, &thread->thread))
|
|
|
|
ListDictionary_Remove(thread_list, &thread->thread);
|
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
free(thread);
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL ThreadCloseHandle(HANDLE handle)
|
|
|
|
{
|
|
|
|
WINPR_THREAD *thread = (WINPR_THREAD *)handle;
|
|
|
|
|
|
|
|
if (!thread_list)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: Thread list does not exist, check call!\n", __FUNCTION__);
|
|
|
|
dump_thread(thread);
|
|
|
|
}
|
|
|
|
else if (!ListDictionary_Contains(thread_list, &thread->thread))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: Thread list does not contain this thread! check call!\n", __FUNCTION__);
|
|
|
|
dump_thread(thread);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ListDictionary_Lock(thread_list);
|
|
|
|
dump_thread(thread);
|
|
|
|
|
2014-07-15 13:11:59 +04:00
|
|
|
if ((thread->started) && (WaitForSingleObject(thread, 0) != WAIT_OBJECT_0))
|
2014-07-14 21:36:31 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: Thread running, setting to detached state!\n", __FUNCTION__);
|
|
|
|
thread->detached = TRUE;
|
|
|
|
pthread_detach(thread->thread);
|
|
|
|
}
|
2014-07-15 13:11:59 +04:00
|
|
|
else
|
|
|
|
cleanup_handle(thread);
|
2014-07-15 14:42:53 +04:00
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
ListDictionary_Unlock(thread_list);
|
|
|
|
|
|
|
|
if (ListDictionary_Count(thread_list) < 1)
|
|
|
|
{
|
|
|
|
ListDictionary_Free(thread_list);
|
|
|
|
thread_list = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2012-09-23 21:54:14 +04:00
|
|
|
HANDLE CreateRemoteThread(HANDLE hProcess, LPSECURITY_ATTRIBUTES lpThreadAttributes, SIZE_T dwStackSize,
|
2014-07-15 13:11:59 +04:00
|
|
|
LPTHREAD_START_ROUTINE lpStartAddress, LPVOID lpParameter, DWORD dwCreationFlags, LPDWORD lpThreadId)
|
2012-09-18 03:17:19 +04:00
|
|
|
{
|
2014-07-14 21:36:31 +04:00
|
|
|
fprintf(stderr, "[%s]: not implemented\n", __FUNCTION__);
|
2012-09-23 21:54:14 +04:00
|
|
|
return NULL;
|
|
|
|
}
|
2012-09-18 03:17:19 +04:00
|
|
|
|
2014-07-16 13:56:25 +04:00
|
|
|
VOID _ExitThread(DWORD dwExitCode)
|
2012-09-23 21:54:14 +04:00
|
|
|
{
|
2014-07-14 21:36:31 +04:00
|
|
|
pthread_t tid = pthread_self();
|
|
|
|
|
|
|
|
if (NULL == thread_list)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: function called without existing thread list!\n", __FUNCTION__);
|
|
|
|
#if defined(WITH_DEBUG_THREADS)
|
|
|
|
DumpThreadHandles();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else if (!ListDictionary_Contains(thread_list, &tid))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: function called, but no matching entry in thread list!\n", __FUNCTION__);
|
|
|
|
#if defined(WITH_DEBUG_THREADS)
|
|
|
|
DumpThreadHandles();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WINPR_THREAD *thread;
|
|
|
|
ListDictionary_Lock(thread_list);
|
|
|
|
thread = ListDictionary_GetItemValue(thread_list, &tid);
|
|
|
|
assert(thread);
|
2014-07-15 14:42:53 +04:00
|
|
|
thread->exited = TRUE;
|
|
|
|
thread->dwExitCode = dwExitCode;
|
|
|
|
#if defined(WITH_DEBUG_THREADS) && defined(HAVE_EXECINFO_H)
|
2014-07-14 21:36:31 +04:00
|
|
|
backtrace(thread->exit_stack, 20);
|
2014-07-15 14:42:53 +04:00
|
|
|
#endif
|
2014-07-14 21:36:31 +04:00
|
|
|
ListDictionary_Unlock(thread_list);
|
|
|
|
}
|
2012-09-18 03:17:19 +04:00
|
|
|
}
|
|
|
|
|
2013-04-02 22:51:12 +04:00
|
|
|
BOOL GetExitCodeThread(HANDLE hThread, LPDWORD lpExitCode)
|
|
|
|
{
|
|
|
|
ULONG Type;
|
|
|
|
PVOID Object;
|
2014-07-15 13:11:59 +04:00
|
|
|
WINPR_THREAD *thread;
|
2013-04-02 22:51:12 +04:00
|
|
|
|
|
|
|
if (!winpr_Handle_GetInfo(hThread, &Type, &Object))
|
|
|
|
return FALSE;
|
|
|
|
|
2014-07-15 13:11:59 +04:00
|
|
|
thread = (WINPR_THREAD *) Object;
|
2013-04-02 22:51:12 +04:00
|
|
|
*lpExitCode = thread->dwExitCode;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2013-01-20 05:32:14 +04:00
|
|
|
HANDLE _GetCurrentThread(VOID)
|
2012-09-18 03:17:19 +04:00
|
|
|
{
|
2014-07-14 21:36:31 +04:00
|
|
|
HANDLE hdl = NULL;
|
|
|
|
pthread_t tid = pthread_self();
|
|
|
|
|
|
|
|
if (NULL == thread_list)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: function called without existing thread list!\n", __FUNCTION__);
|
|
|
|
#if defined(WITH_DEBUG_THREADS)
|
|
|
|
DumpThreadHandles();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else if (!ListDictionary_Contains(thread_list, &tid))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "[%s]: function called, but no matching entry in thread list!\n", __FUNCTION__);
|
|
|
|
#if defined(WITH_DEBUG_THREADS)
|
|
|
|
DumpThreadHandles();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
hdl = ListDictionary_GetItemValue(thread_list, &tid);
|
|
|
|
}
|
|
|
|
|
|
|
|
return hdl;
|
2012-09-18 03:17:19 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
DWORD GetCurrentThreadId(VOID)
|
|
|
|
{
|
2013-01-22 03:33:00 +04:00
|
|
|
pthread_t tid;
|
|
|
|
tid = pthread_self();
|
|
|
|
return (DWORD) tid;
|
2012-09-18 03:17:19 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
DWORD ResumeThread(HANDLE hThread)
|
|
|
|
{
|
2012-09-23 21:54:14 +04:00
|
|
|
ULONG Type;
|
|
|
|
PVOID Object;
|
2014-07-15 13:11:59 +04:00
|
|
|
WINPR_THREAD *thread;
|
2012-09-23 21:54:14 +04:00
|
|
|
|
|
|
|
if (!winpr_Handle_GetInfo(hThread, &Type, &Object))
|
|
|
|
return 0;
|
|
|
|
|
2014-07-15 13:11:59 +04:00
|
|
|
thread = (WINPR_THREAD *) Object;
|
2012-09-23 21:54:14 +04:00
|
|
|
pthread_mutex_lock(&thread->mutex);
|
|
|
|
|
|
|
|
if (!thread->started)
|
|
|
|
winpr_StartThread(thread);
|
2014-07-14 21:36:31 +04:00
|
|
|
else
|
|
|
|
fprintf(stderr, "[%s]: Thread already started!\n", __FUNCTION__);
|
2012-09-23 21:54:14 +04:00
|
|
|
|
|
|
|
pthread_mutex_unlock(&thread->mutex);
|
2012-09-18 03:17:19 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
DWORD SuspendThread(HANDLE hThread)
|
|
|
|
{
|
2014-07-14 21:36:31 +04:00
|
|
|
fprintf(stderr, "[%s]: Function not implemented!\n", __FUNCTION__);
|
2012-09-18 03:17:19 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL SwitchToThread(VOID)
|
|
|
|
{
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL TerminateThread(HANDLE hThread, DWORD dwExitCode)
|
|
|
|
{
|
2012-09-23 21:54:14 +04:00
|
|
|
ULONG Type;
|
|
|
|
PVOID Object;
|
2014-07-15 13:11:59 +04:00
|
|
|
WINPR_THREAD *thread;
|
2012-09-23 21:54:14 +04:00
|
|
|
|
|
|
|
if (!winpr_Handle_GetInfo(hThread, &Type, &Object))
|
|
|
|
return 0;
|
|
|
|
|
2014-07-15 13:11:59 +04:00
|
|
|
thread = (WINPR_THREAD *) Object;
|
2014-07-16 13:56:25 +04:00
|
|
|
thread->dwExitCode = dwExitCode;
|
2012-09-23 21:54:14 +04:00
|
|
|
pthread_mutex_lock(&thread->mutex);
|
2012-10-14 04:31:01 +04:00
|
|
|
#ifndef ANDROID
|
2012-09-23 21:54:14 +04:00
|
|
|
pthread_cancel(thread->thread);
|
2014-07-14 21:36:31 +04:00
|
|
|
#else
|
|
|
|
fprintf(stderr, "[%s]: Function not supported on this platform!\n", __FUNCTION__);
|
2012-10-14 04:31:01 +04:00
|
|
|
#endif
|
2012-09-23 21:54:14 +04:00
|
|
|
pthread_mutex_unlock(&thread->mutex);
|
2012-09-18 03:17:19 +04:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
#if defined(WITH_DEBUG_THREADS)
|
|
|
|
VOID DumpThreadHandles(void)
|
|
|
|
{
|
|
|
|
void *stack[20];
|
|
|
|
#if defined(HAVE_EXECINFO_H)
|
|
|
|
backtrace(stack, 20);
|
|
|
|
#endif
|
2014-07-16 13:56:25 +04:00
|
|
|
fprintf(stderr, "---------------- %s ----------------------\n", __FUNCTION);
|
2014-07-14 21:36:31 +04:00
|
|
|
fprintf(stderr, "---------------- Called from ----------------------------\n");
|
|
|
|
#if defined(HAVE_EXECINFO_H)
|
|
|
|
backtrace_symbols_fd(stack, 20, STDERR_FILENO);
|
|
|
|
#endif
|
|
|
|
fprintf(stderr, "---------------- Start Dumping thread handles -----------\n");
|
|
|
|
|
|
|
|
if (!thread_list)
|
|
|
|
fprintf(stderr, "All threads properly shut down and disposed of.\n");
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ULONG_PTR *keys = NULL;
|
|
|
|
ListDictionary_Lock(thread_list);
|
|
|
|
int x, count = ListDictionary_GetKeys(thread_list, &keys);
|
|
|
|
fprintf(stderr, "Dumping %d elements\n", count);
|
|
|
|
|
|
|
|
for (x=0; x<count; x++)
|
|
|
|
{
|
|
|
|
WINPR_THREAD *thread = ListDictionary_GetItemValue(thread_list, (void *)keys[x]);
|
|
|
|
fprintf(stderr, "Thread [%d] handle created still not closed!\n", x);
|
|
|
|
#if defined(HAVE_EXECINFO_H)
|
|
|
|
backtrace_symbols_fd(thread->create_stack, 20, STDERR_FILENO);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (thread->started)
|
|
|
|
fprintf(stderr, "Thread [%d] still running!\n", x);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Thread [%d] exited at:\n", x);
|
|
|
|
#if defined(HAVE_EXECINFO_H)
|
|
|
|
backtrace_symbols_fd(thread->exit_stack, 20, STDERR_FILENO);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (keys)
|
|
|
|
free(keys);
|
2014-07-15 14:42:53 +04:00
|
|
|
|
2014-07-14 21:36:31 +04:00
|
|
|
ListDictionary_Unlock(thread_list);
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(stderr, "---------------- End Dumping thread handles -------------\n");
|
|
|
|
}
|
|
|
|
#endif
|
2012-09-18 03:17:19 +04:00
|
|
|
#endif
|
|
|
|
|