diff options
| author | Guenter Bartsch <guenter@users.sourceforge.net> | 2003-04-20 16:42:06 +0000 | 
|---|---|---|
| committer | Guenter Bartsch <guenter@users.sourceforge.net> | 2003-04-20 16:42:06 +0000 | 
| commit | ce4f3efd78c110c015c54eab05f201d1caac5f8d (patch) | |
| tree | 668c555d5de70518b93426f45fe3e44d7f06663c /win32/contrib/pthreads/mutex.c | |
| parent | 80f6215b607f0ed76f7f75deaf4e4668bc6f244d (diff) | |
| download | xine-lib-ce4f3efd78c110c015c54eab05f201d1caac5f8d.tar.gz xine-lib-ce4f3efd78c110c015c54eab05f201d1caac5f8d.tar.bz2 | |
importing win32 contrib sources and msvc build environment
CVS patchset: 4641
CVS date: 2003/04/20 16:42:06
Diffstat (limited to 'win32/contrib/pthreads/mutex.c')
| -rw-r--r-- | win32/contrib/pthreads/mutex.c | 784 | 
1 files changed, 784 insertions, 0 deletions
| diff --git a/win32/contrib/pthreads/mutex.c b/win32/contrib/pthreads/mutex.c new file mode 100644 index 000000000..7896570c1 --- /dev/null +++ b/win32/contrib/pthreads/mutex.c @@ -0,0 +1,784 @@ +/*
 + * 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
 + */
 +
 +#include "pthread.h"
 +#include "implement.h"
 +
 +
 +static INLINE 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_INITIALIZER)
 +    {
 +      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;
 +    }
 +
 +  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.
 +
 +
 +#else
 +
 +      result = ENOSYS;
 +      goto FAIL0;
 +
 +#endif /* _POSIX_THREAD_PROCESS_SHARED */
 +
 +    }
 +
 +  mx->lock_idx = PTW32_MUTEX_LOCK_IDX_INIT;
 +  mx->recursive_count = 0;
 +  mx->kind = (attr == NULL || *attr == NULL
 +              ? PTHREAD_MUTEX_DEFAULT
 +              : (*attr)->kind);
 +  mx->ownerThread = NULL;
 +  InitializeCriticalSection( &mx->try_lock_cs );
 +  mx->wait_sema = CreateSemaphore( NULL, 0, 1, NULL );
 +
 +  if( NULL == mx->wait_sema )
 +    {
 +      DeleteCriticalSection( &mx->try_lock_cs );
 +      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_INITIALIZER)
 +    {
 +      mx = *mutex;
 +
 +      result = pthread_mutex_trylock(&mx);
 +
 +      /*
 +       * The mutex type may not be RECURSIVE therefore trylock may return EBUSY if
 +       * we already own the mutex. Here we are assuming that it's OK to destroy
 +       * a mutex that we own and have locked recursively. Is this correct?
 +       *
 +       * For FAST mutexes we record the owner as ANONYMOUS for speed. In this
 +       * case we assume that the thread calling pthread_mutex_destroy() is the
 +       * owner, if the mutex is owned at all.
 +       */
 +      if (result == 0
 +          || mx->ownerThread == (pthread_t) PTW32_MUTEX_OWNER_ANONYMOUS
 +          || pthread_equal( mx->ownerThread, pthread_self() ) )
 +        {
 +          /*
 +           * FIXME!!!
 +           * The mutex isn't held by another thread but we could still
 +           * be too late invalidating the mutex below since another thread
 +           * may alredy have entered mutex_lock and the check for a valid
 +           * *mutex != NULL.
 +           */
 +          *mutex = NULL;
 +
 +          result = pthread_mutex_unlock(&mx);
 +
 +          if (result == 0)
 +            {
 +              DeleteCriticalSection( &mx->try_lock_cs );
 +              CloseHandle( mx->wait_sema );
 +              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_INITIALIZER)
 +        {
 +          /*
 +           * 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;
 +    }
 +
 +  ma->pshared = PTHREAD_PROCESS_PRIVATE;
 +  ma->kind = PTHREAD_MUTEX_DEFAULT;
 +
 +  *attr = ma;
 +
 +  return (result);
 +
 +}                               /* pthread_mutexattr_init */
 +
 +
 +int
 +pthread_mutexattr_destroy (pthread_mutexattr_t * attr)
 +     /*
 +      * ------------------------------------------------------
 +      * DOCPUBLIC
 +      *      Destroys a mutex attributes object. The object can
 +      *      no longer be used.
 +      *
 +      * PARAMETERS
 +      *      attr
 +      *              pointer to an instance of pthread_mutexattr_t
 +      *
 +      *
 +      * DESCRIPTION
 +      *      Destroys a mutex attributes object. The object can
 +      *      no longer be used.
 +      *
 +      *      NOTES:
 +      *              1)      Does not affect mutexes created using 'attr'
 +      *
 +      * RESULTS
 +      *              0               successfully released attr,
 +      *              EINVAL          'attr' is invalid.
 +      *
 +      * ------------------------------------------------------
 +      */
 +{
 +  int result = 0;
 +
 +  if (attr == NULL || *attr == NULL)
 +    {
 +      result = EINVAL;
 +    }
 +  else
 +    {
 +      pthread_mutexattr_t ma = *attr;
 +
 +      *attr = NULL;
 +      free (ma);
 +
 +      result = 0;
 +    }
 +
 +  return (result);
 +
 +}                               /* pthread_mutexattr_destroy */
 +
 +
 +int
 +pthread_mutexattr_getpshared (const pthread_mutexattr_t * attr,
 +			      int *pshared)
 +     /*
 +      * ------------------------------------------------------
 +      * DOCPUBLIC
 +      *      Determine whether mutexes created with 'attr' can be
 +      *      shared between processes.
 +      *
 +      * PARAMETERS
 +      *      attr
 +      *              pointer to an instance of pthread_mutexattr_t
 +      *
 +      *      pshared
 +      *              will be set to one of:
 +      *
 +      *                      PTHREAD_PROCESS_SHARED
 +      *                              May be shared if in shared memory
 +      *
 +      *                      PTHREAD_PROCESS_PRIVATE
 +      *                              Cannot be shared.
 +      *
 +      *
 +      * DESCRIPTION
 +      *      Mutexes creatd with 'attr' can be shared between
 +      *      processes if pthread_mutex_t variable is allocated
 +      *      in memory shared by these processes.
 +      *      NOTES:
 +      *              1)      pshared mutexes MUST be allocated in shared
 +      *                      memory.
 +      *              2)      The following macro is defined if shared mutexes
 +      *                      are supported:
 +      *                              _POSIX_THREAD_PROCESS_SHARED
 +      *
 +      * RESULTS
 +      *              0               successfully retrieved attribute,
 +      *              EINVAL          'attr' is invalid,
 +      *
 +      * ------------------------------------------------------
 +      */
 +{
 +  int result;
 +
 +  if ((attr != NULL && *attr != NULL) &&
 +      (pshared != NULL))
 +    {
 +      *pshared = (*attr)->pshared;
 +      result = 0;
 +    }
 +  else
 +    {
 +      *pshared = PTHREAD_PROCESS_PRIVATE;
 +      result = EINVAL;
 +    }
 +
 +  return (result);
 +
 +}                               /* pthread_mutexattr_getpshared */
 +
 +
 +int
 +pthread_mutexattr_setpshared (pthread_mutexattr_t * attr,
 +			      int pshared)
 +     /*
 +      * ------------------------------------------------------
 +      * DOCPUBLIC
 +      *      Mutexes created with 'attr' can be shared between
 +      *      processes if pthread_mutex_t variable is allocated
 +      *      in memory shared by these processes.
 +      *
 +      * PARAMETERS
 +      *      attr
 +      *              pointer to an instance of pthread_mutexattr_t
 +      *
 +      *      pshared
 +      *              must be one of:
 +      *
 +      *                      PTHREAD_PROCESS_SHARED
 +      *                              May be shared if in shared memory
 +      *
 +      *                      PTHREAD_PROCESS_PRIVATE
 +      *                              Cannot be shared.
 +      *
 +      * DESCRIPTION
 +      *      Mutexes creatd with 'attr' can be shared between
 +      *      processes if pthread_mutex_t variable is allocated
 +      *      in memory shared by these processes.
 +      *
 +      *      NOTES:
 +      *              1)      pshared mutexes MUST be allocated in shared
 +      *                      memory.
 +      *
 +      *              2)      The following macro is defined if shared mutexes
 +      *                      are supported:
 +      *                              _POSIX_THREAD_PROCESS_SHARED
 +      *
 +      * RESULTS
 +      *              0               successfully set attribute,
 +      *              EINVAL          'attr' or pshared is invalid,
 +      *              ENOSYS          PTHREAD_PROCESS_SHARED not supported,
 +      *
 +      * ------------------------------------------------------
 +      */
 +{
 +  int result;
 +
 +  if ((attr != NULL && *attr != NULL) &&
 +      ((pshared == PTHREAD_PROCESS_SHARED) ||
 +       (pshared == PTHREAD_PROCESS_PRIVATE)))
 +    {
 +      if (pshared == PTHREAD_PROCESS_SHARED)
 +        {
 +
 +#if !defined( _POSIX_THREAD_PROCESS_SHARED )
 +
 +          result = ENOSYS;
 +          pshared = PTHREAD_PROCESS_PRIVATE;
 +
 +#else
 +
 +          result = 0;
 +
 +#endif /* _POSIX_THREAD_PROCESS_SHARED */
 +
 +        }
 +      else
 +        {
 +          result = 0;
 +        }
 +
 +      (*attr)->pshared = pshared;
 +    }
 +  else
 +    {
 +      result = EINVAL;
 +    }
 +
 +  return (result);
 +
 +}                               /* pthread_mutexattr_setpshared */
 +
 +
 +int
 +pthread_mutexattr_settype (pthread_mutexattr_t * attr,
 +						   int kind)
 +     /*
 +      * ------------------------------------------------------
 +      *
 +      * DOCPUBLIC
 +      * The pthread_mutexattr_settype() and
 +      * pthread_mutexattr_gettype() functions  respectively set and
 +      * get the mutex type  attribute. This attribute is set in  the
 +      * type parameter to these functions.
 +      *
 +      * PARAMETERS
 +      *      attr
 +      *              pointer to an instance of pthread_mutexattr_t
 +      *
 +      *      type
 +      *              must be one of:
 +      *
 +      *                      PTHREAD_MUTEX_DEFAULT
 +      *
 +      *                      PTHREAD_MUTEX_NORMAL
 +      *
 +      *                      PTHREAD_MUTEX_ERRORCHECK
 +      *
 +      *                      PTHREAD_MUTEX_RECURSIVE
 +      *
 +      * DESCRIPTION
 +      * The pthread_mutexattr_settype() and
 +      * pthread_mutexattr_gettype() functions  respectively set and
 +      * get the mutex type  attribute. This attribute is set in  the
 +      * type  parameter to these functions. The default value of the
 +      * type  attribute is  PTHREAD_MUTEX_DEFAULT.
 +      * 
 +      * The type of mutex is contained in the type  attribute of the
 +      * mutex attributes. Valid mutex types include:
 +      *
 +      * PTHREAD_MUTEX_NORMAL
 +      *          This type of mutex does  not  detect  deadlock.  A
 +      *          thread  attempting  to  relock  this mutex without
 +      *          first unlocking it will  deadlock.  Attempting  to
 +      *          unlock  a  mutex  locked  by  a  different  thread
 +      *          results  in  undefined  behavior.  Attempting   to
 +      *          unlock  an  unlocked  mutex  results  in undefined
 +      *          behavior.
 +      * 
 +      * PTHREAD_MUTEX_ERRORCHECK
 +      *          This type of  mutex  provides  error  checking.  A
 +      *          thread  attempting  to  relock  this mutex without
 +      *          first unlocking it will return with  an  error.  A
 +      *          thread  attempting to unlock a mutex which another
 +      *          thread has locked will return  with  an  error.  A
 +      *          thread attempting to unlock an unlocked mutex will
 +      *          return with an error.
 +      *
 +      * PTHREAD_MUTEX_DEFAULT
 +      *          Same as PTHREAD_MUTEX_NORMAL.
 +      * 
 +      * PTHREAD_MUTEX_RECURSIVE
 +      *          A thread attempting to relock this  mutex  without
 +      *          first  unlocking  it  will  succeed in locking the
 +      *          mutex. The relocking deadlock which can occur with
 +      *          mutexes of type  PTHREAD_MUTEX_NORMAL cannot occur
 +      *          with this type of mutex. Multiple  locks  of  this
 +      *          mutex  require  the  same  number  of  unlocks  to
 +      *          release  the  mutex  before  another  thread   can
 +      *          acquire the mutex. A thread attempting to unlock a
 +      *          mutex which another thread has locked will  return
 +      *          with  an  error. A thread attempting to  unlock an
 +      *          unlocked mutex will return  with  an  error.  This
 +      *          type  of mutex is only supported for mutexes whose
 +      *          process        shared         attribute         is
 +      *          PTHREAD_PROCESS_PRIVATE.
 +      *
 +      * RESULTS
 +      *              0               successfully set attribute,
 +      *              EINVAL          'attr' or 'type' is invalid,
 +      *
 +      * ------------------------------------------------------
 +      */
 +{
 +  int result = 0;
 +
 +  if ((attr != NULL && *attr != NULL))
 +    {
 +      switch (kind)
 +        {
 +        case PTHREAD_MUTEX_FAST_NP:
 +        case PTHREAD_MUTEX_RECURSIVE_NP:
 +        case PTHREAD_MUTEX_ERRORCHECK_NP:
 +          (*attr)->kind = kind;
 +          break;
 +        default:
 +          result = EINVAL;
 +          break;
 +        }
 +    }
 +  else
 +    {
 +      result = EINVAL;
 +    }
 +  
 +  return (result);
 +}                               /* pthread_mutexattr_settype */
 +
 +
 +int
 +pthread_mutexattr_gettype (pthread_mutexattr_t * attr,
 +                           int *kind)
 +{
 +  int result = 0;
 +
 +  if (attr != NULL && *attr != NULL && kind != NULL)
 +    {
 +      *kind = (*attr)->kind;
 +    }
 +  else
 +    {
 +      result = EINVAL;
 +    }
 +
 +  return (result);
 +}
 +
 +
 +int
 +pthread_mutex_lock(pthread_mutex_t *mutex)
 +{
 +  int result = 0;
 +  pthread_mutex_t mx;
 +
 +
 +  if (mutex == NULL || *mutex == NULL)
 +    {
 +      return EINVAL;
 +    }
 +
 +  /*
 +   * We do a quick check to see if we need to do more work
 +   * to initialise a static mutex. We check
 +   * again inside the guarded section of ptw32_mutex_check_need_init()
 +   * to avoid race conditions.
 +   */
 +  if (*mutex == PTHREAD_MUTEX_INITIALIZER)
 +    {
 +      if ((result = ptw32_mutex_check_need_init(mutex)) != 0)
 +        {
 +          return(result);
 +        }
 +    }
 +
 +  mx = *mutex;
 +
 +  if( 0 == InterlockedIncrement( &mx->lock_idx ) )
 +    {
 +      mx->recursive_count = 1;
 +      mx->ownerThread = (mx->kind != PTHREAD_MUTEX_FAST_NP
 +                         ? pthread_self()
 +                         : (pthread_t) PTW32_MUTEX_OWNER_ANONYMOUS);
 +    }
 +  else
 +    {
 +      if( mx->kind != PTHREAD_MUTEX_FAST_NP &&
 +          pthread_equal( mx->ownerThread, pthread_self() ) )
 +	{
 +          mx->lock_idx--;
 +
 +          if( mx->kind == PTHREAD_MUTEX_RECURSIVE_NP )
 +            {
 +              mx->recursive_count++;
 +            }
 +          else
 +            {
 +              result = EDEADLK;
 +            }
 +        }
 +      else
 +        {
 +          WaitForSingleObject( mx->wait_sema, INFINITE );
 +          mx->recursive_count = 1;
 +          mx->ownerThread = (mx->kind != PTHREAD_MUTEX_FAST_NP
 +                             ? pthread_self()
 +                             : (pthread_t) PTW32_MUTEX_OWNER_ANONYMOUS);
 +        }
 +    }
 +
 +  return(result);
 +}
 +
 +int
 +pthread_mutex_unlock(pthread_mutex_t *mutex)
 +{
 +  int result = 0;
 +  pthread_mutex_t mx;
 +
 +  if (mutex == NULL || *mutex == NULL)
 +    {
 +      return EINVAL;
 +    }
 +
 +  mx = *mutex;
 +
 +  /* 
 +   * If the thread calling us holds the mutex then there is no
 +   * race condition. If another thread holds the
 +   * lock then we shouldn't be in here.
 +   */
 +  if (mx != PTHREAD_MUTEX_INITIALIZER)
 +    {
 +      if (mx->ownerThread == (pthread_t) PTW32_MUTEX_OWNER_ANONYMOUS
 +          || pthread_equal(mx->ownerThread, pthread_self()))
 +	{
 +          if( mx->kind != PTHREAD_MUTEX_RECURSIVE_NP
 +              || 0 == --mx->recursive_count )
 +	    {
 +	      mx->ownerThread = NULL;
 +	      
 +              EnterCriticalSection( &mx->try_lock_cs );
 +
 +              if( InterlockedDecrement( &mx->lock_idx ) >= 0 )
 +		{
 +                  /* Someone is waiting on that mutex */
 +                  ReleaseSemaphore( mx->wait_sema, 1, NULL );
 +		}
 +
 +              LeaveCriticalSection( &mx->try_lock_cs );
 +	    }
 +	}
 +      else
 +	{
 +	  result = EPERM;
 +	}
 +    }
 +  else
 +    {
 +      result = EINVAL;
 +    }
 +
 +  return(result);
 +}
 +
 +int
 +pthread_mutex_trylock(pthread_mutex_t *mutex)
 +{
 +  int result = 0;
 +  pthread_mutex_t mx;
 +
 +  if (mutex == NULL || *mutex == NULL)
 +    {
 +      return EINVAL;
 +    }
 +
 +  /*
 +   * We do a quick check to see if we need to do more work
 +   * to initialise a static mutex. We check
 +   * again inside the guarded section of ptw32_mutex_check_need_init()
 +   * to avoid race conditions.
 +   */
 +  if (*mutex == PTHREAD_MUTEX_INITIALIZER)
 +    {
 +      result = ptw32_mutex_check_need_init(mutex);
 +    }
 +
 +  mx = *mutex;
 +
 +  if (result == 0)
 +    {
 +      /* Try to lock only if mutex seems available */
 +      if( PTW32_MUTEX_LOCK_IDX_INIT == mx->lock_idx )
 +	{
 +          EnterCriticalSection( &mx->try_lock_cs );
 +
 +          if( 0 == InterlockedIncrement( &mx->lock_idx ) )
 +	    {
 +              mx->recursive_count = 1;
 +              mx->ownerThread = (mx->kind != PTHREAD_MUTEX_FAST_NP
 +                                 ? pthread_self()
 +                                 : (pthread_t) PTW32_MUTEX_OWNER_ANONYMOUS);
 +	    }
 +          else
 +            {
 +              mx->lock_idx--;
 +              result = EBUSY;
 +            }
 +
 +          LeaveCriticalSection( &mx->try_lock_cs );
 +        }
 +      else
 +        {
 +          result = EBUSY;
 +        }
 +    }
 +
 +  return(result);
 +}
 | 
