forked from urbanserj/entente
-
Notifications
You must be signed in to change notification settings - Fork 1
/
dlist.h
84 lines (78 loc) · 2.17 KB
/
dlist.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
/** \file dlist.h
* A simple circular doubly linked list.
*
* \copyright Copyright (c) 2017 Donovan Baarda <abo@minkirri.apana.org.au>
*
* \licence Licensed under the GPLv3 License. See LICENSE file for details.
*
* This is used by defining ENTRY to the name of your dlist entry type and
* including this header file. The dlist is just a pointer to the head ENTRY of
* the circular dlist, and will be NULL iif the list is empty. Each entry must
* be a struct with *next and *prev ENTRY pointers.
*
* \param ENTRY - the entry type.
*
* Example: \code
* typedef struct myentry_s myentry;
* struct myentry_s {
* myentry *next, *prev;
* ...
* };
*
* #define ENTRY myentry
* #include "dlist.h"
*
* myentry *l = NULL, e;
* myentry_add(&l, e);
* for (e = l; e != NULL; e = myentry_next(&l, e))
* ...;
* \endcode */
#ifndef ENTRY
#error ENTRY needs to be defined
#endif
#define _JOIN2(x, y) x##y
#define _JOIN(x, y) _JOIN2(x, y)
#define ENTRY_add _JOIN(ENTRY, _add)
#define ENTRY_rem _JOIN(ENTRY, _rem)
#define ENTRY_next _JOIN(ENTRY, _next)
#define ENTRY_prev _JOIN(ENTRY, _prev)
/** Add an entry to a circular dlist. */
static inline void ENTRY_add(ENTRY **l, ENTRY *e)
{
if (*l) {
e->next = (*l);
e->prev = (*l)->prev;
(*l)->prev = (*l)->prev->next = e;
} else
/* Adding element to empty dlist as new head. */
*l = e->next = e->prev = e;
}
/** Remove an entry from a circular dlist. */
static inline void ENTRY_rem(ENTRY **l, ENTRY *e)
{
if (*l == e)
/* Removing head, move head to next element. */
*l = e->next;
if (*l == e) {
/* Still removing head, removing last element. */
*l = NULL;
} else {
e->prev->next = e->next;
e->next->prev = e->prev;
}
}
/** Get the next element after e or NULL if it's the last. */
static inline ENTRY *ENTRY_next(ENTRY **l, ENTRY *e)
{
return e->next == *l ? NULL : e->next;
}
/** Get the prev element before e or NULL if it's the first. */
static inline ENTRY *ENTRY_prev(ENTRY **l, ENTRY *e)
{
return e == *l ? NULL : e->prev;
}
#undef ENTRY
#undef ENTRY_add
#undef ENTRY_rem
#undef ENTRY_next
#undef ENTRY_prev