bitarray.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565
  1. /*
  2. * Copyright (c) 2021 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <errno.h>
  7. #include <stddef.h>
  8. #include <stdbool.h>
  9. #include <stdio.h>
  10. #include <sys/bitarray.h>
  11. #include <sys/check.h>
  12. #include <sys/sys_io.h>
  13. /* Number of bits represented by one bundle */
  14. #define bundle_bitness(ba) (sizeof(ba->bundles[0]) * 8)
  15. struct bundle_data {
  16. /* Start and end index of bundles */
  17. size_t sidx, eidx;
  18. /* Offset inside start and end bundles */
  19. size_t soff, eoff;
  20. /* Masks for start/end bundles */
  21. uint32_t smask, emask;
  22. };
  23. static void setup_bundle_data(sys_bitarray_t *bitarray,
  24. struct bundle_data *bd,
  25. size_t offset, size_t num_bits)
  26. {
  27. bd->sidx = offset / bundle_bitness(bitarray);
  28. bd->soff = offset % bundle_bitness(bitarray);
  29. bd->eidx = (offset + num_bits - 1) / bundle_bitness(bitarray);
  30. bd->eoff = (offset + num_bits - 1) % bundle_bitness(bitarray);
  31. bd->smask = ~(BIT(bd->soff) - 1);
  32. bd->emask = (BIT(bd->eoff) - 1) | BIT(bd->eoff);
  33. if (bd->sidx == bd->eidx) {
  34. /* The region lies within the same bundle. So combine the masks. */
  35. bd->smask &= bd->emask;
  36. }
  37. }
  38. /*
  39. * Find out if the bits in a region is all set or all clear.
  40. *
  41. * @param[in] bitarray Bitarray struct
  42. * @param[in] offset Starting bit location
  43. * @param[in] num_bits Number of bits in the region
  44. * @param[in] match_set True if matching all set bits,
  45. * False if matching all cleared bits
  46. * @param[out] bd Data related to matching which can be
  47. * used later to find out where the region
  48. * lies in the bitarray bundles.
  49. * @param[out] mismatch Offset to the mismatched bit.
  50. * Can be NULL.
  51. *
  52. * @retval true If all bits are set or cleared
  53. * @retval false Not all bits are set or cleared
  54. */
  55. static bool match_region(sys_bitarray_t *bitarray, size_t offset,
  56. size_t num_bits, bool match_set,
  57. struct bundle_data *bd,
  58. size_t *mismatch)
  59. {
  60. int idx;
  61. uint32_t bundle;
  62. uint32_t mismatch_bundle;
  63. uint32_t mismatch_mask;
  64. size_t mismatch_bundle_idx;
  65. size_t mismatch_bit_off;
  66. setup_bundle_data(bitarray, bd, offset, num_bits);
  67. if (bd->sidx == bd->eidx) {
  68. bundle = bitarray->bundles[bd->sidx];
  69. if (!match_set) {
  70. bundle = ~bundle;
  71. }
  72. if ((bundle & bd->smask) != bd->smask) {
  73. /* Not matching to mask. */
  74. mismatch_bundle = ~bundle & bd->smask;
  75. mismatch_bundle_idx = bd->sidx;
  76. mismatch_mask = bd->smask;
  77. goto mismatch;
  78. } else {
  79. /* Matching to mask. */
  80. goto out;
  81. }
  82. }
  83. /* Region lies in a number of bundles. Need to loop through them. */
  84. /* Start of bundles */
  85. bundle = bitarray->bundles[bd->sidx];
  86. if (!match_set) {
  87. bundle = ~bundle;
  88. }
  89. if ((bundle & bd->smask) != bd->smask) {
  90. /* Start bundle not matching to mask. */
  91. mismatch_bundle = ~bundle & bd->smask;
  92. mismatch_bundle_idx = bd->sidx;
  93. mismatch_mask = bd->smask;
  94. goto mismatch;
  95. }
  96. /* End of bundles */
  97. bundle = bitarray->bundles[bd->eidx];
  98. if (!match_set) {
  99. bundle = ~bundle;
  100. }
  101. if ((bundle & bd->emask) != bd->emask) {
  102. /* End bundle not matching to mask. */
  103. mismatch_bundle = ~bundle & bd->emask;
  104. mismatch_bundle_idx = bd->eidx;
  105. mismatch_mask = bd->emask;
  106. goto mismatch;
  107. }
  108. /* In-between bundles */
  109. for (idx = bd->sidx + 1; idx < bd->eidx; idx++) {
  110. /* Note that this is opposite from above so that
  111. * we are simply checking if bundle == 0.
  112. */
  113. bundle = bitarray->bundles[idx];
  114. if (match_set) {
  115. bundle = ~bundle;
  116. }
  117. if (bundle != 0U) {
  118. /* Bits in "between bundles" do not match */
  119. mismatch_bundle = ~bundle;
  120. mismatch_bundle_idx = idx;
  121. mismatch_mask = ~0U;
  122. goto mismatch;
  123. }
  124. }
  125. out:
  126. /* All bits in region matched. */
  127. return true;
  128. mismatch:
  129. if (mismatch != NULL) {
  130. /* Must have at least 1 bit set to indicate
  131. * where the mismatch is.
  132. */
  133. __ASSERT_NO_MSG(mismatch_bundle != 0);
  134. mismatch_bit_off = find_lsb_set(mismatch_bundle) - 1;
  135. mismatch_bit_off += mismatch_bundle_idx *
  136. bundle_bitness(bitarray);
  137. *mismatch = (uint32_t)mismatch_bit_off;
  138. }
  139. return false;
  140. }
  141. /*
  142. * Set or clear a region of bits.
  143. *
  144. * @param bitarray Bitarray struct
  145. * @param offset Starting bit location
  146. * @param num_bits Number of bits in the region
  147. * @param to_set True if to set all bits.
  148. * False if to clear all bits.
  149. * @param bd Bundle data. Can reuse the output from
  150. * match_region(). NULL if there is no
  151. * prior call to match_region().
  152. */
  153. static void set_region(sys_bitarray_t *bitarray, size_t offset,
  154. size_t num_bits, bool to_set,
  155. struct bundle_data *bd)
  156. {
  157. int idx;
  158. struct bundle_data bdata;
  159. if (bd == NULL) {
  160. bd = &bdata;
  161. setup_bundle_data(bitarray, bd, offset, num_bits);
  162. }
  163. if (bd->sidx == bd->eidx) {
  164. /* Start/end at same bundle */
  165. if (to_set) {
  166. bitarray->bundles[bd->sidx] |= bd->smask;
  167. } else {
  168. bitarray->bundles[bd->sidx] &= ~bd->smask;
  169. }
  170. } else {
  171. /* Start/end at different bundle.
  172. * So set/clear the bits in start and end bundles
  173. * separately. For in-between bundles,
  174. * set/clear all bits.
  175. */
  176. if (to_set) {
  177. bitarray->bundles[bd->sidx] |= bd->smask;
  178. bitarray->bundles[bd->eidx] |= bd->emask;
  179. for (idx = bd->sidx + 1; idx < bd->eidx; idx++) {
  180. bitarray->bundles[idx] = ~0U;
  181. }
  182. } else {
  183. bitarray->bundles[bd->sidx] &= ~bd->smask;
  184. bitarray->bundles[bd->eidx] &= ~bd->emask;
  185. for (idx = bd->sidx + 1; idx < bd->eidx; idx++) {
  186. bitarray->bundles[idx] = 0U;
  187. }
  188. }
  189. }
  190. }
  191. int sys_bitarray_set_bit(sys_bitarray_t *bitarray, size_t bit)
  192. {
  193. k_spinlock_key_t key;
  194. int ret;
  195. size_t idx, off;
  196. key = k_spin_lock(&bitarray->lock);
  197. __ASSERT_NO_MSG(bitarray->num_bits > 0);
  198. if (bit >= bitarray->num_bits) {
  199. ret = -EINVAL;
  200. goto out;
  201. }
  202. idx = bit / bundle_bitness(bitarray);
  203. off = bit % bundle_bitness(bitarray);
  204. bitarray->bundles[idx] |= BIT(off);
  205. ret = 0;
  206. out:
  207. k_spin_unlock(&bitarray->lock, key);
  208. return ret;
  209. }
  210. int sys_bitarray_clear_bit(sys_bitarray_t *bitarray, size_t bit)
  211. {
  212. k_spinlock_key_t key;
  213. int ret;
  214. size_t idx, off;
  215. key = k_spin_lock(&bitarray->lock);
  216. __ASSERT_NO_MSG(bitarray->num_bits > 0);
  217. if (bit >= bitarray->num_bits) {
  218. ret = -EINVAL;
  219. goto out;
  220. }
  221. idx = bit / bundle_bitness(bitarray);
  222. off = bit % bundle_bitness(bitarray);
  223. bitarray->bundles[idx] &= ~BIT(off);
  224. ret = 0;
  225. out:
  226. k_spin_unlock(&bitarray->lock, key);
  227. return ret;
  228. }
  229. int sys_bitarray_test_bit(sys_bitarray_t *bitarray, size_t bit, int *val)
  230. {
  231. k_spinlock_key_t key;
  232. int ret;
  233. size_t idx, off;
  234. key = k_spin_lock(&bitarray->lock);
  235. __ASSERT_NO_MSG(bitarray->num_bits > 0);
  236. CHECKIF(val == NULL) {
  237. ret = -EINVAL;
  238. goto out;
  239. }
  240. if (bit >= bitarray->num_bits) {
  241. ret = -EINVAL;
  242. goto out;
  243. }
  244. idx = bit / bundle_bitness(bitarray);
  245. off = bit % bundle_bitness(bitarray);
  246. if ((bitarray->bundles[idx] & BIT(off)) != 0) {
  247. *val = 1;
  248. } else {
  249. *val = 0;
  250. }
  251. ret = 0;
  252. out:
  253. k_spin_unlock(&bitarray->lock, key);
  254. return ret;
  255. }
  256. int sys_bitarray_test_and_set_bit(sys_bitarray_t *bitarray, size_t bit, int *prev_val)
  257. {
  258. k_spinlock_key_t key;
  259. int ret;
  260. size_t idx, off;
  261. key = k_spin_lock(&bitarray->lock);
  262. __ASSERT_NO_MSG(bitarray->num_bits > 0);
  263. CHECKIF(prev_val == NULL) {
  264. ret = -EINVAL;
  265. goto out;
  266. }
  267. if (bit >= bitarray->num_bits) {
  268. ret = -EINVAL;
  269. goto out;
  270. }
  271. idx = bit / bundle_bitness(bitarray);
  272. off = bit % bundle_bitness(bitarray);
  273. if ((bitarray->bundles[idx] & BIT(off)) != 0) {
  274. *prev_val = 1;
  275. } else {
  276. *prev_val = 0;
  277. }
  278. bitarray->bundles[idx] |= BIT(off);
  279. ret = 0;
  280. out:
  281. k_spin_unlock(&bitarray->lock, key);
  282. return ret;
  283. }
  284. int sys_bitarray_test_and_clear_bit(sys_bitarray_t *bitarray, size_t bit, int *prev_val)
  285. {
  286. k_spinlock_key_t key;
  287. int ret;
  288. size_t idx, off;
  289. key = k_spin_lock(&bitarray->lock);
  290. __ASSERT_NO_MSG(bitarray->num_bits > 0);
  291. CHECKIF(prev_val == NULL) {
  292. ret = -EINVAL;
  293. goto out;
  294. }
  295. if (bit >= bitarray->num_bits) {
  296. ret = -EINVAL;
  297. goto out;
  298. }
  299. idx = bit / bundle_bitness(bitarray);
  300. off = bit % bundle_bitness(bitarray);
  301. if ((bitarray->bundles[idx] & BIT(off)) != 0) {
  302. *prev_val = 1;
  303. } else {
  304. *prev_val = 0;
  305. }
  306. bitarray->bundles[idx] &= ~BIT(off);
  307. ret = 0;
  308. out:
  309. k_spin_unlock(&bitarray->lock, key);
  310. return ret;
  311. }
  312. int sys_bitarray_alloc(sys_bitarray_t *bitarray, size_t num_bits,
  313. size_t *offset)
  314. {
  315. k_spinlock_key_t key;
  316. uint32_t bit_idx;
  317. int ret;
  318. struct bundle_data bd;
  319. size_t off_start, off_end;
  320. size_t mismatch;
  321. key = k_spin_lock(&bitarray->lock);
  322. __ASSERT_NO_MSG(bitarray->num_bits > 0);
  323. CHECKIF(offset == NULL) {
  324. ret = -EINVAL;
  325. goto out;
  326. }
  327. if ((num_bits == 0) || (num_bits > bitarray->num_bits)) {
  328. ret = -EINVAL;
  329. goto out;
  330. }
  331. bit_idx = 0;
  332. /* Find the first non-allocated bit by looking at bundles
  333. * instead of individual bits.
  334. *
  335. * On RISC-V 64-bit, it complains about undefined reference to `ffs`.
  336. * So don't use this on RISCV64.
  337. */
  338. for (ret = 0; ret < bitarray->num_bundles; ret++) {
  339. if (~bitarray->bundles[ret] == 0U) {
  340. /* bundle is all 1s => all allocated, skip */
  341. bit_idx += bundle_bitness(bitarray);
  342. continue;
  343. }
  344. if (bitarray->bundles[ret] != 0U) {
  345. /* Find the first free bit in bundle if not all free */
  346. off_start = find_lsb_set(~bitarray->bundles[ret]) - 1;
  347. bit_idx += off_start;
  348. }
  349. break;
  350. }
  351. off_end = bitarray->num_bits - num_bits;
  352. ret = -ENOSPC;
  353. while (bit_idx <= off_end) {
  354. if (match_region(bitarray, bit_idx, num_bits, false,
  355. &bd, &mismatch)) {
  356. off_end = bit_idx + num_bits - 1;
  357. set_region(bitarray, bit_idx, num_bits, true, &bd);
  358. *offset = bit_idx;
  359. ret = 0;
  360. break;
  361. }
  362. /* Fast-forward to the bit just after
  363. * the mismatched bit.
  364. */
  365. bit_idx = mismatch + 1;
  366. }
  367. out:
  368. k_spin_unlock(&bitarray->lock, key);
  369. return ret;
  370. }
  371. int sys_bitarray_free(sys_bitarray_t *bitarray, size_t num_bits,
  372. size_t offset)
  373. {
  374. k_spinlock_key_t key;
  375. int ret;
  376. size_t off_end = offset + num_bits - 1;
  377. struct bundle_data bd;
  378. key = k_spin_lock(&bitarray->lock);
  379. __ASSERT_NO_MSG(bitarray->num_bits > 0);
  380. if ((num_bits == 0)
  381. || (num_bits > bitarray->num_bits)
  382. || (offset >= bitarray->num_bits)
  383. || (off_end >= bitarray->num_bits)) {
  384. ret = -EINVAL;
  385. goto out;
  386. }
  387. /* Note that we need to make sure the bits in specified region
  388. * (offset to offset + num_bits) are all allocated before we clear
  389. * them.
  390. */
  391. if (match_region(bitarray, offset, num_bits, true, &bd, NULL)) {
  392. set_region(bitarray, offset, num_bits, false, &bd);
  393. ret = 0;
  394. } else {
  395. ret = -EFAULT;
  396. }
  397. out:
  398. k_spin_unlock(&bitarray->lock, key);
  399. return ret;
  400. }
  401. static bool is_region_set_clear(sys_bitarray_t *bitarray, size_t num_bits,
  402. size_t offset, bool to_set)
  403. {
  404. bool ret;
  405. struct bundle_data bd;
  406. size_t off_end = offset + num_bits - 1;
  407. k_spinlock_key_t key = k_spin_lock(&bitarray->lock);
  408. __ASSERT_NO_MSG(bitarray->num_bits > 0);
  409. if ((num_bits == 0)
  410. || (num_bits > bitarray->num_bits)
  411. || (offset >= bitarray->num_bits)
  412. || (off_end >= bitarray->num_bits)) {
  413. ret = false;
  414. goto out;
  415. }
  416. ret = match_region(bitarray, offset, num_bits, to_set, &bd, NULL);
  417. out:
  418. k_spin_unlock(&bitarray->lock, key);
  419. return ret;
  420. }
  421. bool sys_bitarray_is_region_set(sys_bitarray_t *bitarray, size_t num_bits,
  422. size_t offset)
  423. {
  424. return is_region_set_clear(bitarray, num_bits, offset, true);
  425. }
  426. bool sys_bitarray_is_region_cleared(sys_bitarray_t *bitarray, size_t num_bits,
  427. size_t offset)
  428. {
  429. return is_region_set_clear(bitarray, num_bits, offset, false);
  430. }
  431. static int set_clear_region(sys_bitarray_t *bitarray, size_t num_bits,
  432. size_t offset, bool to_set)
  433. {
  434. int ret;
  435. size_t off_end = offset + num_bits - 1;
  436. k_spinlock_key_t key = k_spin_lock(&bitarray->lock);
  437. __ASSERT_NO_MSG(bitarray->num_bits > 0);
  438. if ((num_bits == 0)
  439. || (num_bits > bitarray->num_bits)
  440. || (offset >= bitarray->num_bits)
  441. || (off_end >= bitarray->num_bits)) {
  442. ret = -EINVAL;
  443. goto out;
  444. }
  445. set_region(bitarray, offset, num_bits, to_set, NULL);
  446. ret = 0;
  447. out:
  448. k_spin_unlock(&bitarray->lock, key);
  449. return ret;
  450. }
  451. int sys_bitarray_set_region(sys_bitarray_t *bitarray, size_t num_bits,
  452. size_t offset)
  453. {
  454. return set_clear_region(bitarray, num_bits, offset, true);
  455. }
  456. int sys_bitarray_clear_region(sys_bitarray_t *bitarray, size_t num_bits,
  457. size_t offset)
  458. {
  459. return set_clear_region(bitarray, num_bits, offset, false);
  460. }