diff options
author | Lennart Poettering <lennart@poettering.net> | 2015-01-28 00:34:58 +0100 |
---|---|---|
committer | Lennart Poettering <lennart@poettering.net> | 2015-01-28 00:34:58 +0100 |
commit | 7663df377016cf7b95001aec893006647175ae4a (patch) | |
tree | 523a2d1bf1787b9a3b08ae516cd54c9a022f9b73 /src | |
parent | 976dec6e7b2d193533191be2969dd4eee95fc6bb (diff) |
list: add macro for iterating through a list an item is in, skipping the item
Diffstat (limited to 'src')
-rw-r--r-- | src/shared/list.h | 10 | ||||
-rw-r--r-- | src/test/test-list.c | 9 |
2 files changed, 18 insertions, 1 deletions
diff --git a/src/shared/list.h b/src/shared/list.h index f0458b54e2..7ed63188ba 100644 --- a/src/shared/list.h +++ b/src/shared/list.h @@ -138,6 +138,16 @@ #define LIST_FOREACH_AFTER(name,i,p) \ for ((i) = (p)->name##_next; (i); (i) = (i)->name##_next) +/* Iterate through all the members of the list p is included in, but skip over p */ +#define LIST_FOREACH_OTHERS(name,i,p) \ + for (({ \ + (i) = (p); \ + while ((i) && (i)->name##_prev) \ + (i) = (i)->name##_prev; \ + }); \ + (i); \ + (i) = (i)->name##_next == (p) ? (p)->name##_next : (i)->name##_next) + /* Loop starting from p->next until p->prev. p can be adjusted meanwhile. */ #define LIST_LOOP_BUT_ONE(name,i,head,p) \ diff --git a/src/test/test-list.c b/src/test/test-list.c index e9d47f0503..399ec8e58e 100644 --- a/src/test/test-list.c +++ b/src/test/test-list.c @@ -38,6 +38,13 @@ int main(int argc, const char *argv[]) { LIST_PREPEND(item, head, &items[i]); } + i = 0; + LIST_FOREACH_OTHERS(item, cursor, &items[2]) { + i++; + assert_se(cursor != &items[2]); + } + assert_se(i == ELEMENTSOF(items)-1); + assert_se(!LIST_JUST_US(item, head)); assert_se(items[0].item_next == NULL); @@ -125,7 +132,7 @@ int main(int argc, const char *argv[]) { assert_se(items[3].item_prev == &items[2]); for (i = 0; i < ELEMENTSOF(items); i++) - LIST_REMOVE(item, head, &items[i]); + LIST_REMOVE(item, head, &items[i]); assert_se(head == NULL); |