| 
									
										
										
										
											1993-01-26 13:33:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | /* Thread package.
 | 
					
						
							|  |  |  |    This is intended to be usable independently from Python. | 
					
						
							|  |  |  |    The implementation for system foobar is in a file thread_foobar.h | 
					
						
							|  |  |  |    which is included by this file dependent on config settings. | 
					
						
							|  |  |  |    Stuff shared by all thread_*.h files is collected here. */ | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-12 11:05:12 +00:00
										 |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											1992-08-17 08:59:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-16 11:35:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifndef _POSIX_THREADS
 | 
					
						
							|  |  |  | /* This means pthreads are not implemented in libc headers, hence the macro
 | 
					
						
							|  |  |  |    not present in unistd.h. But they still can be implemented as an external | 
					
						
							|  |  |  |    library (e.g. gnu pth in pthread emulation) */ | 
					
						
							|  |  |  | # ifdef HAVE_PTHREAD_H
 | 
					
						
							|  |  |  | #  include <pthread.h> /* _POSIX_THREADS */
 | 
					
						
							|  |  |  | # endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-04-07 16:07:23 +00:00
										 |  |  | #ifndef DONT_HAVE_STDIO_H
 | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											1999-04-07 16:07:23 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-12-03 16:54:45 +00:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-04-29 21:48:34 +00:00
										 |  |  | #ifdef __sgi
 | 
					
						
							|  |  |  | #ifndef HAVE_PTHREAD_H /* XXX Need to check in configure.in */
 | 
					
						
							|  |  |  | #undef _POSIX_THREADS
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-10-01 20:42:43 +00:00
										 |  |  | #include "pythread.h"
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifndef _POSIX_THREADS
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef __sgi
 | 
					
						
							|  |  |  | #define SGI_THREADS
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef HAVE_THREAD_H
 | 
					
						
							|  |  |  | #define SOLARIS_THREADS
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(sun) && !defined(SOLARIS_THREADS)
 | 
					
						
							|  |  |  | #define SUN_LWP
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-09-14 17:49:54 +00:00
										 |  |  | /* Check if we're running on HP-UX and _SC_THREADS is defined. If so, then
 | 
					
						
							|  |  |  |    enough of the Posix threads package is implimented to support python  | 
					
						
							|  |  |  |    threads. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    This is valid for HP-UX 11.23 running on an ia64 system. If needed, add | 
					
						
							|  |  |  |    a check of __ia64 to verify that we're running on a ia64 system instead | 
					
						
							|  |  |  |    of a pa-risc system. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | #ifdef __hpux
 | 
					
						
							|  |  |  | #ifdef _SC_THREADS
 | 
					
						
							|  |  |  | #define _POSIX_THREADS
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-12-03 16:54:45 +00:00
										 |  |  | #endif /* _POSIX_THREADS */
 | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-30 16:17:54 +00:00
										 |  |  | #ifdef Py_DEBUG
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | static int thread_debug = 0; | 
					
						
							| 
									
										
										
										
											2002-06-25 19:26:34 +00:00
										 |  |  | #define dprintf(args)	(void)((thread_debug & 1) && printf args)
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | #define d2printf(args)	((thread_debug & 8) && printf args)
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #define dprintf(args)
 | 
					
						
							|  |  |  | #define d2printf(args)
 | 
					
						
							| 
									
										
										
										
											1993-12-03 16:54:45 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1992-09-02 11:25:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | static int initialized; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-07-24 14:39:50 +00:00
										 |  |  | static void PyThread__init_thread(void); /* Forward */ | 
					
						
							| 
									
										
										
										
											1992-09-02 11:25:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-07-22 18:47:25 +00:00
										 |  |  | void PyThread_init_thread(void) | 
					
						
							| 
									
										
										
										
											1992-09-02 11:25:37 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-30 16:17:54 +00:00
										 |  |  | #ifdef Py_DEBUG
 | 
					
						
							| 
									
										
										
										
											1993-12-03 16:54:45 +00:00
										 |  |  | 	char *p = getenv("THREADDEBUG"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p) { | 
					
						
							|  |  |  | 		if (*p) | 
					
						
							|  |  |  | 			thread_debug = atoi(p); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			thread_debug = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-30 16:17:54 +00:00
										 |  |  | #endif /* Py_DEBUG */
 | 
					
						
							| 
									
										
										
										
											1992-09-02 11:25:37 +00:00
										 |  |  | 	if (initialized) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	initialized = 1; | 
					
						
							| 
									
										
										
										
											1998-12-21 19:32:43 +00:00
										 |  |  | 	dprintf(("PyThread_init_thread called\n")); | 
					
						
							|  |  |  | 	PyThread__init_thread(); | 
					
						
							| 
									
										
										
										
											1992-09-02 11:25:37 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | #ifdef SGI_THREADS
 | 
					
						
							|  |  |  | #include "thread_sgi.h"
 | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1992-08-17 08:59:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | #ifdef SOLARIS_THREADS
 | 
					
						
							|  |  |  | #include "thread_solaris.h"
 | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | #ifdef SUN_LWP
 | 
					
						
							|  |  |  | #include "thread_lwp.h"
 | 
					
						
							| 
									
										
										
										
											1993-12-03 16:54:45 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-09-19 00:46:46 +00:00
										 |  |  | #ifdef HAVE_PTH
 | 
					
						
							| 
									
										
										
										
											2000-05-08 13:41:38 +00:00
										 |  |  | #include "thread_pth.h"
 | 
					
						
							| 
									
										
										
										
											2003-09-20 11:13:36 +00:00
										 |  |  | #undef _POSIX_THREADS
 | 
					
						
							| 
									
										
										
										
											2000-09-19 00:46:46 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | #ifdef _POSIX_THREADS
 | 
					
						
							|  |  |  | #include "thread_pthread.h"
 | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | #ifdef C_THREADS
 | 
					
						
							|  |  |  | #include "thread_cthread.h"
 | 
					
						
							| 
									
										
										
										
											1992-08-04 12:41:02 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1995-01-17 16:29:31 +00:00
										 |  |  | #ifdef NT_THREADS
 | 
					
						
							|  |  |  | #include "thread_nt.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-11-22 21:53:48 +00:00
										 |  |  | #ifdef OS2_THREADS
 | 
					
						
							|  |  |  | #include "thread_os2.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-08-04 22:46:29 +00:00
										 |  |  | #ifdef BEOS_THREADS
 | 
					
						
							|  |  |  | #include "thread_beos.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-04-07 16:07:23 +00:00
										 |  |  | #ifdef WINCE_THREADS
 | 
					
						
							|  |  |  | #include "thread_wince.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-03-09 12:10:54 +00:00
										 |  |  | #ifdef PLAN9_THREADS
 | 
					
						
							|  |  |  | #include "thread_plan9.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-11 06:22:31 +00:00
										 |  |  | #ifdef ATHEOS_THREADS
 | 
					
						
							|  |  |  | #include "thread_atheos.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-08-17 08:59:08 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | #ifdef FOOBAR_THREADS
 | 
					
						
							|  |  |  | #include "thread_foobar.h"
 | 
					
						
							| 
									
										
										
										
											1992-08-17 08:59:08 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1994-08-30 08:27:36 +00:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifndef Py_HAVE_NATIVE_TLS
 | 
					
						
							|  |  |  | /* If the platform has not supplied a platform specific
 | 
					
						
							|  |  |  |    TLS implementation, provide our own. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    This code stolen from "thread_sgi.h", where it was the only | 
					
						
							|  |  |  |    implementation of an existing Python TLS API. | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | /* ------------------------------------------------------------------------
 | 
					
						
							|  |  |  | Per-thread data ("key") support. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Use PyThread_create_key() to create a new key.  This is typically shared | 
					
						
							|  |  |  | across threads. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Use PyThread_set_key_value(thekey, value) to associate void* value with | 
					
						
							|  |  |  | thekey in the current thread.  Each thread has a distinct mapping of thekey | 
					
						
							|  |  |  | to a void* value.  Caution:  if the current thread already has a mapping | 
					
						
							|  |  |  | for thekey, value is ignored. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Use PyThread_get_key_value(thekey) to retrieve the void* value associated | 
					
						
							|  |  |  | with thekey in the current thread.  This returns NULL if no value is | 
					
						
							|  |  |  | associated with thekey in the current thread. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Use PyThread_delete_key_value(thekey) to forget the current thread's associated | 
					
						
							|  |  |  | value for thekey.  PyThread_delete_key(thekey) forgets the values associated | 
					
						
							|  |  |  | with thekey across *all* threads. | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | While some of these functions have error-return values, none set any | 
					
						
							|  |  |  | Python exception. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | None of the functions does memory management on behalf of the void* values. | 
					
						
							|  |  |  | You need to allocate and deallocate them yourself.  If the void* values | 
					
						
							|  |  |  | happen to be PyObject*, these functions don't do refcount operations on | 
					
						
							|  |  |  | them either. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The GIL does not need to be held when calling these functions; they supply | 
					
						
							|  |  |  | their own locking.  This isn't true of PyThread_create_key(), though (see | 
					
						
							|  |  |  | next paragraph). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | There's a hidden assumption that PyThread_create_key() will be called before | 
					
						
							|  |  |  | any of the other functions are called.  There's also a hidden assumption | 
					
						
							|  |  |  | that calls to PyThread_create_key() are serialized externally. | 
					
						
							|  |  |  | ------------------------------------------------------------------------ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* A singly-linked list of struct key objects remembers all the key->value
 | 
					
						
							|  |  |  |  * associations.  File static keyhead heads the list.  keymutex is used | 
					
						
							|  |  |  |  * to enforce exclusion internally. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | struct key { | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | 	/* Next record in the list, or NULL if this is the last record. */ | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	struct key *next; | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* The thread id, according to PyThread_get_thread_ident(). */ | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	long id; | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* The key and its associated value. */ | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	int key; | 
					
						
							|  |  |  | 	void *value; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct key *keyhead = NULL; | 
					
						
							|  |  |  | static PyThread_type_lock keymutex = NULL; | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | static int nkeys = 0;  /* PyThread_create_key() hands out nkeys+1 next */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Internal helper.
 | 
					
						
							|  |  |  |  * If the current thread has a mapping for key, the appropriate struct key* | 
					
						
							|  |  |  |  * is returned.  NB:  value is ignored in this case! | 
					
						
							|  |  |  |  * If there is no mapping for key in the current thread, then: | 
					
						
							|  |  |  |  *     If value is NULL, NULL is returned. | 
					
						
							|  |  |  |  *     Else a mapping of key to value is created for the current thread, | 
					
						
							|  |  |  |  *     and a pointer to a new struct key* is returned; except that if | 
					
						
							|  |  |  |  *     malloc() can't find room for a new struct key*, NULL is returned. | 
					
						
							|  |  |  |  * So when value==NULL, this acts like a pure lookup routine, and when | 
					
						
							|  |  |  |  * value!=NULL, this acts like dict.setdefault(), returning an existing | 
					
						
							|  |  |  |  * mapping if one exists, else creating a new mapping. | 
					
						
							| 
									
										
										
										
											2004-10-10 01:58:44 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Caution:  this used to be too clever, trying to hold keymutex only | 
					
						
							|  |  |  |  * around the "p->next = keyhead; keyhead = p" pair.  That allowed | 
					
						
							|  |  |  |  * another thread to mutate the list, via key deletion, concurrent with | 
					
						
							|  |  |  |  * find_key() crawling over the list.  Hilarity ensued.  For example, when | 
					
						
							|  |  |  |  * the for-loop here does "p = p->next", p could end up pointing at a | 
					
						
							|  |  |  |  * record that PyThread_delete_key_value() was concurrently free()'ing. | 
					
						
							|  |  |  |  * That could lead to anything, from failing to find a key that exists, to | 
					
						
							|  |  |  |  * segfaults.  Now we lock the whole routine. | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2004-10-09 17:38:29 +00:00
										 |  |  | static struct key * | 
					
						
							|  |  |  | find_key(int key, void *value) | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct key *p; | 
					
						
							|  |  |  | 	long id = PyThread_get_thread_ident(); | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-10 01:58:44 +00:00
										 |  |  | 	PyThread_acquire_lock(keymutex, 1); | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	for (p = keyhead; p != NULL; p = p->next) { | 
					
						
							|  |  |  | 		if (p->id == id && p->key == key) | 
					
						
							| 
									
										
										
										
											2004-10-10 01:58:44 +00:00
										 |  |  | 			goto Done; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (value == NULL) { | 
					
						
							|  |  |  | 		assert(p == NULL); | 
					
						
							|  |  |  | 		goto Done; | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	p = (struct key *)malloc(sizeof(struct key)); | 
					
						
							|  |  |  | 	if (p != NULL) { | 
					
						
							|  |  |  | 		p->id = id; | 
					
						
							|  |  |  | 		p->key = key; | 
					
						
							|  |  |  | 		p->value = value; | 
					
						
							|  |  |  | 		p->next = keyhead; | 
					
						
							|  |  |  | 		keyhead = p; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-10-10 01:58:44 +00:00
										 |  |  |  Done: | 
					
						
							|  |  |  | 	PyThread_release_lock(keymutex); | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | /* Return a new key.  This must be called before any other functions in
 | 
					
						
							|  |  |  |  * this family, and callers must arrange to serialize calls to this | 
					
						
							|  |  |  |  * function.  No violations are detected. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2004-10-09 17:38:29 +00:00
										 |  |  | int | 
					
						
							|  |  |  | PyThread_create_key(void) | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | 	/* All parts of this function are wrong if it's called by multiple
 | 
					
						
							|  |  |  | 	 * threads simultaneously. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	if (keymutex == NULL) | 
					
						
							|  |  |  | 		keymutex = PyThread_allocate_lock(); | 
					
						
							|  |  |  | 	return ++nkeys; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | /* Forget the associations for key across *all* threads. */ | 
					
						
							| 
									
										
										
										
											2004-10-09 17:38:29 +00:00
										 |  |  | void | 
					
						
							|  |  |  | PyThread_delete_key(int key) | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct key *p, **q; | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	PyThread_acquire_lock(keymutex, 1); | 
					
						
							|  |  |  | 	q = &keyhead; | 
					
						
							|  |  |  | 	while ((p = *q) != NULL) { | 
					
						
							|  |  |  | 		if (p->key == key) { | 
					
						
							|  |  |  | 			*q = p->next; | 
					
						
							|  |  |  | 			free((void *)p); | 
					
						
							|  |  |  | 			/* NB This does *not* free p->value! */ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			q = &p->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	PyThread_release_lock(keymutex); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | /* Confusing:  If the current thread has an association for key,
 | 
					
						
							|  |  |  |  * value is ignored, and 0 is returned.  Else an attempt is made to create | 
					
						
							|  |  |  |  * an association of key to value for the current thread.  0 is returned | 
					
						
							|  |  |  |  * if that succeeds, but -1 is returned if there's not enough memory | 
					
						
							|  |  |  |  * to create the association.  value must not be NULL. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2004-10-09 17:38:29 +00:00
										 |  |  | int | 
					
						
							|  |  |  | PyThread_set_key_value(int key, void *value) | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | 	struct key *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	assert(value != NULL); | 
					
						
							|  |  |  | 	p = find_key(key, value); | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	if (p == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | /* Retrieve the value associated with key in the current thread, or NULL
 | 
					
						
							|  |  |  |  * if the current thread doesn't have an association for key. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2004-10-09 17:38:29 +00:00
										 |  |  | void * | 
					
						
							|  |  |  | PyThread_get_key_value(int key) | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct key *p = find_key(key, NULL); | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	if (p == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return p->value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | /* Forget the current thread's association for key, if any. */ | 
					
						
							| 
									
										
										
										
											2004-10-09 17:38:29 +00:00
										 |  |  | void | 
					
						
							|  |  |  | PyThread_delete_key_value(int key) | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	long id = PyThread_get_thread_ident(); | 
					
						
							|  |  |  | 	struct key *p, **q; | 
					
						
							| 
									
										
										
										
											2004-10-09 22:33:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-19 15:41:53 +00:00
										 |  |  | 	PyThread_acquire_lock(keymutex, 1); | 
					
						
							|  |  |  | 	q = &keyhead; | 
					
						
							|  |  |  | 	while ((p = *q) != NULL) { | 
					
						
							|  |  |  | 		if (p->key == key && p->id == id) { | 
					
						
							|  |  |  | 			*q = p->next; | 
					
						
							|  |  |  | 			free((void *)p); | 
					
						
							|  |  |  | 			/* NB This does *not* free p->value! */ | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			q = &p->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	PyThread_release_lock(keymutex); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* Py_HAVE_NATIVE_TLS */
 |