-
Notifications
You must be signed in to change notification settings - Fork 10
/
tiny_list.h
139 lines (120 loc) · 3.46 KB
/
tiny_list.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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
/*!
* @file
* @brief Linked list that requires nodes to be allocated by clients.
*
* This is an intrusive list so stores node data directly in nodes. See:
* https://www.data-structures-in-practice.com/intrusive-linked-lists/
*
* Nodes can contain arbitrary data by defining a type that contains a
* tiny_list_node_t:
*
* typedef struct client_node_t {
* tiny_list_node_t node;
* int data;
* }
*
* This type must be cast to a tiny_list_node_t to be added but can be cast back
* by the client so that the data can be accessed.
*/
#ifndef tiny_list_h
#define tiny_list_h
#include <stdbool.h>
#include <stddef.h>
#include <stdint.h>
typedef struct tiny_list_node_t {
struct tiny_list_node_t* next;
} tiny_list_node_t;
typedef struct {
tiny_list_node_t head;
} tiny_list_t;
typedef struct {
tiny_list_node_t* current;
} tiny_list_iterator_t;
/*!
* Initializes the list.
*/
inline void tiny_list_init(tiny_list_t* self)
{
self->head.next = &self->head;
}
/*!
* Adds the node to the front of the list.
*/
inline void tiny_list_push_front(tiny_list_t* self, tiny_list_node_t* node)
{
node->next = self->head.next;
self->head.next = node;
}
/*!
* Adds the node to the back of the list.
*/
void tiny_list_push_back(tiny_list_t* self, tiny_list_node_t* node);
/*!
* Inserts a node after the specified node.
*/
inline void tiny_list_insert_after(tiny_list_t* self, tiny_list_node_t* after, tiny_list_node_t* to_insert)
{
(void)self;
to_insert->next = after->next;
after->next = to_insert;
}
/*!
* Removes the node from the front of the list. Returns the node.
*/
inline tiny_list_node_t* tiny_list_pop_front(tiny_list_t* self)
{
tiny_list_node_t* popped = self->head.next;
self->head.next = self->head.next->next;
return popped;
}
/*!
* Removes the node at the back of the list. Returns the node.
*/
tiny_list_node_t* tiny_list_pop_back(tiny_list_t* self);
/*!
* Removes a specified node if present in the list.
*/
void tiny_list_remove(tiny_list_t* self, tiny_list_node_t* node);
/*!
* Returns the number of nodes contained in the list.
*/
uint16_t tiny_list_count(tiny_list_t* self);
/*!
* Returns true if the specified node is in the list and false otherwise.
*/
bool tiny_list_contains(tiny_list_t* self, tiny_list_node_t* node);
/*!
* Gives the index of a given node in the list.
*/
uint16_t tiny_list_index_of(tiny_list_t* self, tiny_list_node_t* node);
/*!
* Initialize an iterator for the provided list.
*/
inline void tiny_list_iterator_init(tiny_list_iterator_t* self, tiny_list_t* list)
{
self->current = list->head.next;
}
/*!
* Return a pointer to the next node or NULL if there are no more nodes.
*/
inline tiny_list_node_t* tiny_list_iterator_next(tiny_list_iterator_t* self, tiny_list_t* list)
{
if(self->current == &list->head) {
return NULL;
}
else {
tiny_list_node_t* item = self->current;
self->current = self->current->next;
return item;
}
}
#define tiny_list_for_each(_list, _type, _item, ...) \
do { \
tiny_list_iterator_t _it; \
tiny_list_iterator_init(&_it, _list); \
_type* _item; \
while((_item = (_type*)tiny_list_iterator_next(&_it, _list))) { \
__VA_ARGS__ \
} \
} while(0)
#endif