blob: a18ce6fd6c7a2b73319e6068d05a22f5a097d5d3 [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
Andrew MacIntyre2bea4742005-01-17 12:16:36 +000017#if !defined(THREAD_STACK_SIZE)
18#define THREAD_STACK_SIZE 0x10000
19#endif
20
Guido van Rossum62332931995-04-10 11:36:14 +000021/*
22 * Initialization of the C package, should not be needed.
23 */
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000024static void
25PyThread__init_thread(void)
Guido van Rossum62332931995-04-10 11:36:14 +000026{
27}
28
29/*
30 * Thread support.
31 */
Guido van Rossum3c288632001-10-16 21:13:49 +000032long
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000033PyThread_start_new_thread(void (*func)(void *), void *arg)
Guido van Rossum62332931995-04-10 11:36:14 +000034{
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000035 int aThread;
36 int success = 0;
Guido van Rossum62332931995-04-10 11:36:14 +000037
Andrew MacIntyre2bea4742005-01-17 12:16:36 +000038 aThread = _beginthread(func, NULL, THREAD_STACK_SIZE, arg);
Guido van Rossum62332931995-04-10 11:36:14 +000039
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000040 if (aThread == -1) {
41 success = -1;
42 fprintf(stderr, "aThread failed == %d", aThread);
43 dprintf(("_beginthread failed. return %ld\n", errno));
44 }
Guido van Rossum62332931995-04-10 11:36:14 +000045
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000046 return success;
Guido van Rossum62332931995-04-10 11:36:14 +000047}
48
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000049long
50PyThread_get_thread_ident(void)
Guido van Rossum62332931995-04-10 11:36:14 +000051{
Andrew MacIntyred9400542002-02-26 11:41:34 +000052#if !defined(PYCC_GCC)
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000053 PPIB pib;
54 PTIB tib;
Andrew MacIntyred9400542002-02-26 11:41:34 +000055#endif
Guido van Rossum62332931995-04-10 11:36:14 +000056
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000057 if (!initialized)
58 PyThread_init_thread();
59
Andrew MacIntyred9400542002-02-26 11:41:34 +000060#if defined(PYCC_GCC)
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000061 return _gettid();
Andrew MacIntyred9400542002-02-26 11:41:34 +000062#else
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000063 DosGetInfoBlocks(&tib, &pib);
64 return tib->tib_ptib2->tib2_ultid;
Andrew MacIntyred9400542002-02-26 11:41:34 +000065#endif
Guido van Rossum62332931995-04-10 11:36:14 +000066}
67
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000068static void
69do_PyThread_exit_thread(int no_cleanup)
Guido van Rossum62332931995-04-10 11:36:14 +000070{
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000071 dprintf(("%ld: PyThread_exit_thread called\n",
72 PyThread_get_thread_ident()));
73 if (!initialized)
74 if (no_cleanup)
75 _exit(0);
76 else
77 exit(0);
78 _endthread();
Guido van Rossum62332931995-04-10 11:36:14 +000079}
80
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000081void
82PyThread_exit_thread(void)
Guido van Rossum62332931995-04-10 11:36:14 +000083{
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000084 do_PyThread_exit_thread(0);
Guido van Rossum62332931995-04-10 11:36:14 +000085}
86
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000087void
88PyThread__exit_thread(void)
Guido van Rossum62332931995-04-10 11:36:14 +000089{
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000090 do_PyThread_exit_thread(1);
Guido van Rossum62332931995-04-10 11:36:14 +000091}
92
93#ifndef NO_EXIT_PROG
Thomas Woutersf70ef4f2000-07-22 18:47:25 +000094static void
95do_PyThread_exit_prog(int status, int no_cleanup)
Guido van Rossum62332931995-04-10 11:36:14 +000096{
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +000097 dprintf(("PyThread_exit_prog(%d) called\n", status));
98 if (!initialized)
99 if (no_cleanup)
100 _exit(status);
101 else
102 exit(status);
Guido van Rossum62332931995-04-10 11:36:14 +0000103}
104
Thomas Woutersf70ef4f2000-07-22 18:47:25 +0000105void
106PyThread_exit_prog(int status)
Guido van Rossum62332931995-04-10 11:36:14 +0000107{
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000108 do_PyThread_exit_prog(status, 0);
Guido van Rossum62332931995-04-10 11:36:14 +0000109}
110
Thomas Woutersf70ef4f2000-07-22 18:47:25 +0000111void
112PyThread__exit_prog(int status)
Guido van Rossum62332931995-04-10 11:36:14 +0000113{
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000114 do_PyThread_exit_prog(status, 1);
Guido van Rossum62332931995-04-10 11:36:14 +0000115}
116#endif /* NO_EXIT_PROG */
117
118/*
Tim Petersa6ca4f42001-10-31 03:50:45 +0000119 * Lock support. This is implemented with an event semaphore and critical
120 * sections to make it behave more like a posix mutex than its OS/2
Andrew MacIntyred9400542002-02-26 11:41:34 +0000121 * counterparts.
Guido van Rossum62332931995-04-10 11:36:14 +0000122 */
Tim Petersa6ca4f42001-10-31 03:50:45 +0000123
124typedef struct os2_lock_t {
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000125 int is_set;
126 HEV changed;
Tim Petersa6ca4f42001-10-31 03:50:45 +0000127} *type_os2_lock;
128
Thomas Woutersf70ef4f2000-07-22 18:47:25 +0000129PyThread_type_lock
130PyThread_allocate_lock(void)
Guido van Rossum62332931995-04-10 11:36:14 +0000131{
Andrew MacIntyred9400542002-02-26 11:41:34 +0000132#if defined(PYCC_GCC)
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000133 _fmutex *sem = malloc(sizeof(_fmutex));
134 if (!initialized)
135 PyThread_init_thread();
136 dprintf(("%ld: PyThread_allocate_lock() -> %lx\n",
137 PyThread_get_thread_ident(),
138 (long)sem));
139 if (_fmutex_create(sem, 0)) {
140 free(sem);
141 sem = NULL;
142 }
143 return (PyThread_type_lock)sem;
Andrew MacIntyred9400542002-02-26 11:41:34 +0000144#else
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000145 APIRET rc;
146 type_os2_lock lock = (type_os2_lock)malloc(sizeof(struct os2_lock_t));
Guido van Rossum62332931995-04-10 11:36:14 +0000147
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000148 dprintf(("PyThread_allocate_lock called\n"));
149 if (!initialized)
150 PyThread_init_thread();
Guido van Rossum62332931995-04-10 11:36:14 +0000151
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000152 lock->is_set = 0;
Guido van Rossum62332931995-04-10 11:36:14 +0000153
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000154 DosCreateEventSem(NULL, &lock->changed, 0, 0);
Guido van Rossum62332931995-04-10 11:36:14 +0000155
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000156 dprintf(("%ld: PyThread_allocate_lock() -> %p\n",
157 PyThread_get_thread_ident(),
158 lock->changed));
Tim Petersa6ca4f42001-10-31 03:50:45 +0000159
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000160 return (PyThread_type_lock)lock;
Andrew MacIntyred9400542002-02-26 11:41:34 +0000161#endif
Guido van Rossum62332931995-04-10 11:36:14 +0000162}
163
Thomas Woutersf70ef4f2000-07-22 18:47:25 +0000164void
165PyThread_free_lock(PyThread_type_lock aLock)
Guido van Rossum62332931995-04-10 11:36:14 +0000166{
Andrew MacIntyred9400542002-02-26 11:41:34 +0000167#if !defined(PYCC_GCC)
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000168 type_os2_lock lock = (type_os2_lock)aLock;
Andrew MacIntyred9400542002-02-26 11:41:34 +0000169#endif
170
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000171 dprintf(("%ld: PyThread_free_lock(%p) called\n",
172 PyThread_get_thread_ident(),aLock));
Guido van Rossum62332931995-04-10 11:36:14 +0000173
Andrew MacIntyred9400542002-02-26 11:41:34 +0000174#if defined(PYCC_GCC)
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000175 if (aLock) {
176 _fmutex_close((_fmutex *)aLock);
177 free((_fmutex *)aLock);
178 }
Andrew MacIntyred9400542002-02-26 11:41:34 +0000179#else
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000180 DosCloseEventSem(lock->changed);
181 free(aLock);
Andrew MacIntyred9400542002-02-26 11:41:34 +0000182#endif
Guido van Rossum62332931995-04-10 11:36:14 +0000183}
184
185/*
186 * Return 1 on success if the lock was acquired
187 *
Tim Petersa6ca4f42001-10-31 03:50:45 +0000188 * and 0 if the lock was not acquired.
Guido van Rossum62332931995-04-10 11:36:14 +0000189 */
Thomas Woutersf70ef4f2000-07-22 18:47:25 +0000190int
191PyThread_acquire_lock(PyThread_type_lock aLock, int waitflag)
Guido van Rossum62332931995-04-10 11:36:14 +0000192{
Andrew MacIntyred9400542002-02-26 11:41:34 +0000193#if !defined(PYCC_GCC)
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000194 int done = 0;
195 ULONG count;
196 PID pid = 0;
197 TID tid = 0;
198 type_os2_lock lock = (type_os2_lock)aLock;
Andrew MacIntyred9400542002-02-26 11:41:34 +0000199#endif
Guido van Rossum62332931995-04-10 11:36:14 +0000200
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000201 dprintf(("%ld: PyThread_acquire_lock(%p, %d) called\n",
202 PyThread_get_thread_ident(),
203 aLock,
204 waitflag));
Guido van Rossum62332931995-04-10 11:36:14 +0000205
Andrew MacIntyred9400542002-02-26 11:41:34 +0000206#if defined(PYCC_GCC)
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000207 /* always successful if the lock doesn't exist */
208 if (aLock &&
209 _fmutex_request((_fmutex *)aLock, waitflag ? 0 : _FMR_NOWAIT))
210 return 0;
Andrew MacIntyred9400542002-02-26 11:41:34 +0000211#else
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000212 while (!done) {
213 /* if the lock is currently set, we have to wait for
214 * the state to change
215 */
216 if (lock->is_set) {
217 if (!waitflag)
218 return 0;
219 DosWaitEventSem(lock->changed, SEM_INDEFINITE_WAIT);
220 }
Tim Petersa6ca4f42001-10-31 03:50:45 +0000221
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000222 /* enter a critical section and try to get the semaphore. If
223 * it is still locked, we will try again.
224 */
225 if (DosEnterCritSec())
226 return 0;
Tim Petersa6ca4f42001-10-31 03:50:45 +0000227
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000228 if (!lock->is_set) {
229 lock->is_set = 1;
230 DosResetEventSem(lock->changed, &count);
231 done = 1;
232 }
233
234 DosExitCritSec();
235 }
Andrew MacIntyred9400542002-02-26 11:41:34 +0000236#endif
Guido van Rossum62332931995-04-10 11:36:14 +0000237
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000238 return 1;
Guido van Rossum62332931995-04-10 11:36:14 +0000239}
240
Tim Petersa6ca4f42001-10-31 03:50:45 +0000241void PyThread_release_lock(PyThread_type_lock aLock)
Guido van Rossum62332931995-04-10 11:36:14 +0000242{
Andrew MacIntyre69049592002-12-04 12:27:06 +0000243#if !defined(PYCC_GCC)
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000244 type_os2_lock lock = (type_os2_lock)aLock;
Andrew MacIntyred9400542002-02-26 11:41:34 +0000245#endif
246
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000247 dprintf(("%ld: PyThread_release_lock(%p) called\n",
248 PyThread_get_thread_ident(),
249 aLock));
Guido van Rossum62332931995-04-10 11:36:14 +0000250
Andrew MacIntyred9400542002-02-26 11:41:34 +0000251#if defined(PYCC_GCC)
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000252 if (aLock)
253 _fmutex_release((_fmutex *)aLock);
Andrew MacIntyred9400542002-02-26 11:41:34 +0000254#else
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000255 if (!lock->is_set) {
256 dprintf(("%ld: Could not PyThread_release_lock(%p) error: %l\n",
257 PyThread_get_thread_ident(),
258 aLock,
259 GetLastError()));
260 return;
261 }
Tim Petersa6ca4f42001-10-31 03:50:45 +0000262
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000263 if (DosEnterCritSec()) {
264 dprintf(("%ld: Could not PyThread_release_lock(%p) error: %l\n",
265 PyThread_get_thread_ident(),
266 aLock,
267 GetLastError()));
268 return;
269 }
Tim Petersa6ca4f42001-10-31 03:50:45 +0000270
Andrew MacIntyrec4c127b2002-12-04 12:29:37 +0000271 lock->is_set = 0;
272 DosPostEventSem(lock->changed);
273
274 DosExitCritSec();
Andrew MacIntyred9400542002-02-26 11:41:34 +0000275#endif
Guido van Rossum62332931995-04-10 11:36:14 +0000276}