openldap/libraries/libldap/thr_nt.c

253 lines
4.7 KiB
C
Raw Normal View History

2003-11-26 15:16:36 +08:00
/* thr_nt.c - wrapper around NT threads */
/* $OpenLDAP$ */
2003-11-26 15:16:36 +08:00
/* This work is part of OpenLDAP Software <http://www.openldap.org/>.
*
2020-01-10 00:50:21 +08:00
* Copyright 1998-2020 The OpenLDAP Foundation.
1999-01-17 23:46:19 +08:00
* All rights reserved.
*
2003-11-26 15:16:36 +08:00
* Redistribution and use in source and binary forms, with or without
* modification, are permitted only as authorized by the OpenLDAP
* Public License.
*
* A copy of this license is available in file LICENSE in the
* top-level directory of the distribution or, alternatively, at
* <http://www.OpenLDAP.org/license.html>.
1999-01-17 23:46:19 +08:00
*/
#include "portable.h"
#if defined( HAVE_NT_THREADS )
2006-09-14 16:06:49 +08:00
#define _WIN32_WINNT 0x0400
#include <windows.h>
#include <process.h>
#include "ldap_pvt_thread.h" /* Get the thread interface */
#define LDAP_THREAD_IMPLEMENTATION
#include "ldap_thr_debug.h" /* May rename the symbols defined below */
typedef struct ldap_int_thread_s {
long tid;
HANDLE thd;
} ldap_int_thread_s;
#ifndef NT_MAX_THREADS
#define NT_MAX_THREADS 1024
#endif
2004-03-17 17:59:03 +08:00
static ldap_int_thread_s tids[NT_MAX_THREADS];
static int ntids;
2001-12-07 12:03:25 +08:00
/* mingw compiler very sensitive about getting prototypes right */
typedef unsigned __stdcall thrfunc_t(void *);
int
2000-06-07 03:59:34 +08:00
ldap_int_thread_initialize( void )
{
return 0;
}
int
2000-06-07 03:59:34 +08:00
ldap_int_thread_destroy( void )
{
return 0;
}
int
ldap_int_mutex_firstcreate( ldap_int_thread_mutex_t *mutex )
{
if ( *mutex == NULL ) {
HANDLE p = CreateMutex( NULL, 0, NULL );
if ( InterlockedCompareExchangePointer((PVOID*)mutex, (PVOID)p, NULL) != NULL)
CloseHandle( p );
}
return 0;
}
int
ldap_pvt_thread_create( ldap_pvt_thread_t * thread,
int detach,
void *(*start_routine)( void *),
void *arg)
{
2001-12-07 12:03:25 +08:00
unsigned tid;
2001-05-07 02:58:45 +08:00
HANDLE thd;
int rc = -1;
2001-05-07 02:58:45 +08:00
thd = (HANDLE) _beginthreadex(NULL, LDAP_PVT_THREAD_STACK_SIZE, (thrfunc_t *) start_routine,
2001-12-07 12:03:25 +08:00
arg, 0, &tid);
2001-05-07 02:58:45 +08:00
if ( thd ) {
*thread = (ldap_pvt_thread_t) tid;
2004-03-17 17:59:03 +08:00
tids[ntids].tid = tid;
tids[ntids].thd = thd;
ntids++;
rc = 0;
}
return rc;
}
void
ldap_pvt_thread_exit( void *retval )
{
_endthread( );
}
int
ldap_pvt_thread_join( ldap_pvt_thread_t thread, void **thread_return )
{
DWORD status;
int i;
for (i=0; i<ntids; i++) {
2004-03-17 17:59:03 +08:00
if ( tids[i].tid == thread )
break;
}
if ( i > ntids ) return -1;
2004-03-17 17:59:03 +08:00
status = WaitForSingleObject( tids[i].thd, INFINITE );
for (; i<ntids; i++) {
2004-03-17 17:59:03 +08:00
tids[i] = tids[i+1];
}
ntids--;
2000-09-13 04:39:13 +08:00
return status == WAIT_FAILED ? -1 : 0;
}
int
ldap_pvt_thread_kill( ldap_pvt_thread_t thread, int signo )
{
return 0;
}
int
ldap_pvt_thread_yield( void )
{
Sleep( 0 );
return 0;
}
int
ldap_pvt_thread_cond_init( ldap_pvt_thread_cond_t *cond )
{
*cond = CreateEvent( NULL, FALSE, FALSE, NULL );
return( 0 );
}
int
ldap_pvt_thread_cond_destroy( ldap_pvt_thread_cond_t *cv )
{
CloseHandle( *cv );
return( 0 );
}
int
ldap_pvt_thread_cond_signal( ldap_pvt_thread_cond_t *cond )
{
SetEvent( *cond );
return( 0 );
}
int
ldap_pvt_thread_cond_wait( ldap_pvt_thread_cond_t *cond,
ldap_pvt_thread_mutex_t *mutex )
{
SignalObjectAndWait( *mutex, *cond, INFINITE, FALSE );
WaitForSingleObject( *mutex, INFINITE );
return( 0 );
}
int
ldap_pvt_thread_cond_broadcast( ldap_pvt_thread_cond_t *cond )
{
while ( WaitForSingleObject( *cond, 0 ) == WAIT_TIMEOUT )
SetEvent( *cond );
return( 0 );
}
int
ldap_pvt_thread_mutex_init( ldap_pvt_thread_mutex_t *mutex )
{
*mutex = CreateMutex( NULL, 0, NULL );
return ( 0 );
}
int
ldap_pvt_thread_mutex_recursive_init( ldap_pvt_thread_mutex_t *mutex )
{
/* All NT mutexes are recursive */
return ldap_pvt_thread_mutex_init( mutex );
}
int
ldap_pvt_thread_mutex_destroy( ldap_pvt_thread_mutex_t *mutex )
{
CloseHandle( *mutex );
return ( 0 );
}
int
ldap_pvt_thread_mutex_lock( ldap_pvt_thread_mutex_t *mutex )
{
2000-09-13 04:39:13 +08:00
DWORD status;
status = WaitForSingleObject( *mutex, INFINITE );
return status == WAIT_FAILED ? -1 : 0;
}
int
ldap_pvt_thread_mutex_unlock( ldap_pvt_thread_mutex_t *mutex )
{
ReleaseMutex( *mutex );
return ( 0 );
}
int
ldap_pvt_thread_mutex_trylock( ldap_pvt_thread_mutex_t *mp )
{
DWORD status;
status = WaitForSingleObject( *mp, 0 );
2000-09-13 04:39:13 +08:00
return status == WAIT_FAILED || status == WAIT_TIMEOUT
? -1 : 0;
}
ldap_pvt_thread_t
ldap_pvt_thread_self( void )
{
return GetCurrentThreadId();
}
int
ldap_pvt_thread_key_create( ldap_pvt_thread_key_t *keyp )
{
DWORD key = TlsAlloc();
if ( key != TLS_OUT_OF_INDEXES ) {
*keyp = key;
return 0;
} else {
return -1;
}
}
int
ldap_pvt_thread_key_destroy( ldap_pvt_thread_key_t key )
{
/* TlsFree returns 0 on failure */
return( TlsFree( key ) == 0 );
}
int
ldap_pvt_thread_key_setdata( ldap_pvt_thread_key_t key, void *data )
{
return ( TlsSetValue( key, data ) == 0 );
}
int
ldap_pvt_thread_key_getdata( ldap_pvt_thread_key_t key, void **data )
{
void *ptr = TlsGetValue( key );
*data = ptr;
return( ptr ? GetLastError() : 0 );
}
#endif