blob: 66f801535e8d41b47e7cea7cec56b21860133ab7 [file] [log] [blame]
Brian Paula360ab22000-02-10 21:54:06 +00001/* $Id: glthread.c,v 1.5 2000/02/10 21:54:06 brianp Exp $ */
Brian Paulc11371a1999-12-16 17:31:06 +00002
3/*
4 * Mesa 3-D graphics library
5 * Version: 3.3
6 *
Brian Paul8d365ab2000-01-28 18:57:56 +00007 * Copyright (C) 1999-2000 Brian Paul All Rights Reserved.
Brian Paulc11371a1999-12-16 17:31:06 +00008 *
9 * Permission is hereby granted, free of charge, to any person obtaining a
10 * copy of this software and associated documentation files (the "Software"),
11 * to deal in the Software without restriction, including without limitation
12 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
13 * and/or sell copies of the Software, and to permit persons to whom the
14 * Software is furnished to do so, subject to the following conditions:
15 *
16 * The above copyright notice and this permission notice shall be included
17 * in all copies or substantial portions of the Software.
18 *
19 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
20 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
21 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
22 * BRIAN PAUL BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
23 * AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
24 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
25 */
26
27
28/*
Brian Paul8d365ab2000-01-28 18:57:56 +000029 * XXX There's probably some work to do in order to make this file
30 * truly reusable outside of Mesa. First, the glheader.h include must go.
Brian Paulc11371a1999-12-16 17:31:06 +000031 */
32
33
34#ifdef PC_ALL
35#include "all.h"
36#else
37#include "glheader.h"
Brian Paula360ab22000-02-10 21:54:06 +000038#include "glthread.h"
Brian Paulc11371a1999-12-16 17:31:06 +000039#endif
40
41
Brian Paulc11371a1999-12-16 17:31:06 +000042/*
43 * This file should still compile even when THREADS is not defined.
44 * This is to make things easier to deal with on the makefile scene..
45 */
46#ifdef THREADS
47#include <errno.h>
Brian Paulc11371a1999-12-16 17:31:06 +000048
49/*
50 * Error messages
51 */
52#define INIT_TSD_ERROR "_glthread_: failed to allocate key for thread specific data"
53#define GET_TSD_ERROR "_glthread_: failed to get thread specific data"
54#define SET_TSD_ERROR "_glthread_: thread failed to set thread specific data"
55
56
57/*
Brian Paula9601f12000-02-10 21:27:25 +000058 * Magic number to determine if a TSD object has been initialized.
59 * Kind of a hack but there doesn't appear to be a better cross-platform
60 * solution.
Brian Paulc11371a1999-12-16 17:31:06 +000061 */
Brian Paula9601f12000-02-10 21:27:25 +000062#define INIT_MAGIC 0xff8adc98
Brian Paulc11371a1999-12-16 17:31:06 +000063
64
65
66/*
67 * POSIX Threads -- The best way to go if your platform supports them.
68 * Solaris >= 2.5 have POSIX threads, IRIX >= 6.4 reportedly
69 * has them, and many of the free Unixes now have them.
70 * Be sure to use appropriate -mt or -D_REENTRANT type
71 * compile flags when building.
72 */
73#ifdef PTHREADS
74
75unsigned long
76_glthread_GetID(void)
77{
78 return (unsigned long) pthread_self();
79}
80
81
82void
83_glthread_InitTSD(_glthread_TSD *tsd)
84{
Brian Paule2b10e71999-12-17 11:13:54 +000085 if (pthread_key_create(&tsd->key, NULL/*free*/) != 0) {
Brian Paulc11371a1999-12-16 17:31:06 +000086 perror(INIT_TSD_ERROR);
87 exit(-1);
88 }
Brian Paula9601f12000-02-10 21:27:25 +000089 tsd->initMagic = INIT_MAGIC;
Brian Paulc11371a1999-12-16 17:31:06 +000090}
91
92
93void *
94_glthread_GetTSD(_glthread_TSD *tsd)
95{
Brian Paula9601f12000-02-10 21:27:25 +000096 if (tsd->initMagic != INIT_MAGIC) {
97 _glthread_InitTSD(tsd);
98 }
Brian Paulc11371a1999-12-16 17:31:06 +000099 return pthread_getspecific(tsd->key);
100}
101
102
103void
Brian Paula9601f12000-02-10 21:27:25 +0000104_glthread_SetTSD(_glthread_TSD *tsd, void *ptr)
Brian Paulc11371a1999-12-16 17:31:06 +0000105{
Brian Paula9601f12000-02-10 21:27:25 +0000106 if (tsd->initMagic != INIT_MAGIC) {
107 _glthread_InitTSD(tsd);
108 }
Brian Paulc11371a1999-12-16 17:31:06 +0000109 if (pthread_setspecific(tsd->key, ptr) != 0) {
110 perror(SET_TSD_ERROR);
111 exit(-1);
Brian Paule2b10e71999-12-17 11:13:54 +0000112 }
Brian Paulc11371a1999-12-16 17:31:06 +0000113}
114
115#endif /* PTHREADS */
116
117
118
119/*
120 * Solaris/Unix International Threads -- Use only if POSIX threads
121 * aren't available on your Unix platform. Solaris 2.[34] are examples
122 * of platforms where this is the case. Be sure to use -mt and/or
123 * -D_REENTRANT when compiling.
124 */
125#ifdef SOLARIS_THREADS
126#define USE_LOCK_FOR_KEY /* undef this to try a version without
127 lock for the global key... */
128
129unsigned long
130_glthread_GetID(void)
131{
132 abort(); /* XXX not implemented yet */
133 return (unsigned long) 0;
134}
135
136
137void
138_glthread_InitTSD(_glthread_TSD *tsd)
139{
140 if ((errno = mutex_init(&tsd->keylock, 0, NULL)) != 0 ||
141 (errno = thr_keycreate(&(tsd->key), free)) != 0) {
142 perror(INIT_TSD_ERROR);
143 exit(-1);
144 }
Brian Paula9601f12000-02-10 21:27:25 +0000145 tsd->initMagic = INIT_MAGIC;
Brian Paulc11371a1999-12-16 17:31:06 +0000146}
147
148
149void *
150_glthread_GetTSD(_glthread_TSD *tsd)
151{
152 void* ret;
Brian Paula9601f12000-02-10 21:27:25 +0000153 if (tsd->initMagic != INIT_MAGIC) {
154 _glthread_InitTSD(tsd);
155 }
Brian Paulc11371a1999-12-16 17:31:06 +0000156#ifdef USE_LOCK_FOR_KEY
157 mutex_lock(&tsd->keylock);
158 thr_getspecific(tsd->key, &ret);
159 mutex_unlock(&tsd->keylock);
160#else
161 if ((errno = thr_getspecific(tsd->key, &ret)) != 0) {
162 perror(GET_TSD_ERROR);
163 exit(-1);
Brian Paule2b10e71999-12-17 11:13:54 +0000164 }
Brian Paulc11371a1999-12-16 17:31:06 +0000165#endif
166 return ret;
167}
168
169
170void
Brian Paula9601f12000-02-10 21:27:25 +0000171_glthread_SetTSD(_glthread_TSD *tsd, void *ptr)
Brian Paulc11371a1999-12-16 17:31:06 +0000172{
Brian Paula9601f12000-02-10 21:27:25 +0000173 if (tsd->initMagic != INIT_MAGIC) {
174 _glthread_InitTSD(tsd);
Brian Paulc11371a1999-12-16 17:31:06 +0000175 }
176 if ((errno = thr_setspecific(tsd->key, ptr)) != 0) {
177 perror(SET_TSD_ERROR);
178 exit(-1);
Brian Paule2b10e71999-12-17 11:13:54 +0000179 }
Brian Paulc11371a1999-12-16 17:31:06 +0000180}
181
182#undef USE_LOCK_FOR_KEY
183#endif /* SOLARIS_THREADS */
184
185
186
187/*
188 * Win32 Threads. The only available option for Windows 95/NT.
189 * Be sure that you compile using the Multithreaded runtime, otherwise
190 * bad things will happen.
191 */
192#ifdef WIN32
193
194unsigned long
195_glthread_GetID(void)
196{
197 abort(); /* XXX not implemented yet */
198 return (unsigned long) 0;
199}
200
201
202void
203_glthread_InitTSD(_glthread_TSD *tsd)
204{
205 tsd->key = TlsAlloc();
206 if (tsd->key == 0xffffffff) {
207 /* Can Windows handle stderr messages for non-console
208 applications? Does Windows have perror? */
209 /* perror(SET_INIT_ERROR);*/
210 exit(-1);
211 }
Brian Paula9601f12000-02-10 21:27:25 +0000212 tsd->initMagic = INIT_MAGIC;
Brian Paulc11371a1999-12-16 17:31:06 +0000213}
214
215
216void *
217_glthread_GetTSD(_glthread_TSD *tsd)
218{
Brian Paula9601f12000-02-10 21:27:25 +0000219 if (tsd->initMagic != INIT_MAGIC) {
220 _glthread_InitTSD(tsd);
221 }
Brian Paulc11371a1999-12-16 17:31:06 +0000222 return TlsGetValue(tsd->key);
223}
224
225
226void
Brian Paula9601f12000-02-10 21:27:25 +0000227_glthread_SetTSD(_glthread_TSD *tsd, void *ptr)
Brian Paulc11371a1999-12-16 17:31:06 +0000228{
229 /* the following code assumes that the _glthread_TSD has been initialized
230 to zero at creation */
Brian Paula9601f12000-02-10 21:27:25 +0000231 if (tsd->initMagic != INIT_MAGIC) {
232 _glthread_InitTSD(tsd);
Brian Paulc11371a1999-12-16 17:31:06 +0000233 }
234 if (TlsSetValue(tsd->key, ptr) == 0) {
235 /* Can Windows handle stderr messages for non-console
236 applications? Does Windows have perror? */
237 /* perror(SET_TSD_ERROR);*/
238 exit(-1);
Brian Paule2b10e71999-12-17 11:13:54 +0000239 }
Brian Paulc11371a1999-12-16 17:31:06 +0000240}
241
242#endif /* WIN32 */
243
Brian Paula9601f12000-02-10 21:27:25 +0000244
245
246/*
247 * XFree86 has its own thread wrapper, Xthreads.h
248 * We wrap it again for GL.
249 */
250#ifdef XTHREADS
251
252unsigned long
253_glthread_GetID(void)
254{
255 return (unsigned long) xthread_self();
256}
257
258
259void
260_glthread_InitTSD(_glthread_TSD *tsd)
261{
262 if (xthread_key_create(&tsd->key, NULL) != 0) {
263 perror(INIT_TSD_ERROR);
264 exit(-1);
265 }
266 tsd->initMagic = INIT_MAGIC;
267}
268
269
270void *
271_glthread_GetTSD(_glthread_TSD *tsd)
272{
273 void *ptr;
274 if (tsd->initMagic != INIT_MAGIC) {
275 _glthread_InitTSD(tsd);
276 }
277 xthread_get_specific(tsd->key, &ptr);
278 return ptr;
279}
280
281
282void
283_glthread_SetTSD(_glthread_TSD *tsd, void *ptr)
284{
285 if (tsd->initMagic != INIT_MAGIC) {
286 _glthread_InitTSD(tsd);
287 }
288 xthread_set_specific(tsd->key, ptr);
289}
290
291#endif /* XTHREAD */
292
293
Brian Paula360ab22000-02-10 21:54:06 +0000294
295#else /* THREADS */
296
297
298/*
299 * no-op functions
300 */
301
302unsigned long
303_glthread_GetID(void)
304{
305 return 0;
306}
307
308
309void
310_glthread_InitTSD(_glthread_TSD *tsd)
311{
312 (void) tsd;
313}
314
315
316void *
317_glthread_GetTSD(_glthread_TSD *tsd)
318{
319 (void) tsd;
320 return NULL;
321}
322
323
324void
325_glthread_SetTSD(_glthread_TSD *tsd, void *ptr)
326{
327 (void) tsd;
328 (void) ptr;
329}
330
331
Brian Paulc11371a1999-12-16 17:31:06 +0000332#endif /* THREADS */