pthread_key.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. /*
  2. * Copyright (c) 2018 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <kernel.h>
  7. #include <posix/pthread.h>
  8. #include <posix/pthread_key.h>
  9. struct k_sem pthread_key_sem;
  10. K_SEM_DEFINE(pthread_key_sem, 1, 1);
  11. /**
  12. * @brief Create a key for thread-specific data
  13. *
  14. * See IEEE 1003.1
  15. */
  16. int pthread_key_create(pthread_key_t *key,
  17. void (*destructor)(void *))
  18. {
  19. pthread_key_obj *new_key;
  20. *key = NULL;
  21. new_key = k_malloc(sizeof(pthread_key_obj));
  22. if (new_key == NULL) {
  23. return ENOMEM;
  24. }
  25. sys_slist_init(&(new_key->key_data_l));
  26. new_key->destructor = destructor;
  27. *key = (void *)new_key;
  28. return 0;
  29. }
  30. /**
  31. * @brief Delete a key for thread-specific data
  32. *
  33. * See IEEE 1003.1
  34. */
  35. int pthread_key_delete(pthread_key_t key)
  36. {
  37. pthread_key_obj *key_obj = (pthread_key_obj *)key;
  38. pthread_key_data *key_data;
  39. sys_snode_t *node_l, *next_node_l;
  40. k_sem_take(&pthread_key_sem, K_FOREVER);
  41. /* Delete thread-specific elements associated with the key */
  42. SYS_SLIST_FOR_EACH_NODE_SAFE(&(key_obj->key_data_l),
  43. node_l, next_node_l) {
  44. /* Remove the object from the list key_data_l */
  45. key_data = (pthread_key_data *)
  46. sys_slist_get(&(key_obj->key_data_l));
  47. /* Deallocate the object's memory */
  48. k_free((void *)key_data);
  49. }
  50. k_free((void *)key_obj);
  51. k_sem_give(&pthread_key_sem);
  52. return 0;
  53. }
  54. /**
  55. * @brief Associate a thread-specific value with a key
  56. *
  57. * See IEEE 1003.1
  58. */
  59. int pthread_setspecific(pthread_key_t key, const void *value)
  60. {
  61. pthread_key_obj *key_obj = (pthread_key_obj *)key;
  62. struct posix_thread *thread = (struct posix_thread *)pthread_self();
  63. pthread_key_data *key_data;
  64. pthread_thread_data *thread_spec_data;
  65. sys_snode_t *node_l;
  66. int retval = 0;
  67. /* Traverse the list of keys set by the thread, looking for key.
  68. * If the key is already in the list, re-assign its value.
  69. * Else add the key to the thread's list.
  70. */
  71. k_sem_take(&pthread_key_sem, K_FOREVER);
  72. SYS_SLIST_FOR_EACH_NODE(&(thread->key_list), node_l) {
  73. thread_spec_data = (pthread_thread_data *)node_l;
  74. if (thread_spec_data->key == key_obj) {
  75. /* Key is already present so
  76. * associate thread specific data
  77. */
  78. thread_spec_data->spec_data = (void *)value;
  79. goto out;
  80. }
  81. }
  82. if (node_l == NULL) {
  83. key_data = k_malloc(sizeof(pthread_key_data));
  84. if (key_data == NULL) {
  85. retval = ENOMEM;
  86. goto out;
  87. } else {
  88. /* Associate thread specific data, initialize new key */
  89. key_data->thread_data.key = key;
  90. key_data->thread_data.spec_data = (void *)value;
  91. /* Append new thread key data to thread's key list */
  92. sys_slist_append((&thread->key_list),
  93. (sys_snode_t *)(&key_data->thread_data));
  94. /* Append new key data to the key object's list */
  95. sys_slist_append(&(key_obj->key_data_l),
  96. (sys_snode_t *)key_data);
  97. }
  98. }
  99. out:
  100. k_sem_give(&pthread_key_sem);
  101. return retval;
  102. }
  103. /**
  104. * @brief Get the thread-specific value associated with the key
  105. *
  106. * See IEEE 1003.1
  107. */
  108. void *pthread_getspecific(pthread_key_t key)
  109. {
  110. pthread_key_obj *key_obj = (pthread_key_obj *)key;
  111. struct posix_thread *thread = (struct posix_thread *)pthread_self();
  112. pthread_thread_data *thread_spec_data;
  113. void *value = NULL;
  114. sys_snode_t *node_l;
  115. k_sem_take(&pthread_key_sem, K_FOREVER);
  116. node_l = sys_slist_peek_head(&(thread->key_list));
  117. /* Traverse the list of keys set by the thread, looking for key */
  118. SYS_SLIST_FOR_EACH_NODE(&(thread->key_list), node_l) {
  119. thread_spec_data = (pthread_thread_data *)node_l;
  120. if (thread_spec_data->key == key_obj) {
  121. /* Key is present, so get the set thread data */
  122. value = thread_spec_data->spec_data;
  123. break;
  124. }
  125. }
  126. k_sem_give(&pthread_key_sem);
  127. return value;
  128. }