rtx_semaphore.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424
  1. /*
  2. * Copyright (c) 2013-2016 ARM Limited. All rights reserved.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Licensed under the Apache License, Version 2.0 (the License); you may
  7. * not use this file except in compliance with the License.
  8. * You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an AS IS BASIS, WITHOUT
  14. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. *
  18. * -----------------------------------------------------------------------------
  19. *
  20. * Project: CMSIS-RTOS RTX
  21. * Title: Semaphore functions
  22. *
  23. * -----------------------------------------------------------------------------
  24. */
  25. #include "rtx_lib.h"
  26. // ==== Helper functions ====
  27. /// Decrement Semaphore tokens.
  28. /// \param[in] semaphore semaphore object.
  29. /// \return 1 - success, 0 - failure.
  30. static uint32_t os_SemaphoreTokenDecrement (os_semaphore_t *semaphore) {
  31. #ifdef __NO_EXCLUSIVE_ACCESS
  32. uint32_t primask = __get_PRIMASK();
  33. #endif
  34. uint32_t ret;
  35. #ifdef __NO_EXCLUSIVE_ACCESS
  36. __disable_irq();
  37. if (semaphore->tokens != 0U) {
  38. semaphore->tokens--;
  39. ret = 1U;
  40. } else {
  41. ret = 0U;
  42. }
  43. if (primask == 0U) {
  44. __enable_irq();
  45. }
  46. #else
  47. if (os_exc_dec16_nz(&semaphore->tokens) != 0U) {
  48. ret = 1U;
  49. } else {
  50. ret = 0U;
  51. }
  52. #endif
  53. return ret;
  54. }
  55. /// Increment Semaphore tokens.
  56. /// \param[in] semaphore semaphore object.
  57. /// \return 1 - success, 0 - failure.
  58. static uint32_t os_SemaphoreTokenIncrement (os_semaphore_t *semaphore) {
  59. #ifdef __NO_EXCLUSIVE_ACCESS
  60. uint32_t primask = __get_PRIMASK();
  61. #endif
  62. uint32_t ret;
  63. #ifdef __NO_EXCLUSIVE_ACCESS
  64. __disable_irq();
  65. if (semaphore->tokens < semaphore->max_tokens) {
  66. semaphore->tokens++;
  67. ret = 1U;
  68. } else {
  69. ret = 0U;
  70. }
  71. if (primask == 0U) {
  72. __enable_irq();
  73. }
  74. #else
  75. if (os_exc_inc16_lt(&semaphore->tokens, semaphore->max_tokens) < semaphore->max_tokens) {
  76. ret = 1U;
  77. } else {
  78. ret = 0U;
  79. }
  80. #endif
  81. return ret;
  82. }
  83. // ==== Library functions ====
  84. /// Semaphore post ISR processing.
  85. /// \param[in] semaphore semaphore object.
  86. void os_SemaphorePostProcess (os_semaphore_t *semaphore) {
  87. os_thread_t *thread;
  88. if (semaphore->state == os_ObjectInactive) {
  89. return;
  90. }
  91. // Check if Thread is waiting for a token
  92. if (semaphore->thread_list != NULL) {
  93. // Try to acquire token
  94. if (os_SemaphoreTokenDecrement(semaphore) != 0U) {
  95. // Wakeup waiting Thread with highest Priority
  96. thread = os_ThreadListGet((os_object_t*)semaphore);
  97. os_ThreadWaitExit(thread, (uint32_t)osOK, false);
  98. }
  99. }
  100. }
  101. // ==== Service Calls ====
  102. // Service Calls definitions
  103. SVC0_3(SemaphoreNew, osSemaphoreId_t, uint32_t, uint32_t, const osSemaphoreAttr_t *)
  104. SVC0_2(SemaphoreAcquire, osStatus_t, osSemaphoreId_t, uint32_t)
  105. SVC0_1(SemaphoreRelease, osStatus_t, osSemaphoreId_t)
  106. SVC0_1(SemaphoreGetCount, uint32_t, osSemaphoreId_t)
  107. SVC0_1(SemaphoreDelete, osStatus_t, osSemaphoreId_t)
  108. /// Create and Initialize a Semaphore object.
  109. /// \note API identical to osSemaphoreNew
  110. osSemaphoreId_t os_svcSemaphoreNew (uint32_t max_count, uint32_t initial_count, const osSemaphoreAttr_t *attr) {
  111. os_semaphore_t *semaphore;
  112. uint8_t flags;
  113. const char *name;
  114. // Check parameters
  115. if ((max_count > os_SemaphoreTokenLimit) ||
  116. (initial_count > max_count)) {
  117. return (osSemaphoreId_t)NULL;
  118. }
  119. // Process attributes
  120. if (attr != NULL) {
  121. name = attr->name;
  122. semaphore = attr->cb_mem;
  123. if (semaphore != NULL) {
  124. if (((uint32_t)semaphore & 3U) || (attr->cb_size < sizeof(os_semaphore_t))) {
  125. return (osSemaphoreId_t)NULL;
  126. }
  127. } else {
  128. if (attr->cb_size != 0U) {
  129. return (osSemaphoreId_t)NULL;
  130. }
  131. }
  132. } else {
  133. name = NULL;
  134. semaphore = NULL;
  135. }
  136. // Allocate object memory if not provided
  137. if (semaphore == NULL) {
  138. if (os_Info.mpi.semaphore != NULL) {
  139. semaphore = os_MemoryPoolAlloc(os_Info.mpi.semaphore);
  140. } else {
  141. semaphore = os_MemoryAlloc(os_Info.mem.cb, sizeof(os_semaphore_t));
  142. }
  143. if (semaphore == NULL) {
  144. return (osSemaphoreId_t)NULL;
  145. }
  146. flags = os_FlagSystemObject;
  147. } else {
  148. flags = 0U;
  149. }
  150. // Initialize control block
  151. semaphore->id = os_IdSemaphore;
  152. semaphore->state = os_ObjectActive;
  153. semaphore->flags = flags;
  154. semaphore->name = name;
  155. semaphore->thread_list = NULL;
  156. semaphore->tokens = (uint16_t)initial_count;
  157. semaphore->max_tokens = (uint16_t)max_count;
  158. // Register post ISR processing function
  159. os_Info.post_process.semaphore = os_SemaphorePostProcess;
  160. return (osSemaphoreId_t)semaphore;
  161. }
  162. /// Acquire a Semaphore token or timeout if no tokens are available.
  163. /// \note API identical to osSemaphoreAcquire
  164. osStatus_t os_svcSemaphoreAcquire (osSemaphoreId_t semaphore_id, uint32_t millisec) {
  165. os_semaphore_t *semaphore = (os_semaphore_t *)semaphore_id;
  166. // Check parameters
  167. if ((semaphore == NULL) ||
  168. (semaphore->id != os_IdSemaphore)) {
  169. return osErrorParameter;
  170. }
  171. // Check object state
  172. if (semaphore->state == os_ObjectInactive) {
  173. return osErrorResource;
  174. }
  175. // Try to acquire token
  176. if (os_SemaphoreTokenDecrement(semaphore) == 0U) {
  177. // No token available
  178. if (millisec != 0U) {
  179. // Suspend current Thread
  180. os_ThreadListPut((os_object_t*)semaphore, os_ThreadGetRunning());
  181. os_ThreadWaitEnter(os_ThreadWaitingSemaphore, millisec);
  182. return osErrorTimeout;
  183. } else {
  184. return osErrorResource;
  185. }
  186. }
  187. return osOK;
  188. }
  189. /// Release a Semaphore token that was acquired by osSemaphoreAcquire.
  190. /// \note API identical to osSemaphoreRelease
  191. osStatus_t os_svcSemaphoreRelease (osSemaphoreId_t semaphore_id) {
  192. os_semaphore_t *semaphore = (os_semaphore_t *)semaphore_id;
  193. os_thread_t *thread;
  194. // Check parameters
  195. if ((semaphore == NULL) ||
  196. (semaphore->id != os_IdSemaphore)) {
  197. return osErrorParameter;
  198. }
  199. // Check object state
  200. if (semaphore->state == os_ObjectInactive) {
  201. return osErrorResource;
  202. }
  203. // Check if Thread is waiting for a token
  204. if (semaphore->thread_list != NULL) {
  205. // Wakeup waiting Thread with highest Priority
  206. thread = os_ThreadListGet((os_object_t*)semaphore);
  207. os_ThreadWaitExit(thread, (uint32_t)osOK, true);
  208. } else {
  209. // Try to release token
  210. if (os_SemaphoreTokenIncrement(semaphore) == 0U) {
  211. return osErrorResource;
  212. }
  213. }
  214. return osOK;
  215. }
  216. /// Get current Semaphore token count.
  217. /// \note API identical to osSemaphoreGetCount
  218. uint32_t os_svcSemaphoreGetCount (osSemaphoreId_t semaphore_id) {
  219. os_semaphore_t *semaphore = (os_semaphore_t *)semaphore_id;
  220. // Check parameters
  221. if ((semaphore == NULL) ||
  222. (semaphore->id != os_IdSemaphore)) {
  223. return 0U;
  224. }
  225. // Check object state
  226. if (semaphore->state == os_ObjectInactive) {
  227. return 0U;
  228. }
  229. return semaphore->tokens;
  230. }
  231. /// Delete a Semaphore object.
  232. /// \note API identical to osSemaphoreDelete
  233. osStatus_t os_svcSemaphoreDelete (osSemaphoreId_t semaphore_id) {
  234. os_semaphore_t *semaphore = (os_semaphore_t *)semaphore_id;
  235. os_thread_t *thread;
  236. // Check parameters
  237. if ((semaphore == NULL) ||
  238. (semaphore->id != os_IdSemaphore)) {
  239. return osErrorParameter;
  240. }
  241. // Check object state
  242. if (semaphore->state == os_ObjectInactive) {
  243. return osErrorResource;
  244. }
  245. // Mark object as inactive
  246. semaphore->state = os_ObjectInactive;
  247. // Unblock waiting threads
  248. if (semaphore->thread_list != NULL) {
  249. do {
  250. thread = os_ThreadListGet((os_object_t*)semaphore);
  251. os_ThreadWaitExit(thread, (uint32_t)osErrorResource, false);
  252. } while (semaphore->thread_list != NULL);
  253. os_ThreadDispatch(NULL);
  254. }
  255. // Free object memory
  256. if (semaphore->flags & os_FlagSystemObject) {
  257. if (os_Info.mpi.semaphore != NULL) {
  258. os_MemoryPoolFree(os_Info.mpi.semaphore, semaphore);
  259. } else {
  260. os_MemoryFree(os_Info.mem.cb, semaphore);
  261. }
  262. }
  263. return osOK;
  264. }
  265. // ==== ISR Calls ====
  266. /// Acquire a Semaphore token or timeout if no tokens are available.
  267. /// \note API identical to osSemaphoreAcquire
  268. __STATIC_INLINE
  269. osStatus_t os_isrSemaphoreAcquire (osSemaphoreId_t semaphore_id, uint32_t millisec) {
  270. os_semaphore_t *semaphore = (os_semaphore_t *)semaphore_id;
  271. // Check parameters
  272. if ((semaphore == NULL) ||
  273. (semaphore->id != os_IdSemaphore)) {
  274. return osErrorParameter;
  275. }
  276. if (millisec != 0U) {
  277. return osErrorParameter;
  278. }
  279. // Check object state
  280. if (semaphore->state == os_ObjectInactive) {
  281. return osErrorResource;
  282. }
  283. // Try to acquire token
  284. if (os_SemaphoreTokenDecrement(semaphore) == 0U) {
  285. // No token available
  286. return osErrorResource;
  287. }
  288. return osOK;
  289. }
  290. /// Release a Semaphore token that was acquired by osSemaphoreAcquire.
  291. /// \note API identical to osSemaphoreRelease
  292. __STATIC_INLINE
  293. osStatus_t os_isrSemaphoreRelease (osSemaphoreId_t semaphore_id) {
  294. os_semaphore_t *semaphore = (os_semaphore_t *)semaphore_id;
  295. // Check parameters
  296. if ((semaphore == NULL) ||
  297. (semaphore->id != os_IdSemaphore)) {
  298. return osErrorParameter;
  299. }
  300. // Check object state
  301. if (semaphore->state == os_ObjectInactive) {
  302. return osErrorResource;
  303. }
  304. // Try to release token
  305. if (os_SemaphoreTokenIncrement(semaphore) != 0U) {
  306. // Register post ISR processing
  307. os_PostProcess((os_object_t *)semaphore);
  308. } else {
  309. return osErrorResource;
  310. }
  311. return osOK;
  312. }
  313. // ==== Public API ====
  314. /// Create and Initialize a Semaphore object.
  315. osSemaphoreId_t osSemaphoreNew (uint32_t max_count, uint32_t initial_count, const osSemaphoreAttr_t *attr) {
  316. if (__get_IPSR() != 0U) {
  317. return (osSemaphoreId_t)NULL; // Not allowed in ISR
  318. }
  319. if ((os_KernelGetState() == os_KernelReady) && ((__get_CONTROL() & 1U) == 0U)) {
  320. // Kernel Ready (not running) and in Privileged mode
  321. return os_svcSemaphoreNew(max_count, initial_count, attr);
  322. } else {
  323. return __svcSemaphoreNew(max_count, initial_count, attr);
  324. }
  325. }
  326. /// Acquire a Semaphore token or timeout if no tokens are available.
  327. osStatus_t osSemaphoreAcquire (osSemaphoreId_t semaphore_id, uint32_t millisec) {
  328. if (__get_IPSR() != 0U) { // in ISR
  329. return os_isrSemaphoreAcquire(semaphore_id, millisec);
  330. } else { // in Thread
  331. return __svcSemaphoreAcquire(semaphore_id, millisec);
  332. }
  333. }
  334. /// Release a Semaphore token that was acquired by osSemaphoreAcquire.
  335. osStatus_t osSemaphoreRelease (osSemaphoreId_t semaphore_id) {
  336. if (__get_IPSR() != 0U) { // in ISR
  337. return os_isrSemaphoreRelease(semaphore_id);
  338. } else { // in Thread
  339. return __svcSemaphoreRelease(semaphore_id);
  340. }
  341. }
  342. /// Get current Semaphore token count.
  343. uint32_t osSemaphoreGetCount (osSemaphoreId_t semaphore_id) {
  344. if (__get_IPSR() != 0U) { // in ISR
  345. return os_svcSemaphoreGetCount(semaphore_id);
  346. } else { // in Thread
  347. return __svcSemaphoreGetCount(semaphore_id);
  348. }
  349. }
  350. /// Delete a Semaphore object.
  351. osStatus_t osSemaphoreDelete (osSemaphoreId_t semaphore_id) {
  352. if (__get_IPSR() != 0U) {
  353. return osErrorISR; // Not allowed in ISR
  354. }
  355. return __svcSemaphoreDelete(semaphore_id);
  356. }