bitarray.h 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. /*
  2. * Copyright (c) 2021 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_SYS_BITARRAY_H_
  7. #define ZEPHYR_INCLUDE_SYS_BITARRAY_H_
  8. #ifdef __cplusplus
  9. extern "C" {
  10. #endif
  11. #include <stddef.h>
  12. #include <stdint.h>
  13. #include <kernel.h>
  14. struct sys_bitarray {
  15. /* Number of bits */
  16. uint32_t num_bits;
  17. /* Number of bundles */
  18. uint32_t num_bundles;
  19. /* Bundle of bits */
  20. uint32_t *bundles;
  21. /* Spinlock guarding access to this bit array */
  22. struct k_spinlock lock;
  23. };
  24. typedef struct sys_bitarray sys_bitarray_t;
  25. /**
  26. * @def SYS_BITARRAY_DEFINE
  27. *
  28. * @brief Create a bitarray object.
  29. *
  30. * @param name Name of the bitarray object.
  31. * @param total_bits Total number of bits in this bitarray object.
  32. */
  33. #define SYS_BITARRAY_DEFINE(name, total_bits) \
  34. uint32_t _sys_bitarray_bundles_##name \
  35. [(((total_bits + 8 - 1) / 8) + sizeof(uint32_t) - 1) \
  36. / sizeof(uint32_t)] = {0U}; \
  37. sys_bitarray_t name = { \
  38. .num_bits = total_bits, \
  39. .num_bundles = (((total_bits + 8 - 1) / 8) \
  40. + sizeof(uint32_t) - 1) \
  41. / sizeof(uint32_t), \
  42. .bundles = _sys_bitarray_bundles_##name, \
  43. }
  44. /**
  45. * Set a bit in a bit array
  46. *
  47. * @param[in] bitarray Bitarray struct
  48. * @param[in] bit The bit to be set
  49. *
  50. * @retval 0 Operation successful
  51. * @retval -EINVAL Invalid argument (e.g. bit to set exceeds
  52. * the number of bits in bit array, etc.)
  53. */
  54. int sys_bitarray_set_bit(sys_bitarray_t *bitarray, size_t bit);
  55. /**
  56. * Clear a bit in a bit array
  57. *
  58. * @param[in] bitarray Bitarray struct
  59. * @param[in] bit The bit to be cleared
  60. *
  61. * @retval 0 Operation successful
  62. * @retval -EINVAL Invalid argument (e.g. bit to clear exceeds
  63. * the number of bits in bit array, etc.)
  64. */
  65. int sys_bitarray_clear_bit(sys_bitarray_t *bitarray, size_t bit);
  66. /**
  67. * Test whether a bit is set or not
  68. *
  69. * @param[in] bitarray Bitarray struct
  70. * @param[in] bit The bit to be tested
  71. * @param[out] val The value of the bit (0 or 1)
  72. *
  73. * @retval 0 Operation successful
  74. * @retval -EINVAL Invalid argument (e.g. bit to test exceeds
  75. * the number of bits in bit array, etc.)
  76. */
  77. int sys_bitarray_test_bit(sys_bitarray_t *bitarray, size_t bit, int *val);
  78. /**
  79. * Test the bit and set it
  80. *
  81. * @param[in] bitarray Bitarray struct
  82. * @param[in] bit The bit to be tested and set
  83. * @param[out] prev_val Previous value of the bit (0 or 1)
  84. *
  85. * @retval 0 Operation successful
  86. * @retval -EINVAL Invalid argument (e.g. bit to test exceeds
  87. * the number of bits in bit array, etc.)
  88. */
  89. int sys_bitarray_test_and_set_bit(sys_bitarray_t *bitarray, size_t bit, int *prev_val);
  90. /**
  91. * Test the bit and clear it
  92. *
  93. * @param[in] bitarray Bitarray struct
  94. * @param[in] bit The bit to be tested and cleared
  95. * @param[out] prev_val Previous value of the bit (0 or 1)
  96. *
  97. * @retval 0 Operation successful
  98. * @retval -EINVAL Invalid argument (e.g. bit to test exceeds
  99. * the number of bits in bit array, etc.)
  100. */
  101. int sys_bitarray_test_and_clear_bit(sys_bitarray_t *bitarray, size_t bit, int *prev_val);
  102. /**
  103. * Allocate bits in a bit array
  104. *
  105. * This finds a number of bits (@p num_bits) in a contiguous of
  106. * previosly unallocated region. If such a region exists, the bits are
  107. * marked as allocated and the offset to the start of this region is
  108. * returned via @p offset.
  109. *
  110. * @param[in] bitarray Bitarray struct
  111. * @param[in] num_bits Number of bits to allocate
  112. * @param[out] offset Offset to the start of allocated region if
  113. * successful
  114. *
  115. * @retval 0 Allocation successful
  116. * @retval -EINVAL Invalid argument (e.g. allocating more bits than
  117. * the bitarray has, trying to allocate 0 bits, etc.)
  118. * @retval -ENOSPC No contiguous region big enough to accommodate
  119. * the allocation
  120. */
  121. int sys_bitarray_alloc(sys_bitarray_t *bitarray, size_t num_bits,
  122. size_t *offset);
  123. /**
  124. * Free bits in a bit array
  125. *
  126. * This marks the number of bits (@p num_bits) starting from @p offset
  127. * as no longer allocated.
  128. *
  129. * @param bitarray Bitarray struct
  130. * @param num_bits Number of bits to free
  131. * @param offset Starting bit position to free
  132. *
  133. * @retval 0 Free is successful
  134. * @retval -EINVAL Invalid argument (e.g. try to free more bits than
  135. * the bitarray has, trying to free 0 bits, etc.)
  136. * @retval -EFAULT The bits in the indicated region are not all allocated.
  137. */
  138. int sys_bitarray_free(sys_bitarray_t *bitarray, size_t num_bits,
  139. size_t offset);
  140. /**
  141. * Test if bits in a region is all set.
  142. *
  143. * This tests if the number of bits (@p num_bits) in region starting
  144. * from @p offset are all set.
  145. *
  146. * @param bitarray Bitarray struct
  147. * @param num_bits Number of bits to test
  148. * @param offset Starting bit position to test
  149. *
  150. * @retval true All bits are set.
  151. * @retval false Not all bits are set.
  152. */
  153. bool sys_bitarray_is_region_set(sys_bitarray_t *bitarray, size_t num_bits,
  154. size_t offset);
  155. /**
  156. * Test if bits in a region is all cleared.
  157. *
  158. * This tests if the number of bits (@p num_bits) in region starting
  159. * from @p offset are all cleared.
  160. *
  161. * @param bitarray Bitarray struct
  162. * @param num_bits Number of bits to test
  163. * @param offset Starting bit position to test
  164. *
  165. * @retval true All bits are cleared.
  166. * @retval false Not all bits are cleared.
  167. */
  168. bool sys_bitarray_is_region_cleared(sys_bitarray_t *bitarray, size_t num_bits,
  169. size_t offset);
  170. /**
  171. * Set all bits in a region.
  172. *
  173. * This sets the number of bits (@p num_bits) in region starting
  174. * from @p offset.
  175. *
  176. * @param bitarray Bitarray struct
  177. * @param num_bits Number of bits to test
  178. * @param offset Starting bit position to test
  179. *
  180. * @retval 0 Operation successful
  181. * @retval -EINVAL Invalid argument (e.g. bit to set exceeds
  182. * the number of bits in bit array, etc.)
  183. */
  184. int sys_bitarray_set_region(sys_bitarray_t *bitarray, size_t num_bits,
  185. size_t offset);
  186. /**
  187. * Clear all bits in a region.
  188. *
  189. * This clears the number of bits (@p num_bits) in region starting
  190. * from @p offset.
  191. *
  192. * @param bitarray Bitarray struct
  193. * @param num_bits Number of bits to test
  194. * @param offset Starting bit position to test
  195. *
  196. * @retval 0 Operation successful
  197. * @retval -EINVAL Invalid argument (e.g. bit to set exceeds
  198. * the number of bits in bit array, etc.)
  199. */
  200. int sys_bitarray_clear_region(sys_bitarray_t *bitarray, size_t num_bits,
  201. size_t offset);
  202. #ifdef __cplusplus
  203. }
  204. #endif
  205. #endif /* ZEPHYR_INCLUDE_SYS_BITARRAY_H_ */