2 * Copyright (c) 2000-2003 Apple Computer, Inc. All rights reserved.
4 * @APPLE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. Please obtain a copy of the License at
10 * http://www.opensource.apple.com/apsl/ and read it before using this
13 * The Original Code and all software distributed under the License are
14 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
15 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
16 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
18 * Please see the License for the specific language governing rights and
19 * limitations under the License.
21 * @APPLE_LICENSE_HEADER_END@
24 * Copyright 1996 1995 by Open Software Foundation, Inc. 1997 1996 1995 1994 1993 1992 1991
27 * Permission to use, copy, modify, and distribute this software and
28 * its documentation for any purpose and without fee is hereby granted,
29 * provided that the above copyright notice appears in all copies and
30 * that both the copyright notice and this permission notice appear in
31 * supporting documentation.
33 * OSF DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE
34 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
35 * FOR A PARTICULAR PURPOSE.
37 * IN NO EVENT SHALL OSF BE LIABLE FOR ANY SPECIAL, INDIRECT, OR
38 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
39 * LOSS OF USE, DATA OR PROFITS, WHETHER IN ACTION OF CONTRACT,
40 * NEGLIGENCE, OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
41 * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
49 * POSIX Pthread Library
50 * Thread Specific Data support
51 * NB: pthread_getspecific() is in a separate assembly file
54 #include "pthread_internals.h"
58 int created
; /* Set TRUE if 'create_key' used this slot */
59 void (*destructor
)(void *);
60 } _pthread_keys
[_POSIX_THREAD_KEYS_MAX
];
61 static pthread_lock_t tds_lock
= LOCK_INITIALIZER
;
63 * Partition _pthread_keys in a lower part that dyld can use, and an upper
64 * part for libSystem. The libSystem part starts at __pthread_tsd_first = 4.
65 * dyld will set this value to 1.
67 __private_extern__
int __pthread_tsd_first
= 4;
70 * Create a new key for thread specific data
73 pthread_key_create(pthread_key_t
*key
,
74 void (*destructor
)(void *))
78 res
= ENOMEM
; /* No 'free' keys */
79 /* The first slot is reserved for pthread_self() */
80 for (i
= __pthread_tsd_first
; i
< _POSIX_THREAD_KEYS_MAX
; i
++)
82 if (_pthread_keys
[i
].created
== FALSE
)
84 _pthread_keys
[i
].created
= TRUE
;
85 _pthread_keys
[i
].destructor
= destructor
;
96 * Destroy a thread specific data key
99 pthread_key_delete(pthread_key_t key
)
103 /* The first slot is reserved for pthread_self() */
104 if ((key
>= __pthread_tsd_first
) && (key
< _POSIX_THREAD_KEYS_MAX
))
106 if (_pthread_keys
[key
].created
)
110 _pthread_keys
[key
].created
= FALSE
;
111 LOCK(_pthread_list_lock
);
112 LIST_FOREACH(p
, &__pthread_head
, plist
) {
113 /* It is an 32bit value no lock needed */
116 UNLOCK(_pthread_list_lock
);
131 * Set the thread private value for a given key.
132 * We do not take the spinlock for this or pthread_getspecific.
133 * The assignment to self->tsd[] is thread-safe because we never
134 * refer to the tsd[] of a thread other than pthread_self().
135 * The reference to _pthread_keys[...].created could race with a
136 * pthread_key_delete() but in this case the behaviour is allowed
140 pthread_setspecific(pthread_key_t key
,
145 /* The first slot is reserved for pthread_self() */
146 if ((key
>= __pthread_tsd_first
) && (key
< _POSIX_THREAD_KEYS_MAX
))
148 if (_pthread_keys
[key
].created
)
150 self
= pthread_self();
151 self
->tsd
[key
] = (void *) value
;
165 * Clean up thread specific data as thread 'dies'
168 _pthread_tsd_cleanup(pthread_t self
)
172 for (j
= 0; j
< PTHREAD_DESTRUCTOR_ITERATIONS
; j
++)
174 /* The first slot is reserved for pthread_self() */
175 for (i
= __pthread_tsd_first
; i
< _POSIX_THREAD_KEYS_MAX
; i
++)
177 if (_pthread_keys
[i
].created
&& (param
= self
->tsd
[i
]))
179 self
->tsd
[i
] = (void *)NULL
;
180 if (_pthread_keys
[i
].destructor
)
182 (_pthread_keys
[i
].destructor
)(param
);