|
|
|
|
|
|
|
/* 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. */
|
|
|
|
|
|
|
|
#include "Python.h"
|
|
|
|
|
|
|
|
#ifndef DONT_HAVE_STDIO_H
|
|
|
|
#include <stdio.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAVE_STDLIB_H
|
|
|
|
#include <stdlib.h>
|
|
|
|
#else
|
|
|
|
#ifdef Py_DEBUG
|
|
|
|
extern char *getenv(const char *);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef __DGUX
|
|
|
|
#define _USING_POSIX4A_DRAFT6
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef __sgi
|
|
|
|
#ifndef HAVE_PTHREAD_H /* XXX Need to check in configure.in */
|
|
|
|
#undef _POSIX_THREADS
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "pythread.h"
|
|
|
|
|
|
|
|
#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
|
|
|
|
|
|
|
|
#if defined(__MWERKS__) && !defined(__BEOS__)
|
|
|
|
#define _POSIX_THREADS
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* _POSIX_THREADS */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef Py_DEBUG
|
|
|
|
static int thread_debug = 0;
|
|
|
|
#define dprintf(args) (void)((thread_debug & 1) && printf args)
|
|
|
|
#define d2printf(args) ((thread_debug & 8) && printf args)
|
|
|
|
#else
|
|
|
|
#define dprintf(args)
|
|
|
|
#define d2printf(args)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static int initialized;
|
|
|
|
|
|
|
|
static void PyThread__init_thread(void); /* Forward */
|
|
|
|
|
|
|
|
void PyThread_init_thread(void)
|
|
|
|
{
|
|
|
|
#ifdef Py_DEBUG
|
|
|
|
char *p = Py_GETENV("THREADDEBUG");
|
|
|
|
|
|
|
|
if (p) {
|
|
|
|
if (*p)
|
|
|
|
thread_debug = atoi(p);
|
|
|
|
else
|
|
|
|
thread_debug = 1;
|
|
|
|
}
|
|
|
|
#endif /* Py_DEBUG */
|
|
|
|
if (initialized)
|
|
|
|
return;
|
|
|
|
initialized = 1;
|
|
|
|
dprintf(("PyThread_init_thread called\n"));
|
|
|
|
PyThread__init_thread();
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SGI_THREADS
|
|
|
|
#include "thread_sgi.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef SOLARIS_THREADS
|
|
|
|
#include "thread_solaris.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef SUN_LWP
|
|
|
|
#include "thread_lwp.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAVE_PTH
|
|
|
|
#include "thread_pth.h"
|
|
|
|
#undef _POSIX_THREADS
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef _POSIX_THREADS
|
|
|
|
#include "thread_pthread.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef C_THREADS
|
|
|
|
#include "thread_cthread.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef NT_THREADS
|
|
|
|
#include "thread_nt.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef OS2_THREADS
|
|
|
|
#include "thread_os2.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef BEOS_THREADS
|
|
|
|
#include "thread_beos.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef WINCE_THREADS
|
|
|
|
#include "thread_wince.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef PLAN9_THREADS
|
|
|
|
#include "thread_plan9.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef ATHEOS_THREADS
|
|
|
|
#include "thread_atheos.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
#ifdef FOOBAR_THREADS
|
|
|
|
#include "thread_foobar.h"
|
|
|
|
#endif
|
|
|
|
*/
|
|
|
|
|
|
|
|
#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.
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
* Per-thread data ("key") support.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct key {
|
|
|
|
struct key *next;
|
|
|
|
long id;
|
|
|
|
int key;
|
|
|
|
void *value;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct key *keyhead = NULL;
|
|
|
|
static int nkeys = 0;
|
|
|
|
static PyThread_type_lock keymutex = NULL;
|
|
|
|
|
|
|
|
static struct key *find_key(int key, void *value)
|
|
|
|
{
|
|
|
|
struct key *p;
|
|
|
|
long id = PyThread_get_thread_ident();
|
|
|
|
for (p = keyhead; p != NULL; p = p->next) {
|
|
|
|
if (p->id == id && p->key == key)
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
if (value == NULL)
|
|
|
|
return NULL;
|
|
|
|
p = (struct key *)malloc(sizeof(struct key));
|
|
|
|
if (p != NULL) {
|
|
|
|
p->id = id;
|
|
|
|
p->key = key;
|
|
|
|
p->value = value;
|
|
|
|
PyThread_acquire_lock(keymutex, 1);
|
|
|
|
p->next = keyhead;
|
|
|
|
keyhead = p;
|
|
|
|
PyThread_release_lock(keymutex);
|
|
|
|
}
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
int PyThread_create_key(void)
|
|
|
|
{
|
|
|
|
if (keymutex == NULL)
|
|
|
|
keymutex = PyThread_allocate_lock();
|
|
|
|
return ++nkeys;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PyThread_delete_key(int key)
|
|
|
|
{
|
|
|
|
struct key *p, **q;
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
int PyThread_set_key_value(int key, void *value)
|
|
|
|
{
|
|
|
|
struct key *p = find_key(key, value);
|
|
|
|
if (p == NULL)
|
|
|
|
return -1;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *PyThread_get_key_value(int key)
|
|
|
|
{
|
|
|
|
struct key *p = find_key(key, NULL);
|
|
|
|
if (p == NULL)
|
|
|
|
return NULL;
|
|
|
|
else
|
|
|
|
return p->value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PyThread_delete_key_value(int key)
|
|
|
|
{
|
|
|
|
long id = PyThread_get_thread_ident();
|
|
|
|
struct key *p, **q;
|
|
|
|
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 */
|