PipeWire  0.3.33
list.h
Go to the documentation of this file.
1 /* Simple Plugin API
2  *
3  * Copyright © 2018 Wim Taymans
4  *
5  * Permission is hereby granted, free of charge, to any person obtaining a
6  * copy of this software and associated documentation files (the "Software"),
7  * to deal in the Software without restriction, including without limitation
8  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9  * and/or sell copies of the Software, and to permit persons to whom the
10  * Software is furnished to do so, subject to the following conditions:
11  *
12  * The above copyright notice and this permission notice (including the next
13  * paragraph) shall be included in all copies or substantial portions of the
14  * Software.
15  *
16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22  * DEALINGS IN THE SOFTWARE.
23  */
24 
25 #ifndef SPA_LIST_H
26 #define SPA_LIST_H
27 
28 #ifdef __cplusplus
29 extern "C" {
30 #endif
31 
37 struct spa_list {
38  struct spa_list *next;
39  struct spa_list *prev;
40 };
41 
42 #define SPA_LIST_INIT(list) (struct spa_list){ list, list }
43 
44 /* static */ inline void spa_list_init(struct spa_list *list)
45 {
46  *list = SPA_LIST_INIT(list);
47 }
48 
49 #define spa_list_is_empty(l) ((l)->next == (l))
50 
51 /* static */ inline void spa_list_insert(struct spa_list *list, struct spa_list *elem)
52 {
53  elem->prev = list;
54  elem->next = list->next;
55  list->next = elem;
56  elem->next->prev = elem;
57 }
58 
59 /* static */ inline void spa_list_insert_list(struct spa_list *list, struct spa_list *other)
60 {
61  if (spa_list_is_empty(other))
62  return;
63  other->next->prev = list;
64  other->prev->next = list->next;
65  list->next->prev = other->prev;
66  list->next = other->next;
67 }
68 
69 /* static */ inline void spa_list_remove(struct spa_list *elem)
70 {
71  elem->prev->next = elem->next;
72  elem->next->prev = elem->prev;
73 }
74 
75 #define spa_list_first(head, type, member) \
76  SPA_CONTAINER_OF((head)->next, type, member)
77 
78 #define spa_list_last(head, type, member) \
79  SPA_CONTAINER_OF((head)->prev, type, member)
80 
81 #define spa_list_append(list, item) \
82  spa_list_insert((list)->prev, item)
83 
84 #define spa_list_prepend(list, item) \
85  spa_list_insert(list, item)
86 
87 #define spa_list_is_end(pos, head, member) \
88  (&(pos)->member == (head))
89 
90 #define spa_list_next(pos, member) \
91  SPA_CONTAINER_OF((pos)->member.next, __typeof__(*pos), member)
92 
93 #define spa_list_prev(pos, member) \
94  SPA_CONTAINER_OF((pos)->member.prev, __typeof__(*pos), member)
95 
96 #define spa_list_consume(pos, head, member) \
97  for (pos = spa_list_first(head, __typeof__(*pos), member); \
98  !spa_list_is_empty(head); \
99  pos = spa_list_first(head, __typeof__(*pos), member))
100 
101 #define spa_list_for_each_next(pos, head, curr, member) \
102  for (pos = spa_list_first(curr, __typeof__(*pos), member); \
103  !spa_list_is_end(pos, head, member); \
104  pos = spa_list_next(pos, member))
105 
106 #define spa_list_for_each_prev(pos, head, curr, member) \
107  for (pos = spa_list_last(curr, __typeof__(*pos), member); \
108  !spa_list_is_end(pos, head, member); \
109  pos = spa_list_prev(pos, member))
110 
111 #define spa_list_for_each(pos, head, member) \
112  spa_list_for_each_next(pos, head, head, member)
113 
114 #define spa_list_for_each_reverse(pos, head, member) \
115  spa_list_for_each_prev(pos, head, head, member)
116 
117 #define spa_list_for_each_safe_next(pos, tmp, head, curr, member) \
118  for (pos = spa_list_first(curr, __typeof__(*pos), member); \
119  tmp = spa_list_next(pos, member), \
120  !spa_list_is_end(pos, head, member); \
121  pos = tmp)
122 
123 #define spa_list_for_each_safe_prev(pos, tmp, head, curr, member) \
124  for (pos = spa_list_last(curr, __typeof__(*pos), member); \
125  tmp = spa_list_prev(pos, member), \
126  !spa_list_is_end(pos, head, member); \
127  pos = tmp)
128 
129 #define spa_list_for_each_safe(pos, tmp, head, member) \
130  spa_list_for_each_safe_next(pos, tmp, head, head, member)
131 
132 #define spa_list_for_each_safe_reverse(pos, tmp, head, member) \
133  spa_list_for_each_safe_prev(pos, tmp, head, head, member)
134 
135 #define spa_list_cursor_start(cursor, head, member) \
136  spa_list_prepend(head, &(cursor).member)
137 
138 #define spa_list_for_each_cursor(pos, cursor, head, member) \
139  for(pos = spa_list_first(&(cursor).member, __typeof__(*(pos)), member); \
140  spa_list_remove(&(pos)->member), \
141  spa_list_append(&(cursor).member, &(pos)->member), \
142  !spa_list_is_end(pos, head, member); \
143  pos = spa_list_next(&cursor, member))
144 
145 #define spa_list_cursor_end(cursor, member) \
146  spa_list_remove(&(cursor).member)
147 
152 #ifdef __cplusplus
153 } /* extern "C" */
154 #endif
155 
156 #endif /* SPA_LIST_H */
spa_list_insert_list
void spa_list_insert_list(struct spa_list *list, struct spa_list *other)
Definition: list.h:59
spa_list
Definition: list.h:37
SPA_LIST_INIT
#define SPA_LIST_INIT(list)
Definition: list.h:42
spa_list_remove
void spa_list_remove(struct spa_list *elem)
Definition: list.h:69
spa_list::next
struct spa_list * next
Definition: list.h:38
spa_list_init
void spa_list_init(struct spa_list *list)
Definition: list.h:44
spa_list_insert
void spa_list_insert(struct spa_list *list, struct spa_list *elem)
Definition: list.h:51
spa_list_is_empty
#define spa_list_is_empty(l)
Definition: list.h:49
spa_list::prev
struct spa_list * prev
Definition: list.h:39