pthread_mutex.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. /*
  2. * Copyright (c) 2017 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <kernel.h>
  7. #include <ksched.h>
  8. #include <wait_q.h>
  9. #include <posix/pthread.h>
  10. struct k_spinlock z_pthread_spinlock;
  11. int64_t timespec_to_timeoutms(const struct timespec *abstime);
  12. #define MUTEX_MAX_REC_LOCK 32767
  13. /*
  14. * Default mutex attrs.
  15. */
  16. static const pthread_mutexattr_t def_attr = {
  17. .type = PTHREAD_MUTEX_DEFAULT,
  18. };
  19. static int acquire_mutex(pthread_mutex_t *m, k_timeout_t timeout)
  20. {
  21. int rc = 0;
  22. k_spinlock_key_t key = k_spin_lock(&z_pthread_spinlock);
  23. if (m->lock_count == 0U && m->owner == NULL) {
  24. m->lock_count++;
  25. m->owner = pthread_self();
  26. k_spin_unlock(&z_pthread_spinlock, key);
  27. return 0;
  28. } else if (m->owner == pthread_self()) {
  29. if (m->type == PTHREAD_MUTEX_RECURSIVE &&
  30. m->lock_count < MUTEX_MAX_REC_LOCK) {
  31. m->lock_count++;
  32. rc = 0;
  33. } else if (m->type == PTHREAD_MUTEX_ERRORCHECK) {
  34. rc = EDEADLK;
  35. } else {
  36. rc = EINVAL;
  37. }
  38. k_spin_unlock(&z_pthread_spinlock, key);
  39. return rc;
  40. }
  41. if (K_TIMEOUT_EQ(timeout, K_NO_WAIT)) {
  42. k_spin_unlock(&z_pthread_spinlock, key);
  43. return EINVAL;
  44. }
  45. rc = z_pend_curr(&z_pthread_spinlock, key, &m->wait_q, timeout);
  46. if (rc != 0) {
  47. rc = ETIMEDOUT;
  48. }
  49. return rc;
  50. }
  51. /**
  52. * @brief Lock POSIX mutex with non-blocking call.
  53. *
  54. * See IEEE 1003.1
  55. */
  56. int pthread_mutex_trylock(pthread_mutex_t *m)
  57. {
  58. return acquire_mutex(m, K_NO_WAIT);
  59. }
  60. /**
  61. * @brief Lock POSIX mutex with timeout.
  62. *
  63. *
  64. * See IEEE 1003.1
  65. */
  66. int pthread_mutex_timedlock(pthread_mutex_t *m,
  67. const struct timespec *abstime)
  68. {
  69. int32_t timeout = (int32_t)timespec_to_timeoutms(abstime);
  70. return acquire_mutex(m, K_MSEC(timeout));
  71. }
  72. /**
  73. * @brief Intialize POSIX mutex.
  74. *
  75. * See IEEE 1003.1
  76. */
  77. int pthread_mutex_init(pthread_mutex_t *m,
  78. const pthread_mutexattr_t *attr)
  79. {
  80. const pthread_mutexattr_t *mattr;
  81. m->owner = NULL;
  82. m->lock_count = 0U;
  83. mattr = (attr == NULL) ? &def_attr : attr;
  84. m->type = mattr->type;
  85. z_waitq_init(&m->wait_q);
  86. return 0;
  87. }
  88. /**
  89. * @brief Lock POSIX mutex with blocking call.
  90. *
  91. * See IEEE 1003.1
  92. */
  93. int pthread_mutex_lock(pthread_mutex_t *m)
  94. {
  95. return acquire_mutex(m, K_FOREVER);
  96. }
  97. /**
  98. * @brief Unlock POSIX mutex.
  99. *
  100. * See IEEE 1003.1
  101. */
  102. int pthread_mutex_unlock(pthread_mutex_t *m)
  103. {
  104. k_spinlock_key_t key = k_spin_lock(&z_pthread_spinlock);
  105. k_tid_t thread;
  106. if (m->owner != pthread_self()) {
  107. k_spin_unlock(&z_pthread_spinlock, key);
  108. return EPERM;
  109. }
  110. if (m->lock_count == 0U) {
  111. k_spin_unlock(&z_pthread_spinlock, key);
  112. return EINVAL;
  113. }
  114. m->lock_count--;
  115. if (m->lock_count == 0U) {
  116. thread = z_unpend_first_thread(&m->wait_q);
  117. if (thread) {
  118. m->owner = (pthread_t)thread;
  119. m->lock_count++;
  120. arch_thread_return_value_set(thread, 0);
  121. z_ready_thread(thread);
  122. z_reschedule(&z_pthread_spinlock, key);
  123. return 0;
  124. }
  125. m->owner = NULL;
  126. }
  127. k_spin_unlock(&z_pthread_spinlock, key);
  128. return 0;
  129. }
  130. /**
  131. * @brief Destroy POSIX mutex.
  132. *
  133. * See IEEE 1003.1
  134. */
  135. int pthread_mutex_destroy(pthread_mutex_t *m)
  136. {
  137. ARG_UNUSED(m);
  138. return 0;
  139. }
  140. /**
  141. * @brief Read protocol attribute for mutex.
  142. *
  143. * See IEEE 1003.1
  144. */
  145. int pthread_mutexattr_getprotocol(const pthread_mutexattr_t *attr,
  146. int *protocol)
  147. {
  148. *protocol = PTHREAD_PRIO_NONE;
  149. return 0;
  150. }
  151. /**
  152. * @brief Read type attribute for mutex.
  153. *
  154. * See IEEE 1003.1
  155. */
  156. int pthread_mutexattr_gettype(const pthread_mutexattr_t *attr, int *type)
  157. {
  158. *type = attr->type;
  159. return 0;
  160. }
  161. /**
  162. * @brief Set type attribute for mutex.
  163. *
  164. * See IEEE 1003.1
  165. */
  166. int pthread_mutexattr_settype(pthread_mutexattr_t *attr, int type)
  167. {
  168. int retc = EINVAL;
  169. if ((type == PTHREAD_MUTEX_NORMAL) ||
  170. (type == PTHREAD_MUTEX_RECURSIVE) ||
  171. (type == PTHREAD_MUTEX_ERRORCHECK)) {
  172. attr->type = type;
  173. retc = 0;
  174. }
  175. return retc;
  176. }