atomic_builtin.h 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. /* atomic operations */
  2. /*
  3. * Copyright (c) 1997-2015, Wind River Systems, Inc.
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. #ifndef ZEPHYR_INCLUDE_SYS_ATOMIC_BUILTIN_H_
  8. #define ZEPHYR_INCLUDE_SYS_ATOMIC_BUILTIN_H_
  9. #ifdef __cplusplus
  10. extern "C" {
  11. #endif
  12. /* Included from <atomic.h> */
  13. /**
  14. * @addtogroup atomic_apis Atomic Services APIs
  15. * @ingroup kernel_apis
  16. * @{
  17. */
  18. /**
  19. * @brief Atomic compare-and-set.
  20. *
  21. * This routine performs an atomic compare-and-set on @a target. If the current
  22. * value of @a target equals @a old_value, @a target is set to @a new_value.
  23. * If the current value of @a target does not equal @a old_value, @a target
  24. * is left unchanged.
  25. *
  26. * @param target Address of atomic variable.
  27. * @param old_value Original value to compare against.
  28. * @param new_value New value to store.
  29. * @return true if @a new_value is written, false otherwise.
  30. */
  31. static inline bool atomic_cas(atomic_t *target, atomic_val_t old_value,
  32. atomic_val_t new_value)
  33. {
  34. return __atomic_compare_exchange_n(target, &old_value, new_value,
  35. 0, __ATOMIC_SEQ_CST,
  36. __ATOMIC_SEQ_CST);
  37. }
  38. /**
  39. * @brief Atomic compare-and-set with pointer values
  40. *
  41. * This routine performs an atomic compare-and-set on @a target. If the current
  42. * value of @a target equals @a old_value, @a target is set to @a new_value.
  43. * If the current value of @a target does not equal @a old_value, @a target
  44. * is left unchanged.
  45. *
  46. * @param target Address of atomic variable.
  47. * @param old_value Original value to compare against.
  48. * @param new_value New value to store.
  49. * @return true if @a new_value is written, false otherwise.
  50. */
  51. static inline bool atomic_ptr_cas(atomic_ptr_t *target, atomic_ptr_val_t old_value,
  52. atomic_ptr_val_t new_value)
  53. {
  54. return __atomic_compare_exchange_n(target, &old_value, new_value,
  55. 0, __ATOMIC_SEQ_CST,
  56. __ATOMIC_SEQ_CST);
  57. }
  58. /**
  59. *
  60. * @brief Atomic addition.
  61. *
  62. * This routine performs an atomic addition on @a target.
  63. *
  64. * @param target Address of atomic variable.
  65. * @param value Value to add.
  66. *
  67. * @return Previous value of @a target.
  68. */
  69. static inline atomic_val_t atomic_add(atomic_t *target, atomic_val_t value)
  70. {
  71. return __atomic_fetch_add(target, value, __ATOMIC_SEQ_CST);
  72. }
  73. /**
  74. *
  75. * @brief Atomic subtraction.
  76. *
  77. * This routine performs an atomic subtraction on @a target.
  78. *
  79. * @param target Address of atomic variable.
  80. * @param value Value to subtract.
  81. *
  82. * @return Previous value of @a target.
  83. */
  84. static inline atomic_val_t atomic_sub(atomic_t *target, atomic_val_t value)
  85. {
  86. return __atomic_fetch_sub(target, value, __ATOMIC_SEQ_CST);
  87. }
  88. /**
  89. *
  90. * @brief Atomic increment.
  91. *
  92. * This routine performs an atomic increment by 1 on @a target.
  93. *
  94. * @param target Address of atomic variable.
  95. *
  96. * @return Previous value of @a target.
  97. */
  98. static inline atomic_val_t atomic_inc(atomic_t *target)
  99. {
  100. return atomic_add(target, 1);
  101. }
  102. /**
  103. *
  104. * @brief Atomic decrement.
  105. *
  106. * This routine performs an atomic decrement by 1 on @a target.
  107. *
  108. * @param target Address of atomic variable.
  109. *
  110. * @return Previous value of @a target.
  111. */
  112. static inline atomic_val_t atomic_dec(atomic_t *target)
  113. {
  114. return atomic_sub(target, 1);
  115. }
  116. /**
  117. *
  118. * @brief Atomic get.
  119. *
  120. * This routine performs an atomic read on @a target.
  121. *
  122. * @param target Address of atomic variable.
  123. *
  124. * @return Value of @a target.
  125. */
  126. static inline atomic_val_t atomic_get(const atomic_t *target)
  127. {
  128. return __atomic_load_n(target, __ATOMIC_SEQ_CST);
  129. }
  130. /**
  131. *
  132. * @brief Atomic get a pointer value
  133. *
  134. * This routine performs an atomic read on @a target.
  135. *
  136. * @param target Address of pointer variable.
  137. *
  138. * @return Value of @a target.
  139. */
  140. static inline atomic_ptr_val_t atomic_ptr_get(const atomic_ptr_t *target)
  141. {
  142. return __atomic_load_n(target, __ATOMIC_SEQ_CST);
  143. }
  144. /**
  145. *
  146. * @brief Atomic get-and-set.
  147. *
  148. * This routine atomically sets @a target to @a value and returns
  149. * the previous value of @a target.
  150. *
  151. * @param target Address of atomic variable.
  152. * @param value Value to write to @a target.
  153. *
  154. * @return Previous value of @a target.
  155. */
  156. static inline atomic_val_t atomic_set(atomic_t *target, atomic_val_t value)
  157. {
  158. /* This builtin, as described by Intel, is not a traditional
  159. * test-and-set operation, but rather an atomic exchange operation. It
  160. * writes value into *ptr, and returns the previous contents of *ptr.
  161. */
  162. return __atomic_exchange_n(target, value, __ATOMIC_SEQ_CST);
  163. }
  164. /**
  165. *
  166. * @brief Atomic get-and-set for pointer values
  167. *
  168. * This routine atomically sets @a target to @a value and returns
  169. * the previous value of @a target.
  170. *
  171. * @param target Address of atomic variable.
  172. * @param value Value to write to @a target.
  173. *
  174. * @return Previous value of @a target.
  175. */
  176. static inline atomic_ptr_val_t atomic_ptr_set(atomic_ptr_t *target, atomic_ptr_val_t value)
  177. {
  178. return __atomic_exchange_n(target, value, __ATOMIC_SEQ_CST);
  179. }
  180. /**
  181. *
  182. * @brief Atomic clear.
  183. *
  184. * This routine atomically sets @a target to zero and returns its previous
  185. * value. (Hence, it is equivalent to atomic_set(target, 0).)
  186. *
  187. * @param target Address of atomic variable.
  188. *
  189. * @return Previous value of @a target.
  190. */
  191. static inline atomic_val_t atomic_clear(atomic_t *target)
  192. {
  193. return atomic_set(target, 0);
  194. }
  195. /**
  196. *
  197. * @brief Atomic clear of a pointer value
  198. *
  199. * This routine atomically sets @a target to zero and returns its previous
  200. * value. (Hence, it is equivalent to atomic_set(target, 0).)
  201. *
  202. * @param target Address of atomic variable.
  203. *
  204. * @return Previous value of @a target.
  205. */
  206. static inline atomic_ptr_val_t atomic_ptr_clear(atomic_ptr_t *target)
  207. {
  208. return atomic_ptr_set(target, NULL);
  209. }
  210. /**
  211. *
  212. * @brief Atomic bitwise inclusive OR.
  213. *
  214. * This routine atomically sets @a target to the bitwise inclusive OR of
  215. * @a target and @a value.
  216. *
  217. * @param target Address of atomic variable.
  218. * @param value Value to OR.
  219. *
  220. * @return Previous value of @a target.
  221. */
  222. static inline atomic_val_t atomic_or(atomic_t *target, atomic_val_t value)
  223. {
  224. return __atomic_fetch_or(target, value, __ATOMIC_SEQ_CST);
  225. }
  226. /**
  227. *
  228. * @brief Atomic bitwise exclusive OR (XOR).
  229. *
  230. * This routine atomically sets @a target to the bitwise exclusive OR (XOR) of
  231. * @a target and @a value.
  232. *
  233. * @param target Address of atomic variable.
  234. * @param value Value to XOR
  235. *
  236. * @return Previous value of @a target.
  237. */
  238. static inline atomic_val_t atomic_xor(atomic_t *target, atomic_val_t value)
  239. {
  240. return __atomic_fetch_xor(target, value, __ATOMIC_SEQ_CST);
  241. }
  242. /**
  243. *
  244. * @brief Atomic bitwise AND.
  245. *
  246. * This routine atomically sets @a target to the bitwise AND of @a target
  247. * and @a value.
  248. *
  249. * @param target Address of atomic variable.
  250. * @param value Value to AND.
  251. *
  252. * @return Previous value of @a target.
  253. */
  254. static inline atomic_val_t atomic_and(atomic_t *target, atomic_val_t value)
  255. {
  256. return __atomic_fetch_and(target, value, __ATOMIC_SEQ_CST);
  257. }
  258. /**
  259. *
  260. * @brief Atomic bitwise NAND.
  261. *
  262. * This routine atomically sets @a target to the bitwise NAND of @a target
  263. * and @a value. (This operation is equivalent to target = ~(target & value).)
  264. *
  265. * @param target Address of atomic variable.
  266. * @param value Value to NAND.
  267. *
  268. * @return Previous value of @a target.
  269. */
  270. static inline atomic_val_t atomic_nand(atomic_t *target, atomic_val_t value)
  271. {
  272. return __atomic_fetch_nand(target, value, __ATOMIC_SEQ_CST);
  273. }
  274. /** @} */
  275. #ifdef __cplusplus
  276. }
  277. #endif
  278. #endif /* ZEPHYR_INCLUDE_SYS_ATOMIC_BUILTIN_H_ */