root/native_client_sdk/src/libraries/third_party/pthreads-win32/ptw32_callUserDestroyRoutines.c

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. ptw32_callUserDestroyRoutines

/*
 * ptw32_callUserDestroyRoutines.c
 *
 * Description:
 * This translation unit implements routines which are private to
 * the implementation and may be used throughout it.
 *
 * --------------------------------------------------------------------------
 *
 *      Pthreads-win32 - POSIX Threads Library for Win32
 *      Copyright(C) 1998 John E. Bossom
 *      Copyright(C) 1999,2005 Pthreads-win32 contributors
 * 
 *      Contact Email: rpj@callisto.canberra.edu.au
 * 
 *      The current list of contributors is contained
 *      in the file CONTRIBUTORS included with the source
 *      code distribution. The list can also be seen at the
 *      following World Wide Web location:
 *      http://sources.redhat.com/pthreads-win32/contributors.html
 * 
 *      This library is free software; you can redistribute it and/or
 *      modify it under the terms of the GNU Lesser 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
 *      Lesser General Public License for more details.
 * 
 *      You should have received a copy of the GNU Lesser General Public
 *      License along with this library in the file COPYING.LIB;
 *      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"

#if defined(__CLEANUP_CXX)
# if defined(_MSC_VER)
#  include <eh.h>
# elif defined(__WATCOMC__)
#  include <eh.h>
#  include <exceptio.h>
# else
#  if defined(__GNUC__) && __GNUC__ < 3
#    include <new.h>
#  else
#    include <new>
     using
       std::terminate;
#  endif
# endif
#endif

void
ptw32_callUserDestroyRoutines (pthread_t thread)
     /*
      * -------------------------------------------------------------------
      * DOCPRIVATE
      *
      * This the routine runs through all thread keys and calls
      * the destroy routines on the user's data for the current thread.
      * It simulates the behaviour of POSIX Threads.
      *
      * PARAMETERS
      *              thread
      *                      an instance of pthread_t
      *
      * RETURNS
      *              N/A
      * -------------------------------------------------------------------
      */
{
  ThreadKeyAssoc * assoc;

  if (thread.p != NULL)
    {
      ptw32_mcs_local_node_t threadLock;
      ptw32_mcs_local_node_t keyLock;
      int assocsRemaining;
      int iterations = 0;
      ptw32_thread_t * sp = (ptw32_thread_t *) thread.p;

      /*
       * Run through all Thread<-->Key associations
       * for the current thread.
       *
       * Do this process at most PTHREAD_DESTRUCTOR_ITERATIONS times.
       */
      do
        {
          assocsRemaining = 0;
          iterations++;

          ptw32_mcs_lock_acquire(&(sp->threadLock), &threadLock);
          /*
           * The pointer to the next assoc is stored in the thread struct so that
           * the assoc destructor in pthread_key_delete can adjust it
           * if it deletes this assoc. This can happen if we fail to acquire
           * both locks below, and are forced to release all of our locks,
           * leaving open the opportunity for pthread_key_delete to get in
           * before us.
           */
          sp->nextAssoc = sp->keys;
          ptw32_mcs_lock_release(&threadLock);

          for (;;)
            {
              void * value;
              pthread_key_t k;
              void (*destructor) (void *);

              /*
               * First we need to serialise with pthread_key_delete by locking
               * both assoc guards, but in the reverse order to our convention,
               * so we must be careful to avoid deadlock.
               */
              ptw32_mcs_lock_acquire(&(sp->threadLock), &threadLock);

              if ((assoc = (ThreadKeyAssoc *)sp->nextAssoc) == NULL)
                {
                  /* Finished */
                  ptw32_mcs_lock_release(&threadLock);
                  break;
                }
              else
                {
                  /*
                   * assoc->key must be valid because assoc can't change or be
                   * removed from our chain while we hold at least one lock. If
                   * the assoc was on our key chain then the key has not been
                   * deleted yet.
                   *
                   * Now try to acquire the second lock without deadlocking.
                   * If we fail, we need to relinquish the first lock and the
                   * processor and then try to acquire them all again.
                   */
                  if (ptw32_mcs_lock_try_acquire(&(assoc->key->keyLock), &keyLock) == EBUSY)
                    {
                      ptw32_mcs_lock_release(&threadLock);
                      Sleep(0);
                      /*
                       * Go around again.
                       * If pthread_key_delete has removed this assoc in the meantime,
                       * sp->nextAssoc will point to a new assoc.
                       */
                      continue;
                    }
                }

              /* We now hold both locks */

              sp->nextAssoc = assoc->nextKey;

              /*
               * Key still active; pthread_key_delete
               * will block on these same mutexes before
               * it can release actual key; therefore,
               * key is valid and we can call the destroy
               * routine;
               */
              k = assoc->key;
              destructor = k->destructor;
              value = TlsGetValue(k->key);
              TlsSetValue (k->key, NULL);

              // Every assoc->key exists and has a destructor
              if (value != NULL && iterations <= PTHREAD_DESTRUCTOR_ITERATIONS)
                {
                  /*
                   * Unlock both locks before the destructor runs.
                   * POSIX says pthread_key_delete can be run from destructors,
                   * and that probably includes with this key as target.
                   * pthread_setspecific can also be run from destructors and
                   * also needs to be able to access the assocs.
                   */
                  ptw32_mcs_lock_release(&threadLock);
                  ptw32_mcs_lock_release(&keyLock);

                  assocsRemaining++;

#if defined(__cplusplus)

                  try
                    {
                      /*
                       * Run the caller's cleanup routine.
                       */
                      destructor (value);
                    }
                  catch (...)
                    {
                      /*
                       * A system unexpected exception has occurred
                       * running the user's destructor.
                       * We get control back within this block in case
                       * the application has set up it's own terminate
                       * handler. Since we are leaving the thread we
                       * should not get any internal pthreads
                       * exceptions.
                       */
                      terminate ();
                    }

#else /* __cplusplus */

                  /*
                   * Run the caller's cleanup routine.
                   */
                  destructor (value);

#endif /* __cplusplus */

                }
              else
                {
                  /*
                   * Remove association from both the key and thread chains
                   * and reclaim it's memory resources.
                   */
                  ptw32_tkAssocDestroy (assoc);
                  ptw32_mcs_lock_release(&threadLock);
                  ptw32_mcs_lock_release(&keyLock);
                }
            }
        }
      while (assocsRemaining);
    }
}                               /* ptw32_callUserDestroyRoutines */

/* [<][>][^][v][top][bottom][index][help] */