Re-introduce the changes from change 68 (runops becomes a
[p5sagit/p5-mst-13.2.git] / thread.h
CommitLineData
11343788 1#ifndef USE_THREADS
2#define MUTEX_LOCK(m)
3#define MUTEX_UNLOCK(m)
4#define MUTEX_INIT(m)
5#define MUTEX_DESTROY(m)
6#define COND_INIT(c)
7#define COND_SIGNAL(c)
8#define COND_BROADCAST(c)
9#define COND_WAIT(c, m)
10#define COND_DESTROY(c)
11
12#define THR
13/* Rats: if dTHR is just blank then the subsequent ";" throws an error */
14#define dTHR extern int errno
15#else
11343788 16
12ca11f6 17#ifdef FAKE_THREADS
18typedef struct thread *perl_thread;
19/* With fake threads, thr is global(ish) so we don't need dTHR */
20#define dTHR extern int errno
21
22/*
23 * Note that SCHEDULE() is only callable from pp code (which
24 * must be expecting to be restarted). We'll have to do
25 * something a bit different for XS code.
26 */
27#define SCHEDULE() return schedule(), op
28
29#define MUTEX_LOCK(m)
30#define MUTEX_UNLOCK(m)
31#define MUTEX_INIT(m)
32#define MUTEX_DESTROY(m)
33#define COND_INIT(c) perl_cond_init(c)
34#define COND_SIGNAL(c) perl_cond_signal(c)
35#define COND_BROADCAST(c) perl_cond_broadcast(c)
36#define COND_WAIT(c, m) STMT_START { \
37 perl_cond_wait(c); \
38 SCHEDULE(); \
39 } STMT_END
40#define COND_DESTROY(c)
41
42#else
43/* POSIXish threads */
44typedef pthread_t perl_thread;
11343788 45#ifdef OLD_PTHREADS_API
46#define pthread_mutexattr_init(a) pthread_mutexattr_create(a)
47#define pthread_mutexattr_settype(a,t) pthread_mutexattr_setkind_np(a,t)
48#define pthread_key_create(k,d) pthread_keycreate(k,(pthread_destructor_t)(d))
49#else
50#define pthread_mutexattr_default NULL
51#endif /* OLD_PTHREADS_API */
52
53#define MUTEX_INIT(m) \
54 if (pthread_mutex_init((m), pthread_mutexattr_default)) \
55 croak("panic: MUTEX_INIT"); \
56 else 1
57#define MUTEX_LOCK(m) \
58 if (pthread_mutex_lock((m))) croak("panic: MUTEX_LOCK"); else 1
59#define MUTEX_UNLOCK(m) \
60 if (pthread_mutex_unlock((m))) croak("panic: MUTEX_UNLOCK"); else 1
61#define MUTEX_DESTROY(m) \
62 if (pthread_mutex_destroy((m))) croak("panic: MUTEX_DESTROY"); else 1
63#define COND_INIT(c) \
64 if (pthread_cond_init((c), NULL)) croak("panic: COND_INIT"); else 1
65#define COND_SIGNAL(c) \
66 if (pthread_cond_signal((c))) croak("panic: COND_SIGNAL"); else 1
67#define COND_BROADCAST(c) \
68 if (pthread_cond_broadcast((c))) croak("panic: COND_BROADCAST"); else 1
69#define COND_WAIT(c, m) \
70 if (pthread_cond_wait((c), (m))) croak("panic: COND_WAIT"); else 1
71#define COND_DESTROY(c) \
72 if (pthread_cond_destroy((c))) croak("panic: COND_DESTROY"); else 1
73/* XXX Add "old" (?) POSIX draft interface too */
74#ifdef OLD_PTHREADS_API
75struct thread *getTHR _((void));
76#define THR getTHR()
77#else
78#define THR ((struct thread *) pthread_getspecific(thr_key))
79#endif /* OLD_PTHREADS_API */
80#define dTHR struct thread *thr = THR
12ca11f6 81#endif /* FAKE_THREADS */
11343788 82
83struct thread {
12ca11f6 84 perl_thread Tself;
07b73707 85 SV * Toursv;
11343788 86
87 /* The fields that used to be global */
88 SV ** Tstack_base;
89 SV ** Tstack_sp;
90 SV ** Tstack_max;
91
462e5cf6 92#ifdef OP_IN_REGISTER
93 OP * Topsave;
94#else
11343788 95 OP * Top;
462e5cf6 96#endif
11343788 97
98 I32 * Tscopestack;
99 I32 Tscopestack_ix;
100 I32 Tscopestack_max;
101
102 ANY * Tsavestack;
103 I32 Tsavestack_ix;
104 I32 Tsavestack_max;
105
106 OP ** Tretstack;
107 I32 Tretstack_ix;
108 I32 Tretstack_max;
109
110 I32 * Tmarkstack;
111 I32 * Tmarkstack_ptr;
112 I32 * Tmarkstack_max;
113
114 SV ** Tcurpad;
115
116 SV * TSv;
117 XPV * TXpv;
118 char Tbuf[2048]; /* should be a global locked by a mutex */
119 char Ttokenbuf[256]; /* should be a global locked by a mutex */
120 struct stat Tstatbuf;
121 struct tms Ttimesbuf;
122
123 /* XXX What about regexp stuff? */
124
125 /* Now the fields that used to be "per interpreter" (even when global) */
126
127 /* XXX What about magic variables such as $/, $? and so on? */
128 HV * Tdefstash;
129 HV * Tcurstash;
11343788 130
131 SV ** Ttmps_stack;
132 I32 Ttmps_ix;
133 I32 Ttmps_floor;
134 I32 Ttmps_max;
135
136 int Tin_eval;
137 OP * Trestartop;
138 int Tdelaymagic;
139 bool Tdirty;
140 U8 Tlocalizing;
0f15f207 141 COP * Tcurcop;
11343788 142
143 CONTEXT * Tcxstack;
144 I32 Tcxstack_ix;
145 I32 Tcxstack_max;
146
0f15f207 147 AV * Tcurstack;
11343788 148 AV * Tmainstack;
e858de61 149 JMPENV * Ttop_env;
11343788 150 I32 Trunlevel;
151
152 /* XXX Sort stuff, firstgv, secongv and so on? */
153
12ca11f6 154 perl_mutex *Tthreadstart_mutexp;
11343788 155 HV * Tcvcache;
f93b4edd 156 U32 Tthrflags;
12ca11f6 157
158#ifdef FAKE_THREADS
159 perl_thread next, prev; /* Linked list of all threads */
160 perl_thread next_run, prev_run; /* Linked list of runnable threads */
161 perl_cond wait_queue; /* Wait queue that we are waiting on */
162 IV private; /* Holds data across time slices */
0f15f207 163 I32 savemark; /* Holds MARK for thread join values */
12ca11f6 164#endif /* FAKE_THREADS */
11343788 165};
166
167typedef struct thread *Thread;
168
f93b4edd 169/* Values and macros for thrflags */
07b73707 170#define THRf_STATE_MASK 3
171#define THRf_NORMAL 0
172#define THRf_DETACHED 1
173#define THRf_JOINED 2
174#define THRf_DEAD 3
f93b4edd 175
07b73707 176#define THRf_DIE_FATAL 4
177
178#define ThrSTATE(t) (t->Tthrflags & THRf_STATE_MASK)
f93b4edd 179#define ThrSETSTATE(t, s) STMT_START { \
07b73707 180 (t)->Tthrflags &= ~THRf_STATE_MASK; \
f93b4edd 181 (t)->Tthrflags |= (s); \
182 DEBUG_L(fprintf(stderr, "thread 0x%lx set to state %d\n", \
183 (unsigned long)(t), (s))); \
184 } STMT_END
185
186typedef struct condpair {
12ca11f6 187 perl_mutex mutex;
188 perl_cond owner_cond;
189 perl_cond cond;
190 Thread owner;
f93b4edd 191} condpair_t;
192
193#define MgMUTEXP(mg) (&((condpair_t *)(mg->mg_ptr))->mutex)
194#define MgOWNERCONDP(mg) (&((condpair_t *)(mg->mg_ptr))->owner_cond)
195#define MgCONDP(mg) (&((condpair_t *)(mg->mg_ptr))->cond)
196#define MgOWNER(mg) ((condpair_t *)(mg->mg_ptr))->owner
197
11343788 198#undef stack_base
199#undef stack_sp
200#undef stack_max
0f15f207 201#undef curstack
11343788 202#undef mainstack
203#undef markstack
204#undef markstack_ptr
205#undef markstack_max
206#undef scopestack
207#undef scopestack_ix
208#undef scopestack_max
209#undef savestack
210#undef savestack_ix
211#undef savestack_max
212#undef retstack
213#undef retstack_ix
214#undef retstack_max
0f15f207 215#undef curcop
11343788 216#undef cxstack
217#undef cxstack_ix
218#undef cxstack_max
809a5acc 219#undef defstash
220#undef curstash
6d4ff0d2 221#undef tmps_stack
222#undef tmps_floor
223#undef tmps_ix
224#undef tmps_max
11343788 225#undef curpad
226#undef Sv
227#undef Xpv
11343788 228#undef top_env
229#undef runlevel
230#undef in_eval
809a5acc 231#undef restartop
232#undef delaymagic
233#undef dirty
234#undef localizing
11343788 235
236#define self (thr->Tself)
07b73707 237#define oursv (thr->Toursv)
11343788 238#define stack_base (thr->Tstack_base)
239#define stack_sp (thr->Tstack_sp)
240#define stack_max (thr->Tstack_max)
462e5cf6 241#ifdef OP_IN_REGISTER
242#define opsave (thr->Topsave)
243#else
244#undef op
11343788 245#define op (thr->Top)
462e5cf6 246#endif
0f15f207 247#define curcop (thr->Tcurcop)
11343788 248#define stack (thr->Tstack)
809a5acc 249#define curstack (thr->Tcurstack)
11343788 250#define mainstack (thr->Tmainstack)
251#define markstack (thr->Tmarkstack)
252#define markstack_ptr (thr->Tmarkstack_ptr)
253#define markstack_max (thr->Tmarkstack_max)
254#define scopestack (thr->Tscopestack)
255#define scopestack_ix (thr->Tscopestack_ix)
256#define scopestack_max (thr->Tscopestack_max)
257
258#define savestack (thr->Tsavestack)
259#define savestack_ix (thr->Tsavestack_ix)
260#define savestack_max (thr->Tsavestack_max)
261
262#define retstack (thr->Tretstack)
263#define retstack_ix (thr->Tretstack_ix)
264#define retstack_max (thr->Tretstack_max)
265
266#define cxstack (thr->Tcxstack)
267#define cxstack_ix (thr->Tcxstack_ix)
268#define cxstack_max (thr->Tcxstack_max)
269
270#define curpad (thr->Tcurpad)
271#define Sv (thr->TSv)
272#define Xpv (thr->TXpv)
273#define defstash (thr->Tdefstash)
274#define curstash (thr->Tcurstash)
11343788 275
276#define tmps_stack (thr->Ttmps_stack)
277#define tmps_ix (thr->Ttmps_ix)
278#define tmps_floor (thr->Ttmps_floor)
279#define tmps_max (thr->Ttmps_max)
280
281#define in_eval (thr->Tin_eval)
282#define restartop (thr->Trestartop)
283#define delaymagic (thr->Tdelaymagic)
284#define dirty (thr->Tdirty)
285#define localizing (thr->Tlocalizing)
286
287#define top_env (thr->Ttop_env)
288#define runlevel (thr->Trunlevel)
289
290#define threadstart_mutexp (thr->Tthreadstart_mutexp)
f93b4edd 291#define cvcache (thr->Tcvcache)
292#define thrflags (thr->Tthrflags)
11343788 293#endif /* USE_THREADS */