sem.h 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. /*
  2. * Copyright (c) 2019 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. *
  9. * @brief public sys_sem APIs.
  10. */
  11. #ifndef ZEPHYR_INCLUDE_SYS_SEM_H_
  12. #define ZEPHYR_INCLUDE_SYS_SEM_H_
  13. /*
  14. * sys_sem exists in user memory working as counter semaphore for
  15. * user mode thread when user mode enabled. When user mode isn't
  16. * enabled, sys_sem behaves like k_sem.
  17. */
  18. #include <kernel.h>
  19. #include <sys/atomic.h>
  20. #include <zephyr/types.h>
  21. #ifdef __cplusplus
  22. extern "C" {
  23. #endif
  24. /**
  25. * sys_sem structure
  26. */
  27. struct sys_sem {
  28. #ifdef CONFIG_USERSPACE
  29. struct k_futex futex;
  30. int limit;
  31. #else
  32. struct k_sem kernel_sem;
  33. #endif
  34. };
  35. /**
  36. * @defgroup user_semaphore_apis User mode semaphore APIs
  37. * @ingroup kernel_apis
  38. * @{
  39. */
  40. /**
  41. * @brief Statically define and initialize a sys_sem
  42. *
  43. * The semaphore can be accessed outside the module where it is defined using:
  44. *
  45. * @code extern struct sys_sem <name>; @endcode
  46. *
  47. * Route this to memory domains using K_APP_DMEM().
  48. *
  49. * @param _name Name of the semaphore.
  50. * @param _initial_count Initial semaphore count.
  51. * @param _count_limit Maximum permitted semaphore count.
  52. */
  53. #ifdef CONFIG_USERSPACE
  54. #define SYS_SEM_DEFINE(_name, _initial_count, _count_limit) \
  55. struct sys_sem _name = { \
  56. .futex = { _initial_count }, \
  57. .limit = _count_limit \
  58. }; \
  59. BUILD_ASSERT(((_count_limit) != 0) && \
  60. ((_initial_count) <= (_count_limit)))
  61. #else
  62. /* Stuff this in the section with the rest of the k_sem objects, since they
  63. * are identical and can be treated as a k_sem in the boot initialization code
  64. */
  65. #define SYS_SEM_DEFINE(_name, _initial_count, _count_limit) \
  66. STRUCT_SECTION_ITERABLE_ALTERNATE(k_sem, sys_sem, _name) = { \
  67. .kernel_sem = Z_SEM_INITIALIZER(_name.kernel_sem, \
  68. _initial_count, _count_limit) \
  69. }; \
  70. BUILD_ASSERT(((_count_limit) != 0) && \
  71. ((_initial_count) <= (_count_limit)))
  72. #endif
  73. /**
  74. * @brief Initialize a semaphore.
  75. *
  76. * This routine initializes a semaphore instance, prior to its first use.
  77. *
  78. * @param sem Address of the semaphore.
  79. * @param initial_count Initial semaphore count.
  80. * @param limit Maximum permitted semaphore count.
  81. *
  82. * @retval 0 Initial success.
  83. * @retval -EINVAL Bad parameters, the value of limit should be located in
  84. * (0, INT_MAX] and initial_count shouldn't be greater than limit.
  85. */
  86. int sys_sem_init(struct sys_sem *sem, unsigned int initial_count,
  87. unsigned int limit);
  88. /**
  89. * @brief Give a semaphore.
  90. *
  91. * This routine gives @a sem, unless the semaphore is already at its
  92. * maximum permitted count.
  93. *
  94. * @param sem Address of the semaphore.
  95. *
  96. * @retval 0 Semaphore given.
  97. * @retval -EINVAL Parameter address not recognized.
  98. * @retval -EACCES Caller does not have enough access.
  99. * @retval -EAGAIN Count reached Maximum permitted count and try again.
  100. */
  101. int sys_sem_give(struct sys_sem *sem);
  102. /**
  103. * @brief Take a sys_sem.
  104. *
  105. * This routine takes @a sem.
  106. *
  107. * @param sem Address of the sys_sem.
  108. * @param timeout Waiting period to take the sys_sem,
  109. * or one of the special values K_NO_WAIT and K_FOREVER.
  110. *
  111. * @retval 0 sys_sem taken.
  112. * @retval -EINVAL Parameter address not recognized.
  113. * @retval -ETIMEDOUT Waiting period timed out.
  114. * @retval -EACCES Caller does not have enough access.
  115. */
  116. int sys_sem_take(struct sys_sem *sem, k_timeout_t timeout);
  117. /**
  118. * @brief Get sys_sem's value
  119. *
  120. * This routine returns the current value of @a sem.
  121. *
  122. * @param sem Address of the sys_sem.
  123. *
  124. * @return Current value of sys_sem.
  125. */
  126. unsigned int sys_sem_count_get(struct sys_sem *sem);
  127. /**
  128. * @}
  129. */
  130. #ifdef __cplusplus
  131. }
  132. #endif
  133. #endif