1 /* 2 * Copyright (c) 2016 Intel Corporation 3 * 4 * SPDX-License-Identifier: Apache-2.0 5 */ 6 7 #ifndef ZEPHYR_INCLUDE_SYS_LIST_GEN_H_ 8 #define ZEPHYR_INCLUDE_SYS_LIST_GEN_H_ 9 10 #include <stddef.h> 11 #include <stdbool.h> 12 #include <zephyr/sys/util.h> 13 14 #define Z_GENLIST_FOR_EACH_NODE(__lname, __l, __sn) \ 15 for ((__sn) = sys_ ## __lname ## _peek_head(__l); (__sn) != NULL; \ 16 (__sn) = sys_ ## __lname ## _peek_next(__sn)) 17 18 19 #define Z_GENLIST_ITERATE_FROM_NODE(__lname, __l, __sn) \ 20 for ((__sn) = (__sn) ? sys_ ## __lname ## _peek_next_no_check(__sn) \ 21 : sys_ ## __lname ## _peek_head(__l); \ 22 (__sn) != NULL; \ 23 (__sn) = sys_ ## __lname ## _peek_next(__sn)) 24 25 #define Z_GENLIST_FOR_EACH_NODE_SAFE(__lname, __l, __sn, __sns) \ 26 for ((__sn) = sys_ ## __lname ## _peek_head(__l), \ 27 (__sns) = sys_ ## __lname ## _peek_next(__sn); \ 28 (__sn) != NULL ; (__sn) = (__sns), \ 29 (__sns) = sys_ ## __lname ## _peek_next(__sn)) 30 31 #define Z_GENLIST_CONTAINER(__ln, __cn, __n) \ 32 ((__ln) ? CONTAINER_OF((__ln), __typeof__(*(__cn)), __n) : NULL) 33 34 #define Z_GENLIST_PEEK_HEAD_CONTAINER(__lname, __l, __cn, __n) \ 35 Z_GENLIST_CONTAINER(sys_ ## __lname ## _peek_head(__l), __cn, __n) 36 37 #define Z_GENLIST_PEEK_TAIL_CONTAINER(__lname, __l, __cn, __n) \ 38 Z_GENLIST_CONTAINER(sys_ ## __lname ## _peek_tail(__l), __cn, __n) 39 40 #define Z_GENLIST_PEEK_NEXT_CONTAINER(__lname, __cn, __n) \ 41 ((__cn) ? Z_GENLIST_CONTAINER( \ 42 sys_ ## __lname ## _peek_next(&((__cn)->__n)), \ 43 __cn, __n) : NULL) 44 45 #define Z_GENLIST_FOR_EACH_CONTAINER(__lname, __l, __cn, __n) \ 46 for ((__cn) = Z_GENLIST_PEEK_HEAD_CONTAINER(__lname, __l, __cn, \ 47 __n); \ 48 (__cn) != NULL; \ 49 (__cn) = Z_GENLIST_PEEK_NEXT_CONTAINER(__lname, __cn, __n)) 50 51 #define Z_GENLIST_FOR_EACH_CONTAINER_SAFE(__lname, __l, __cn, __cns, __n) \ 52 for ((__cn) = Z_GENLIST_PEEK_HEAD_CONTAINER(__lname, __l, __cn, __n), \ 53 (__cns) = Z_GENLIST_PEEK_NEXT_CONTAINER(__lname, __cn, __n); \ 54 (__cn) != NULL; (__cn) = (__cns), \ 55 (__cns) = Z_GENLIST_PEEK_NEXT_CONTAINER(__lname, __cn, __n)) 56 57 #define Z_GENLIST_IS_EMPTY(__lname) \ 58 static inline bool \ 59 sys_ ## __lname ## _is_empty(sys_ ## __lname ## _t *list) \ 60 { \ 61 return (sys_ ## __lname ## _peek_head(list) == NULL); \ 62 } 63 64 #define Z_GENLIST_PEEK_NEXT_NO_CHECK(__lname, __nname) \ 65 static inline sys_ ## __nname ## _t * \ 66 sys_ ## __lname ## _peek_next_no_check(sys_ ## __nname ## _t *node) \ 67 { \ 68 return z_ ## __nname ## _next_peek(node); \ 69 } 70 71 #define Z_GENLIST_PEEK_NEXT(__lname, __nname) \ 72 static inline sys_ ## __nname ## _t * \ 73 sys_ ## __lname ## _peek_next(sys_ ## __nname ## _t *node) \ 74 { \ 75 return (node != NULL) ? \ 76 sys_ ## __lname ## _peek_next_no_check(node) : \ 77 NULL; \ 78 } 79 80 #define Z_GENLIST_PREPEND(__lname, __nname) \ 81 static inline void \ 82 sys_ ## __lname ## _prepend(sys_ ## __lname ## _t *list, \ 83 sys_ ## __nname ## _t *node) \ 84 { \ 85 z_ ## __nname ## _next_set(node, \ 86 sys_ ## __lname ## _peek_head(list)); \ 87 z_ ## __lname ## _head_set(list, node); \ 88 \ 89 if (sys_ ## __lname ## _peek_tail(list) == NULL) { \ 90 z_ ## __lname ## _tail_set(list, \ 91 sys_ ## __lname ## _peek_head(list)); \ 92 } \ 93 } 94 95 #define Z_GENLIST_APPEND(__lname, __nname) \ 96 static inline void \ 97 sys_ ## __lname ## _append(sys_ ## __lname ## _t *list, \ 98 sys_ ## __nname ## _t *node) \ 99 { \ 100 z_ ## __nname ## _next_set(node, NULL); \ 101 \ 102 if (sys_ ## __lname ## _peek_tail(list) == NULL) { \ 103 z_ ## __lname ## _tail_set(list, node); \ 104 z_ ## __lname ## _head_set(list, node); \ 105 } else { \ 106 z_ ## __nname ## _next_set( \ 107 sys_ ## __lname ## _peek_tail(list), \ 108 node); \ 109 z_ ## __lname ## _tail_set(list, node); \ 110 } \ 111 } 112 113 #define Z_GENLIST_APPEND_LIST(__lname, __nname) \ 114 static inline void \ 115 sys_ ## __lname ## _append_list(sys_ ## __lname ## _t *list, \ 116 void *head, void *tail) \ 117 { \ 118 if (head != NULL && tail != NULL) { \ 119 if (sys_ ## __lname ## _peek_tail(list) == NULL) { \ 120 z_ ## __lname ## _head_set(list, \ 121 (sys_ ## __nname ## _t *)head); \ 122 } else { \ 123 z_ ## __nname ## _next_set( \ 124 sys_ ## __lname ## _peek_tail(list), \ 125 (sys_ ## __nname ## _t *)head); \ 126 } \ 127 z_ ## __lname ## _tail_set(list, \ 128 (sys_ ## __nname ## _t *)tail); \ 129 } \ 130 } 131 132 #define Z_GENLIST_MERGE_LIST(__lname, __nname) \ 133 static inline void \ 134 sys_ ## __lname ## _merge_ ## __lname ( \ 135 sys_ ## __lname ## _t *list, \ 136 sys_ ## __lname ## _t *list_to_append) \ 137 { \ 138 sys_ ## __nname ## _t *head, *tail; \ 139 head = sys_ ## __lname ## _peek_head(list_to_append); \ 140 tail = sys_ ## __lname ## _peek_tail(list_to_append); \ 141 sys_ ## __lname ## _append_list(list, head, tail); \ 142 sys_ ## __lname ## _init(list_to_append); \ 143 } 144 145 #define Z_GENLIST_INSERT(__lname, __nname) \ 146 static inline void \ 147 sys_ ## __lname ## _insert(sys_ ## __lname ## _t *list, \ 148 sys_ ## __nname ## _t *prev, \ 149 sys_ ## __nname ## _t *node) \ 150 { \ 151 if (prev == NULL) { \ 152 sys_ ## __lname ## _prepend(list, node); \ 153 } else if (z_ ## __nname ## _next_peek(prev) == NULL) { \ 154 sys_ ## __lname ## _append(list, node); \ 155 } else { \ 156 z_ ## __nname ## _next_set(node, \ 157 z_ ## __nname ## _next_peek(prev)); \ 158 z_ ## __nname ## _next_set(prev, node); \ 159 } \ 160 } 161 162 #define Z_GENLIST_GET_NOT_EMPTY(__lname, __nname) \ 163 static inline sys_ ## __nname ## _t * \ 164 sys_ ## __lname ## _get_not_empty(sys_ ## __lname ## _t *list) \ 165 { \ 166 sys_ ## __nname ## _t *node = \ 167 sys_ ## __lname ## _peek_head(list); \ 168 \ 169 z_ ## __lname ## _head_set(list, \ 170 z_ ## __nname ## _next_peek(node)); \ 171 if (sys_ ## __lname ## _peek_tail(list) == node) { \ 172 z_ ## __lname ## _tail_set(list, \ 173 sys_ ## __lname ## _peek_head(list)); \ 174 } \ 175 \ 176 return node; \ 177 } 178 179 #define Z_GENLIST_GET(__lname, __nname) \ 180 static inline sys_ ## __nname ## _t * \ 181 sys_ ## __lname ## _get(sys_ ## __lname ## _t *list) \ 182 { \ 183 return sys_ ## __lname ## _is_empty(list) ? NULL : \ 184 sys_ ## __lname ## _get_not_empty(list); \ 185 } 186 187 #define Z_GENLIST_REMOVE(__lname, __nname) \ 188 static inline void \ 189 sys_ ## __lname ## _remove(sys_ ## __lname ## _t *list, \ 190 sys_ ## __nname ## _t *prev_node, \ 191 sys_ ## __nname ## _t *node) \ 192 { \ 193 if (prev_node == NULL) { \ 194 z_ ## __lname ## _head_set(list, \ 195 z_ ## __nname ## _next_peek(node)); \ 196 \ 197 /* Was node also the tail? */ \ 198 if (sys_ ## __lname ## _peek_tail(list) == node) { \ 199 z_ ## __lname ## _tail_set(list, \ 200 sys_ ## __lname ## _peek_head(list)); \ 201 } \ 202 } else { \ 203 z_ ## __nname ## _next_set(prev_node, \ 204 z_ ## __nname ## _next_peek(node)); \ 205 \ 206 /* Was node the tail? */ \ 207 if (sys_ ## __lname ## _peek_tail(list) == node) { \ 208 z_ ## __lname ## _tail_set(list, \ 209 prev_node); \ 210 } \ 211 } \ 212 \ 213 z_ ## __nname ## _next_set(node, NULL); \ 214 } 215 216 #define Z_GENLIST_FIND_AND_REMOVE(__lname, __nname) \ 217 static inline bool \ 218 sys_ ## __lname ## _find_and_remove(sys_ ## __lname ## _t *list, \ 219 sys_ ## __nname ## _t *node) \ 220 { \ 221 sys_ ## __nname ## _t *prev = NULL; \ 222 sys_ ## __nname ## _t *test; \ 223 \ 224 Z_GENLIST_FOR_EACH_NODE(__lname, list, test) { \ 225 if (test == node) { \ 226 sys_ ## __lname ## _remove(list, prev, \ 227 node); \ 228 return true; \ 229 } \ 230 \ 231 prev = test; \ 232 } \ 233 \ 234 return false; \ 235 } 236 237 #define Z_GENLIST_FIND(__lname, __nname) \ 238 static inline bool sys_##__lname##_find( \ 239 sys_##__lname##_t *list, sys_##__nname##_t *node, sys_##__nname##_t **prev) \ 240 { \ 241 sys_##__nname##_t *current = NULL; \ 242 sys_##__nname##_t *previous = NULL; \ 243 \ 244 Z_GENLIST_FOR_EACH_NODE(__lname, list, current) { \ 245 if (current == node) { \ 246 if (prev != NULL) { \ 247 *prev = previous; \ 248 } \ 249 return true; \ 250 } \ 251 \ 252 previous = current; \ 253 } \ 254 \ 255 if (prev != NULL) { \ 256 *prev = previous; \ 257 } \ 258 \ 259 return false; \ 260 } 261 262 #define Z_GENLIST_LEN(__lname, __nname) \ 263 static inline size_t sys_##__lname##_len(sys_##__lname##_t * list) \ 264 { \ 265 size_t len = 0; \ 266 static sys_##__nname##_t * node; \ 267 Z_GENLIST_FOR_EACH_NODE(__lname, list, node) { \ 268 len++; \ 269 } \ 270 return len; \ 271 } 272 273 #endif /* ZEPHYR_INCLUDE_SYS_LIST_GEN_H_ */ 274