233 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			233 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  * 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 */
 |