Coverage Report

Created: 2025-08-26 07:09

/src/curl/lib/llist.c
Line
Count
Source (jump to first uncovered line)
1
/***************************************************************************
2
 *                                  _   _ ____  _
3
 *  Project                     ___| | | |  _ \| |
4
 *                             / __| | | | |_) | |
5
 *                            | (__| |_| |  _ <| |___
6
 *                             \___|\___/|_| \_\_____|
7
 *
8
 * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
9
 *
10
 * This software is licensed as described in the file COPYING, which
11
 * you should have received as part of this distribution. The terms
12
 * are also available at https://curl.se/docs/copyright.html.
13
 *
14
 * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15
 * copies of the Software, and permit persons to whom the Software is
16
 * furnished to do so, under the terms of the COPYING file.
17
 *
18
 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19
 * KIND, either express or implied.
20
 *
21
 * SPDX-License-Identifier: curl
22
 *
23
 ***************************************************************************/
24
25
#include "curl_setup.h"
26
27
#include <curl/curl.h>
28
29
#include "llist.h"
30
#include "curl_memory.h"
31
32
/* this must be the last include file */
33
#include "memdebug.h"
34
35
#ifdef DEBUGBUILD
36
271k
#define LLISTINIT 0x100cc001 /* random pattern */
37
1.39k
#define NODEINIT  0x12344321 /* random pattern */
38
1.39k
#define NODEREM   0x54321012 /* random pattern */
39
40
680k
#define VERIFYNODE(x) verifynode(x)
41
static struct Curl_llist_node *verifynode(struct Curl_llist_node *n)
42
680k
{
43
680k
  DEBUGASSERT(!n || (n->_init == NODEINIT));
44
680k
  return n;
45
680k
}
46
#else
47
#define VERIFYNODE(x) x
48
#endif
49
/*
50
 * @unittest: 1300
51
 */
52
void
53
Curl_llist_init(struct Curl_llist *l, Curl_llist_dtor dtor)
54
271k
{
55
271k
  l->_size = 0;
56
271k
  l->_dtor = dtor;
57
271k
  l->_head = NULL;
58
271k
  l->_tail = NULL;
59
271k
#ifdef DEBUGBUILD
60
271k
  l->_init = LLISTINIT;
61
271k
#endif
62
271k
}
63
64
/*
65
 * Curl_llist_insert_next()
66
 *
67
 * Inserts a new list element after the given one 'e'. If the given existing
68
 * entry is NULL and the list already has elements, the new one will be
69
 * inserted first in the list.
70
 *
71
 * The 'ne' argument should be a pointer into the object to store.
72
 *
73
 * @unittest: 1300
74
 */
75
void
76
Curl_llist_insert_next(struct Curl_llist *list,
77
                       struct Curl_llist_node *e, /* may be NULL */
78
                       const void *p,
79
                       struct Curl_llist_node *ne)
80
1.39k
{
81
1.39k
  DEBUGASSERT(list);
82
1.39k
  DEBUGASSERT(list->_init == LLISTINIT);
83
1.39k
  DEBUGASSERT(ne);
84
85
1.39k
#ifdef DEBUGBUILD
86
1.39k
  ne->_init = NODEINIT;
87
1.39k
#endif
88
1.39k
  ne->_ptr = CURL_UNCONST(p);
89
1.39k
  ne->_list = list;
90
1.39k
  if(list->_size == 0) {
91
1.39k
    list->_head = ne;
92
1.39k
    list->_head->_prev = NULL;
93
1.39k
    list->_head->_next = NULL;
94
1.39k
    list->_tail = ne;
95
1.39k
  }
96
0
  else {
97
    /* if 'e' is NULL here, we insert the new element first in the list */
98
0
    ne->_next = e ? e->_next : list->_head;
99
0
    ne->_prev = e;
100
0
    if(!e) {
101
0
      list->_head->_prev = ne;
102
0
      list->_head = ne;
103
0
    }
104
0
    else if(e->_next) {
105
0
      e->_next->_prev = ne;
106
0
    }
107
0
    else {
108
0
      list->_tail = ne;
109
0
    }
110
0
    if(e)
111
0
      e->_next = ne;
112
0
  }
113
114
1.39k
  ++list->_size;
115
1.39k
}
116
117
/*
118
 * Curl_llist_append()
119
 *
120
 * Adds a new list element to the end of the list.
121
 *
122
 * The 'ne' argument should be a pointer into the object to store.
123
 *
124
 * @unittest: 1300
125
 */
126
void
127
Curl_llist_append(struct Curl_llist *list, const void *p,
128
                  struct Curl_llist_node *ne)
129
1.39k
{
130
1.39k
  DEBUGASSERT(list);
131
1.39k
  DEBUGASSERT(list->_init == LLISTINIT);
132
1.39k
  DEBUGASSERT(ne);
133
1.39k
  Curl_llist_insert_next(list, list->_tail, p, ne);
134
1.39k
}
135
136
void *Curl_node_take_elem(struct Curl_llist_node *e)
137
1.39k
{
138
1.39k
  void *ptr;
139
1.39k
  struct Curl_llist *list;
140
1.39k
  if(!e)
141
0
    return NULL;
142
143
1.39k
  list = e->_list;
144
1.39k
  DEBUGASSERT(list);
145
1.39k
  DEBUGASSERT(list->_init == LLISTINIT);
146
1.39k
  DEBUGASSERT(list->_size);
147
1.39k
  DEBUGASSERT(e->_init == NODEINIT);
148
1.39k
  if(list) {
149
1.39k
    if(e == list->_head) {
150
1.39k
      list->_head = e->_next;
151
152
1.39k
      if(!list->_head)
153
1.39k
        list->_tail = NULL;
154
0
      else
155
0
        e->_next->_prev = NULL;
156
1.39k
    }
157
0
    else {
158
0
      if(e->_prev)
159
0
        e->_prev->_next = e->_next;
160
161
0
      if(!e->_next)
162
0
        list->_tail = e->_prev;
163
0
      else
164
0
        e->_next->_prev = e->_prev;
165
0
    }
166
1.39k
    --list->_size;
167
1.39k
  }
168
1.39k
  ptr = e->_ptr;
169
170
1.39k
  e->_list = NULL;
171
1.39k
  e->_ptr  = NULL;
172
1.39k
  e->_prev = NULL;
173
1.39k
  e->_next = NULL;
174
1.39k
#ifdef DEBUGBUILD
175
1.39k
  e->_init = NODEREM; /* specific pattern on remove - not zero */
176
1.39k
#endif
177
178
1.39k
  return ptr;
179
1.39k
}
180
181
/*
182
 * @unittest: 1300
183
 */
184
UNITTEST void Curl_node_uremove(struct Curl_llist_node *, void *);
185
UNITTEST void
186
Curl_node_uremove(struct Curl_llist_node *e, void *user)
187
1.39k
{
188
1.39k
  struct Curl_llist *list;
189
1.39k
  void *ptr;
190
1.39k
  if(!e)
191
0
    return;
192
193
1.39k
  list = e->_list;
194
1.39k
  DEBUGASSERT(list);
195
1.39k
  if(list) {
196
1.39k
    ptr = Curl_node_take_elem(e);
197
1.39k
    if(list->_dtor)
198
0
      list->_dtor(user, ptr);
199
1.39k
  }
200
1.39k
}
201
202
void Curl_node_remove(struct Curl_llist_node *e)
203
1.39k
{
204
1.39k
  Curl_node_uremove(e, NULL);
205
1.39k
}
206
207
void
208
Curl_llist_destroy(struct Curl_llist *list, void *user)
209
0
{
210
0
  if(list) {
211
0
    DEBUGASSERT(list->_init == LLISTINIT);
212
0
    while(list->_size > 0)
213
0
      Curl_node_uremove(list->_tail, user);
214
0
  }
215
0
}
216
217
/* Curl_llist_head() returns the first 'struct Curl_llist_node *', which
218
   might be NULL */
219
struct Curl_llist_node *Curl_llist_head(struct Curl_llist *list)
220
677k
{
221
677k
  DEBUGASSERT(list);
222
677k
  DEBUGASSERT(list->_init == LLISTINIT);
223
677k
  return VERIFYNODE(list->_head);
224
677k
}
225
226
#ifdef UNITTESTS
227
/* Curl_llist_tail() returns the last 'struct Curl_llist_node *', which
228
   might be NULL */
229
struct Curl_llist_node *Curl_llist_tail(struct Curl_llist *list)
230
{
231
  DEBUGASSERT(list);
232
  DEBUGASSERT(list->_init == LLISTINIT);
233
  return VERIFYNODE(list->_tail);
234
}
235
#endif
236
237
/* Curl_llist_count() returns a size_t the number of nodes in the list */
238
size_t Curl_llist_count(struct Curl_llist *list)
239
0
{
240
0
  DEBUGASSERT(list);
241
0
  DEBUGASSERT(list->_init == LLISTINIT);
242
0
  return list->_size;
243
0
}
244
245
/* Curl_node_elem() returns the custom data from a Curl_llist_node */
246
void *Curl_node_elem(struct Curl_llist_node *n)
247
3.67k
{
248
3.67k
  DEBUGASSERT(n);
249
3.67k
  DEBUGASSERT(n->_init == NODEINIT);
250
3.67k
  return n->_ptr;
251
3.67k
}
252
253
/* Curl_node_next() returns the next element in a list from a given
254
   Curl_llist_node */
255
struct Curl_llist_node *Curl_node_next(struct Curl_llist_node *n)
256
3.67k
{
257
3.67k
  DEBUGASSERT(n);
258
3.67k
  DEBUGASSERT(n->_init == NODEINIT);
259
3.67k
  return VERIFYNODE(n->_next);
260
3.67k
}
261
262
#ifdef UNITTESTS
263
264
/* Curl_node_prev() returns the previous element in a list from a given
265
   Curl_llist_node */
266
struct Curl_llist_node *Curl_node_prev(struct Curl_llist_node *n)
267
{
268
  DEBUGASSERT(n);
269
  DEBUGASSERT(n->_init == NODEINIT);
270
  return VERIFYNODE(n->_prev);
271
}
272
273
#endif
274
275
struct Curl_llist *Curl_node_llist(struct Curl_llist_node *n)
276
0
{
277
0
  DEBUGASSERT(n);
278
0
  DEBUGASSERT(!n->_list || n->_init == NODEINIT);
279
0
  return n->_list;
280
0
}