-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathn_list_remove_ex.c
79 lines (60 loc) · 1.52 KB
/
n_list_remove_ex.c
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
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include "n_list_int.h"
int n_list_remove_ex(tn_list *l, const void *data, t_fn_cmp cmpf)
{
register struct list_node *node, *prev_node;
int removed = 0;
if (l->head == NULL)
return 0;
if (cmpf == NULL)
cmpf = l->cmp_fn;
n_assert(cmpf != NULL);
if (cmpf == NULL) {
trurl_die("n_list_remove_ex: compare function is NULL");
return -1;
}
node = l->head;
/* remove from the beginning */
while (node != NULL && cmpf(node->data, data) == 0) {
if (l->free_fn != NULL)
l->free_fn(node->data);
l->head = l->head->next;
free(node);
node = l->head;
l->items--;
removed++;
if (l->items > 1)
n_assert(l->head->next);
}
if (l->head == NULL) {
n_assert(l->items == 0);
return removed;
} else if (removed > 0 && (l->flags & TN_LIST_UNIQ)) {
return removed;
} else {
for (prev_node = l->head, node = l->head->next; node != NULL;
prev_node = node, node = node->next) {
if (cmpf(node->data, data) == 0) {
if (node->data != NULL && l->free_fn != NULL)
l->free_fn(node->data);
n_assert(prev_node != NULL);
prev_node->next = node->next;
if (node == l->tail)
l->tail = prev_node;
free(node);
n_assert(l->items > 0);
l->items--;
removed++;
if (l->flags & TN_LIST_UNIQ) {
break;
} else { /* duplicates allowed */
n_assert(prev_node != NULL);
node = prev_node;
}
}
}
}
return removed;
}