pthread_rwlock.c 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261
  1. /*
  2. * Copyright (c) 2018 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <kernel.h>
  7. #include <errno.h>
  8. #include <posix/time.h>
  9. #include <posix/posix_types.h>
  10. #define INITIALIZED 1
  11. #define NOT_INITIALIZED 0
  12. #define CONCURRENT_READER_LIMIT (CONFIG_MAX_PTHREAD_COUNT + 1)
  13. int64_t timespec_to_timeoutms(const struct timespec *abstime);
  14. static uint32_t read_lock_acquire(pthread_rwlock_t *rwlock, int32_t timeout);
  15. static uint32_t write_lock_acquire(pthread_rwlock_t *rwlock, int32_t timeout);
  16. /**
  17. * @brief Initialize read-write lock object.
  18. *
  19. * See IEEE 1003.1
  20. */
  21. int pthread_rwlock_init(pthread_rwlock_t *rwlock,
  22. const pthread_rwlockattr_t *attr)
  23. {
  24. k_sem_init(&rwlock->rd_sem, CONCURRENT_READER_LIMIT,
  25. CONCURRENT_READER_LIMIT);
  26. k_sem_init(&rwlock->wr_sem, 1, 1);
  27. k_sem_init(&rwlock->reader_active, 1, 1);
  28. rwlock->wr_owner = NULL;
  29. rwlock->status = INITIALIZED;
  30. return 0;
  31. }
  32. /**
  33. * @brief Destroy read-write lock object.
  34. *
  35. * See IEEE 1003.1
  36. */
  37. int pthread_rwlock_destroy(pthread_rwlock_t *rwlock)
  38. {
  39. if (rwlock->status == NOT_INITIALIZED) {
  40. return EINVAL;
  41. }
  42. if (rwlock->wr_owner != NULL) {
  43. return EBUSY;
  44. }
  45. if (rwlock->status == INITIALIZED) {
  46. rwlock->status = NOT_INITIALIZED;
  47. return 0;
  48. }
  49. return EINVAL;
  50. }
  51. /**
  52. * @brief Lock a read-write lock object for reading.
  53. *
  54. * API behaviour is unpredictable if number of concurrent reader
  55. * lock held is greater than CONCURRENT_READER_LIMIT.
  56. *
  57. * See IEEE 1003.1
  58. */
  59. int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock)
  60. {
  61. if (rwlock->status == NOT_INITIALIZED) {
  62. return EINVAL;
  63. }
  64. return read_lock_acquire(rwlock, SYS_FOREVER_MS);
  65. }
  66. /**
  67. * @brief Lock a read-write lock object for reading within specific time.
  68. *
  69. * API behaviour is unpredictable if number of concurrent reader
  70. * lock held is greater than CONCURRENT_READER_LIMIT.
  71. *
  72. * See IEEE 1003.1
  73. */
  74. int pthread_rwlock_timedrdlock(pthread_rwlock_t *rwlock,
  75. const struct timespec *abstime)
  76. {
  77. int32_t timeout;
  78. uint32_t ret = 0U;
  79. if (rwlock->status == NOT_INITIALIZED || abstime->tv_nsec < 0 ||
  80. abstime->tv_nsec > NSEC_PER_SEC) {
  81. return EINVAL;
  82. }
  83. timeout = (int32_t) timespec_to_timeoutms(abstime);
  84. if (read_lock_acquire(rwlock, timeout) != 0U) {
  85. ret = ETIMEDOUT;
  86. }
  87. return ret;
  88. }
  89. /**
  90. * @brief Lock a read-write lock object for reading immedately.
  91. *
  92. * API behaviour is unpredictable if number of concurrent reader
  93. * lock held is greater than CONCURRENT_READER_LIMIT.
  94. *
  95. * See IEEE 1003.1
  96. */
  97. int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock)
  98. {
  99. if (rwlock->status == NOT_INITIALIZED) {
  100. return EINVAL;
  101. }
  102. return read_lock_acquire(rwlock, 0);
  103. }
  104. /**
  105. * @brief Lock a read-write lock object for writing.
  106. *
  107. * Write lock does not have priority over reader lock,
  108. * threads get lock based on priority.
  109. *
  110. * See IEEE 1003.1
  111. */
  112. int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock)
  113. {
  114. if (rwlock->status == NOT_INITIALIZED) {
  115. return EINVAL;
  116. }
  117. return write_lock_acquire(rwlock, SYS_FOREVER_MS);
  118. }
  119. /**
  120. * @brief Lock a read-write lock object for writing within specific time.
  121. *
  122. * Write lock does not have priority over reader lock,
  123. * threads get lock based on priority.
  124. *
  125. * See IEEE 1003.1
  126. */
  127. int pthread_rwlock_timedwrlock(pthread_rwlock_t *rwlock,
  128. const struct timespec *abstime)
  129. {
  130. int32_t timeout;
  131. uint32_t ret = 0U;
  132. if (rwlock->status == NOT_INITIALIZED || abstime->tv_nsec < 0 ||
  133. abstime->tv_nsec > NSEC_PER_SEC) {
  134. return EINVAL;
  135. }
  136. timeout = (int32_t) timespec_to_timeoutms(abstime);
  137. if (write_lock_acquire(rwlock, timeout) != 0U) {
  138. ret = ETIMEDOUT;
  139. }
  140. return ret;
  141. }
  142. /**
  143. * @brief Lock a read-write lock object for writing immedately.
  144. *
  145. * Write lock does not have priority over reader lock,
  146. * threads get lock based on priority.
  147. *
  148. * See IEEE 1003.1
  149. */
  150. int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock)
  151. {
  152. if (rwlock->status == NOT_INITIALIZED) {
  153. return EINVAL;
  154. }
  155. return write_lock_acquire(rwlock, 0);
  156. }
  157. /**
  158. *
  159. * @brief Unlock a read-write lock object.
  160. *
  161. * See IEEE 1003.1
  162. */
  163. int pthread_rwlock_unlock(pthread_rwlock_t *rwlock)
  164. {
  165. if (rwlock->status == NOT_INITIALIZED) {
  166. return EINVAL;
  167. }
  168. if (k_current_get() == rwlock->wr_owner) {
  169. /* Write unlock */
  170. rwlock->wr_owner = NULL;
  171. k_sem_give(&rwlock->reader_active);
  172. k_sem_give(&rwlock->wr_sem);
  173. } else {
  174. /* Read unlock */
  175. if (k_sem_count_get(&rwlock->rd_sem) ==
  176. (CONCURRENT_READER_LIMIT - 1)) {
  177. /* Last read lock, unlock writer */
  178. k_sem_give(&rwlock->reader_active);
  179. }
  180. k_sem_give(&rwlock->rd_sem);
  181. }
  182. return 0;
  183. }
  184. static uint32_t read_lock_acquire(pthread_rwlock_t *rwlock, int32_t timeout)
  185. {
  186. uint32_t ret = 0U;
  187. if (k_sem_take(&rwlock->wr_sem, SYS_TIMEOUT_MS(timeout)) == 0) {
  188. k_sem_take(&rwlock->reader_active, K_NO_WAIT);
  189. k_sem_take(&rwlock->rd_sem, K_NO_WAIT);
  190. k_sem_give(&rwlock->wr_sem);
  191. } else {
  192. ret = EBUSY;
  193. }
  194. return ret;
  195. }
  196. static uint32_t write_lock_acquire(pthread_rwlock_t *rwlock, int32_t timeout)
  197. {
  198. uint32_t ret = 0U;
  199. int64_t elapsed_time, st_time = k_uptime_get();
  200. k_timeout_t k_timeout;
  201. k_timeout = SYS_TIMEOUT_MS(timeout);
  202. /* waiting for release of write lock */
  203. if (k_sem_take(&rwlock->wr_sem, k_timeout) == 0) {
  204. /* update remaining timeout time for 2nd sem */
  205. if (timeout != SYS_FOREVER_MS) {
  206. elapsed_time = k_uptime_get() - st_time;
  207. timeout = timeout <= elapsed_time ? 0 :
  208. timeout - elapsed_time;
  209. }
  210. k_timeout = SYS_TIMEOUT_MS(timeout);
  211. /* waiting for reader to complete operation */
  212. if (k_sem_take(&rwlock->reader_active, k_timeout) == 0) {
  213. rwlock->wr_owner = k_current_get();
  214. } else {
  215. k_sem_give(&rwlock->wr_sem);
  216. ret = EBUSY;
  217. }
  218. } else {
  219. ret = EBUSY;
  220. }
  221. return ret;
  222. }