Add some more Alpha asm code and Alpha-specific parts.
[wine/multimedia.git] / loader / kthread.c
blob6fe101ffa8eb080fac6a9951e4fffa07108f7cdb
1 /*
2 * pthread emulation based on kernel threads
4 * We can't use pthreads directly, so why not let libcs
5 * that want pthreads use Wine's own threading instead...
7 * Copyright 1999 Ove Kåven
8 * Copyright 2003 Alexandre Julliard
10 * This library is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU Lesser General Public
12 * License as published by the Free Software Foundation; either
13 * version 2.1 of the License, or (at your option) any later version.
15 * This library is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 * Lesser General Public License for more details.
20 * You should have received a copy of the GNU Lesser General Public
21 * License along with this library; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
25 #include "config.h"
26 #include "wine/port.h"
28 struct _pthread_cleanup_buffer;
30 #include <assert.h>
31 #include <errno.h>
32 #include <stdlib.h>
33 #include <signal.h>
34 #include <setjmp.h>
35 #ifdef HAVE_UNISTD_H
36 # include <unistd.h>
37 #endif
38 #include <string.h>
39 #include <sys/types.h>
40 #ifdef HAVE_SYS_SOCKET_H
41 # include <sys/socket.h>
42 #endif
43 #ifdef HAVE_SYS_MMAN_H
44 #include <sys/mman.h>
45 #endif
46 #ifdef HAVE_NETINET_IN_H
47 # include <netinet/in.h>
48 #endif
49 #ifdef HAVE_ARPA_NAMESER_H
50 # include <arpa/nameser.h>
51 #endif
52 #ifdef HAVE_RESOLV_H
53 # include <resolv.h>
54 #endif
55 #ifdef HAVE_VALGRIND_MEMCHECK_H
56 #include <valgrind/memcheck.h>
57 #endif
58 #ifdef HAVE_SYS_SYSCALL_H
59 # include <sys/syscall.h>
60 #endif
61 #ifdef HAVE_SYS_LWP_H
62 # include <sys/lwp.h>
63 #endif
64 #ifdef HAVE_UCONTEXT_H
65 # include <ucontext.h>
66 #endif
67 #ifdef HAVE_SCHED_H
68 #include <sched.h>
69 #endif
71 #include "wine/library.h"
72 #include "wine/pthread.h"
74 #define P_OUTPUT(stuff) write(2,stuff,strlen(stuff))
76 #define PSTR(str) __ASM_NAME(#str)
78 static struct wine_pthread_functions funcs;
80 /* thread descriptor */
82 #define FIRST_KEY 0
83 #define MAX_KEYS 16 /* libc6 doesn't use that many, but... */
84 #define MAX_TSD 16
86 struct pthread_descr_struct
88 char dummy[2048];
89 int thread_errno;
90 int thread_h_errno;
91 int cancel_state;
92 int cancel_type;
93 struct __res_state res_state;
94 const void *key_data[MAX_KEYS]; /* for normal pthread keys */
95 const void *tsd_data[MAX_TSD]; /* for libc internal tsd variables */
98 typedef struct pthread_descr_struct *pthread_descr;
100 static struct pthread_descr_struct initial_descr;
102 pthread_descr __pthread_thread_self(void)
104 struct pthread_descr_struct *descr;
105 if (!funcs.ptr_get_thread_data) return &initial_descr;
106 descr = funcs.ptr_get_thread_data();
107 if (!descr) return &initial_descr;
108 return descr;
111 static int (*libc_uselocale)(int set);
112 static int *libc_multiple_threads;
114 /***********************************************************************
115 * __errno_location/__error/__errno/___errno/__thr_errno
117 * Get the per-thread errno location.
119 int *__errno_location(void) /* Linux */
121 pthread_descr descr = __pthread_thread_self();
122 return &descr->thread_errno;
124 int *__error(void) { return __errno_location(); } /* FreeBSD */
125 int *__errno(void) { return __errno_location(); } /* NetBSD */
126 int *___errno(void) { return __errno_location(); } /* Solaris */
127 int *__thr_errno(void) { return __errno_location(); } /* UnixWare */
129 /***********************************************************************
130 * __h_errno_location
132 * Get the per-thread h_errno location.
134 int *__h_errno_location(void)
136 pthread_descr descr = __pthread_thread_self();
137 return &descr->thread_h_errno;
140 struct __res_state *__res_state(void)
142 pthread_descr descr = __pthread_thread_self();
143 return &descr->res_state;
146 static inline void writejump( const char *symbol, void *dest )
148 #if defined(__GLIBC__) && defined(__i386__)
149 unsigned char *addr = wine_dlsym( RTLD_NEXT, symbol, NULL, 0 );
151 if (!addr) return;
153 /* write a relative jump at the function address */
154 mprotect((void*)((unsigned int)addr & ~(getpagesize()-1)), 5, PROT_READ|PROT_EXEC|PROT_WRITE);
155 addr[0] = 0xe9;
156 *(int *)(addr+1) = (unsigned char *)dest - (addr + 5);
157 mprotect((void*)((unsigned int)addr & ~(getpagesize()-1)), 5, PROT_READ|PROT_EXEC);
159 # ifdef HAVE_VALGRIND_MEMCHECK_H
160 VALGRIND_DISCARD_TRANSLATIONS( addr, 5 );
161 # endif
162 #endif /* __GLIBC__ && __i386__ */
165 /* temporary stacks used on thread exit */
166 #define TEMP_STACK_SIZE 1024
167 #define NB_TEMP_STACKS 8
168 static char temp_stacks[NB_TEMP_STACKS][TEMP_STACK_SIZE];
169 static LONG next_temp_stack; /* next temp stack to use */
171 /***********************************************************************
172 * get_temp_stack
174 * Get a temporary stack address to run the thread exit code on.
176 inline static char *get_temp_stack(void)
178 unsigned int next = interlocked_xchg_add( &next_temp_stack, 1 );
179 return temp_stacks[next % NB_TEMP_STACKS] + TEMP_STACK_SIZE;
183 /***********************************************************************
184 * cleanup_thread
186 * Cleanup the remains of a thread. Runs on a temporary stack.
188 static void cleanup_thread( void *ptr )
190 /* copy the info structure since it is on the stack we will free */
191 struct wine_pthread_thread_info info = *(struct wine_pthread_thread_info *)ptr;
192 wine_ldt_free_fs( info.teb_sel );
193 munmap( info.stack_base, info.stack_size );
194 munmap( info.teb_base, info.teb_size );
195 #ifdef HAVE__LWP_CREATE
196 _lwp_exit();
197 #endif
198 _exit( info.exit_status );
202 /***********************************************************************
203 * wine_pthread_init_process
205 * Initialization for a newly created process.
207 void wine_pthread_init_process( const struct wine_pthread_functions *functions )
209 memcpy( &funcs, functions, min(functions->size,sizeof(funcs)) );
210 funcs.ptr_set_thread_data( &initial_descr );
214 /***********************************************************************
215 * wine_pthread_init_thread
217 * Initialization for a newly created thread.
219 void wine_pthread_init_thread( struct wine_pthread_thread_info *info )
221 struct pthread_descr_struct *descr;
223 if (funcs.ptr_set_thread_data)
225 descr = calloc( 1, sizeof(*descr) );
226 funcs.ptr_set_thread_data( descr );
227 if (libc_multiple_threads) *libc_multiple_threads = 1;
229 else /* first thread */
231 descr = &initial_descr;
232 writejump( "__errno_location", __errno_location );
233 writejump( "__h_errno_location", __h_errno_location );
234 writejump( "__res_state", __res_state );
236 descr->cancel_state = PTHREAD_CANCEL_ENABLE;
237 descr->cancel_type = PTHREAD_CANCEL_ASYNCHRONOUS;
238 if (libc_uselocale) libc_uselocale( -1 /*LC_GLOBAL_LOCALE*/ );
242 /***********************************************************************
243 * wine_pthread_create_thread
245 int wine_pthread_create_thread( struct wine_pthread_thread_info *info )
247 if (!info->stack_base)
249 info->stack_base = wine_anon_mmap( NULL, info->stack_size,
250 PROT_READ | PROT_WRITE | PROT_EXEC, 0 );
251 if (info->stack_base == (void *)-1) return -1;
253 #ifdef HAVE_CLONE
254 if (clone( (int (*)(void *))info->entry, (char *)info->stack_base + info->stack_size,
255 CLONE_VM | CLONE_FS | CLONE_FILES | SIGCHLD, info ) < 0)
256 return -1;
257 return 0;
258 #elif defined(HAVE_RFORK)
260 void **sp = (void **)((char *)info->stack_base + info->stack_size);
261 *--sp = info;
262 *--sp = 0;
263 *--sp = info->entry;
264 __asm__ __volatile__(
265 "pushl %2;\n\t" /* flags */
266 "pushl $0;\n\t" /* 0 ? */
267 "movl %1,%%eax;\n\t" /* SYS_rfork */
268 ".byte 0x9a; .long 0; .word 7;\n\t" /* lcall 7:0... FreeBSD syscall */
269 "cmpl $0, %%edx;\n\t"
270 "je 1f;\n\t"
271 "movl %0,%%esp;\n\t" /* child -> new thread */
272 "ret;\n"
273 "1:\n\t" /* parent -> caller thread */
274 "addl $8,%%esp" :
275 : "r" (sp), "g" (SYS_rfork), "g" (RFPROC | RFMEM | RFTHREAD)
276 : "eax", "edx");
277 return 0;
279 #elif defined(HAVE__LWP_CREATE)
281 ucontext_t context;
282 _lwp_makecontext( &context, (void(*)(void *))info->entry, info,
283 NULL, info->stack_base, info->stack_size );
284 if ( _lwp_create( &context, 0, NULL ) )
285 return -1;
286 return 0;
288 #endif
289 return -1;
293 /***********************************************************************
294 * wine_pthread_init_current_teb
296 * Set the current TEB for a new thread.
298 void wine_pthread_init_current_teb( struct wine_pthread_thread_info *info )
300 #ifdef __i386__
301 /* On the i386, the current thread is in the %fs register */
302 LDT_ENTRY fs_entry;
304 wine_ldt_set_base( &fs_entry, info->teb_base );
305 wine_ldt_set_limit( &fs_entry, info->teb_size - 1 );
306 wine_ldt_set_flags( &fs_entry, WINE_LDT_FLAGS_DATA|WINE_LDT_FLAGS_32BIT );
307 wine_ldt_init_fs( info->teb_sel, &fs_entry );
308 #elif defined(__powerpc__)
309 /* On PowerPC, the current TEB is in the gpr13 register */
310 # ifdef __APPLE__
311 __asm__ __volatile__("mr r13, %0" : : "r" (info->teb_base));
312 # else
313 __asm__ __volatile__("mr 2, %0" : : "r" (info->teb_base));
314 # endif
315 #elif defined(__ALPHA__)
316 /* FIXME: On Alpha, the current TEB is not accessible to user-space */
317 /* __asm__ __volatile__();*/
318 #else
319 # error You must implement wine_pthread_init_current_teb for your platform
320 #endif
322 /* set pid and tid */
323 info->pid = getpid();
324 #ifdef HAVE__LWP_SELF
325 info->tid = _lwp_self();
326 #else
327 info->tid = -1;
328 #endif
332 /***********************************************************************
333 * wine_pthread_get_current_teb
335 void *wine_pthread_get_current_teb(void)
337 void *ret;
339 #ifdef __i386__
340 __asm__( ".byte 0x64\n\tmovl 0x18,%0" : "=r" (ret) );
341 #elif defined(HAVE__LWP_CREATE)
342 ret = _lwp_getprivate();
343 #elif defined(__powerpc__)
344 # ifdef __APPLE__
345 __asm__( "mr %0,r13" : "=r" (ret) );
346 # else
347 __asm__( "mr %0,2" : "=r" (ret) );
348 # endif
349 #elif defined(__ALPHA__)
350 /* 0x00ab is the PAL opcode for rdteb */
351 __asm__( "lda $30,8($30)\n\t"
352 "stq $0,0($30)\n\t"
353 "call_pal 0x00ab\n\t"
354 "mov $0,%0\n\t"
355 "ldq $0,0($30)\n\t"
356 "lda $30,-8($30)" : "=r" (ret) );
357 #else
358 # error wine_pthread_get_current_teb not defined for this architecture
359 #endif /* __i386__ */
361 return ret;
365 /***********************************************************************
366 * wine_pthread_exit_thread
368 void wine_pthread_exit_thread( struct wine_pthread_thread_info *info )
370 wine_switch_to_stack( cleanup_thread, info, get_temp_stack() );
374 /***********************************************************************
375 * wine_pthread_abort_thread
377 void wine_pthread_abort_thread( int status )
379 #ifdef HAVE__LWP_CREATE
380 _lwp_exit();
381 #endif
382 _exit( status );
386 /* Currently this probably works only for glibc2,
387 * which checks for the presence of double-underscore-prepended
388 * pthread primitives, and use them if available.
389 * If they are not available, the libc defaults to
390 * non-threadsafe operation (not good). */
392 #if defined(__GLIBC__) || defined(__FreeBSD__)
394 /* adapt as necessary (a construct like this is used in glibc sources) */
395 #define strong_alias(orig, alias) \
396 asm(".globl " PSTR(alias) "\n" \
397 "\t.set " PSTR(alias) "," PSTR(orig))
399 struct fork_block;
401 /* pthread functions redirection */
403 struct pthread_functions
405 pid_t (*ptr_pthread_fork) (struct fork_block *);
406 int (*ptr_pthread_attr_destroy) (pthread_attr_t *);
407 int (*ptr___pthread_attr_init_2_0) (pthread_attr_t *);
408 int (*ptr___pthread_attr_init_2_1) (pthread_attr_t *);
409 int (*ptr_pthread_attr_getdetachstate) (const pthread_attr_t *, int *);
410 int (*ptr_pthread_attr_setdetachstate) (pthread_attr_t *, int);
411 int (*ptr_pthread_attr_getinheritsched) (const pthread_attr_t *, int *);
412 int (*ptr_pthread_attr_setinheritsched) (pthread_attr_t *, int);
413 int (*ptr_pthread_attr_getschedparam) (const pthread_attr_t *, struct sched_param *);
414 int (*ptr_pthread_attr_setschedparam) (pthread_attr_t *, const struct sched_param *);
415 int (*ptr_pthread_attr_getschedpolicy) (const pthread_attr_t *, int *);
416 int (*ptr_pthread_attr_setschedpolicy) (pthread_attr_t *, int);
417 int (*ptr_pthread_attr_getscope) (const pthread_attr_t *, int *);
418 int (*ptr_pthread_attr_setscope) (pthread_attr_t *, int);
419 int (*ptr_pthread_condattr_destroy) (pthread_condattr_t *);
420 int (*ptr_pthread_condattr_init) (pthread_condattr_t *);
421 int (*ptr___pthread_cond_broadcast) (pthread_cond_t *);
422 int (*ptr___pthread_cond_destroy) (pthread_cond_t *);
423 int (*ptr___pthread_cond_init) (pthread_cond_t *, const pthread_condattr_t *);
424 int (*ptr___pthread_cond_signal) (pthread_cond_t *);
425 int (*ptr___pthread_cond_wait) (pthread_cond_t *, pthread_mutex_t *);
426 int (*ptr_pthread_equal) (pthread_t, pthread_t);
427 void (*ptr___pthread_exit) (void *);
428 int (*ptr_pthread_getschedparam) (pthread_t, int *, struct sched_param *);
429 int (*ptr_pthread_setschedparam) (pthread_t, int, const struct sched_param *);
430 int (*ptr_pthread_mutex_destroy) (pthread_mutex_t *);
431 int (*ptr_pthread_mutex_init) (pthread_mutex_t *, const pthread_mutexattr_t *);
432 int (*ptr_pthread_mutex_lock) (pthread_mutex_t *);
433 int (*ptr_pthread_mutex_trylock) (pthread_mutex_t *);
434 int (*ptr_pthread_mutex_unlock) (pthread_mutex_t *);
435 pthread_t (*ptr_pthread_self) (void);
436 int (*ptr_pthread_setcancelstate) (int, int *);
437 int (*ptr_pthread_setcanceltype) (int, int *);
438 void (*ptr_pthread_do_exit) (void *retval, char *currentframe);
439 void (*ptr_pthread_cleanup_upto) (jmp_buf target, char *targetframe);
440 pthread_descr (*ptr_pthread_thread_self) (void);
441 int (*ptr_pthread_internal_tsd_set) (int key, const void *pointer);
442 void * (*ptr_pthread_internal_tsd_get) (int key);
443 void ** __attribute__ ((__const__)) (*ptr_pthread_internal_tsd_address) (int key);
444 int (*ptr_pthread_sigaction) (int sig, const struct sigaction * act, struct sigaction *oact);
445 int (*ptr_pthread_sigwait) (const sigset_t *set, int *sig);
446 int (*ptr_pthread_raise) (int sig);
447 int (*ptr___pthread_cond_timedwait) (pthread_cond_t *, pthread_mutex_t *, const struct timespec *);
448 void (*ptr__pthread_cleanup_push) (struct _pthread_cleanup_buffer * buffer, void (*routine)(void *), void * arg);
449 void (*ptr__pthread_cleanup_pop) (struct _pthread_cleanup_buffer * buffer, int execute);
452 static pid_t (*libc_fork)(void);
453 static int (*libc_sigaction)(int signum, const struct sigaction *act, struct sigaction *oldact);
454 static int *(*libc_pthread_init)( const struct pthread_functions *funcs );
456 static struct pthread_functions libc_pthread_functions;
458 strong_alias(__pthread_thread_self, pthread_thread_self);
460 /* redefine this to prevent libpthread from overriding our function pointers */
461 int *__libc_pthread_init( const struct pthread_functions *funcs )
463 return libc_multiple_threads;
466 typedef struct _wine_cleanup {
467 void (*routine)(void *);
468 void *arg;
469 } *wine_cleanup;
471 int pthread_create(pthread_t* thread, const pthread_attr_t* attr, void*
472 (*start_routine)(void *), void* arg)
474 assert( funcs.ptr_pthread_create );
475 return funcs.ptr_pthread_create( thread, attr, start_routine, arg );
478 int pthread_cancel(pthread_t thread)
480 assert( funcs.ptr_pthread_cancel );
481 return funcs.ptr_pthread_cancel( thread );
484 int pthread_join(pthread_t thread, void **value_ptr)
486 assert( funcs.ptr_pthread_join );
487 return funcs.ptr_pthread_join( thread, value_ptr );
490 int pthread_detach(pthread_t thread)
492 assert( funcs.ptr_pthread_detach );
493 return funcs.ptr_pthread_detach( thread );
496 /* FIXME: we have no equivalents in win32 for the policys */
497 /* so just keep this as a stub */
498 int pthread_attr_setschedpolicy(pthread_attr_t *attr, int policy)
500 P_OUTPUT("FIXME:pthread_attr_setschedpolicy\n");
501 return 0;
504 /* FIXME: no win32 equivalent for scope */
505 int pthread_attr_setscope(pthread_attr_t *attr, int scope)
507 P_OUTPUT("FIXME:pthread_attr_setscope\n");
508 return 0; /* return success */
511 /* FIXME: no win32 equivalent for schedule param */
512 int pthread_attr_setschedparam(pthread_attr_t *attr,
513 const struct sched_param *param)
515 P_OUTPUT("FIXME:pthread_attr_setschedparam\n");
516 return 0; /* return success */
519 /* FIXME */
520 int pthread_attr_setstack(pthread_attr_t *attr, void *addr, size_t size)
522 return 0; /* return success */
525 int __pthread_once(pthread_once_t *once_control, void (*init_routine)(void))
527 static pthread_once_t the_once = PTHREAD_ONCE_INIT;
528 long once_now;
530 memcpy(&once_now,&the_once,sizeof(once_now));
531 if (interlocked_cmpxchg((long*)once_control, once_now+1, once_now) == once_now)
532 (*init_routine)();
533 return 0;
535 strong_alias(__pthread_once, pthread_once);
537 void __pthread_kill_other_threads_np(void)
539 /* we don't need to do anything here */
541 strong_alias(__pthread_kill_other_threads_np, pthread_kill_other_threads_np);
543 /***** atfork *****/
545 #define MAX_ATFORK 8 /* libc doesn't need that many anyway */
547 static pthread_mutex_t atfork_mutex = PTHREAD_MUTEX_INITIALIZER;
549 typedef void (*atfork_handler)();
550 static atfork_handler atfork_prepare[MAX_ATFORK];
551 static atfork_handler atfork_parent[MAX_ATFORK];
552 static atfork_handler atfork_child[MAX_ATFORK];
553 static int atfork_count;
555 int __pthread_atfork(void (*prepare)(void), void (*parent)(void), void (*child)(void))
557 pthread_mutex_lock( &atfork_mutex );
558 assert( atfork_count < MAX_ATFORK );
559 atfork_prepare[atfork_count] = prepare;
560 atfork_parent[atfork_count] = parent;
561 atfork_child[atfork_count] = child;
562 atfork_count++;
563 pthread_mutex_unlock( &atfork_mutex );
564 return 0;
566 strong_alias(__pthread_atfork, pthread_atfork);
568 pid_t __fork(void)
570 pid_t pid;
571 int i;
573 if (!libc_fork)
575 libc_fork = wine_dlsym( RTLD_NEXT, "fork", NULL, 0 );
576 assert( libc_fork );
578 pthread_mutex_lock( &atfork_mutex );
579 /* prepare handlers are called in reverse insertion order */
580 for (i = atfork_count - 1; i >= 0; i--) if (atfork_prepare[i]) atfork_prepare[i]();
581 if (!(pid = libc_fork()))
583 pthread_mutex_init( &atfork_mutex, NULL );
584 for (i = 0; i < atfork_count; i++) if (atfork_child[i]) atfork_child[i]();
586 else
588 for (i = 0; i < atfork_count; i++) if (atfork_parent[i]) atfork_parent[i]();
589 pthread_mutex_unlock( &atfork_mutex );
591 return pid;
593 strong_alias(__fork, fork);
595 /***** MUTEXES *****/
597 int __pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *mutexattr)
599 if (!funcs.ptr_pthread_mutex_init) return 0;
600 return funcs.ptr_pthread_mutex_init( mutex, mutexattr );
602 strong_alias(__pthread_mutex_init, pthread_mutex_init);
604 int __pthread_mutex_lock(pthread_mutex_t *mutex)
606 if (!funcs.ptr_pthread_mutex_lock) return 0;
607 return funcs.ptr_pthread_mutex_lock( mutex );
609 strong_alias(__pthread_mutex_lock, pthread_mutex_lock);
611 int __pthread_mutex_trylock(pthread_mutex_t *mutex)
613 if (!funcs.ptr_pthread_mutex_trylock) return 0;
614 return funcs.ptr_pthread_mutex_trylock( mutex );
616 strong_alias(__pthread_mutex_trylock, pthread_mutex_trylock);
618 int __pthread_mutex_unlock(pthread_mutex_t *mutex)
620 if (!funcs.ptr_pthread_mutex_unlock) return 0;
621 return funcs.ptr_pthread_mutex_unlock( mutex );
623 strong_alias(__pthread_mutex_unlock, pthread_mutex_unlock);
625 int __pthread_mutex_destroy(pthread_mutex_t *mutex)
627 if (!funcs.ptr_pthread_mutex_destroy) return 0;
628 return funcs.ptr_pthread_mutex_destroy( mutex );
630 strong_alias(__pthread_mutex_destroy, pthread_mutex_destroy);
633 /***** MUTEX ATTRIBUTES *****/
634 /* just dummies, since critical sections are always recursive */
636 int __pthread_mutexattr_init(pthread_mutexattr_t *attr)
638 return 0;
640 strong_alias(__pthread_mutexattr_init, pthread_mutexattr_init);
642 int __pthread_mutexattr_destroy(pthread_mutexattr_t *attr)
644 return 0;
646 strong_alias(__pthread_mutexattr_destroy, pthread_mutexattr_destroy);
648 int __pthread_mutexattr_setkind_np(pthread_mutexattr_t *attr, int kind)
650 return 0;
652 strong_alias(__pthread_mutexattr_setkind_np, pthread_mutexattr_setkind_np);
654 int __pthread_mutexattr_getkind_np(pthread_mutexattr_t *attr, int *kind)
656 *kind = PTHREAD_MUTEX_RECURSIVE;
657 return 0;
659 strong_alias(__pthread_mutexattr_getkind_np, pthread_mutexattr_getkind_np);
661 int __pthread_mutexattr_settype(pthread_mutexattr_t *attr, int kind)
663 return 0;
665 strong_alias(__pthread_mutexattr_settype, pthread_mutexattr_settype);
667 int __pthread_mutexattr_gettype(const pthread_mutexattr_t *attr, int *kind)
669 *kind = PTHREAD_MUTEX_RECURSIVE;
670 return 0;
672 strong_alias(__pthread_mutexattr_gettype, pthread_mutexattr_gettype);
675 /***** THREAD-SPECIFIC VARIABLES (KEYS) *****/
677 int __pthread_key_create(pthread_key_t *key, void (*destr_function)(void *))
679 static long keycnt = FIRST_KEY;
680 *key = interlocked_xchg_add(&keycnt, 1);
681 return 0;
683 strong_alias(__pthread_key_create, pthread_key_create);
685 int __pthread_key_delete(pthread_key_t key)
687 return 0;
689 strong_alias(__pthread_key_delete, pthread_key_delete);
691 int __pthread_setspecific(pthread_key_t key, const void *pointer)
693 pthread_descr descr = __pthread_thread_self();
694 descr->key_data[key] = pointer;
695 return 0;
697 strong_alias(__pthread_setspecific, pthread_setspecific);
699 void *__pthread_getspecific(pthread_key_t key)
701 pthread_descr descr = __pthread_thread_self();
702 return (void *)descr->key_data[key];
704 strong_alias(__pthread_getspecific, pthread_getspecific);
706 static int pthread_internal_tsd_set( int key, const void *pointer )
708 pthread_descr descr = __pthread_thread_self();
709 descr->tsd_data[key] = pointer;
710 return 0;
712 int (*__libc_internal_tsd_set)(int, const void *) = pthread_internal_tsd_set;
714 static void *pthread_internal_tsd_get( int key )
716 pthread_descr descr = __pthread_thread_self();
717 return (void *)descr->tsd_data[key];
719 void* (*__libc_internal_tsd_get)(int) = pthread_internal_tsd_get;
721 static void ** __attribute__((const)) pthread_internal_tsd_address( int key )
723 pthread_descr descr = __pthread_thread_self();
724 return (void **)&descr->tsd_data[key];
726 void** (*__libc_internal_tsd_address)(int) = pthread_internal_tsd_address;
728 /***** "EXCEPTION" FRAMES *****/
729 /* not implemented right now */
731 void _pthread_cleanup_push(struct _pthread_cleanup_buffer *buffer, void (*routine)(void *), void *arg)
733 ((wine_cleanup)buffer)->routine = routine;
734 ((wine_cleanup)buffer)->arg = arg;
737 void _pthread_cleanup_pop(struct _pthread_cleanup_buffer *buffer, int execute)
739 if (execute) (*(((wine_cleanup)buffer)->routine))(((wine_cleanup)buffer)->arg);
742 void _pthread_cleanup_push_defer(struct _pthread_cleanup_buffer *buffer, void (*routine)(void *), void *arg)
744 _pthread_cleanup_push(buffer, routine, arg);
747 void _pthread_cleanup_pop_restore(struct _pthread_cleanup_buffer *buffer, int execute)
749 _pthread_cleanup_pop(buffer, execute);
752 void __pthread_cleanup_upto(jmp_buf target, char *frame)
754 /* FIXME */
757 /***** CONDITIONS *****/
759 int __pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *cond_attr)
761 if (!funcs.ptr_pthread_cond_init) return 0;
762 return funcs.ptr_pthread_cond_init(cond, cond_attr);
764 strong_alias(__pthread_cond_init, pthread_cond_init);
766 int __pthread_cond_destroy(pthread_cond_t *cond)
768 if (!funcs.ptr_pthread_cond_destroy) return 0;
769 return funcs.ptr_pthread_cond_destroy(cond);
771 strong_alias(__pthread_cond_destroy, pthread_cond_destroy);
773 int __pthread_cond_signal(pthread_cond_t *cond)
775 if (!funcs.ptr_pthread_cond_signal) return 0;
776 return funcs.ptr_pthread_cond_signal(cond);
778 strong_alias(__pthread_cond_signal, pthread_cond_signal);
780 int __pthread_cond_broadcast(pthread_cond_t *cond)
782 if (!funcs.ptr_pthread_cond_broadcast) return 0;
783 return funcs.ptr_pthread_cond_broadcast(cond);
785 strong_alias(__pthread_cond_broadcast, pthread_cond_broadcast);
787 int __pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex)
789 if (!funcs.ptr_pthread_cond_wait) return 0;
790 return funcs.ptr_pthread_cond_wait(cond, mutex);
792 strong_alias(__pthread_cond_wait, pthread_cond_wait);
794 int __pthread_cond_timedwait(pthread_cond_t *cond, pthread_mutex_t *mutex, const struct timespec *abstime)
796 if (!funcs.ptr_pthread_cond_timedwait) return 0;
797 return funcs.ptr_pthread_cond_timedwait(cond, mutex, abstime);
799 strong_alias(__pthread_cond_timedwait, pthread_cond_timedwait);
801 /**** CONDITION ATTRIBUTES *****/
802 /* not implemented right now */
804 int pthread_condattr_init(pthread_condattr_t *attr)
806 return 0;
809 int pthread_condattr_destroy(pthread_condattr_t *attr)
811 return 0;
814 /***** READ-WRITE LOCKS *****/
816 int __pthread_rwlock_init(pthread_rwlock_t *rwlock, const pthread_rwlockattr_t *rwlock_attr)
818 assert( funcs.ptr_pthread_rwlock_init );
819 return funcs.ptr_pthread_rwlock_init( rwlock, rwlock_attr );
821 strong_alias(__pthread_rwlock_init, pthread_rwlock_init);
823 int __pthread_rwlock_destroy(pthread_rwlock_t *rwlock)
825 assert( funcs.ptr_pthread_rwlock_destroy );
826 return funcs.ptr_pthread_rwlock_destroy( rwlock );
828 strong_alias(__pthread_rwlock_destroy, pthread_rwlock_destroy);
830 int __pthread_rwlock_rdlock(pthread_rwlock_t *rwlock)
832 if (!funcs.ptr_pthread_rwlock_rdlock) return 0;
833 return funcs.ptr_pthread_rwlock_rdlock( rwlock );
835 strong_alias(__pthread_rwlock_rdlock, pthread_rwlock_rdlock);
837 int __pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock)
839 assert( funcs.ptr_pthread_rwlock_tryrdlock );
840 return funcs.ptr_pthread_rwlock_tryrdlock( rwlock );
842 strong_alias(__pthread_rwlock_tryrdlock, pthread_rwlock_tryrdlock);
844 int __pthread_rwlock_wrlock(pthread_rwlock_t *rwlock)
846 assert( funcs.ptr_pthread_rwlock_wrlock );
847 return funcs.ptr_pthread_rwlock_wrlock( rwlock );
849 strong_alias(__pthread_rwlock_wrlock, pthread_rwlock_wrlock);
851 int __pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock)
853 assert( funcs.ptr_pthread_rwlock_trywrlock );
854 return funcs.ptr_pthread_rwlock_trywrlock( rwlock );
856 strong_alias(__pthread_rwlock_trywrlock, pthread_rwlock_trywrlock);
858 int __pthread_rwlock_unlock(pthread_rwlock_t *rwlock)
860 if (!funcs.ptr_pthread_rwlock_unlock) return 0;
861 return funcs.ptr_pthread_rwlock_unlock( rwlock );
863 strong_alias(__pthread_rwlock_unlock, pthread_rwlock_unlock);
865 /**** READ-WRITE LOCK ATTRIBUTES *****/
866 /* not implemented right now */
868 int pthread_rwlockattr_init(pthread_rwlockattr_t *attr)
870 return 0;
873 int __pthread_rwlockattr_destroy(pthread_rwlockattr_t *attr)
875 return 0;
877 strong_alias(__pthread_rwlockattr_destroy, pthread_rwlockattr_destroy);
879 int pthread_rwlockattr_getkind_np(const pthread_rwlockattr_t *attr, int *pref)
881 *pref = 0;
882 return 0;
885 int pthread_rwlockattr_setkind_np(pthread_rwlockattr_t *attr, int pref)
887 return 0;
890 /***** MISC *****/
892 pthread_t pthread_self(void)
894 assert( funcs.ptr_pthread_self );
895 return funcs.ptr_pthread_self();
898 int pthread_equal(pthread_t thread1, pthread_t thread2)
900 assert( funcs.ptr_pthread_equal );
901 return funcs.ptr_pthread_equal( thread1, thread2 );
904 void __pthread_do_exit(void *retval, char *currentframe)
906 assert( funcs.ptr_pthread_exit );
907 return funcs.ptr_pthread_exit( retval, currentframe );
910 void __pthread_exit(void *retval)
912 __pthread_do_exit( retval, NULL );
914 strong_alias(__pthread_exit, pthread_exit);
916 int pthread_setcancelstate(int state, int *oldstate)
918 pthread_descr descr = __pthread_thread_self();
919 if (oldstate) *oldstate = descr->cancel_state;
920 descr->cancel_state = state;
921 return 0;
924 int pthread_setcanceltype(int type, int *oldtype)
926 pthread_descr descr = __pthread_thread_self();
927 if (oldtype) *oldtype = descr->cancel_type;
928 descr->cancel_type = type;
929 return 0;
932 /***** ANTI-OVERRIDES *****/
933 /* pthreads tries to override these, point them back to libc */
935 int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact)
937 if (!libc_sigaction)
939 libc_sigaction = wine_dlsym( RTLD_NEXT, "sigaction", NULL, 0 );
940 assert( libc_sigaction );
942 return libc_sigaction(signum, act, oldact);
945 void __pthread_initialize(void)
947 static int done;
949 if (!done)
951 done = 1;
952 libc_fork = wine_dlsym( RTLD_NEXT, "fork", NULL, 0 );
953 libc_sigaction = wine_dlsym( RTLD_NEXT, "sigaction", NULL, 0 );
954 libc_uselocale = wine_dlsym( RTLD_DEFAULT, "uselocale", NULL, 0 );
955 libc_pthread_init = wine_dlsym( RTLD_NEXT, "__libc_pthread_init", NULL, 0 );
956 if (libc_pthread_init) libc_multiple_threads = libc_pthread_init( &libc_pthread_functions );
959 DECL_GLOBAL_CONSTRUCTOR(init) { __pthread_initialize(); }
961 static struct pthread_functions libc_pthread_functions =
963 NULL, /* ptr_pthread_fork */
964 NULL, /* FIXME */ /* ptr_pthread_attr_destroy */
965 NULL, /* FIXME */ /* ptr___pthread_attr_init_2_0 */
966 NULL, /* FIXME */ /* ptr___pthread_attr_init_2_1 */
967 NULL, /* FIXME */ /* ptr_pthread_attr_getdetachstate */
968 NULL, /* FIXME */ /* ptr_pthread_attr_setdetachstate */
969 NULL, /* FIXME */ /* ptr_pthread_attr_getinheritsched */
970 NULL, /* FIXME */ /* ptr_pthread_attr_setinheritsched */
971 NULL, /* FIXME */ /* ptr_pthread_attr_getschedparam */
972 pthread_attr_setschedparam, /* ptr_pthread_attr_setschedparam */
973 NULL, /* FIXME */ /* ptr_pthread_attr_getschedpolicy */
974 NULL, /* FIXME */ /* ptr_pthread_attr_setschedpolicy */
975 NULL, /* FIXME */ /* ptr_pthread_attr_getscope */
976 NULL, /* FIXME */ /* ptr_pthread_attr_setscope */
977 pthread_condattr_destroy, /* ptr_pthread_condattr_destroy */
978 pthread_condattr_init, /* ptr_pthread_condattr_init */
979 __pthread_cond_broadcast, /* ptr___pthread_cond_broadcast */
980 __pthread_cond_destroy, /* ptr___pthread_cond_destroy */
981 __pthread_cond_init, /* ptr___pthread_cond_init */
982 __pthread_cond_signal, /* ptr___pthread_cond_signal */
983 __pthread_cond_wait, /* ptr___pthread_cond_wait */
984 pthread_equal, /* ptr_pthread_equal */
985 __pthread_exit, /* ptr___pthread_exit */
986 NULL, /* FIXME */ /* ptr_pthread_getschedparam */
987 NULL, /* FIXME */ /* ptr_pthread_setschedparam */
988 __pthread_mutex_destroy, /* ptr_pthread_mutex_destroy */
989 __pthread_mutex_init, /* ptr_pthread_mutex_init */
990 __pthread_mutex_lock, /* ptr_pthread_mutex_lock */
991 __pthread_mutex_trylock, /* ptr_pthread_mutex_trylock */
992 __pthread_mutex_unlock, /* ptr_pthread_mutex_unlock */
993 pthread_self, /* ptr_pthread_self */
994 pthread_setcancelstate, /* ptr_pthread_setcancelstate */
995 pthread_setcanceltype, /* ptr_pthread_setcanceltype */
996 __pthread_do_exit, /* ptr_pthread_do_exit */
997 __pthread_cleanup_upto, /* ptr_pthread_cleanup_upto */
998 __pthread_thread_self, /* ptr_pthread_thread_self */
999 pthread_internal_tsd_set, /* ptr_pthread_internal_tsd_set */
1000 pthread_internal_tsd_get, /* ptr_pthread_internal_tsd_get */
1001 pthread_internal_tsd_address, /* ptr_pthread_internal_tsd_address */
1002 NULL, /* ptr_pthread_sigaction */
1003 NULL, /* ptr_pthread_sigwait */
1004 NULL, /* ptr_pthread_raise */
1005 __pthread_cond_timedwait, /* ptr___pthread_cond_timedwait */
1006 _pthread_cleanup_push, /* ptr__pthread_cleanup_push */
1007 _pthread_cleanup_pop /* ptr__pthread_cleanup_pop */
1010 #endif /* __GLIBC__ || __FREEBSD__ */