📄 mutex.c
字号:
/*
* mutex.c
*
* Description:
* This translation unit implements mutual exclusion (mutex) primitives.
*
* Pthreads-win32 - POSIX Threads Library for Win32
* Copyright (C) 1998
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the Free
* Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
* MA 02111-1307, USA
*/
/* errno.h or a replacement file is included by pthread.h */
//#include <errno.h>
#include "pthread.h"
#include "implement.h"
static int
ptw32_mutex_check_need_init(pthread_mutex_t *mutex)
{
int result = 0;
/*
* The following guarded test is specifically for statically
* initialised mutexes (via PTHREAD_MUTEX_INITIALIZER).
*
* Note that by not providing this synchronisation we risk
* introducing race conditions into applications which are
* correctly written.
*
* Approach
* --------
* We know that static mutexes will not be PROCESS_SHARED
* so we can serialise access to internal state using
* Win32 Critical Sections rather than Win32 Mutexes.
*
* If using a single global lock slows applications down too much,
* multiple global locks could be created and hashed on some random
* value associated with each mutex, the pointer perhaps. At a guess,
* a good value for the optimal number of global locks might be
* the number of processors + 1.
*
*/
EnterCriticalSection(&ptw32_mutex_test_init_lock);
/*
* We got here possibly under race
* conditions. Check again inside the critical section
* and only initialise if the mutex is valid (not been destroyed).
* If a static mutex has been destroyed, the application can
* re-initialise it only by calling pthread_mutex_init()
* explicitly.
*/
if (*mutex == (pthread_mutex_t) PTW32_OBJECT_AUTO_INIT)
{
result = pthread_mutex_init(mutex, NULL);
}
else if (*mutex == NULL)
{
/*
* The mutex has been destroyed while we were waiting to
* initialise it, so the operation that caused the
* auto-initialisation should fail.
*/
result = EINVAL;
}
LeaveCriticalSection(&ptw32_mutex_test_init_lock);
return(result);
}
int
pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *attr)
{
int result = 0;
pthread_mutex_t mx;
if (mutex == NULL)
{
return EINVAL;
}
mx = (pthread_mutex_t) calloc(1, sizeof(*mx));
if (mx == NULL)
{
result = ENOMEM;
goto FAIL0;
}
mx->mutex = 0;
if (attr != NULL
&& *attr != NULL
&& (*attr)->pshared == PTHREAD_PROCESS_SHARED
)
{
/*
* Creating mutex that can be shared between
* processes.
*/
#if _POSIX_THREAD_PROCESS_SHARED
/*
* Not implemented yet.
*/
#error ERROR [__FILE__, line __LINE__]: Process shared mutexes are not supported yet.
mx->mutex = CreateMutex(NULL, FALSE, "FIXME FIXME FIXME");
if (mx->mutex == 0)
{
result = EAGAIN;
}
#else
result = ENOSYS;
#endif /* _POSIX_THREAD_PROCESS_SHARED */
}
else
{
if (ptw32_try_enter_critical_section != NULL
|| (attr != NULL
&& *attr != NULL
&& (*attr)->forcecs == 1)
)
{
/*
* Create a critical section.
*/
InitializeCriticalSection(&mx->cs);
/*
* Check that it works ok - since InitializeCriticalSection doesn't
* return success or failure.
*/
if (TryEnterCriticalSection(&mx->cs))
{
LeaveCriticalSection(&mx->cs);
}
else
{
DeleteCriticalSection(&mx->cs);
result = EAGAIN;
}
}
else
{
/*
* Create a mutex that can only be used within the
* current process
*/
mx->mutex = CreateMutex (NULL,
FALSE,
NULL);
if (mx->mutex == 0)
{
result = EAGAIN;
}
}
}
if (result != 0 && mx != NULL)
{
free(mx);
mx = NULL;
}
FAIL0:
*mutex = mx;
return(result);
}
int
pthread_mutex_destroy(pthread_mutex_t *mutex)
{
int result = 0;
pthread_mutex_t mx;
if (mutex == NULL
|| *mutex == NULL)
{
return EINVAL;
}
/*
* Check to see if we have something to delete.
*/
if (*mutex != (pthread_mutex_t) PTW32_OBJECT_AUTO_INIT)
{
mx = *mutex;
if ((result = pthread_mutex_trylock(&mx)) == 0)
{
/*
* FIXME!!!
* The mutex isn't held by another thread but we could still
* be too late invalidating the mutex below. Yet we can't do it
* earlier in case another thread needs to unlock the mutex
* that it's holding.
*/
*mutex = NULL;
pthread_mutex_unlock(&mx);
if (mx->mutex == 0)
{
DeleteCriticalSection(&mx->cs);
}
else
{
result = (CloseHandle (mx->mutex) ? 0 : EINVAL);
}
if (result == 0)
{
mx->mutex = 0;
free(mx);
}
else
{
/*
* Restore the mutex before we return the error.
*/
*mutex = mx;
}
}
}
else
{
/*
* See notes in ptw32_mutex_check_need_init() above also.
*/
EnterCriticalSection(&ptw32_mutex_test_init_lock);
/*
* Check again.
*/
if (*mutex == (pthread_mutex_t) PTW32_OBJECT_AUTO_INIT)
{
/*
* This is all we need to do to destroy a statically
* initialised mutex that has not yet been used (initialised).
* If we get to here, another thread
* waiting to initialise this mutex will get an EINVAL.
*/
*mutex = NULL;
}
else
{
/*
* The mutex has been initialised while we were waiting
* so assume it's in use.
*/
result = EBUSY;
}
LeaveCriticalSection(&ptw32_mutex_test_init_lock);
}
return(result);
}
int
pthread_mutexattr_init (pthread_mutexattr_t * attr)
/*
* ------------------------------------------------------
* DOCPUBLIC
* Initializes a mutex attributes object with default
* attributes.
*
* PARAMETERS
* attr
* pointer to an instance of pthread_mutexattr_t
*
*
* DESCRIPTION
* Initializes a mutex attributes object with default
* attributes.
*
* NOTES:
* 1) Used to define mutex types
*
* RESULTS
* 0 successfully initialized attr,
* ENOMEM insufficient memory for attr.
*
* ------------------------------------------------------
*/
{
pthread_mutexattr_t ma;
int result = 0;
ma = (pthread_mutexattr_t) calloc (1, sizeof (*ma));
if (ma == NULL)
{
result = ENOMEM;
}
*attr = ma;
return (result);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -