Commit fc56ef08b9bb8cc1478fa0758c2bc65f1b6a9a79
1 parent
e2832619
Remove CRs
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@6021 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
1 changed file
with
343 additions
and
343 deletions
sys-queue.h
1 | -/* $NetBSD: queue.h,v 1.45.14.1 2007/07/18 20:13:24 liamjfoy Exp $ */ | |
2 | - | |
3 | -/* | |
4 | - * Qemu version: Copy from netbsd, removed debug code, removed some of | |
5 | - * the implementations. Left in lists, tail queues and circular queues. | |
6 | - */ | |
7 | - | |
8 | -/* | |
9 | - * Copyright (c) 1991, 1993 | |
10 | - * The Regents of the University of California. All rights reserved. | |
11 | - * | |
12 | - * Redistribution and use in source and binary forms, with or without | |
13 | - * modification, are permitted provided that the following conditions | |
14 | - * are met: | |
15 | - * 1. Redistributions of source code must retain the above copyright | |
16 | - * notice, this list of conditions and the following disclaimer. | |
17 | - * 2. Redistributions in binary form must reproduce the above copyright | |
18 | - * notice, this list of conditions and the following disclaimer in the | |
19 | - * documentation and/or other materials provided with the distribution. | |
20 | - * 3. Neither the name of the University nor the names of its contributors | |
21 | - * may be used to endorse or promote products derived from this software | |
22 | - * without specific prior written permission. | |
23 | - * | |
24 | - * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND | |
25 | - * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
26 | - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | |
27 | - * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE | |
28 | - * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | |
29 | - * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS | |
30 | - * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) | |
31 | - * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT | |
32 | - * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY | |
33 | - * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |
34 | - * SUCH DAMAGE. | |
35 | - * | |
36 | - * @(#)queue.h 8.5 (Berkeley) 8/20/94 | |
37 | - */ | |
38 | - | |
39 | -#ifndef _SYS_QUEUE_H_ | |
40 | -#define _SYS_QUEUE_H_ | |
41 | - | |
42 | -/* | |
43 | - * This file defines three types of data structures: | |
44 | - * lists, tail queues, and circular queues. | |
45 | - * | |
46 | - * A list is headed by a single forward pointer (or an array of forward | |
47 | - * pointers for a hash table header). The elements are doubly linked | |
48 | - * so that an arbitrary element can be removed without a need to | |
49 | - * traverse the list. New elements can be added to the list before | |
50 | - * or after an existing element or at the head of the list. A list | |
51 | - * may only be traversed in the forward direction. | |
52 | - * | |
53 | - * A tail queue is headed by a pair of pointers, one to the head of the | |
54 | - * list and the other to the tail of the list. The elements are doubly | |
55 | - * linked so that an arbitrary element can be removed without a need to | |
56 | - * traverse the list. New elements can be added to the list before or | |
57 | - * after an existing element, at the head of the list, or at the end of | |
58 | - * the list. A tail queue may be traversed in either direction. | |
59 | - * | |
60 | - * A circle queue is headed by a pair of pointers, one to the head of the | |
61 | - * list and the other to the tail of the list. The elements are doubly | |
62 | - * linked so that an arbitrary element can be removed without a need to | |
63 | - * traverse the list. New elements can be added to the list before or after | |
64 | - * an existing element, at the head of the list, or at the end of the list. | |
65 | - * A circle queue may be traversed in either direction, but has a more | |
66 | - * complex end of list detection. | |
67 | - * | |
68 | - * For details on the use of these macros, see the queue(3) manual page. | |
69 | - */ | |
70 | - | |
71 | -/* | |
72 | - * List definitions. | |
73 | - */ | |
74 | -#define LIST_HEAD(name, type) \ | |
75 | -struct name { \ | |
76 | - struct type *lh_first; /* first element */ \ | |
77 | -} | |
78 | - | |
79 | -#define LIST_HEAD_INITIALIZER(head) \ | |
80 | - { NULL } | |
81 | - | |
82 | -#define LIST_ENTRY(type) \ | |
83 | -struct { \ | |
84 | - struct type *le_next; /* next element */ \ | |
85 | - struct type **le_prev; /* address of previous next element */ \ | |
86 | -} | |
87 | - | |
88 | -/* | |
89 | - * List functions. | |
90 | - */ | |
91 | -#define LIST_INIT(head) do { \ | |
92 | - (head)->lh_first = NULL; \ | |
93 | -} while (/*CONSTCOND*/0) | |
94 | - | |
95 | -#define LIST_INSERT_AFTER(listelm, elm, field) do { \ | |
96 | - if (((elm)->field.le_next = (listelm)->field.le_next) != NULL) \ | |
97 | - (listelm)->field.le_next->field.le_prev = \ | |
98 | - &(elm)->field.le_next; \ | |
99 | - (listelm)->field.le_next = (elm); \ | |
100 | - (elm)->field.le_prev = &(listelm)->field.le_next; \ | |
101 | -} while (/*CONSTCOND*/0) | |
102 | - | |
103 | -#define LIST_INSERT_BEFORE(listelm, elm, field) do { \ | |
104 | - (elm)->field.le_prev = (listelm)->field.le_prev; \ | |
105 | - (elm)->field.le_next = (listelm); \ | |
106 | - *(listelm)->field.le_prev = (elm); \ | |
107 | - (listelm)->field.le_prev = &(elm)->field.le_next; \ | |
108 | -} while (/*CONSTCOND*/0) | |
109 | - | |
110 | -#define LIST_INSERT_HEAD(head, elm, field) do { \ | |
111 | - if (((elm)->field.le_next = (head)->lh_first) != NULL) \ | |
112 | - (head)->lh_first->field.le_prev = &(elm)->field.le_next;\ | |
113 | - (head)->lh_first = (elm); \ | |
114 | - (elm)->field.le_prev = &(head)->lh_first; \ | |
115 | -} while (/*CONSTCOND*/0) | |
116 | - | |
117 | -#define LIST_REMOVE(elm, field) do { \ | |
118 | - if ((elm)->field.le_next != NULL) \ | |
119 | - (elm)->field.le_next->field.le_prev = \ | |
120 | - (elm)->field.le_prev; \ | |
121 | - *(elm)->field.le_prev = (elm)->field.le_next; \ | |
122 | -} while (/*CONSTCOND*/0) | |
123 | - | |
124 | -#define LIST_FOREACH(var, head, field) \ | |
125 | - for ((var) = ((head)->lh_first); \ | |
126 | - (var); \ | |
127 | - (var) = ((var)->field.le_next)) | |
128 | - | |
129 | -/* | |
130 | - * List access methods. | |
131 | - */ | |
132 | -#define LIST_EMPTY(head) ((head)->lh_first == NULL) | |
133 | -#define LIST_FIRST(head) ((head)->lh_first) | |
134 | -#define LIST_NEXT(elm, field) ((elm)->field.le_next) | |
135 | - | |
136 | - | |
137 | -/* | |
138 | - * Tail queue definitions. | |
139 | - */ | |
140 | -#define _TAILQ_HEAD(name, type, qual) \ | |
141 | -struct name { \ | |
142 | - qual type *tqh_first; /* first element */ \ | |
143 | - qual type *qual *tqh_last; /* addr of last next element */ \ | |
144 | -} | |
145 | -#define TAILQ_HEAD(name, type) _TAILQ_HEAD(name, struct type,) | |
146 | - | |
147 | -#define TAILQ_HEAD_INITIALIZER(head) \ | |
148 | - { NULL, &(head).tqh_first } | |
149 | - | |
150 | -#define _TAILQ_ENTRY(type, qual) \ | |
151 | -struct { \ | |
152 | - qual type *tqe_next; /* next element */ \ | |
153 | - qual type *qual *tqe_prev; /* address of previous next element */\ | |
154 | -} | |
155 | -#define TAILQ_ENTRY(type) _TAILQ_ENTRY(struct type,) | |
156 | - | |
157 | -/* | |
158 | - * Tail queue functions. | |
159 | - */ | |
160 | -#define TAILQ_INIT(head) do { \ | |
161 | - (head)->tqh_first = NULL; \ | |
162 | - (head)->tqh_last = &(head)->tqh_first; \ | |
163 | -} while (/*CONSTCOND*/0) | |
164 | - | |
165 | -#define TAILQ_INSERT_HEAD(head, elm, field) do { \ | |
166 | - if (((elm)->field.tqe_next = (head)->tqh_first) != NULL) \ | |
167 | - (head)->tqh_first->field.tqe_prev = \ | |
168 | - &(elm)->field.tqe_next; \ | |
169 | - else \ | |
170 | - (head)->tqh_last = &(elm)->field.tqe_next; \ | |
171 | - (head)->tqh_first = (elm); \ | |
172 | - (elm)->field.tqe_prev = &(head)->tqh_first; \ | |
173 | -} while (/*CONSTCOND*/0) | |
174 | - | |
175 | -#define TAILQ_INSERT_TAIL(head, elm, field) do { \ | |
176 | - (elm)->field.tqe_next = NULL; \ | |
177 | - (elm)->field.tqe_prev = (head)->tqh_last; \ | |
178 | - *(head)->tqh_last = (elm); \ | |
179 | - (head)->tqh_last = &(elm)->field.tqe_next; \ | |
180 | -} while (/*CONSTCOND*/0) | |
181 | - | |
182 | -#define TAILQ_INSERT_AFTER(head, listelm, elm, field) do { \ | |
183 | - if (((elm)->field.tqe_next = (listelm)->field.tqe_next) != NULL)\ | |
184 | - (elm)->field.tqe_next->field.tqe_prev = \ | |
185 | - &(elm)->field.tqe_next; \ | |
186 | - else \ | |
187 | - (head)->tqh_last = &(elm)->field.tqe_next; \ | |
188 | - (listelm)->field.tqe_next = (elm); \ | |
189 | - (elm)->field.tqe_prev = &(listelm)->field.tqe_next; \ | |
190 | -} while (/*CONSTCOND*/0) | |
191 | - | |
192 | -#define TAILQ_INSERT_BEFORE(listelm, elm, field) do { \ | |
193 | - (elm)->field.tqe_prev = (listelm)->field.tqe_prev; \ | |
194 | - (elm)->field.tqe_next = (listelm); \ | |
195 | - *(listelm)->field.tqe_prev = (elm); \ | |
196 | - (listelm)->field.tqe_prev = &(elm)->field.tqe_next; \ | |
197 | -} while (/*CONSTCOND*/0) | |
198 | - | |
199 | -#define TAILQ_REMOVE(head, elm, field) do { \ | |
200 | - if (((elm)->field.tqe_next) != NULL) \ | |
201 | - (elm)->field.tqe_next->field.tqe_prev = \ | |
202 | - (elm)->field.tqe_prev; \ | |
203 | - else \ | |
204 | - (head)->tqh_last = (elm)->field.tqe_prev; \ | |
205 | - *(elm)->field.tqe_prev = (elm)->field.tqe_next; \ | |
206 | -} while (/*CONSTCOND*/0) | |
207 | - | |
208 | -#define TAILQ_FOREACH(var, head, field) \ | |
209 | - for ((var) = ((head)->tqh_first); \ | |
210 | - (var); \ | |
211 | - (var) = ((var)->field.tqe_next)) | |
212 | - | |
213 | -#define TAILQ_FOREACH_SAFE(var, head, field, next_var) \ | |
214 | - for ((var) = ((head)->tqh_first); \ | |
215 | - (var) && ((next_var) = ((var)->field.tqe_next), 1); \ | |
216 | - (var) = (next_var)) | |
217 | - | |
218 | -#define TAILQ_FOREACH_REVERSE(var, head, headname, field) \ | |
219 | - for ((var) = (*(((struct headname *)((head)->tqh_last))->tqh_last)); \ | |
220 | - (var); \ | |
221 | - (var) = (*(((struct headname *)((var)->field.tqe_prev))->tqh_last))) | |
222 | - | |
223 | -/* | |
224 | - * Tail queue access methods. | |
225 | - */ | |
226 | -#define TAILQ_EMPTY(head) ((head)->tqh_first == NULL) | |
227 | -#define TAILQ_FIRST(head) ((head)->tqh_first) | |
228 | -#define TAILQ_NEXT(elm, field) ((elm)->field.tqe_next) | |
229 | - | |
230 | -#define TAILQ_LAST(head, headname) \ | |
231 | - (*(((struct headname *)((head)->tqh_last))->tqh_last)) | |
232 | -#define TAILQ_PREV(elm, headname, field) \ | |
233 | - (*(((struct headname *)((elm)->field.tqe_prev))->tqh_last)) | |
234 | - | |
235 | - | |
236 | -/* | |
237 | - * Circular queue definitions. | |
238 | - */ | |
239 | -#define CIRCLEQ_HEAD(name, type) \ | |
240 | -struct name { \ | |
241 | - struct type *cqh_first; /* first element */ \ | |
242 | - struct type *cqh_last; /* last element */ \ | |
243 | -} | |
244 | - | |
245 | -#define CIRCLEQ_HEAD_INITIALIZER(head) \ | |
246 | - { (void *)&head, (void *)&head } | |
247 | - | |
248 | -#define CIRCLEQ_ENTRY(type) \ | |
249 | -struct { \ | |
250 | - struct type *cqe_next; /* next element */ \ | |
251 | - struct type *cqe_prev; /* previous element */ \ | |
252 | -} | |
253 | - | |
254 | -/* | |
255 | - * Circular queue functions. | |
256 | - */ | |
257 | -#define CIRCLEQ_INIT(head) do { \ | |
258 | - (head)->cqh_first = (void *)(head); \ | |
259 | - (head)->cqh_last = (void *)(head); \ | |
260 | -} while (/*CONSTCOND*/0) | |
261 | - | |
262 | -#define CIRCLEQ_INSERT_AFTER(head, listelm, elm, field) do { \ | |
263 | - (elm)->field.cqe_next = (listelm)->field.cqe_next; \ | |
264 | - (elm)->field.cqe_prev = (listelm); \ | |
265 | - if ((listelm)->field.cqe_next == (void *)(head)) \ | |
266 | - (head)->cqh_last = (elm); \ | |
267 | - else \ | |
268 | - (listelm)->field.cqe_next->field.cqe_prev = (elm); \ | |
269 | - (listelm)->field.cqe_next = (elm); \ | |
270 | -} while (/*CONSTCOND*/0) | |
271 | - | |
272 | -#define CIRCLEQ_INSERT_BEFORE(head, listelm, elm, field) do { \ | |
273 | - (elm)->field.cqe_next = (listelm); \ | |
274 | - (elm)->field.cqe_prev = (listelm)->field.cqe_prev; \ | |
275 | - if ((listelm)->field.cqe_prev == (void *)(head)) \ | |
276 | - (head)->cqh_first = (elm); \ | |
277 | - else \ | |
278 | - (listelm)->field.cqe_prev->field.cqe_next = (elm); \ | |
279 | - (listelm)->field.cqe_prev = (elm); \ | |
280 | -} while (/*CONSTCOND*/0) | |
281 | - | |
282 | -#define CIRCLEQ_INSERT_HEAD(head, elm, field) do { \ | |
283 | - (elm)->field.cqe_next = (head)->cqh_first; \ | |
284 | - (elm)->field.cqe_prev = (void *)(head); \ | |
285 | - if ((head)->cqh_last == (void *)(head)) \ | |
286 | - (head)->cqh_last = (elm); \ | |
287 | - else \ | |
288 | - (head)->cqh_first->field.cqe_prev = (elm); \ | |
289 | - (head)->cqh_first = (elm); \ | |
290 | -} while (/*CONSTCOND*/0) | |
291 | - | |
292 | -#define CIRCLEQ_INSERT_TAIL(head, elm, field) do { \ | |
293 | - (elm)->field.cqe_next = (void *)(head); \ | |
294 | - (elm)->field.cqe_prev = (head)->cqh_last; \ | |
295 | - if ((head)->cqh_first == (void *)(head)) \ | |
296 | - (head)->cqh_first = (elm); \ | |
297 | - else \ | |
298 | - (head)->cqh_last->field.cqe_next = (elm); \ | |
299 | - (head)->cqh_last = (elm); \ | |
300 | -} while (/*CONSTCOND*/0) | |
301 | - | |
302 | -#define CIRCLEQ_REMOVE(head, elm, field) do { \ | |
303 | - if ((elm)->field.cqe_next == (void *)(head)) \ | |
304 | - (head)->cqh_last = (elm)->field.cqe_prev; \ | |
305 | - else \ | |
306 | - (elm)->field.cqe_next->field.cqe_prev = \ | |
307 | - (elm)->field.cqe_prev; \ | |
308 | - if ((elm)->field.cqe_prev == (void *)(head)) \ | |
309 | - (head)->cqh_first = (elm)->field.cqe_next; \ | |
310 | - else \ | |
311 | - (elm)->field.cqe_prev->field.cqe_next = \ | |
312 | - (elm)->field.cqe_next; \ | |
313 | -} while (/*CONSTCOND*/0) | |
314 | - | |
315 | -#define CIRCLEQ_FOREACH(var, head, field) \ | |
316 | - for ((var) = ((head)->cqh_first); \ | |
317 | - (var) != (const void *)(head); \ | |
318 | - (var) = ((var)->field.cqe_next)) | |
319 | - | |
320 | -#define CIRCLEQ_FOREACH_REVERSE(var, head, field) \ | |
321 | - for ((var) = ((head)->cqh_last); \ | |
322 | - (var) != (const void *)(head); \ | |
323 | - (var) = ((var)->field.cqe_prev)) | |
324 | - | |
325 | -/* | |
326 | - * Circular queue access methods. | |
327 | - */ | |
328 | -#define CIRCLEQ_EMPTY(head) ((head)->cqh_first == (void *)(head)) | |
329 | -#define CIRCLEQ_FIRST(head) ((head)->cqh_first) | |
330 | -#define CIRCLEQ_LAST(head) ((head)->cqh_last) | |
331 | -#define CIRCLEQ_NEXT(elm, field) ((elm)->field.cqe_next) | |
332 | -#define CIRCLEQ_PREV(elm, field) ((elm)->field.cqe_prev) | |
333 | - | |
334 | -#define CIRCLEQ_LOOP_NEXT(head, elm, field) \ | |
335 | - (((elm)->field.cqe_next == (void *)(head)) \ | |
336 | - ? ((head)->cqh_first) \ | |
337 | - : (elm->field.cqe_next)) | |
338 | -#define CIRCLEQ_LOOP_PREV(head, elm, field) \ | |
339 | - (((elm)->field.cqe_prev == (void *)(head)) \ | |
340 | - ? ((head)->cqh_last) \ | |
341 | - : (elm->field.cqe_prev)) | |
342 | - | |
343 | -#endif /* !_SYS_QUEUE_H_ */ | |
1 | +/* $NetBSD: queue.h,v 1.45.14.1 2007/07/18 20:13:24 liamjfoy Exp $ */ | |
2 | + | |
3 | +/* | |
4 | + * Qemu version: Copy from netbsd, removed debug code, removed some of | |
5 | + * the implementations. Left in lists, tail queues and circular queues. | |
6 | + */ | |
7 | + | |
8 | +/* | |
9 | + * Copyright (c) 1991, 1993 | |
10 | + * The Regents of the University of California. All rights reserved. | |
11 | + * | |
12 | + * Redistribution and use in source and binary forms, with or without | |
13 | + * modification, are permitted provided that the following conditions | |
14 | + * are met: | |
15 | + * 1. Redistributions of source code must retain the above copyright | |
16 | + * notice, this list of conditions and the following disclaimer. | |
17 | + * 2. Redistributions in binary form must reproduce the above copyright | |
18 | + * notice, this list of conditions and the following disclaimer in the | |
19 | + * documentation and/or other materials provided with the distribution. | |
20 | + * 3. Neither the name of the University nor the names of its contributors | |
21 | + * may be used to endorse or promote products derived from this software | |
22 | + * without specific prior written permission. | |
23 | + * | |
24 | + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND | |
25 | + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
26 | + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | |
27 | + * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE | |
28 | + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | |
29 | + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS | |
30 | + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) | |
31 | + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT | |
32 | + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY | |
33 | + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |
34 | + * SUCH DAMAGE. | |
35 | + * | |
36 | + * @(#)queue.h 8.5 (Berkeley) 8/20/94 | |
37 | + */ | |
38 | + | |
39 | +#ifndef _SYS_QUEUE_H_ | |
40 | +#define _SYS_QUEUE_H_ | |
41 | + | |
42 | +/* | |
43 | + * This file defines three types of data structures: | |
44 | + * lists, tail queues, and circular queues. | |
45 | + * | |
46 | + * A list is headed by a single forward pointer (or an array of forward | |
47 | + * pointers for a hash table header). The elements are doubly linked | |
48 | + * so that an arbitrary element can be removed without a need to | |
49 | + * traverse the list. New elements can be added to the list before | |
50 | + * or after an existing element or at the head of the list. A list | |
51 | + * may only be traversed in the forward direction. | |
52 | + * | |
53 | + * A tail queue is headed by a pair of pointers, one to the head of the | |
54 | + * list and the other to the tail of the list. The elements are doubly | |
55 | + * linked so that an arbitrary element can be removed without a need to | |
56 | + * traverse the list. New elements can be added to the list before or | |
57 | + * after an existing element, at the head of the list, or at the end of | |
58 | + * the list. A tail queue may be traversed in either direction. | |
59 | + * | |
60 | + * A circle queue is headed by a pair of pointers, one to the head of the | |
61 | + * list and the other to the tail of the list. The elements are doubly | |
62 | + * linked so that an arbitrary element can be removed without a need to | |
63 | + * traverse the list. New elements can be added to the list before or after | |
64 | + * an existing element, at the head of the list, or at the end of the list. | |
65 | + * A circle queue may be traversed in either direction, but has a more | |
66 | + * complex end of list detection. | |
67 | + * | |
68 | + * For details on the use of these macros, see the queue(3) manual page. | |
69 | + */ | |
70 | + | |
71 | +/* | |
72 | + * List definitions. | |
73 | + */ | |
74 | +#define LIST_HEAD(name, type) \ | |
75 | +struct name { \ | |
76 | + struct type *lh_first; /* first element */ \ | |
77 | +} | |
78 | + | |
79 | +#define LIST_HEAD_INITIALIZER(head) \ | |
80 | + { NULL } | |
81 | + | |
82 | +#define LIST_ENTRY(type) \ | |
83 | +struct { \ | |
84 | + struct type *le_next; /* next element */ \ | |
85 | + struct type **le_prev; /* address of previous next element */ \ | |
86 | +} | |
87 | + | |
88 | +/* | |
89 | + * List functions. | |
90 | + */ | |
91 | +#define LIST_INIT(head) do { \ | |
92 | + (head)->lh_first = NULL; \ | |
93 | +} while (/*CONSTCOND*/0) | |
94 | + | |
95 | +#define LIST_INSERT_AFTER(listelm, elm, field) do { \ | |
96 | + if (((elm)->field.le_next = (listelm)->field.le_next) != NULL) \ | |
97 | + (listelm)->field.le_next->field.le_prev = \ | |
98 | + &(elm)->field.le_next; \ | |
99 | + (listelm)->field.le_next = (elm); \ | |
100 | + (elm)->field.le_prev = &(listelm)->field.le_next; \ | |
101 | +} while (/*CONSTCOND*/0) | |
102 | + | |
103 | +#define LIST_INSERT_BEFORE(listelm, elm, field) do { \ | |
104 | + (elm)->field.le_prev = (listelm)->field.le_prev; \ | |
105 | + (elm)->field.le_next = (listelm); \ | |
106 | + *(listelm)->field.le_prev = (elm); \ | |
107 | + (listelm)->field.le_prev = &(elm)->field.le_next; \ | |
108 | +} while (/*CONSTCOND*/0) | |
109 | + | |
110 | +#define LIST_INSERT_HEAD(head, elm, field) do { \ | |
111 | + if (((elm)->field.le_next = (head)->lh_first) != NULL) \ | |
112 | + (head)->lh_first->field.le_prev = &(elm)->field.le_next;\ | |
113 | + (head)->lh_first = (elm); \ | |
114 | + (elm)->field.le_prev = &(head)->lh_first; \ | |
115 | +} while (/*CONSTCOND*/0) | |
116 | + | |
117 | +#define LIST_REMOVE(elm, field) do { \ | |
118 | + if ((elm)->field.le_next != NULL) \ | |
119 | + (elm)->field.le_next->field.le_prev = \ | |
120 | + (elm)->field.le_prev; \ | |
121 | + *(elm)->field.le_prev = (elm)->field.le_next; \ | |
122 | +} while (/*CONSTCOND*/0) | |
123 | + | |
124 | +#define LIST_FOREACH(var, head, field) \ | |
125 | + for ((var) = ((head)->lh_first); \ | |
126 | + (var); \ | |
127 | + (var) = ((var)->field.le_next)) | |
128 | + | |
129 | +/* | |
130 | + * List access methods. | |
131 | + */ | |
132 | +#define LIST_EMPTY(head) ((head)->lh_first == NULL) | |
133 | +#define LIST_FIRST(head) ((head)->lh_first) | |
134 | +#define LIST_NEXT(elm, field) ((elm)->field.le_next) | |
135 | + | |
136 | + | |
137 | +/* | |
138 | + * Tail queue definitions. | |
139 | + */ | |
140 | +#define _TAILQ_HEAD(name, type, qual) \ | |
141 | +struct name { \ | |
142 | + qual type *tqh_first; /* first element */ \ | |
143 | + qual type *qual *tqh_last; /* addr of last next element */ \ | |
144 | +} | |
145 | +#define TAILQ_HEAD(name, type) _TAILQ_HEAD(name, struct type,) | |
146 | + | |
147 | +#define TAILQ_HEAD_INITIALIZER(head) \ | |
148 | + { NULL, &(head).tqh_first } | |
149 | + | |
150 | +#define _TAILQ_ENTRY(type, qual) \ | |
151 | +struct { \ | |
152 | + qual type *tqe_next; /* next element */ \ | |
153 | + qual type *qual *tqe_prev; /* address of previous next element */\ | |
154 | +} | |
155 | +#define TAILQ_ENTRY(type) _TAILQ_ENTRY(struct type,) | |
156 | + | |
157 | +/* | |
158 | + * Tail queue functions. | |
159 | + */ | |
160 | +#define TAILQ_INIT(head) do { \ | |
161 | + (head)->tqh_first = NULL; \ | |
162 | + (head)->tqh_last = &(head)->tqh_first; \ | |
163 | +} while (/*CONSTCOND*/0) | |
164 | + | |
165 | +#define TAILQ_INSERT_HEAD(head, elm, field) do { \ | |
166 | + if (((elm)->field.tqe_next = (head)->tqh_first) != NULL) \ | |
167 | + (head)->tqh_first->field.tqe_prev = \ | |
168 | + &(elm)->field.tqe_next; \ | |
169 | + else \ | |
170 | + (head)->tqh_last = &(elm)->field.tqe_next; \ | |
171 | + (head)->tqh_first = (elm); \ | |
172 | + (elm)->field.tqe_prev = &(head)->tqh_first; \ | |
173 | +} while (/*CONSTCOND*/0) | |
174 | + | |
175 | +#define TAILQ_INSERT_TAIL(head, elm, field) do { \ | |
176 | + (elm)->field.tqe_next = NULL; \ | |
177 | + (elm)->field.tqe_prev = (head)->tqh_last; \ | |
178 | + *(head)->tqh_last = (elm); \ | |
179 | + (head)->tqh_last = &(elm)->field.tqe_next; \ | |
180 | +} while (/*CONSTCOND*/0) | |
181 | + | |
182 | +#define TAILQ_INSERT_AFTER(head, listelm, elm, field) do { \ | |
183 | + if (((elm)->field.tqe_next = (listelm)->field.tqe_next) != NULL)\ | |
184 | + (elm)->field.tqe_next->field.tqe_prev = \ | |
185 | + &(elm)->field.tqe_next; \ | |
186 | + else \ | |
187 | + (head)->tqh_last = &(elm)->field.tqe_next; \ | |
188 | + (listelm)->field.tqe_next = (elm); \ | |
189 | + (elm)->field.tqe_prev = &(listelm)->field.tqe_next; \ | |
190 | +} while (/*CONSTCOND*/0) | |
191 | + | |
192 | +#define TAILQ_INSERT_BEFORE(listelm, elm, field) do { \ | |
193 | + (elm)->field.tqe_prev = (listelm)->field.tqe_prev; \ | |
194 | + (elm)->field.tqe_next = (listelm); \ | |
195 | + *(listelm)->field.tqe_prev = (elm); \ | |
196 | + (listelm)->field.tqe_prev = &(elm)->field.tqe_next; \ | |
197 | +} while (/*CONSTCOND*/0) | |
198 | + | |
199 | +#define TAILQ_REMOVE(head, elm, field) do { \ | |
200 | + if (((elm)->field.tqe_next) != NULL) \ | |
201 | + (elm)->field.tqe_next->field.tqe_prev = \ | |
202 | + (elm)->field.tqe_prev; \ | |
203 | + else \ | |
204 | + (head)->tqh_last = (elm)->field.tqe_prev; \ | |
205 | + *(elm)->field.tqe_prev = (elm)->field.tqe_next; \ | |
206 | +} while (/*CONSTCOND*/0) | |
207 | + | |
208 | +#define TAILQ_FOREACH(var, head, field) \ | |
209 | + for ((var) = ((head)->tqh_first); \ | |
210 | + (var); \ | |
211 | + (var) = ((var)->field.tqe_next)) | |
212 | + | |
213 | +#define TAILQ_FOREACH_SAFE(var, head, field, next_var) \ | |
214 | + for ((var) = ((head)->tqh_first); \ | |
215 | + (var) && ((next_var) = ((var)->field.tqe_next), 1); \ | |
216 | + (var) = (next_var)) | |
217 | + | |
218 | +#define TAILQ_FOREACH_REVERSE(var, head, headname, field) \ | |
219 | + for ((var) = (*(((struct headname *)((head)->tqh_last))->tqh_last)); \ | |
220 | + (var); \ | |
221 | + (var) = (*(((struct headname *)((var)->field.tqe_prev))->tqh_last))) | |
222 | + | |
223 | +/* | |
224 | + * Tail queue access methods. | |
225 | + */ | |
226 | +#define TAILQ_EMPTY(head) ((head)->tqh_first == NULL) | |
227 | +#define TAILQ_FIRST(head) ((head)->tqh_first) | |
228 | +#define TAILQ_NEXT(elm, field) ((elm)->field.tqe_next) | |
229 | + | |
230 | +#define TAILQ_LAST(head, headname) \ | |
231 | + (*(((struct headname *)((head)->tqh_last))->tqh_last)) | |
232 | +#define TAILQ_PREV(elm, headname, field) \ | |
233 | + (*(((struct headname *)((elm)->field.tqe_prev))->tqh_last)) | |
234 | + | |
235 | + | |
236 | +/* | |
237 | + * Circular queue definitions. | |
238 | + */ | |
239 | +#define CIRCLEQ_HEAD(name, type) \ | |
240 | +struct name { \ | |
241 | + struct type *cqh_first; /* first element */ \ | |
242 | + struct type *cqh_last; /* last element */ \ | |
243 | +} | |
244 | + | |
245 | +#define CIRCLEQ_HEAD_INITIALIZER(head) \ | |
246 | + { (void *)&head, (void *)&head } | |
247 | + | |
248 | +#define CIRCLEQ_ENTRY(type) \ | |
249 | +struct { \ | |
250 | + struct type *cqe_next; /* next element */ \ | |
251 | + struct type *cqe_prev; /* previous element */ \ | |
252 | +} | |
253 | + | |
254 | +/* | |
255 | + * Circular queue functions. | |
256 | + */ | |
257 | +#define CIRCLEQ_INIT(head) do { \ | |
258 | + (head)->cqh_first = (void *)(head); \ | |
259 | + (head)->cqh_last = (void *)(head); \ | |
260 | +} while (/*CONSTCOND*/0) | |
261 | + | |
262 | +#define CIRCLEQ_INSERT_AFTER(head, listelm, elm, field) do { \ | |
263 | + (elm)->field.cqe_next = (listelm)->field.cqe_next; \ | |
264 | + (elm)->field.cqe_prev = (listelm); \ | |
265 | + if ((listelm)->field.cqe_next == (void *)(head)) \ | |
266 | + (head)->cqh_last = (elm); \ | |
267 | + else \ | |
268 | + (listelm)->field.cqe_next->field.cqe_prev = (elm); \ | |
269 | + (listelm)->field.cqe_next = (elm); \ | |
270 | +} while (/*CONSTCOND*/0) | |
271 | + | |
272 | +#define CIRCLEQ_INSERT_BEFORE(head, listelm, elm, field) do { \ | |
273 | + (elm)->field.cqe_next = (listelm); \ | |
274 | + (elm)->field.cqe_prev = (listelm)->field.cqe_prev; \ | |
275 | + if ((listelm)->field.cqe_prev == (void *)(head)) \ | |
276 | + (head)->cqh_first = (elm); \ | |
277 | + else \ | |
278 | + (listelm)->field.cqe_prev->field.cqe_next = (elm); \ | |
279 | + (listelm)->field.cqe_prev = (elm); \ | |
280 | +} while (/*CONSTCOND*/0) | |
281 | + | |
282 | +#define CIRCLEQ_INSERT_HEAD(head, elm, field) do { \ | |
283 | + (elm)->field.cqe_next = (head)->cqh_first; \ | |
284 | + (elm)->field.cqe_prev = (void *)(head); \ | |
285 | + if ((head)->cqh_last == (void *)(head)) \ | |
286 | + (head)->cqh_last = (elm); \ | |
287 | + else \ | |
288 | + (head)->cqh_first->field.cqe_prev = (elm); \ | |
289 | + (head)->cqh_first = (elm); \ | |
290 | +} while (/*CONSTCOND*/0) | |
291 | + | |
292 | +#define CIRCLEQ_INSERT_TAIL(head, elm, field) do { \ | |
293 | + (elm)->field.cqe_next = (void *)(head); \ | |
294 | + (elm)->field.cqe_prev = (head)->cqh_last; \ | |
295 | + if ((head)->cqh_first == (void *)(head)) \ | |
296 | + (head)->cqh_first = (elm); \ | |
297 | + else \ | |
298 | + (head)->cqh_last->field.cqe_next = (elm); \ | |
299 | + (head)->cqh_last = (elm); \ | |
300 | +} while (/*CONSTCOND*/0) | |
301 | + | |
302 | +#define CIRCLEQ_REMOVE(head, elm, field) do { \ | |
303 | + if ((elm)->field.cqe_next == (void *)(head)) \ | |
304 | + (head)->cqh_last = (elm)->field.cqe_prev; \ | |
305 | + else \ | |
306 | + (elm)->field.cqe_next->field.cqe_prev = \ | |
307 | + (elm)->field.cqe_prev; \ | |
308 | + if ((elm)->field.cqe_prev == (void *)(head)) \ | |
309 | + (head)->cqh_first = (elm)->field.cqe_next; \ | |
310 | + else \ | |
311 | + (elm)->field.cqe_prev->field.cqe_next = \ | |
312 | + (elm)->field.cqe_next; \ | |
313 | +} while (/*CONSTCOND*/0) | |
314 | + | |
315 | +#define CIRCLEQ_FOREACH(var, head, field) \ | |
316 | + for ((var) = ((head)->cqh_first); \ | |
317 | + (var) != (const void *)(head); \ | |
318 | + (var) = ((var)->field.cqe_next)) | |
319 | + | |
320 | +#define CIRCLEQ_FOREACH_REVERSE(var, head, field) \ | |
321 | + for ((var) = ((head)->cqh_last); \ | |
322 | + (var) != (const void *)(head); \ | |
323 | + (var) = ((var)->field.cqe_prev)) | |
324 | + | |
325 | +/* | |
326 | + * Circular queue access methods. | |
327 | + */ | |
328 | +#define CIRCLEQ_EMPTY(head) ((head)->cqh_first == (void *)(head)) | |
329 | +#define CIRCLEQ_FIRST(head) ((head)->cqh_first) | |
330 | +#define CIRCLEQ_LAST(head) ((head)->cqh_last) | |
331 | +#define CIRCLEQ_NEXT(elm, field) ((elm)->field.cqe_next) | |
332 | +#define CIRCLEQ_PREV(elm, field) ((elm)->field.cqe_prev) | |
333 | + | |
334 | +#define CIRCLEQ_LOOP_NEXT(head, elm, field) \ | |
335 | + (((elm)->field.cqe_next == (void *)(head)) \ | |
336 | + ? ((head)->cqh_first) \ | |
337 | + : (elm->field.cqe_next)) | |
338 | +#define CIRCLEQ_LOOP_PREV(head, elm, field) \ | |
339 | + (((elm)->field.cqe_prev == (void *)(head)) \ | |
340 | + ? ((head)->cqh_last) \ | |
341 | + : (elm->field.cqe_prev)) | |
342 | + | |
343 | +#endif /* !_SYS_QUEUE_H_ */ | ... | ... |