VirtualBox

source: vbox/trunk/include/iprt/list.h@ 36638

Last change on this file since 36638 was 34406, checked in by vboxsync, 14 years ago

iprt/list.h: RTListNodeGetFirst/Last -> RTListGetFirst/Last; added RTListGetNext, RTListGetPrev, RTListNodeInsertAfter and RTListNodeInsertBefore.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 11.4 KB
Line 
1/** @file
2 * IPRT - Generic Doubly Linked List.
3 */
4
5/*
6 * Copyright (C) 2010 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26#ifndef ___iprt_list_h
27#define ___iprt_list_h
28
29#include <iprt/types.h>
30
31/** @defgroup grp_rt_list RTList - Generic Doubly Linked List
32 * @ingroup grp_rt
33 *
34 * The list implementation is circular without any type wise distintion between
35 * the list and its nodes. This can be confusing since the list head usually
36 * resides in a different structure than the nodes, so care must be taken when
37 * walking the list.
38 *
39 * @{
40 */
41
42RT_C_DECLS_BEGIN
43
44/**
45 * A list node of a doubly linked list.
46 */
47typedef struct RTLISTNODE
48{
49 /** Pointer to the next list node. */
50 struct RTLISTNODE *pNext;
51 /** Pointer to the previous list node. */
52 struct RTLISTNODE *pPrev;
53} RTLISTNODE;
54/** Pointer to a list node. */
55typedef RTLISTNODE *PRTLISTNODE;
56/** Pointer to a list node pointer. */
57typedef PRTLISTNODE *PPRTLISTNODE;
58
59/**
60 * Initialize a list.
61 *
62 * @param pList Pointer to an unitialised list.
63 */
64DECLINLINE(void) RTListInit(PRTLISTNODE pList)
65{
66 pList->pNext = pList;
67 pList->pPrev = pList;
68}
69
70/**
71 * Append a node to the end of the list.
72 *
73 * @param pList The list to append the node to.
74 * @param pNode The node to append.
75 */
76DECLINLINE(void) RTListAppend(PRTLISTNODE pList, PRTLISTNODE pNode)
77{
78 pList->pPrev->pNext = pNode;
79 pNode->pPrev = pList->pPrev;
80 pNode->pNext = pList;
81 pList->pPrev = pNode;
82}
83
84/**
85 * Add a node as the first element of the list.
86 *
87 * @param pList The list to prepend the node to.
88 * @param pNode The node to prepend.
89 */
90DECLINLINE(void) RTListPrepend(PRTLISTNODE pList, PRTLISTNODE pNode)
91{
92 pList->pNext->pPrev = pNode;
93 pNode->pNext = pList->pNext;
94 pNode->pPrev = pList;
95 pList->pNext = pNode;
96}
97
98/**
99 * Inserts a node after the specified one.
100 *
101 * @param pCurNode The current node.
102 * @param pNewNode The node to insert.
103 */
104DECLINLINE(void) RTListNodeInsertAfter(PRTLISTNODE pCurNode, PRTLISTNODE pNewNode)
105{
106 RTListPrepend(pCurNode, pNewNode);
107}
108
109/**
110 * Inserts a node before the specified one.
111 *
112 * @param pCurNode The current node.
113 * @param pNewNode The node to insert.
114 */
115DECLINLINE(void) RTListNodeInsertBefore(PRTLISTNODE pCurNode, PRTLISTNODE pNewNode)
116{
117 RTListAppend(pCurNode, pNewNode);
118}
119
120/**
121 * Remove a node from a list.
122 *
123 * @param pNode The node to remove.
124 */
125DECLINLINE(void) RTListNodeRemove(PRTLISTNODE pNode)
126{
127 PRTLISTNODE pPrev = pNode->pPrev;
128 PRTLISTNODE pNext = pNode->pNext;
129
130 pPrev->pNext = pNext;
131 pNext->pPrev = pPrev;
132
133 /* poison */
134 pNode->pNext = NULL;
135 pNode->pPrev = NULL;
136}
137
138/**
139 * Checks if a node is the last element in the list.
140 *
141 * @retval @c true if the node is the last element in the list.
142 * @retval @c false otherwise
143 *
144 * @param pList The list.
145 * @param pNode The node to check.
146 */
147#define RTListNodeIsLast(pList, pNode) ((pNode)->pNext == (pList))
148
149/**
150 * Checks if a node is the first element in the list.
151 *
152 * @retval @c true if the node is the first element in the list.
153 * @retval @c false otherwise.
154 *
155 * @param pList The list.
156 * @param pNode The node to check.
157 */
158#define RTListNodeIsFirst(pList, pNode) ((pNode)->pPrev == (pList))
159
160/**
161 * Checks if a type converted node is actually the dummy element (@a pList).
162 *
163 * @retval @c true if the node is the dummy element in the list.
164 * @retval @c false otherwise.
165 *
166 * @param pList The list.
167 * @param pNodeStruct The node structure to check. Typically
168 * something obtained from RTListNodeGetNext() or
169 * RTListNodeGetPrev(). This is NOT a PRTLISTNODE
170 * but something that contains a RTLISTNODE member!
171 * @param Type Structure the list node is a member of.
172 * @param Member The list node member.
173 */
174#define RTListNodeIsDummy(pList, pNode, Type, Member) \
175 ( (pNode) == RT_FROM_MEMBER((pList), Type, Member) )
176
177/**
178 * Checks if a list is empty.
179 *
180 * @retval @c true if the list is empty.
181 * @retval @c false otherwise.
182 *
183 * @param pList The list to check.
184 */
185#define RTListIsEmpty(pList) ((pList)->pPrev == (pList))
186
187/**
188 * Returns the next node in the list.
189 *
190 * @returns The next node.
191 *
192 * @param pCurNode The current node.
193 * @param Type Structure the list node is a member of.
194 * @param Member The list node member.
195 */
196#define RTListNodeGetNext(pCurNode, Type, Member) \
197 RT_FROM_MEMBER((pCurNode)->pNext, Type, Member)
198
199/**
200 * Returns the previous node in the list.
201 *
202 * @returns The previous node.
203 *
204 * @param pCurNode The current node.
205 * @param Type Structure the list node is a member of.
206 * @param Member The list node member.
207 */
208#define RTListNodeGetPrev(pCurNode, Type, Member) \
209 RT_FROM_MEMBER((pCurNode)->pPrev, Type, Member)
210
211/**
212 * Returns the first element in the list (checks for empty list).
213 *
214 * @retval Pointer to the first list element.
215 * @retval NULL if the list is empty.
216 *
217 * @param pList List to get the first element from.
218 * @param Type Structure the list node is a member of.
219 * @param Member The list node member.
220 */
221#define RTListGetFirst(pList, Type, Member) \
222 (!RTListIsEmpty(pList) ? RTListNodeGetNext(pList, Type, Member) : NULL)
223
224/**
225 * Returns the last element in the list (checks for empty list).
226 *
227 * @retval Pointer to the last list element.
228 * @retval NULL if the list is empty.
229 *
230 * @param pList List to get the last element from.
231 * @param Type Structure the list node is a member of.
232 * @param Member The list node member.
233 */
234#define RTListGetLast(pList, Type, Member) \
235 (!RTListIsEmpty(pList) ? RTListNodeGetPrev(pList, Type, Member) : NULL)
236
237/**
238 * Returns the next node in the list or NULL if the end has been reached.
239 *
240 * @returns The next node or NULL.
241 *
242 * @param pList The list @a pCurNode is linked on.
243 * @param pCurNode The current node, of type @a Type.
244 * @param Type Structure the list node is a member of.
245 * @param Member The list node member.
246 */
247#define RTListGetNext(pList, pCurNode, Type, Member) \
248 ( (pCurNode)->Member.pNext != (pList) ? RT_FROM_MEMBER((pCurNode)->Member.pNext, Type, Member) : NULL )
249
250/**
251 * Returns the previous node in the list or NULL if the start has been reached.
252 *
253 * @returns The previous node or NULL.
254 *
255 * @param pList The list @a pCurNode is linked on.
256 * @param pCurNode The current node, of type @a Type.
257 * @param Type Structure the list node is a member of.
258 * @param Member The list node member.
259 */
260#define RTListGetPrev(pList, pCurNode, Type, Member) \
261 ( (pCurNode)->Member.pPrev != (pList) ? RT_FROM_MEMBER((pCurNode)->Member.pPrev, Type, Member) : NULL )
262
263/**
264 * Enumerate the list in head to tail order.
265 *
266 * @param pList List to enumerate.
267 * @param pIterator The iterator variable name.
268 * @param Type Structure the list node is a member of.
269 * @param Member The list node member name.
270 */
271#define RTListForEach(pList, pIterator, Type, Member) \
272 for (pIterator = RTListNodeGetNext(pList, Type, Member); \
273 !RTListNodeIsDummy(pList, pIterator, Type, Member); \
274 pIterator = RT_FROM_MEMBER((pIterator)->Member.pNext, Type, Member) )
275
276
277/**
278 * Enumerate the list in head to tail order, safe against removal of the
279 * current node.
280 *
281 * @param pList List to enumerate.
282 * @param pIterator The iterator variable name.
283 * @param pIterNext The name of the variable saving the pointer to
284 * the next element.
285 * @param Type Structure the list node is a member of.
286 * @param Member The list node member name.
287 */
288#define RTListForEachSafe(pList, pIterator, pIterNext, Type, Member) \
289 for (pIterator = RTListNodeGetNext(pList, Type, Member), \
290 pIterNext = RT_FROM_MEMBER((pIterator)->Member.pNext, Type, Member); \
291 !RTListNodeIsDummy(pList, pIterator, Type, Member); \
292 pIterator = pIterNext, \
293 pIterNext = RT_FROM_MEMBER((pIterator)->Member.pNext, Type, Member) )
294
295
296/**
297 * Enumerate the list in reverse order (tail to head).
298 *
299 * @param pList List to enumerate.
300 * @param pIterator The iterator variable name.
301 * @param Type Structure the list node is a member of.
302 * @param Member The list node member name.
303 */
304#define RTListForEachReverse(pList, pIterator, Type, Member) \
305 for (pIterator = RTListNodeGetPrev(pList, Type, Member); \
306 !RTListNodeIsDummy(pList, pIterator, Type, Member); \
307 pIterator = RT_FROM_MEMBER((pIterator)->Member.pPrev, Type, Member) )
308
309
310/**
311 * Enumerate the list in reverse order (tail to head).
312 *
313 * @param pList List to enumerate.
314 * @param pIterator The iterator variable name.
315 * @param pIterPrev The name of the variable saving the pointer to
316 * the previous element.
317 * @param Type Structure the list node is a member of.
318 * @param Member The list node member name.
319 */
320#define RTListForEachReverseSafe(pList, pIterator, pIterPrev, Type, Member) \
321 for (pIterator = RTListNodeGetPrev(pList, Type, Member), \
322 pIterPrev = RT_FROM_MEMBER((pIterator)->Member.pPrev, Type, Member); \
323 !RTListNodeIsDummy(pList, pIterator, Type, Member); \
324 pIterator = pIterPrev, \
325 pIterPrev = RT_FROM_MEMBER((pIterator)->Member.pPrev, Type, Member) )
326
327
328/**
329 * Move the given list to a new list header.
330 *
331 * @param pListDst The new list.
332 * @param pListSrc The list to move.
333 */
334DECLINLINE(void) RTListMove(PRTLISTNODE pListDst, PRTLISTNODE pListSrc)
335{
336 if (!RTListIsEmpty(pListSrc))
337 {
338 pListDst->pNext = pListSrc->pNext;
339 pListDst->pPrev = pListSrc->pPrev;
340
341 /* Adjust the first and last element links */
342 pListDst->pNext->pPrev = pListDst;
343 pListDst->pPrev->pNext = pListDst;
344
345 /* Finally remove the elements from the source list */
346 RTListInit(pListSrc);
347 }
348}
349
350RT_C_DECLS_END
351
352/** @} */
353
354#endif
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette