cpython/Python/thread_os2.h

268 lines
5.0 KiB
C
Raw Normal View History

1995-04-10 08:36:14 -03:00
/* This code implemented by cvale@netcom.com */
#define INCL_DOSPROCESS
#define INCL_DOSSEMAPHORES
#include "os2.h"
#include "limits.h"
#include "process.h"
#if defined(PYCC_GCC)
#include <sys/builtin.h>
#include <sys/fmutex.h>
#else
long PyThread_get_thread_ident(void);
#endif
1995-04-10 08:36:14 -03:00
/* default thread stack size of 64kB */
#if !defined(THREAD_STACK_SIZE)
#define THREAD_STACK_SIZE 0x10000
#endif
#define OS2_STACKSIZE(x) (x ? x : THREAD_STACK_SIZE)
1995-04-10 08:36:14 -03:00
/*
* Initialization of the C package, should not be needed.
*/
static void
PyThread__init_thread(void)
1995-04-10 08:36:14 -03:00
{
}
/*
* Thread support.
*/
long
PyThread_start_new_thread(void (*func)(void *), void *arg)
1995-04-10 08:36:14 -03:00
{
int thread_id;
1995-04-10 08:36:14 -03:00
thread_id = _beginthread(func,
NULL,
OS2_STACKSIZE(_pythread_stacksize),
arg);
1995-04-10 08:36:14 -03:00
if (thread_id == -1) {
2002-12-04 08:29:37 -04:00
dprintf(("_beginthread failed. return %ld\n", errno));
}
1995-04-10 08:36:14 -03:00
return thread_id;
1995-04-10 08:36:14 -03:00
}
long
PyThread_get_thread_ident(void)
1995-04-10 08:36:14 -03:00
{
#if !defined(PYCC_GCC)
2002-12-04 08:29:37 -04:00
PPIB pib;
PTIB tib;
#endif
1995-04-10 08:36:14 -03:00
2002-12-04 08:29:37 -04:00
if (!initialized)
PyThread_init_thread();
#if defined(PYCC_GCC)
2002-12-04 08:29:37 -04:00
return _gettid();
#else
2002-12-04 08:29:37 -04:00
DosGetInfoBlocks(&tib, &pib);
return tib->tib_ptib2->tib2_ultid;
#endif
1995-04-10 08:36:14 -03:00
}
void
PyThread_exit_thread(void)
1995-04-10 08:36:14 -03:00
{
2002-12-04 08:29:37 -04:00
dprintf(("%ld: PyThread_exit_thread called\n",
PyThread_get_thread_ident()));
if (!initialized)
exit(0);
2002-12-04 08:29:37 -04:00
_endthread();
1995-04-10 08:36:14 -03:00
}
/*
* Lock support. This is implemented with an event semaphore and critical
* sections to make it behave more like a posix mutex than its OS/2
* counterparts.
1995-04-10 08:36:14 -03:00
*/
typedef struct os2_lock_t {
2002-12-04 08:29:37 -04:00
int is_set;
HEV changed;
} *type_os2_lock;
PyThread_type_lock
PyThread_allocate_lock(void)
1995-04-10 08:36:14 -03:00
{
#if defined(PYCC_GCC)
2002-12-04 08:29:37 -04:00
_fmutex *sem = malloc(sizeof(_fmutex));
if (!initialized)
PyThread_init_thread();
dprintf(("%ld: PyThread_allocate_lock() -> %lx\n",
PyThread_get_thread_ident(),
(long)sem));
if (_fmutex_create(sem, 0)) {
free(sem);
sem = NULL;
}
return (PyThread_type_lock)sem;
#else
2002-12-04 08:29:37 -04:00
APIRET rc;
type_os2_lock lock = (type_os2_lock)malloc(sizeof(struct os2_lock_t));
1995-04-10 08:36:14 -03:00
2002-12-04 08:29:37 -04:00
dprintf(("PyThread_allocate_lock called\n"));
if (!initialized)
PyThread_init_thread();
1995-04-10 08:36:14 -03:00
2002-12-04 08:29:37 -04:00
lock->is_set = 0;
1995-04-10 08:36:14 -03:00
2002-12-04 08:29:37 -04:00
DosCreateEventSem(NULL, &lock->changed, 0, 0);
1995-04-10 08:36:14 -03:00
2002-12-04 08:29:37 -04:00
dprintf(("%ld: PyThread_allocate_lock() -> %p\n",
PyThread_get_thread_ident(),
lock->changed));
2002-12-04 08:29:37 -04:00
return (PyThread_type_lock)lock;
#endif
1995-04-10 08:36:14 -03:00
}
void
PyThread_free_lock(PyThread_type_lock aLock)
1995-04-10 08:36:14 -03:00
{
#if !defined(PYCC_GCC)
2002-12-04 08:29:37 -04:00
type_os2_lock lock = (type_os2_lock)aLock;
#endif
2002-12-04 08:29:37 -04:00
dprintf(("%ld: PyThread_free_lock(%p) called\n",
PyThread_get_thread_ident(),aLock));
1995-04-10 08:36:14 -03:00
#if defined(PYCC_GCC)
2002-12-04 08:29:37 -04:00
if (aLock) {
_fmutex_close((_fmutex *)aLock);
free((_fmutex *)aLock);
}
#else
2002-12-04 08:29:37 -04:00
DosCloseEventSem(lock->changed);
free(aLock);
#endif
1995-04-10 08:36:14 -03:00
}
/*
* Return 1 on success if the lock was acquired
*
* and 0 if the lock was not acquired.
1995-04-10 08:36:14 -03:00
*/
int
PyThread_acquire_lock(PyThread_type_lock aLock, int waitflag)
1995-04-10 08:36:14 -03:00
{
#if !defined(PYCC_GCC)
2002-12-04 08:29:37 -04:00
int done = 0;
ULONG count;
PID pid = 0;
TID tid = 0;
type_os2_lock lock = (type_os2_lock)aLock;
#endif
1995-04-10 08:36:14 -03:00
2002-12-04 08:29:37 -04:00
dprintf(("%ld: PyThread_acquire_lock(%p, %d) called\n",
PyThread_get_thread_ident(),
aLock,
waitflag));
1995-04-10 08:36:14 -03:00
#if defined(PYCC_GCC)
2002-12-04 08:29:37 -04:00
/* always successful if the lock doesn't exist */
if (aLock &&
_fmutex_request((_fmutex *)aLock, waitflag ? 0 : _FMR_NOWAIT))
return 0;
#else
2002-12-04 08:29:37 -04:00
while (!done) {
/* if the lock is currently set, we have to wait for
* the state to change
*/
if (lock->is_set) {
if (!waitflag)
return 0;
DosWaitEventSem(lock->changed, SEM_INDEFINITE_WAIT);
}
/* enter a critical section and try to get the semaphore. If
* it is still locked, we will try again.
*/
if (DosEnterCritSec())
return 0;
if (!lock->is_set) {
lock->is_set = 1;
DosResetEventSem(lock->changed, &count);
done = 1;
}
DosExitCritSec();
}
#endif
1995-04-10 08:36:14 -03:00
2002-12-04 08:29:37 -04:00
return 1;
1995-04-10 08:36:14 -03:00
}
void
PyThread_release_lock(PyThread_type_lock aLock)
1995-04-10 08:36:14 -03:00
{
#if !defined(PYCC_GCC)
2002-12-04 08:29:37 -04:00
type_os2_lock lock = (type_os2_lock)aLock;
#endif
2002-12-04 08:29:37 -04:00
dprintf(("%ld: PyThread_release_lock(%p) called\n",
PyThread_get_thread_ident(),
aLock));
1995-04-10 08:36:14 -03:00
#if defined(PYCC_GCC)
2002-12-04 08:29:37 -04:00
if (aLock)
_fmutex_release((_fmutex *)aLock);
#else
2002-12-04 08:29:37 -04:00
if (!lock->is_set) {
dprintf(("%ld: Could not PyThread_release_lock(%p) error: %l\n",
PyThread_get_thread_ident(),
aLock,
GetLastError()));
return;
}
if (DosEnterCritSec()) {
dprintf(("%ld: Could not PyThread_release_lock(%p) error: %l\n",
PyThread_get_thread_ident(),
aLock,
GetLastError()));
return;
}
lock->is_set = 0;
DosPostEventSem(lock->changed);
DosExitCritSec();
#endif
1995-04-10 08:36:14 -03:00
}
/* minimum/maximum thread stack sizes supported */
#define THREAD_MIN_STACKSIZE 0x8000 /* 32kB */
#define THREAD_MAX_STACKSIZE 0x2000000 /* 32MB */
/* set the thread stack size.
* Return 0 if size is valid, -1 otherwise.
*/
static int
_pythread_os2_set_stacksize(size_t size)
{
/* set to default */
if (size == 0) {
_pythread_stacksize = 0;
return 0;
}
/* valid range? */
if (size >= THREAD_MIN_STACKSIZE && size < THREAD_MAX_STACKSIZE) {
_pythread_stacksize = size;
return 0;
}
return -1;
}
#define THREAD_SET_STACKSIZE(x) _pythread_os2_set_stacksize(x)