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
;
64 * Create a new key for thread specific data
67 pthread_key_create(pthread_key_t
*key
,
68 void (*destructor
)(void *))
72 res
= ENOMEM
; /* No 'free' keys */
73 /* The first slot is reserved for pthread_self() */
74 for (i
= 1; i
< _POSIX_THREAD_KEYS_MAX
; i
++)
76 if (_pthread_keys
[i
].created
== FALSE
)
78 _pthread_keys
[i
].created
= TRUE
;
79 _pthread_keys
[i
].destructor
= destructor
;
90 * Destroy a thread specific data key
93 pthread_key_delete(pthread_key_t key
)
97 /* The first slot is reserved for pthread_self() */
98 if ((key
> 0) && (key
< _POSIX_THREAD_KEYS_MAX
))
100 if (_pthread_keys
[key
].created
)
104 _pthread_keys
[key
].created
= FALSE
;
105 LOCK(_pthread_list_lock
);
106 LIST_FOREACH(p
, &__pthread_head
, plist
) {
107 /* It is an 32bit value no lock needed */
110 UNLOCK(_pthread_list_lock
);
125 * Set the thread private value for a given key.
126 * We do not take the spinlock for this or pthread_getspecific.
127 * The assignment to self->tsd[] is thread-safe because we never
128 * refer to the tsd[] of a thread other than pthread_self().
129 * The reference to _pthread_keys[...].created could race with a
130 * pthread_key_delete() but in this case the behaviour is allowed
134 pthread_setspecific(pthread_key_t key
,
139 /* The first slot is reserved for pthread_self() */
140 if ((key
> 0) && (key
< _POSIX_THREAD_KEYS_MAX
))
142 if (_pthread_keys
[key
].created
)
144 self
= pthread_self();
145 self
->tsd
[key
] = (void *) value
;
159 * Clean up thread specific data as thread 'dies'
162 _pthread_tsd_cleanup(pthread_t self
)
166 for (j
= 0; j
< PTHREAD_DESTRUCTOR_ITERATIONS
; j
++)
168 /* The first slot is reserved for pthread_self() */
169 for (i
= 1; i
< _POSIX_THREAD_KEYS_MAX
; i
++)
171 if (_pthread_keys
[i
].created
&& (param
= self
->tsd
[i
]))
173 self
->tsd
[i
] = (void *)NULL
;
174 if (_pthread_keys
[i
].destructor
)
176 (_pthread_keys
[i
].destructor
)(param
);