dlist.h 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546
  1. /*
  2. * Copyright (c) 2013-2015 Wind River Systems, Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Doubly-linked list implementation
  9. *
  10. * Doubly-linked list implementation using inline macros/functions.
  11. * This API is not thread safe, and thus if a list is used across threads,
  12. * calls to functions must be protected with synchronization primitives.
  13. *
  14. * The lists are expected to be initialized such that both the head and tail
  15. * pointers point to the list itself. Initializing the lists in such a fashion
  16. * simplifies the adding and removing of nodes to/from the list.
  17. */
  18. #ifndef ZEPHYR_INCLUDE_SYS_DLIST_H_
  19. #define ZEPHYR_INCLUDE_SYS_DLIST_H_
  20. #include <stddef.h>
  21. #include <stdbool.h>
  22. #include <toolchain.h>
  23. #ifdef __cplusplus
  24. extern "C" {
  25. #endif
  26. /**
  27. * @defgroup doubly-linked-list_apis Doubly-linked list
  28. * @ingroup datastructure_apis
  29. * @{
  30. */
  31. struct _dnode {
  32. union {
  33. struct _dnode *head; /* ptr to head of list (sys_dlist_t) */
  34. struct _dnode *next; /* ptr to next node (sys_dnode_t) */
  35. };
  36. union {
  37. struct _dnode *tail; /* ptr to tail of list (sys_dlist_t) */
  38. struct _dnode *prev; /* ptr to previous node (sys_dnode_t) */
  39. };
  40. };
  41. typedef struct _dnode sys_dlist_t;
  42. typedef struct _dnode sys_dnode_t;
  43. /**
  44. * @brief Provide the primitive to iterate on a list
  45. * Note: the loop is unsafe and thus __dn should not be removed
  46. *
  47. * User _MUST_ add the loop statement curly braces enclosing its own code:
  48. *
  49. * SYS_DLIST_FOR_EACH_NODE(l, n) {
  50. * <user code>
  51. * }
  52. *
  53. * This and other SYS_DLIST_*() macros are not thread safe.
  54. *
  55. * @param __dl A pointer on a sys_dlist_t to iterate on
  56. * @param __dn A sys_dnode_t pointer to peek each node of the list
  57. */
  58. #define SYS_DLIST_FOR_EACH_NODE(__dl, __dn) \
  59. for (__dn = sys_dlist_peek_head(__dl); __dn != NULL; \
  60. __dn = sys_dlist_peek_next(__dl, __dn))
  61. /**
  62. * @brief Provide the primitive to iterate on a list, from a node in the list
  63. * Note: the loop is unsafe and thus __dn should not be removed
  64. *
  65. * User _MUST_ add the loop statement curly braces enclosing its own code:
  66. *
  67. * SYS_DLIST_ITERATE_FROM_NODE(l, n) {
  68. * <user code>
  69. * }
  70. *
  71. * Like SYS_DLIST_FOR_EACH_NODE(), but __dn already contains a node in the list
  72. * where to start searching for the next entry from. If NULL, it starts from
  73. * the head.
  74. *
  75. * This and other SYS_DLIST_*() macros are not thread safe.
  76. *
  77. * @param __dl A pointer on a sys_dlist_t to iterate on
  78. * @param __dn A sys_dnode_t pointer to peek each node of the list;
  79. * it contains the starting node, or NULL to start from the head
  80. */
  81. #define SYS_DLIST_ITERATE_FROM_NODE(__dl, __dn) \
  82. for (__dn = __dn ? sys_dlist_peek_next_no_check(__dl, __dn) \
  83. : sys_dlist_peek_head(__dl); \
  84. __dn != NULL; \
  85. __dn = sys_dlist_peek_next(__dl, __dn))
  86. /**
  87. * @brief Provide the primitive to safely iterate on a list
  88. * Note: __dn can be removed, it will not break the loop.
  89. *
  90. * User _MUST_ add the loop statement curly braces enclosing its own code:
  91. *
  92. * SYS_DLIST_FOR_EACH_NODE_SAFE(l, n, s) {
  93. * <user code>
  94. * }
  95. *
  96. * This and other SYS_DLIST_*() macros are not thread safe.
  97. *
  98. * @param __dl A pointer on a sys_dlist_t to iterate on
  99. * @param __dn A sys_dnode_t pointer to peek each node of the list
  100. * @param __dns A sys_dnode_t pointer for the loop to run safely
  101. */
  102. #define SYS_DLIST_FOR_EACH_NODE_SAFE(__dl, __dn, __dns) \
  103. for (__dn = sys_dlist_peek_head(__dl), \
  104. __dns = sys_dlist_peek_next(__dl, __dn); \
  105. __dn != NULL; __dn = __dns, \
  106. __dns = sys_dlist_peek_next(__dl, __dn))
  107. /*
  108. * @brief Provide the primitive to resolve the container of a list node
  109. * Note: it is safe to use with NULL pointer nodes
  110. *
  111. * @param __dn A pointer on a sys_dnode_t to get its container
  112. * @param __cn Container struct type pointer
  113. * @param __n The field name of sys_dnode_t within the container struct
  114. */
  115. #define SYS_DLIST_CONTAINER(__dn, __cn, __n) \
  116. ((__dn != NULL) ? CONTAINER_OF(__dn, __typeof__(*__cn), __n) : NULL)
  117. /*
  118. * @brief Provide the primitive to peek container of the list head
  119. *
  120. * @param __dl A pointer on a sys_dlist_t to peek
  121. * @param __cn Container struct type pointer
  122. * @param __n The field name of sys_dnode_t within the container struct
  123. */
  124. #define SYS_DLIST_PEEK_HEAD_CONTAINER(__dl, __cn, __n) \
  125. SYS_DLIST_CONTAINER(sys_dlist_peek_head(__dl), __cn, __n)
  126. /*
  127. * @brief Provide the primitive to peek the next container
  128. *
  129. * @param __dl A pointer on a sys_dlist_t to peek
  130. * @param __cn Container struct type pointer
  131. * @param __n The field name of sys_dnode_t within the container struct
  132. */
  133. #define SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n) \
  134. ((__cn != NULL) ? \
  135. SYS_DLIST_CONTAINER(sys_dlist_peek_next(__dl, &(__cn->__n)), \
  136. __cn, __n) : NULL)
  137. /**
  138. * @brief Provide the primitive to iterate on a list under a container
  139. * Note: the loop is unsafe and thus __cn should not be detached
  140. *
  141. * User _MUST_ add the loop statement curly braces enclosing its own code:
  142. *
  143. * SYS_DLIST_FOR_EACH_CONTAINER(l, c, n) {
  144. * <user code>
  145. * }
  146. *
  147. * @param __dl A pointer on a sys_dlist_t to iterate on
  148. * @param __cn A pointer to peek each entry of the list
  149. * @param __n The field name of sys_dnode_t within the container struct
  150. */
  151. #define SYS_DLIST_FOR_EACH_CONTAINER(__dl, __cn, __n) \
  152. for (__cn = SYS_DLIST_PEEK_HEAD_CONTAINER(__dl, __cn, __n); \
  153. __cn != NULL; \
  154. __cn = SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n))
  155. /**
  156. * @brief Provide the primitive to safely iterate on a list under a container
  157. * Note: __cn can be detached, it will not break the loop.
  158. *
  159. * User _MUST_ add the loop statement curly braces enclosing its own code:
  160. *
  161. * SYS_DLIST_FOR_EACH_CONTAINER_SAFE(l, c, cn, n) {
  162. * <user code>
  163. * }
  164. *
  165. * @param __dl A pointer on a sys_dlist_t to iterate on
  166. * @param __cn A pointer to peek each entry of the list
  167. * @param __cns A pointer for the loop to run safely
  168. * @param __n The field name of sys_dnode_t within the container struct
  169. */
  170. #define SYS_DLIST_FOR_EACH_CONTAINER_SAFE(__dl, __cn, __cns, __n) \
  171. for (__cn = SYS_DLIST_PEEK_HEAD_CONTAINER(__dl, __cn, __n), \
  172. __cns = SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n); \
  173. __cn != NULL; __cn = __cns, \
  174. __cns = SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n))
  175. /**
  176. * @brief initialize list to its empty state
  177. *
  178. * @param list the doubly-linked list
  179. *
  180. * @return N/A
  181. */
  182. static inline void sys_dlist_init(sys_dlist_t *list)
  183. {
  184. list->head = (sys_dnode_t *)list;
  185. list->tail = (sys_dnode_t *)list;
  186. }
  187. #define SYS_DLIST_STATIC_INIT(ptr_to_list) { {(ptr_to_list)}, {(ptr_to_list)} }
  188. /**
  189. * @brief initialize node to its state when not in a list
  190. *
  191. * @param node the node
  192. *
  193. * @return N/A
  194. */
  195. static inline void sys_dnode_init(sys_dnode_t *node)
  196. {
  197. node->next = NULL;
  198. node->prev = NULL;
  199. }
  200. /**
  201. * @brief check if a node is a member of any list
  202. *
  203. * @param node the node
  204. *
  205. * @return true if node is linked into a list, false if it is not
  206. */
  207. static inline bool sys_dnode_is_linked(const sys_dnode_t *node)
  208. {
  209. return node->next != NULL;
  210. }
  211. /**
  212. * @brief check if a node is the list's head
  213. *
  214. * @param list the doubly-linked list to operate on
  215. * @param node the node to check
  216. *
  217. * @return true if node is the head, false otherwise
  218. */
  219. static inline bool sys_dlist_is_head(sys_dlist_t *list, sys_dnode_t *node)
  220. {
  221. return list->head == node;
  222. }
  223. /**
  224. * @brief check if a node is the list's tail
  225. *
  226. * @param list the doubly-linked list to operate on
  227. * @param node the node to check
  228. *
  229. * @return true if node is the tail, false otherwise
  230. */
  231. static inline bool sys_dlist_is_tail(sys_dlist_t *list, sys_dnode_t *node)
  232. {
  233. return list->tail == node;
  234. }
  235. /**
  236. * @brief check if the list is empty
  237. *
  238. * @param list the doubly-linked list to operate on
  239. *
  240. * @return true if empty, false otherwise
  241. */
  242. static inline bool sys_dlist_is_empty(sys_dlist_t *list)
  243. {
  244. return list->head == list;
  245. }
  246. /**
  247. * @brief check if more than one node present
  248. *
  249. * This and other sys_dlist_*() functions are not thread safe.
  250. *
  251. * @param list the doubly-linked list to operate on
  252. *
  253. * @return true if multiple nodes, false otherwise
  254. */
  255. static inline bool sys_dlist_has_multiple_nodes(sys_dlist_t *list)
  256. {
  257. return list->head != list->tail;
  258. }
  259. /**
  260. * @brief get a reference to the head item in the list
  261. *
  262. * @param list the doubly-linked list to operate on
  263. *
  264. * @return a pointer to the head element, NULL if list is empty
  265. */
  266. static inline sys_dnode_t *sys_dlist_peek_head(sys_dlist_t *list)
  267. {
  268. return sys_dlist_is_empty(list) ? NULL : list->head;
  269. }
  270. /**
  271. * @brief get a reference to the head item in the list
  272. *
  273. * The list must be known to be non-empty.
  274. *
  275. * @param list the doubly-linked list to operate on
  276. *
  277. * @return a pointer to the head element
  278. */
  279. static inline sys_dnode_t *sys_dlist_peek_head_not_empty(sys_dlist_t *list)
  280. {
  281. return list->head;
  282. }
  283. /**
  284. * @brief get a reference to the next item in the list, node is not NULL
  285. *
  286. * Faster than sys_dlist_peek_next() if node is known not to be NULL.
  287. *
  288. * @param list the doubly-linked list to operate on
  289. * @param node the node from which to get the next element in the list
  290. *
  291. * @return a pointer to the next element from a node, NULL if node is the tail
  292. */
  293. static inline sys_dnode_t *sys_dlist_peek_next_no_check(sys_dlist_t *list,
  294. sys_dnode_t *node)
  295. {
  296. return (node == list->tail) ? NULL : node->next;
  297. }
  298. /**
  299. * @brief get a reference to the next item in the list
  300. *
  301. * @param list the doubly-linked list to operate on
  302. * @param node the node from which to get the next element in the list
  303. *
  304. * @return a pointer to the next element from a node, NULL if node is the tail
  305. * or NULL (when node comes from reading the head of an empty list).
  306. */
  307. static inline sys_dnode_t *sys_dlist_peek_next(sys_dlist_t *list,
  308. sys_dnode_t *node)
  309. {
  310. return (node != NULL) ? sys_dlist_peek_next_no_check(list, node) : NULL;
  311. }
  312. /**
  313. * @brief get a reference to the previous item in the list, node is not NULL
  314. *
  315. * Faster than sys_dlist_peek_prev() if node is known not to be NULL.
  316. *
  317. * @param list the doubly-linked list to operate on
  318. * @param node the node from which to get the previous element in the list
  319. *
  320. * @return a pointer to the previous element from a node, NULL if node is the
  321. * tail
  322. */
  323. static inline sys_dnode_t *sys_dlist_peek_prev_no_check(sys_dlist_t *list,
  324. sys_dnode_t *node)
  325. {
  326. return (node == list->head) ? NULL : node->prev;
  327. }
  328. /**
  329. * @brief get a reference to the previous item in the list
  330. *
  331. * @param list the doubly-linked list to operate on
  332. * @param node the node from which to get the previous element in the list
  333. *
  334. * @return a pointer to the previous element from a node, NULL if node is the
  335. * tail or NULL (when node comes from reading the head of an empty
  336. * list).
  337. */
  338. static inline sys_dnode_t *sys_dlist_peek_prev(sys_dlist_t *list,
  339. sys_dnode_t *node)
  340. {
  341. return (node != NULL) ? sys_dlist_peek_prev_no_check(list, node) : NULL;
  342. }
  343. /**
  344. * @brief get a reference to the tail item in the list
  345. *
  346. * @param list the doubly-linked list to operate on
  347. *
  348. * @return a pointer to the tail element, NULL if list is empty
  349. */
  350. static inline sys_dnode_t *sys_dlist_peek_tail(sys_dlist_t *list)
  351. {
  352. return sys_dlist_is_empty(list) ? NULL : list->tail;
  353. }
  354. /**
  355. * @brief add node to tail of list
  356. *
  357. * This and other sys_dlist_*() functions are not thread safe.
  358. *
  359. * @param list the doubly-linked list to operate on
  360. * @param node the element to append
  361. *
  362. * @return N/A
  363. */
  364. static inline void sys_dlist_append(sys_dlist_t *list, sys_dnode_t *node)
  365. {
  366. sys_dnode_t *const tail = list->tail;
  367. node->next = list;
  368. node->prev = tail;
  369. tail->next = node;
  370. list->tail = node;
  371. }
  372. /**
  373. * @brief add node to head of list
  374. *
  375. * This and other sys_dlist_*() functions are not thread safe.
  376. *
  377. * @param list the doubly-linked list to operate on
  378. * @param node the element to append
  379. *
  380. * @return N/A
  381. */
  382. static inline void sys_dlist_prepend(sys_dlist_t *list, sys_dnode_t *node)
  383. {
  384. sys_dnode_t *const head = list->head;
  385. node->next = head;
  386. node->prev = list;
  387. head->prev = node;
  388. list->head = node;
  389. }
  390. /**
  391. * @brief Insert a node into a list
  392. *
  393. * Insert a node before a specified node in a dlist.
  394. *
  395. * @param successor the position before which "node" will be inserted
  396. * @param node the element to insert
  397. */
  398. static inline void sys_dlist_insert(sys_dnode_t *successor, sys_dnode_t *node)
  399. {
  400. sys_dnode_t *const prev = successor->prev;
  401. node->prev = prev;
  402. node->next = successor;
  403. prev->next = node;
  404. successor->prev = node;
  405. }
  406. /**
  407. * @brief insert node at position
  408. *
  409. * Insert a node in a location depending on a external condition. The cond()
  410. * function checks if the node is to be inserted _before_ the current node
  411. * against which it is checked.
  412. * This and other sys_dlist_*() functions are not thread safe.
  413. *
  414. * @param list the doubly-linked list to operate on
  415. * @param node the element to insert
  416. * @param cond a function that determines if the current node is the correct
  417. * insert point
  418. * @param data parameter to cond()
  419. *
  420. * @return N/A
  421. */
  422. static inline void sys_dlist_insert_at(sys_dlist_t *list, sys_dnode_t *node,
  423. int (*cond)(sys_dnode_t *node, void *data), void *data)
  424. {
  425. if (sys_dlist_is_empty(list)) {
  426. sys_dlist_append(list, node);
  427. } else {
  428. sys_dnode_t *pos = sys_dlist_peek_head(list);
  429. while ((pos != NULL) && (cond(pos, data) == 0)) {
  430. pos = sys_dlist_peek_next(list, pos);
  431. }
  432. if (pos != NULL) {
  433. sys_dlist_insert(pos, node);
  434. } else {
  435. sys_dlist_append(list, node);
  436. }
  437. }
  438. }
  439. /**
  440. * @brief remove a specific node from a list
  441. *
  442. * The list is implicit from the node. The node must be part of a list.
  443. * This and other sys_dlist_*() functions are not thread safe.
  444. *
  445. * @param node the node to remove
  446. *
  447. * @return N/A
  448. */
  449. static inline void sys_dlist_remove(sys_dnode_t *node)
  450. {
  451. sys_dnode_t *const prev = node->prev;
  452. sys_dnode_t *const next = node->next;
  453. prev->next = next;
  454. next->prev = prev;
  455. sys_dnode_init(node);
  456. }
  457. /**
  458. * @brief get the first node in a list
  459. *
  460. * This and other sys_dlist_*() functions are not thread safe.
  461. *
  462. * @param list the doubly-linked list to operate on
  463. *
  464. * @return the first node in the list, NULL if list is empty
  465. */
  466. static inline sys_dnode_t *sys_dlist_get(sys_dlist_t *list)
  467. {
  468. sys_dnode_t *node = NULL;
  469. if (!sys_dlist_is_empty(list)) {
  470. node = list->head;
  471. sys_dlist_remove(node);
  472. }
  473. return node;
  474. }
  475. /** @} */
  476. #ifdef __cplusplus
  477. }
  478. #endif
  479. #endif /* ZEPHYR_INCLUDE_SYS_DLIST_H_ */