blob: 98199d3acace8d4f22053f9738e160c2ebb94e8c [file] [log] [blame]
The Android Open Source Project1dc9e472009-03-03 19:28:35 -08001/*
2 * Copyright (C) 2008 The Android Open Source Project
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in
12 * the documentation and/or other materials provided with the
13 * distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
18 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
19 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
21 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
22 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
23 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
25 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * SUCH DAMAGE.
27 */
28#ifndef _PTHREAD_INTERNAL_H_
29#define _PTHREAD_INTERNAL_H_
30
31#include <pthread.h>
Elliott Hughesbfeab1b2012-09-05 17:47:37 -070032#include <stdbool.h>
Elliott Hughes36fa67b2013-06-05 17:51:20 -070033#include <sys/cdefs.h>
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080034
35__BEGIN_DECLS
36
37typedef struct pthread_internal_t
38{
39 struct pthread_internal_t* next;
Elliott Hughes4f251be2012-11-01 16:33:29 -070040 struct pthread_internal_t* prev;
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080041 pthread_attr_t attr;
Elliott Hughes40eabe22013-02-14 18:59:37 -080042 pid_t tid;
Elliott Hughes4f251be2012-11-01 16:33:29 -070043 bool allocated_on_heap;
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080044 pthread_cond_t join_cond;
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080045 void* return_value;
Pierre Peifferd0c884d2012-02-22 16:40:15 +010046 int internal_flags;
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080047 __pthread_cleanup_t* cleanup_stack;
48 void** tls; /* thread-local storage area */
Elliott Hughes5419b942012-10-16 15:54:46 -070049
Elliott Hughes84114c82013-07-17 13:33:19 -070050 void* alternate_signal_stack;
51
Elliott Hughes5419b942012-10-16 15:54:46 -070052 /*
53 * The dynamic linker implements dlerror(3), which makes it hard for us to implement this
54 * per-thread buffer by simply using malloc(3) and free(3).
55 */
56#define __BIONIC_DLERROR_BUFFER_SIZE 512
57 char dlerror_buffer[__BIONIC_DLERROR_BUFFER_SIZE];
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080058} pthread_internal_t;
59
Elliott Hughes40eabe22013-02-14 18:59:37 -080060int _init_thread(pthread_internal_t* thread, bool add_to_thread_list);
61void __init_tls(pthread_internal_t* thread);
Elliott Hughes9d23e042013-02-15 19:21:51 -080062void _pthread_internal_add(pthread_internal_t* thread);
Evgeniy Stepanov1a78fbb2012-03-22 18:01:53 +040063pthread_internal_t* __get_thread(void);
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080064
Elliott Hughes44b53ad2013-02-11 20:18:47 +000065__LIBC_HIDDEN__ void pthread_key_clean_all(void);
Elliott Hughes9d23e042013-02-15 19:21:51 -080066__LIBC_HIDDEN__ void _pthread_internal_remove_locked(pthread_internal_t* thread);
Elliott Hughes44b53ad2013-02-11 20:18:47 +000067
msg5550f020d12013-06-06 14:59:28 -040068/* Has the thread been detached by a pthread_join or pthread_detach call? */
Elliott Hughes3e898472013-02-12 16:40:24 +000069#define PTHREAD_ATTR_FLAG_DETACHED 0x00000001
msg5550f020d12013-06-06 14:59:28 -040070
71/* Was the thread's stack allocated by the user rather than by us? */
Elliott Hughes3e898472013-02-12 16:40:24 +000072#define PTHREAD_ATTR_FLAG_USER_STACK 0x00000002
73
msg5550f020d12013-06-06 14:59:28 -040074/* Has the thread been joined by another thread? */
75#define PTHREAD_ATTR_FLAG_JOINED 0x00000004
76
77/* Has the thread already exited but not been joined? */
78#define PTHREAD_ATTR_FLAG_ZOMBIE 0x00000008
79
Elliott Hughes9d23e042013-02-15 19:21:51 -080080__LIBC_HIDDEN__ extern pthread_internal_t* gThreadList;
81__LIBC_HIDDEN__ extern pthread_mutex_t gThreadListLock;
Elliott Hughes44b53ad2013-02-11 20:18:47 +000082
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080083/* needed by posix-timers.c */
84
85static __inline__ void timespec_add( struct timespec* a, const struct timespec* b )
86{
87 a->tv_sec += b->tv_sec;
88 a->tv_nsec += b->tv_nsec;
89 if (a->tv_nsec >= 1000000000) {
90 a->tv_nsec -= 1000000000;
91 a->tv_sec += 1;
92 }
93}
94
95static __inline__ void timespec_sub( struct timespec* a, const struct timespec* b )
96{
97 a->tv_sec -= b->tv_sec;
98 a->tv_nsec -= b->tv_nsec;
99 if (a->tv_nsec < 0) {
100 a->tv_nsec += 1000000000;
101 a->tv_sec -= 1;
102 }
103}
104
105static __inline__ void timespec_zero( struct timespec* a )
106{
107 a->tv_sec = a->tv_nsec = 0;
108}
109
110static __inline__ int timespec_is_zero( const struct timespec* a )
111{
112 return (a->tv_sec == 0 && a->tv_nsec == 0);
113}
114
115static __inline__ int timespec_cmp( const struct timespec* a, const struct timespec* b )
116{
117 if (a->tv_sec < b->tv_sec) return -1;
118 if (a->tv_sec > b->tv_sec) return +1;
119 if (a->tv_nsec < b->tv_nsec) return -1;
120 if (a->tv_nsec > b->tv_nsec) return +1;
121 return 0;
122}
123
124static __inline__ int timespec_cmp0( const struct timespec* a )
125{
126 if (a->tv_sec < 0) return -1;
127 if (a->tv_sec > 0) return +1;
128 if (a->tv_nsec < 0) return -1;
129 if (a->tv_nsec > 0) return +1;
130 return 0;
131}
132
Elliott Hughesbfeab1b2012-09-05 17:47:37 -0700133extern int __pthread_cond_timedwait(pthread_cond_t*,
The Android Open Source Project1dc9e472009-03-03 19:28:35 -0800134 pthread_mutex_t*,
Elliott Hughesbfeab1b2012-09-05 17:47:37 -0700135 const struct timespec*,
The Android Open Source Project1dc9e472009-03-03 19:28:35 -0800136 clockid_t);
137
138extern int __pthread_cond_timedwait_relative(pthread_cond_t*,
139 pthread_mutex_t*,
140 const struct timespec*);
141
142/* needed by fork.c */
143extern void __timer_table_start_stop(int stop);
Matt Fischer4f086ae2010-06-25 14:36:39 -0500144extern void __bionic_atfork_run_prepare();
145extern void __bionic_atfork_run_child();
146extern void __bionic_atfork_run_parent();
The Android Open Source Project1dc9e472009-03-03 19:28:35 -0800147
148__END_DECLS
149
150#endif /* _PTHREAD_INTERNAL_H_ */