qdf_ptr_hash.h 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341
  1. /*
  2. * Copyright (c) 2019 The Linux Foundation. All rights reserved.
  3. *
  4. * Permission to use, copy, modify, and/or distribute this software for
  5. * any purpose with or without fee is hereby granted, provided that the
  6. * above copyright notice and this permission notice appear in all
  7. * copies.
  8. *
  9. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
  10. * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
  11. * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
  12. * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  13. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
  14. * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  15. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  16. * PERFORMANCE OF THIS SOFTWARE.
  17. */
  18. /**
  19. * DOC: qdf_ptr_hash.h
  20. *
  21. * A minimal hashtable implementation for doing fast lookups via pointer.
  22. *
  23. * qdf_ptr_hash also has the benefit of knowing its own size, allowing a pointer
  24. * to the hashtable to be passed around and embedded in other structs. Since
  25. * every hashtable is not necessarily of the same size, this allows using hash
  26. * tables in a lot of new places which would be impossible with the current
  27. * kernel hashtable implementation.
  28. *
  29. * Because the size of the hashtable varies with the number of bits used in the
  30. * hash, declaring a qdf_ptr_hash is a bit different. If you want to embed a
  31. * qdf_ptr_hash in another type, use a combination of qdf_ptr_hash_declare() and
  32. * qdf_ptr_hash_ptr(). If you just want to declare and use a qdf_ptr_hash, use
  33. * qdf_ptr_hash_declare_ptr() instead. Either method will ensure the appropriate
  34. * number of bytes is accounted for using an internal union, and provides the
  35. * consumer with a pointer to a qdf_ptr_hash type which can be used with all of
  36. * the other qdf_ptr_hash APIs. Alternatively, you can skip these complexities
  37. * by simply dynamically allocating the qdf_ptr_hash via qdf_ptr_hash_create().
  38. */
  39. #ifndef __QDF_PTR_HASH_H
  40. #define __QDF_PTR_HASH_H
  41. #include "i_qdf_ptr_hash.h"
  42. #include "qdf_mem.h"
  43. #include "qdf_slist.h"
  44. #include "qdf_types.h"
  45. #include "qdf_util.h"
  46. /**
  47. * struct qdf_ptr_hash_bucket - a type representing a hash bucket
  48. * @list: the list used for hash chaining
  49. */
  50. struct qdf_ptr_hash_bucket {
  51. struct qdf_slist list;
  52. };
  53. /**
  54. * struct qdf_ptr_hash - a hash table type for doing fast lookups via pointer
  55. * @bits: the number of bits to use when hashing keys
  56. * @count: the number of buckets, always equal to 2^@bits
  57. * @buckets: empty bucket array for accessing a variable length array of buckets
  58. */
  59. struct qdf_ptr_hash {
  60. int8_t bits;
  61. int16_t count;
  62. struct qdf_ptr_hash_bucket buckets[0];
  63. };
  64. /**
  65. * struct qdf_ptr_hash_entry - entry type of membership in a qdf_ptr_hash
  66. * @key: the value used as the key for insertion/lookup
  67. * @node: the list node used for hash chaining
  68. */
  69. struct qdf_ptr_hash_entry {
  70. uintptr_t key;
  71. struct qdf_slist_node node;
  72. };
  73. #define __qdf_ptr_hash_size(bits) (sizeof(struct qdf_ptr_hash) + \
  74. sizeof(((struct qdf_ptr_hash *)0)->buckets[0]) * (1 << bits))
  75. /**
  76. * qdf_ptr_hash_declare() - declare a new qdf_ptr_hash
  77. * @name: the C identifier to use for the new hash table
  78. * @bits: The number of bits to use for hashing
  79. *
  80. * Return: None
  81. */
  82. #define qdf_ptr_hash_declare(name, _bits) \
  83. union { \
  84. struct qdf_ptr_hash ht; \
  85. uint8_t __raw[__qdf_ptr_hash_size(_bits)]; \
  86. } __##name = { .ht = { .bits = _bits, .count = (1 << _bits) } }
  87. /**
  88. * qdf_ptr_hash_ptr() - get a pointer to a declared qdf_ptr_hash
  89. * @name: the C identifier of the declared qdf_ptr_hash
  90. *
  91. * Return: pointer to a qdf_ptr_hash
  92. */
  93. #define qdf_ptr_hash_ptr(name) &__##name.ht
  94. /**
  95. * qdf_ptr_hash_declare_ptr() - declare a pointer to a new qdf_ptr_hash
  96. * @name: the C identifier to use for the pointer to the new qdf_ptr_hash
  97. * @bits: The number of bits to use for hashing
  98. *
  99. * Return: None
  100. */
  101. #define qdf_ptr_hash_declare_ptr(name, bits) \
  102. qdf_ptr_hash_declare(name, bits); \
  103. struct qdf_ptr_hash *name = qdf_ptr_hash_ptr(name)
  104. #define __qdf_ptr_hash_for_each_bucket(ht, bkt) \
  105. for ((bkt) = (ht)->buckets; \
  106. (bkt) < (ht)->buckets + (ht)->count; \
  107. (bkt)++)
  108. /**
  109. * qdf_ptr_hash_init() - initialize a qdf_ptr_hash
  110. * @ht: the hash table to initialize
  111. *
  112. * Return: None
  113. */
  114. static inline void qdf_ptr_hash_init(struct qdf_ptr_hash *ht)
  115. {
  116. struct qdf_ptr_hash_bucket *bucket;
  117. __qdf_ptr_hash_for_each_bucket(ht, bucket)
  118. qdf_slist_init(&bucket->list);
  119. }
  120. /**
  121. * qdf_ptr_hash_deinit() - de-initialize a qdf_ptr_hash
  122. * @ht: the hash table to de-initialize
  123. *
  124. * Return: None
  125. */
  126. static inline void qdf_ptr_hash_deinit(struct qdf_ptr_hash *ht)
  127. {
  128. struct qdf_ptr_hash_bucket *bucket;
  129. __qdf_ptr_hash_for_each_bucket(ht, bucket)
  130. qdf_slist_deinit(&bucket->list);
  131. }
  132. /**
  133. * qdf_ptr_hash_create() - allocate and initialize a qdf_ptr_hash
  134. * @bits: the number of bits to use for hashing
  135. *
  136. * Return: qdf_ptr_hash pointer on succes, NULL on allocation failure
  137. */
  138. static inline struct qdf_ptr_hash *qdf_ptr_hash_create(uint8_t bits)
  139. {
  140. struct qdf_ptr_hash *ht = qdf_mem_malloc(__qdf_ptr_hash_size(bits));
  141. if (!ht)
  142. return NULL;
  143. ht->bits = bits;
  144. ht->count = 1 << bits;
  145. qdf_ptr_hash_init(ht);
  146. return ht;
  147. }
  148. /**
  149. * qdf_ptr_hash_destroy() - de-initialize and de-allocate a qdf_ptr_hash
  150. * @ht: the qdf_ptr_hash to destroy
  151. *
  152. * Return: None
  153. */
  154. static inline void qdf_ptr_hash_destroy(struct qdf_ptr_hash *ht)
  155. {
  156. qdf_ptr_hash_deinit(ht);
  157. qdf_mem_free(ht);
  158. }
  159. /**
  160. * qdf_ptr_hash_empty() - check if a qdf_ptr_hash has any entries
  161. * @ht: the qdf_ptr_hash to check
  162. *
  163. * Return: true if @ht contains no entries
  164. */
  165. static inline bool qdf_ptr_hash_empty(struct qdf_ptr_hash *ht)
  166. {
  167. struct qdf_ptr_hash_bucket *bucket;
  168. __qdf_ptr_hash_for_each_bucket(ht, bucket)
  169. if (!qdf_slist_empty(&bucket->list))
  170. return false;
  171. return true;
  172. }
  173. #ifdef ENABLE_QDF_PTR_HASH_DEBUG
  174. /**
  175. * qdf_ptr_hash_dup_check_in_bucket() - check if a hash_entry is duplicated
  176. in hash_bucket
  177. * @bucket: qdf_ptr_hash_bucket pointer
  178. * cmp_entry: the hash_entry to be checked
  179. *
  180. * if the cmp_entry is found in bucket list, then trigger
  181. * assert to report duplication.
  182. *
  183. * Return: None
  184. */
  185. static inline void qdf_ptr_hash_dup_check_in_bucket(
  186. struct qdf_ptr_hash_bucket *bucket,
  187. struct qdf_ptr_hash_entry *cmp_entry)
  188. {
  189. struct qdf_ptr_hash_entry *tmp_entry;
  190. qdf_slist_for_each(&bucket->list, tmp_entry, node)
  191. qdf_assert_always(tmp_entry != cmp_entry);
  192. }
  193. #else
  194. #define qdf_ptr_hash_dup_check_in_bucket(_bucket, _cmp_entry) /* no op */
  195. #endif
  196. static inline struct qdf_ptr_hash_bucket *
  197. __qdf_ptr_hash_get_bucket(struct qdf_ptr_hash *ht, uintptr_t key)
  198. {
  199. return ht->buckets + __qdf_ptr_hash_key(key, ht->bits);
  200. }
  201. /**
  202. * qdf_ptr_hash_add() - insert an entry into a qdf_ptr_hash
  203. * @ht: the qdf_ptr_hash to insert into
  204. * @key: the pointer to use as an insertion/lookup key
  205. * @item: a pointer to a type that contains a qdf_ptr_hash_entry
  206. * @entry_field: C identifier for the qdf_ptr_hash_entry field in @item
  207. *
  208. * Return: None
  209. */
  210. #define qdf_ptr_hash_add(ht, key, item, entry_field) \
  211. __qdf_ptr_hash_add(ht, (uintptr_t)key, &(item)->entry_field)
  212. static inline void __qdf_ptr_hash_add(struct qdf_ptr_hash *ht, uintptr_t key,
  213. struct qdf_ptr_hash_entry *entry)
  214. {
  215. entry->key = key;
  216. /* check hash_enrty exist or not before push */
  217. qdf_ptr_hash_dup_check_in_bucket(__qdf_ptr_hash_get_bucket(ht, key),
  218. entry);
  219. qdf_slist_push(&__qdf_ptr_hash_get_bucket(ht, key)->list, entry, node);
  220. }
  221. /**
  222. * qdf_ptr_hash_remove() - remove an entry from a qdf_ptr_hash
  223. * @ht: the qdf_ptr_hash to remove from
  224. * @key: the pointer to use as a lookup key
  225. * @cursor: a pointer to a type that contains a qdf_ptr_hash_entry
  226. * @entry_field: C identifier for the qdf_ptr_hash_entry field in @cursor
  227. *
  228. * Return: removed item of type @cursor on success, NULL otherwise
  229. */
  230. #define qdf_ptr_hash_remove(ht, key, cursor, entry_field) ({ \
  231. struct qdf_ptr_hash_entry *_e = \
  232. __qdf_ptr_hash_remove(ht, (uintptr_t)key); \
  233. cursor = _e ? qdf_container_of(_e, typeof(*(cursor)), \
  234. entry_field) : NULL; \
  235. cursor; })
  236. static inline struct qdf_ptr_hash_entry *
  237. __qdf_ptr_hash_remove(struct qdf_ptr_hash *ht, uintptr_t key)
  238. {
  239. struct qdf_ptr_hash_bucket *bucket = __qdf_ptr_hash_get_bucket(ht, key);
  240. struct qdf_ptr_hash_entry *prev;
  241. struct qdf_ptr_hash_entry *entry;
  242. qdf_slist_for_each_del(&bucket->list, prev, entry, node) {
  243. if (entry->key == key) {
  244. qdf_slist_remove(&bucket->list, prev, node);
  245. /* check hash_enrty exist or not after remove */
  246. qdf_ptr_hash_dup_check_in_bucket(bucket, entry);
  247. entry->key = 0;
  248. return entry;
  249. }
  250. }
  251. return NULL;
  252. }
  253. #define __qdf_ptr_hash_for_each_in_bucket(bucket, cursor, entry_field) \
  254. qdf_slist_for_each(&(bucket)->list, cursor, entry_field.node)
  255. /**
  256. * qdf_ptr_hash_for_each() - qdf_ptr_hash item iterator for all items
  257. * @ht: the qdf_ptr_hash to iterate over
  258. * @bucket: qdf_ptr_hash_bucket cursor pointer
  259. * @cursor: a pointer to a type that contains a qdf_ptr_hash_entry
  260. * @entry_field: C identifier for the qdf_ptr_hash_entry field in @cursor
  261. */
  262. #define qdf_ptr_hash_for_each(ht, bucket, cursor, entry_field) \
  263. __qdf_ptr_hash_for_each_bucket(ht, bucket) \
  264. __qdf_ptr_hash_for_each_in_bucket(bucket, cursor, entry_field)
  265. /**
  266. * qdf_ptr_hash_for_each_by_hash() - qdf_ptr_hash item iterator for items which
  267. * hash to the same value as @key
  268. * @ht: the qdf_ptr_hash to iterate over
  269. * @key: the pointer to use as a lookup key
  270. * @cursor: a pointer to a type that contains a qdf_ptr_hash_entry
  271. * @entry_field: C identifier for the qdf_ptr_hash_entry field in @cursor
  272. */
  273. #define qdf_ptr_hash_for_each_by_hash(ht, key, cursor, entry_field) \
  274. __qdf_ptr_hash_for_each_in_bucket( \
  275. __qdf_ptr_hash_get_bucket(ht, (uintptr_t)key), \
  276. cursor, entry_field)
  277. /**
  278. * qdf_ptr_hash_for_each_by_key() - qdf_ptr_hash item iterator for items whose
  279. * keys equal @key
  280. * @ht: the qdf_ptr_hash to iterate over
  281. * @key: the pointer to use as a lookup key
  282. * @cursor: a pointer to a type that contains a qdf_ptr_hash_entry
  283. * @entry_field: C identifier for the qdf_ptr_hash_entry field in @cursor
  284. */
  285. #define qdf_ptr_hash_for_each_by_key(ht, _key, cursor, entry_field) \
  286. qdf_ptr_hash_for_each_by_hash(ht, _key, cursor, entry_field) \
  287. if ((cursor)->entry_field.key == (uintptr_t)_key)
  288. /**
  289. * qdf_ptr_hash_get() - get the first item whose key matches @key
  290. * @ht: the qdf_ptr_hash to look in
  291. * @key: the pointer to use as a lookup key
  292. * @cursor: a pointer to a type that contains a qdf_ptr_hash_entry
  293. * @entry_field: C identifier for the qdf_ptr_hash_entry field in @cursor
  294. *
  295. * Return: first item matching @key of type @cursor on success, NULL otherwise
  296. */
  297. #define qdf_ptr_hash_get(ht, key, cursor, entry_field) ({ \
  298. cursor = NULL; \
  299. qdf_ptr_hash_for_each_by_key(ht, key, cursor, entry_field) \
  300. break; \
  301. cursor; })
  302. #endif /* __QDF_PTR_HASH_H */