blob: 69bcd2ce504ff539a230b199dad09f11453ceb9e [file] [log] [blame]
Guido van Rossum62332931995-04-10 11:36:14 +00001/* This code implemented by cvale@netcom.com */
2
3#define INCL_DOSPROCESS
4#define INCL_DOSSEMAPHORES
5#include "os2.h"
6#include "limits.h"
7
8#include "process.h"
9
Andrew MacIntyred9400542002-02-26 11:41:34 +000010#if defined(PYCC_GCC)
11#include <sys/builtin.h>
12#include <sys/fmutex.h>
13#else
Guido van Rossum65d5b571998-12-21 19:32:43 +000014long PyThread_get_thread_ident(void);
Andrew MacIntyred9400542002-02-26 11:41:34 +000015#endif
Guido van Rossum62332931995-04-10 11:36:14 +000016
17/*
18 * Initialization of the C package, should not be needed.
19 */
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000020static void
21PyThread__init_thread(void)
Guido van Rossum62332931995-04-10 11:36:14 +000022{
23}
24
25/*
26 * Thread support.
27 */
Guido van Rossum3c288632001-10-16 21:13:49 +000028long
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000029PyThread_start_new_thread(void (*func)(void *), void *arg)
Guido van Rossum62332931995-04-10 11:36:14 +000030{
31 int aThread;
Guido van Rossum3c288632001-10-16 21:13:49 +000032 int success = 0;
Guido van Rossum62332931995-04-10 11:36:14 +000033
Guido van Rossum8e9ebfd1997-11-22 21:53:48 +000034 aThread = _beginthread(func,NULL,65536,arg);
Guido van Rossum62332931995-04-10 11:36:14 +000035
36 if( aThread == -1 ) {
Guido van Rossum3c288632001-10-16 21:13:49 +000037 success = -1;
Guido van Rossum62332931995-04-10 11:36:14 +000038 fprintf(stderr,"aThread failed == %d",aThread);
39 dprintf(("_beginthread failed. return %ld\n", errno));
40 }
41
42 return success;
43}
44
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000045long
46PyThread_get_thread_ident(void)
Guido van Rossum62332931995-04-10 11:36:14 +000047{
Andrew MacIntyred9400542002-02-26 11:41:34 +000048#if !defined(PYCC_GCC)
Guido van Rossum62332931995-04-10 11:36:14 +000049 PPIB pib;
50 PTIB tib;
Andrew MacIntyred9400542002-02-26 11:41:34 +000051#endif
Guido van Rossum62332931995-04-10 11:36:14 +000052
53 if (!initialized)
Guido van Rossum65d5b571998-12-21 19:32:43 +000054 PyThread_init_thread();
Guido van Rossum62332931995-04-10 11:36:14 +000055
Andrew MacIntyred9400542002-02-26 11:41:34 +000056#if defined(PYCC_GCC)
57 return _gettid();
58#else
Guido van Rossum62332931995-04-10 11:36:14 +000059 DosGetInfoBlocks(&tib,&pib);
60 return tib->tib_ptib2->tib2_ultid;
Andrew MacIntyred9400542002-02-26 11:41:34 +000061#endif
Guido van Rossum62332931995-04-10 11:36:14 +000062}
63
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000064static void
65do_PyThread_exit_thread(int no_cleanup)
Guido van Rossum62332931995-04-10 11:36:14 +000066{
Guido van Rossum65d5b571998-12-21 19:32:43 +000067 dprintf(("%ld: PyThread_exit_thread called\n", PyThread_get_thread_ident()));
Guido van Rossum62332931995-04-10 11:36:14 +000068 if (!initialized)
69 if (no_cleanup)
70 _exit(0);
71 else
72 exit(0);
73 _endthread();
74}
75
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000076void
77PyThread_exit_thread(void)
Guido van Rossum62332931995-04-10 11:36:14 +000078{
Guido van Rossum65d5b571998-12-21 19:32:43 +000079 do_PyThread_exit_thread(0);
Guido van Rossum62332931995-04-10 11:36:14 +000080}
81
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000082void
83PyThread__exit_thread(void)
Guido van Rossum62332931995-04-10 11:36:14 +000084{
Guido van Rossum65d5b571998-12-21 19:32:43 +000085 do_PyThread_exit_thread(1);
Guido van Rossum62332931995-04-10 11:36:14 +000086}
87
88#ifndef NO_EXIT_PROG
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000089static void
90do_PyThread_exit_prog(int status, int no_cleanup)
Guido van Rossum62332931995-04-10 11:36:14 +000091{
Guido van Rossum65d5b571998-12-21 19:32:43 +000092 dprintf(("PyThread_exit_prog(%d) called\n", status));
Guido van Rossum62332931995-04-10 11:36:14 +000093 if (!initialized)
94 if (no_cleanup)
95 _exit(status);
96 else
97 exit(status);
98}
99
Thomas Woutersf70ef4f2000-07-22 18:47:25 +0000100void
101PyThread_exit_prog(int status)
Guido van Rossum62332931995-04-10 11:36:14 +0000102{
Guido van Rossum65d5b571998-12-21 19:32:43 +0000103 do_PyThread_exit_prog(status, 0);
Guido van Rossum62332931995-04-10 11:36:14 +0000104}
105
Thomas Woutersf70ef4f2000-07-22 18:47:25 +0000106void
107PyThread__exit_prog(int status)
Guido van Rossum62332931995-04-10 11:36:14 +0000108{
Guido van Rossum65d5b571998-12-21 19:32:43 +0000109 do_PyThread_exit_prog(status, 1);
Guido van Rossum62332931995-04-10 11:36:14 +0000110}
111#endif /* NO_EXIT_PROG */
112
113/*
Tim Petersa6ca4f42001-10-31 03:50:45 +0000114 * Lock support. This is implemented with an event semaphore and critical
115 * sections to make it behave more like a posix mutex than its OS/2
Andrew MacIntyred9400542002-02-26 11:41:34 +0000116 * counterparts.
Guido van Rossum62332931995-04-10 11:36:14 +0000117 */
Tim Petersa6ca4f42001-10-31 03:50:45 +0000118
119typedef struct os2_lock_t {
120 int is_set;
121 HEV changed;
122} *type_os2_lock;
123
Thomas Woutersf70ef4f2000-07-22 18:47:25 +0000124PyThread_type_lock
125PyThread_allocate_lock(void)
Guido van Rossum62332931995-04-10 11:36:14 +0000126{
Andrew MacIntyred9400542002-02-26 11:41:34 +0000127#if defined(PYCC_GCC)
128 _fmutex *sem = malloc(sizeof(_fmutex));
129 if (!initialized)
130 PyThread_init_thread();
131 dprintf(("%ld: PyThread_allocate_lock() -> %lx\n",
132 PyThread_get_thread_ident(),
133 (long)sem));
134 if (_fmutex_create(sem, 0)) {
135 free(sem);
136 sem = NULL;
137 }
138 return (PyThread_type_lock) sem;
139#else
Guido van Rossum62332931995-04-10 11:36:14 +0000140 APIRET rc;
Tim Petersa6ca4f42001-10-31 03:50:45 +0000141 type_os2_lock lock = (type_os2_lock)malloc(sizeof(struct os2_lock_t));
Guido van Rossum62332931995-04-10 11:36:14 +0000142
Guido van Rossum65d5b571998-12-21 19:32:43 +0000143 dprintf(("PyThread_allocate_lock called\n"));
Guido van Rossum62332931995-04-10 11:36:14 +0000144 if (!initialized)
Guido van Rossum65d5b571998-12-21 19:32:43 +0000145 PyThread_init_thread();
Guido van Rossum62332931995-04-10 11:36:14 +0000146
Tim Petersa6ca4f42001-10-31 03:50:45 +0000147 lock->is_set = 0;
Guido van Rossum62332931995-04-10 11:36:14 +0000148
Tim Petersa6ca4f42001-10-31 03:50:45 +0000149 DosCreateEventSem(NULL, &lock->changed, 0, 0);
Guido van Rossum62332931995-04-10 11:36:14 +0000150
Tim Petersa6ca4f42001-10-31 03:50:45 +0000151 dprintf(("%ld: PyThread_allocate_lock() -> %p\n",
152 PyThread_get_thread_ident(),
153 lock->changed));
154
155 return (PyThread_type_lock) lock;
Andrew MacIntyred9400542002-02-26 11:41:34 +0000156#endif
Guido van Rossum62332931995-04-10 11:36:14 +0000157}
158
Thomas Woutersf70ef4f2000-07-22 18:47:25 +0000159void
160PyThread_free_lock(PyThread_type_lock aLock)
Guido van Rossum62332931995-04-10 11:36:14 +0000161{
Andrew MacIntyred9400542002-02-26 11:41:34 +0000162#if !defined(PYCC_GCC)
Tim Petersa6ca4f42001-10-31 03:50:45 +0000163 type_os2_lock lock = (type_os2_lock)aLock;
Andrew MacIntyred9400542002-02-26 11:41:34 +0000164#endif
165
Fred Drakea44d3532000-06-30 15:01:00 +0000166 dprintf(("%ld: PyThread_free_lock(%p) called\n", PyThread_get_thread_ident(),aLock));
Guido van Rossum62332931995-04-10 11:36:14 +0000167
Andrew MacIntyred9400542002-02-26 11:41:34 +0000168#if defined(PYCC_GCC)
169 if (aLock) {
170 _fmutex_close((_fmutex *)aLock);
171 free((_fmutex *)aLock);
172 }
173#else
Tim Petersa6ca4f42001-10-31 03:50:45 +0000174 DosCloseEventSem(lock->changed);
175 free(aLock);
Andrew MacIntyred9400542002-02-26 11:41:34 +0000176#endif
Guido van Rossum62332931995-04-10 11:36:14 +0000177}
178
179/*
180 * Return 1 on success if the lock was acquired
181 *
Tim Petersa6ca4f42001-10-31 03:50:45 +0000182 * and 0 if the lock was not acquired.
Guido van Rossum62332931995-04-10 11:36:14 +0000183 */
Thomas Woutersf70ef4f2000-07-22 18:47:25 +0000184int
185PyThread_acquire_lock(PyThread_type_lock aLock, int waitflag)
Guido van Rossum62332931995-04-10 11:36:14 +0000186{
Andrew MacIntyred9400542002-02-26 11:41:34 +0000187#if !defined(PYCC_GCC)
Tim Petersa6ca4f42001-10-31 03:50:45 +0000188 int done = 0;
189 ULONG count;
Guido van Rossum62332931995-04-10 11:36:14 +0000190 PID pid = 0;
191 TID tid = 0;
Tim Petersa6ca4f42001-10-31 03:50:45 +0000192 type_os2_lock lock = (type_os2_lock)aLock;
Andrew MacIntyred9400542002-02-26 11:41:34 +0000193#endif
Guido van Rossum62332931995-04-10 11:36:14 +0000194
Fred Drakea44d3532000-06-30 15:01:00 +0000195 dprintf(("%ld: PyThread_acquire_lock(%p, %d) called\n", PyThread_get_thread_ident(),
196 aLock, waitflag));
Guido van Rossum62332931995-04-10 11:36:14 +0000197
Andrew MacIntyred9400542002-02-26 11:41:34 +0000198#if defined(PYCC_GCC)
199 /* always successful if the lock doesn't exist */
200 if (aLock && _fmutex_request((_fmutex *)aLock, waitflag ? 0 : _FMR_NOWAIT))
201 return 0;
202#else
Tim Petersa6ca4f42001-10-31 03:50:45 +0000203 while (!done) {
204 /* if the lock is currently set, we have to wait for the state to change */
205 if (lock->is_set) {
206 if (!waitflag)
207 return 0;
208 DosWaitEventSem(lock->changed, SEM_INDEFINITE_WAIT);
Guido van Rossum62332931995-04-10 11:36:14 +0000209 }
Tim Petersa6ca4f42001-10-31 03:50:45 +0000210
211 /*
212 * enter a critical section and try to get the semaphore. If
213 * it is still locked, we will try again.
214 */
215 if (DosEnterCritSec())
216 return 0;
217
218 if (!lock->is_set) {
219 lock->is_set = 1;
220 DosResetEventSem(lock->changed, &count);
221 done = 1;
222 }
223
224 DosExitCritSec();
Guido van Rossum62332931995-04-10 11:36:14 +0000225 }
Andrew MacIntyred9400542002-02-26 11:41:34 +0000226#endif
Guido van Rossum62332931995-04-10 11:36:14 +0000227
Tim Petersa6ca4f42001-10-31 03:50:45 +0000228 return 1;
Guido van Rossum62332931995-04-10 11:36:14 +0000229}
230
Tim Petersa6ca4f42001-10-31 03:50:45 +0000231void PyThread_release_lock(PyThread_type_lock aLock)
Guido van Rossum62332931995-04-10 11:36:14 +0000232{
Andrew MacIntyred9400542002-02-26 11:41:34 +0000233#if defined(PYCC_GCC)
Tim Petersa6ca4f42001-10-31 03:50:45 +0000234 type_os2_lock lock = (type_os2_lock)aLock;
Andrew MacIntyred9400542002-02-26 11:41:34 +0000235#endif
236
Fred Drakea44d3532000-06-30 15:01:00 +0000237 dprintf(("%ld: PyThread_release_lock(%p) called\n", PyThread_get_thread_ident(),aLock));
Guido van Rossum62332931995-04-10 11:36:14 +0000238
Andrew MacIntyred9400542002-02-26 11:41:34 +0000239#if defined(PYCC_GCC)
240 if (aLock)
241 _fmutex_release((_fmutex *)aLock);
242#else
Tim Petersa6ca4f42001-10-31 03:50:45 +0000243 if (!lock->is_set) {
Fred Drakea44d3532000-06-30 15:01:00 +0000244 dprintf(("%ld: Could not PyThread_release_lock(%p) error: %l\n",
245 PyThread_get_thread_ident(), aLock, GetLastError()));
Tim Petersa6ca4f42001-10-31 03:50:45 +0000246 return;
Guido van Rossum62332931995-04-10 11:36:14 +0000247 }
Tim Petersa6ca4f42001-10-31 03:50:45 +0000248
249
250 if (DosEnterCritSec()) {
251 dprintf(("%ld: Could not PyThread_release_lock(%p) error: %l\n",
252 PyThread_get_thread_ident(), aLock, GetLastError()));
253 return;
254 }
255
256 lock->is_set = 0;
257 DosPostEventSem(lock->changed);
258
259 DosExitCritSec();
Andrew MacIntyred9400542002-02-26 11:41:34 +0000260#endif
Guido van Rossum62332931995-04-10 11:36:14 +0000261}