xref: /nrf52832-nimble/rt-thread/components/libc/pthreads/pthread_tls.c (revision 104654410c56c573564690304ae786df310c91fc)
1*10465441SEvalZero /*
2*10465441SEvalZero  * Copyright (c) 2006-2018, RT-Thread Development Team
3*10465441SEvalZero  *
4*10465441SEvalZero  * SPDX-License-Identifier: Apache-2.0
5*10465441SEvalZero  *
6*10465441SEvalZero  * Change Logs:
7*10465441SEvalZero  * Date           Author       Notes
8*10465441SEvalZero  * 2010-10-26     Bernard      the first version
9*10465441SEvalZero  */
10*10465441SEvalZero 
11*10465441SEvalZero #include <pthread.h>
12*10465441SEvalZero #include "pthread_internal.h"
13*10465441SEvalZero 
14*10465441SEvalZero _pthread_key_data_t _thread_keys[PTHREAD_KEY_MAX];
15*10465441SEvalZero 
pthread_key_system_init()16*10465441SEvalZero void pthread_key_system_init()
17*10465441SEvalZero {
18*10465441SEvalZero     rt_memset(&_thread_keys[0], 0, sizeof(_thread_keys));
19*10465441SEvalZero }
20*10465441SEvalZero 
pthread_getspecific(pthread_key_t key)21*10465441SEvalZero void *pthread_getspecific(pthread_key_t key)
22*10465441SEvalZero {
23*10465441SEvalZero     struct _pthread_data* ptd;
24*10465441SEvalZero 
25*10465441SEvalZero     ptd = _pthread_get_data(rt_thread_self());
26*10465441SEvalZero     RT_ASSERT(ptd != NULL);
27*10465441SEvalZero 
28*10465441SEvalZero     if (ptd->tls == NULL)
29*10465441SEvalZero         return NULL;
30*10465441SEvalZero 
31*10465441SEvalZero     if ((key < PTHREAD_KEY_MAX) && (_thread_keys[key].is_used))
32*10465441SEvalZero         return ptd->tls[key];
33*10465441SEvalZero 
34*10465441SEvalZero     return NULL;
35*10465441SEvalZero }
36*10465441SEvalZero RTM_EXPORT(pthread_getspecific);
37*10465441SEvalZero 
pthread_setspecific(pthread_key_t key,const void * value)38*10465441SEvalZero int pthread_setspecific(pthread_key_t key, const void *value)
39*10465441SEvalZero {
40*10465441SEvalZero     struct _pthread_data* ptd;
41*10465441SEvalZero 
42*10465441SEvalZero     ptd = _pthread_get_data(rt_thread_self());
43*10465441SEvalZero     RT_ASSERT(ptd != NULL);
44*10465441SEvalZero 
45*10465441SEvalZero     /* check tls area */
46*10465441SEvalZero     if (ptd->tls == NULL)
47*10465441SEvalZero     {
48*10465441SEvalZero         ptd->tls = (void**)rt_malloc(sizeof(void*) * PTHREAD_KEY_MAX);
49*10465441SEvalZero     }
50*10465441SEvalZero 
51*10465441SEvalZero     if ((key < PTHREAD_KEY_MAX) && _thread_keys[key].is_used)
52*10465441SEvalZero     {
53*10465441SEvalZero         ptd->tls[key] = (void *)value;
54*10465441SEvalZero 
55*10465441SEvalZero         return 0;
56*10465441SEvalZero     }
57*10465441SEvalZero 
58*10465441SEvalZero     return EINVAL;
59*10465441SEvalZero }
60*10465441SEvalZero RTM_EXPORT(pthread_setspecific);
61*10465441SEvalZero 
pthread_key_create(pthread_key_t * key,void (* destructor)(void *))62*10465441SEvalZero int pthread_key_create(pthread_key_t *key, void (*destructor)(void*))
63*10465441SEvalZero {
64*10465441SEvalZero     rt_uint32_t index;
65*10465441SEvalZero 
66*10465441SEvalZero     rt_enter_critical();
67*10465441SEvalZero     for (index = 0; index < PTHREAD_KEY_MAX; index ++)
68*10465441SEvalZero     {
69*10465441SEvalZero         if (_thread_keys[index].is_used == 0)
70*10465441SEvalZero         {
71*10465441SEvalZero             _thread_keys[index].is_used = 1;
72*10465441SEvalZero             _thread_keys[index].destructor = destructor;
73*10465441SEvalZero 
74*10465441SEvalZero             *key = index;
75*10465441SEvalZero 
76*10465441SEvalZero             rt_exit_critical();
77*10465441SEvalZero 
78*10465441SEvalZero             return 0;
79*10465441SEvalZero         }
80*10465441SEvalZero     }
81*10465441SEvalZero 
82*10465441SEvalZero     rt_exit_critical();
83*10465441SEvalZero 
84*10465441SEvalZero     return EAGAIN;
85*10465441SEvalZero }
86*10465441SEvalZero RTM_EXPORT(pthread_key_create);
87*10465441SEvalZero 
pthread_key_delete(pthread_key_t key)88*10465441SEvalZero int pthread_key_delete(pthread_key_t key)
89*10465441SEvalZero {
90*10465441SEvalZero     if (key >= PTHREAD_KEY_MAX)
91*10465441SEvalZero         return EINVAL;
92*10465441SEvalZero 
93*10465441SEvalZero     rt_enter_critical();
94*10465441SEvalZero     _thread_keys[key].is_used = 0;
95*10465441SEvalZero     _thread_keys[key].destructor = 0;
96*10465441SEvalZero     rt_exit_critical();
97*10465441SEvalZero 
98*10465441SEvalZero     return 0;
99*10465441SEvalZero }
100*10465441SEvalZero RTM_EXPORT(pthread_key_delete);
101*10465441SEvalZero 
102