mutex.h 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. /*
  2. * Copyright (c) 2019 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_SYS_MUTEX_H_
  7. #define ZEPHYR_INCLUDE_SYS_MUTEX_H_
  8. /*
  9. * sys_mutex behaves almost exactly like k_mutex, with the added advantage
  10. * that a sys_mutex instance can reside in user memory.
  11. *
  12. * Further enhancements will support locking/unlocking uncontended sys_mutexes
  13. * with simple atomic ops instead of syscalls, similar to Linux's
  14. * FUTEX_LOCK_PI and FUTEX_UNLOCK_PI
  15. */
  16. #ifdef __cplusplus
  17. extern "C" {
  18. #endif
  19. #ifdef CONFIG_USERSPACE
  20. #include <sys/atomic.h>
  21. #include <zephyr/types.h>
  22. #include <sys_clock.h>
  23. struct sys_mutex {
  24. /* Currently unused, but will be used to store state for fast mutexes
  25. * that can be locked/unlocked with atomic ops if there is no
  26. * contention
  27. */
  28. atomic_t val;
  29. };
  30. /**
  31. * @defgroup user_mutex_apis User mode mutex APIs
  32. * @ingroup kernel_apis
  33. * @{
  34. */
  35. /**
  36. * @brief Statically define and initialize a sys_mutex
  37. *
  38. * The mutex can be accessed outside the module where it is defined using:
  39. *
  40. * @code extern struct sys_mutex <name>; @endcode
  41. *
  42. * Route this to memory domains using K_APP_DMEM().
  43. *
  44. * @param name Name of the mutex.
  45. */
  46. #define SYS_MUTEX_DEFINE(name) \
  47. struct sys_mutex name
  48. /**
  49. * @brief Initialize a mutex.
  50. *
  51. * This routine initializes a mutex object, prior to its first use.
  52. *
  53. * Upon completion, the mutex is available and does not have an owner.
  54. *
  55. * This routine is only necessary to call when userspace is disabled
  56. * and the mutex was not created with SYS_MUTEX_DEFINE().
  57. *
  58. * @param mutex Address of the mutex.
  59. *
  60. * @return N/A
  61. */
  62. static inline void sys_mutex_init(struct sys_mutex *mutex)
  63. {
  64. ARG_UNUSED(mutex);
  65. /* Nothing to do, kernel-side data structures are initialized at
  66. * boot
  67. */
  68. }
  69. __syscall int z_sys_mutex_kernel_lock(struct sys_mutex *mutex,
  70. k_timeout_t timeout);
  71. __syscall int z_sys_mutex_kernel_unlock(struct sys_mutex *mutex);
  72. /**
  73. * @brief Lock a mutex.
  74. *
  75. * This routine locks @a mutex. If the mutex is locked by another thread,
  76. * the calling thread waits until the mutex becomes available or until
  77. * a timeout occurs.
  78. *
  79. * A thread is permitted to lock a mutex it has already locked. The operation
  80. * completes immediately and the lock count is increased by 1.
  81. *
  82. * @param mutex Address of the mutex, which may reside in user memory
  83. * @param timeout Waiting period to lock the mutex,
  84. * or one of the special values K_NO_WAIT and K_FOREVER.
  85. *
  86. * @retval 0 Mutex locked.
  87. * @retval -EBUSY Returned without waiting.
  88. * @retval -EAGAIN Waiting period timed out.
  89. * @retval -EACCES Caller has no access to provided mutex address
  90. * @retval -EINVAL Provided mutex not recognized by the kernel
  91. */
  92. static inline int sys_mutex_lock(struct sys_mutex *mutex, k_timeout_t timeout)
  93. {
  94. /* For now, make the syscall unconditionally */
  95. return z_sys_mutex_kernel_lock(mutex, timeout);
  96. }
  97. /**
  98. * @brief Unlock a mutex.
  99. *
  100. * This routine unlocks @a mutex. The mutex must already be locked by the
  101. * calling thread.
  102. *
  103. * The mutex cannot be claimed by another thread until it has been unlocked by
  104. * the calling thread as many times as it was previously locked by that
  105. * thread.
  106. *
  107. * @param mutex Address of the mutex, which may reside in user memory
  108. * @retval 0 Mutex unlocked
  109. * @retval -EACCES Caller has no access to provided mutex address
  110. * @retval -EINVAL Provided mutex not recognized by the kernel or mutex wasn't
  111. * locked
  112. * @retval -EPERM Caller does not own the mutex
  113. */
  114. static inline int sys_mutex_unlock(struct sys_mutex *mutex)
  115. {
  116. /* For now, make the syscall unconditionally */
  117. return z_sys_mutex_kernel_unlock(mutex);
  118. }
  119. #include <syscalls/mutex.h>
  120. #else
  121. #include <kernel.h>
  122. #include <kernel_structs.h>
  123. struct sys_mutex {
  124. struct k_mutex kernel_mutex;
  125. };
  126. #define SYS_MUTEX_DEFINE(name) \
  127. struct sys_mutex name = { \
  128. .kernel_mutex = Z_MUTEX_INITIALIZER(name.kernel_mutex) \
  129. }
  130. static inline void sys_mutex_init(struct sys_mutex *mutex)
  131. {
  132. k_mutex_init(&mutex->kernel_mutex);
  133. }
  134. static inline int sys_mutex_lock(struct sys_mutex *mutex, k_timeout_t timeout)
  135. {
  136. return k_mutex_lock(&mutex->kernel_mutex, timeout);
  137. }
  138. static inline int sys_mutex_unlock(struct sys_mutex *mutex)
  139. {
  140. return k_mutex_unlock(&mutex->kernel_mutex);
  141. }
  142. #endif /* CONFIG_USERSPACE */
  143. /**
  144. * @}
  145. */
  146. #ifdef __cplusplus
  147. }
  148. #endif
  149. #endif /* ZEPHYR_INCLUDE_SYS_MUTEX_H_ */