mariadb/mysys/my_winthread.c

151 lines
3.4 KiB
C
Raw Normal View History

/* Copyright (C) 2000 MySQL AB, 2008-2009 Sun Microsystems, Inc
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; version 2 of the License.
This program is distributed in the hope that it will be useful,
2000-07-31 21:29:14 +02:00
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
2000-07-31 21:29:14 +02:00
/*****************************************************************************
** Simulation of posix threads calls for Windows
2000-07-31 21:29:14 +02:00
*****************************************************************************/
#if defined (_WIN32)
2000-09-21 00:58:38 +02:00
/* SAFE_MUTEX will not work until the thread structure is up to date */
#undef SAFE_MUTEX
2000-07-31 21:29:14 +02:00
#include "mysys_priv.h"
#include <process.h>
#include <signal.h>
2000-07-31 21:29:14 +02:00
static void install_sigabrt_handler(void);
2000-07-31 21:29:14 +02:00
struct thread_start_parameter
2000-07-31 21:29:14 +02:00
{
pthread_handler func;
void *arg;
2000-07-31 21:29:14 +02:00
};
/**
Adapter to @c pthread_mutex_trylock()
@retval 0 Mutex was acquired
@retval EBUSY Mutex was already locked by a thread
*/
int
win_pthread_mutex_trylock(pthread_mutex_t *mutex)
{
if (TryEnterCriticalSection(mutex))
{
/* Don't allow recursive lock */
if (mutex->RecursionCount > 1){
LeaveCriticalSection(mutex);
return EBUSY;
}
return 0;
}
return EBUSY;
}
static unsigned int __stdcall pthread_start(void *p)
2000-07-31 21:29:14 +02:00
{
struct thread_start_parameter *par= (struct thread_start_parameter *)p;
pthread_handler func= par->func;
void *arg= par->arg;
free(p);
(*func)(arg);
return 0;
2000-07-31 21:29:14 +02:00
}
int pthread_create(pthread_t *thread_id, const pthread_attr_t *attr,
pthread_handler func, void *param)
2000-07-31 21:29:14 +02:00
{
uintptr_t handle;
struct thread_start_parameter *par;
unsigned int stack_size;
2000-07-31 21:29:14 +02:00
DBUG_ENTER("pthread_create");
par= (struct thread_start_parameter *)malloc(sizeof(*par));
if (!par)
goto error_return;
2000-07-31 21:29:14 +02:00
par->func= func;
par->arg= param;
stack_size= attr?attr->dwStackSize:0;
handle= _beginthreadex(NULL, stack_size , pthread_start, par, 0, thread_id);
if (!handle)
goto error_return;
DBUG_PRINT("info", ("thread id=%u",*thread_id));
/* Do not need thread handle, close it */
CloseHandle((HANDLE)handle);
2000-07-31 21:29:14 +02:00
DBUG_RETURN(0);
error_return:
DBUG_PRINT("error",
("Can't create thread to handle request (error %d)",errno));
DBUG_RETURN(-1);
2000-07-31 21:29:14 +02:00
}
void pthread_exit(void *a)
2000-07-31 21:29:14 +02:00
{
_endthreadex(0);
2000-07-31 21:29:14 +02:00
}
int pthread_join(pthread_t thread, void **value_ptr)
{
DWORD ret;
HANDLE handle;
handle= OpenThread(SYNCHRONIZE, FALSE, thread);
if (!handle)
{
errno= EINVAL;
goto error_return;
}
ret= WaitForSingleObject(handle, INFINITE);
if(ret != WAIT_OBJECT_0)
{
errno= EINVAL;
goto error_return;
}
CloseHandle(handle);
return 0;
error_return:
if(handle)
CloseHandle(handle);
return -1;
}
2000-07-31 21:29:14 +02:00
int pthread_cancel(pthread_t thread)
{
HANDLE handle= 0;
BOOL ok= FALSE;
handle= OpenThread(THREAD_TERMINATE, FALSE, thread);
if (handle)
{
ok= TerminateThread(handle,0);
CloseHandle(handle);
}
if (ok)
return 0;
errno= EINVAL;
return -1;
}
2000-07-31 21:29:14 +02:00
#endif