Coverage Report

Created: 2026-01-10 07:08

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/curl/lib/bufq.c
Line
Count
Source
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
#include "curl_setup.h"
25
26
#include "bufq.h"
27
28
static bool chunk_is_empty(const struct buf_chunk *chunk)
29
86.8M
{
30
86.8M
  return chunk->r_offset >= chunk->w_offset;
31
86.8M
}
32
33
static bool chunk_is_full(const struct buf_chunk *chunk)
34
28.3M
{
35
28.3M
  return chunk->w_offset >= chunk->dlen;
36
28.3M
}
37
38
static size_t chunk_len(const struct buf_chunk *chunk)
39
778M
{
40
778M
  return chunk->w_offset - chunk->r_offset;
41
778M
}
42
43
static void chunk_reset(struct buf_chunk *chunk)
44
16.9M
{
45
16.9M
  chunk->next = NULL;
46
16.9M
  chunk->r_offset = chunk->w_offset = 0;
47
16.9M
}
48
49
static size_t chunk_append(struct buf_chunk *chunk,
50
                           const uint8_t *buf, size_t len)
51
12.3M
{
52
12.3M
  uint8_t *p = &chunk->x.data[chunk->w_offset];
53
12.3M
  size_t n = chunk->dlen - chunk->w_offset;
54
12.3M
  DEBUGASSERT(chunk->dlen >= chunk->w_offset);
55
12.3M
  if(n) {
56
12.3M
    n = CURLMIN(n, len);
57
12.3M
    memcpy(p, buf, n);
58
12.3M
    chunk->w_offset += n;
59
12.3M
  }
60
12.3M
  return n;
61
12.3M
}
62
63
static size_t chunk_read(struct buf_chunk *chunk,
64
                         uint8_t *buf, size_t len)
65
2.00M
{
66
2.00M
  uint8_t *p = &chunk->x.data[chunk->r_offset];
67
2.00M
  size_t n = chunk->w_offset - chunk->r_offset;
68
2.00M
  DEBUGASSERT(chunk->w_offset >= chunk->r_offset);
69
2.00M
  if(!n) {
70
331
    return 0;
71
331
  }
72
2.00M
  else if(n <= len) {
73
2.00M
    memcpy(buf, p, n);
74
2.00M
    chunk->r_offset = chunk->w_offset = 0;
75
2.00M
    return n;
76
2.00M
  }
77
3.32k
  else {
78
3.32k
    memcpy(buf, p, len);
79
3.32k
    chunk->r_offset += len;
80
3.32k
    return len;
81
3.32k
  }
82
2.00M
}
83
84
static CURLcode chunk_slurpn(struct buf_chunk *chunk, size_t max_len,
85
                             Curl_bufq_reader *reader,
86
                             void *reader_ctx, size_t *pnread)
87
10.6M
{
88
10.6M
  uint8_t *p = &chunk->x.data[chunk->w_offset];
89
10.6M
  size_t n = chunk->dlen - chunk->w_offset; /* free amount */
90
10.6M
  CURLcode result;
91
92
10.6M
  *pnread = 0;
93
10.6M
  DEBUGASSERT(chunk->dlen >= chunk->w_offset);
94
10.6M
  if(!n)
95
0
    return CURLE_AGAIN;
96
10.6M
  if(max_len && n > max_len)
97
3.43k
    n = max_len;
98
10.6M
  result = reader(reader_ctx, p, n, pnread);
99
10.6M
  if(!result) {
100
3.94M
    DEBUGASSERT(*pnread <= n);
101
3.94M
    chunk->w_offset += *pnread;
102
3.94M
  }
103
10.6M
  return result;
104
10.6M
}
105
106
static void chunk_peek(const struct buf_chunk *chunk,
107
                       const uint8_t **pbuf, size_t *plen)
108
3.45M
{
109
3.45M
  DEBUGASSERT(chunk->w_offset >= chunk->r_offset);
110
3.45M
  *pbuf = &chunk->x.data[chunk->r_offset];
111
3.45M
  *plen = chunk->w_offset - chunk->r_offset;
112
3.45M
}
113
114
static void chunk_peek_at(const struct buf_chunk *chunk, size_t offset,
115
                          const uint8_t **pbuf, size_t *plen)
116
4.02k
{
117
4.02k
  offset += chunk->r_offset;
118
4.02k
  DEBUGASSERT(chunk->w_offset >= offset);
119
4.02k
  *pbuf = &chunk->x.data[offset];
120
4.02k
  *plen = chunk->w_offset - offset;
121
4.02k
}
122
123
static size_t chunk_skip(struct buf_chunk *chunk, size_t amount)
124
4.69M
{
125
4.69M
  size_t n = chunk->w_offset - chunk->r_offset;
126
4.69M
  DEBUGASSERT(chunk->w_offset >= chunk->r_offset);
127
4.69M
  if(n) {
128
4.69M
    n = CURLMIN(n, amount);
129
4.69M
    chunk->r_offset += n;
130
4.69M
    if(chunk->r_offset == chunk->w_offset)
131
4.68M
      chunk->r_offset = chunk->w_offset = 0;
132
4.69M
  }
133
4.69M
  return n;
134
4.69M
}
135
136
static void chunk_list_free(struct buf_chunk **anchor)
137
468k
{
138
468k
  struct buf_chunk *chunk;
139
759k
  while(*anchor) {
140
290k
    chunk = *anchor;
141
290k
    *anchor = chunk->next;
142
290k
    curlx_free(chunk);
143
290k
  }
144
468k
}
145
146
void Curl_bufcp_init(struct bufc_pool *pool,
147
                     size_t chunk_size, size_t spare_max)
148
19.9k
{
149
19.9k
  DEBUGASSERT(chunk_size > 0);
150
19.9k
  DEBUGASSERT(spare_max > 0);
151
19.9k
  memset(pool, 0, sizeof(*pool));
152
19.9k
  pool->chunk_size = chunk_size;
153
19.9k
  pool->spare_max = spare_max;
154
19.9k
}
155
156
static CURLcode bufcp_take(struct bufc_pool *pool,
157
                           struct buf_chunk **pchunk)
158
3.60M
{
159
3.60M
  struct buf_chunk *chunk = NULL;
160
161
3.60M
  if(pool->spare) {
162
3.36M
    chunk = pool->spare;
163
3.36M
    pool->spare = chunk->next;
164
3.36M
    --pool->spare_count;
165
3.36M
    chunk_reset(chunk);
166
3.36M
    *pchunk = chunk;
167
3.36M
    return CURLE_OK;
168
3.36M
  }
169
170
  /* Check for integer overflow before allocation */
171
243k
  if(pool->chunk_size > SIZE_MAX - sizeof(*chunk)) {
172
0
    *pchunk = NULL;
173
0
    return CURLE_OUT_OF_MEMORY;
174
0
  }
175
176
243k
  chunk = curlx_calloc(1, sizeof(*chunk) + pool->chunk_size);
177
243k
  if(!chunk) {
178
0
    *pchunk = NULL;
179
0
    return CURLE_OUT_OF_MEMORY;
180
0
  }
181
243k
  chunk->dlen = pool->chunk_size;
182
243k
  *pchunk = chunk;
183
243k
  return CURLE_OK;
184
243k
}
185
186
static void bufcp_put(struct bufc_pool *pool,
187
                      struct buf_chunk *chunk)
188
3.48M
{
189
3.48M
  if(pool->spare_count >= pool->spare_max) {
190
78.3k
    curlx_free(chunk);
191
78.3k
  }
192
3.40M
  else {
193
3.40M
    chunk_reset(chunk);
194
3.40M
    chunk->next = pool->spare;
195
3.40M
    pool->spare = chunk;
196
3.40M
    ++pool->spare_count;
197
3.40M
  }
198
3.48M
}
199
200
void Curl_bufcp_free(struct bufc_pool *pool)
201
19.9k
{
202
19.9k
  chunk_list_free(&pool->spare);
203
19.9k
  pool->spare_count = 0;
204
19.9k
}
205
206
static void bufq_init(struct bufq *q, struct bufc_pool *pool,
207
                      size_t chunk_size, size_t max_chunks, int opts)
208
224k
{
209
224k
  DEBUGASSERT(chunk_size > 0);
210
224k
  DEBUGASSERT(max_chunks > 0);
211
224k
  memset(q, 0, sizeof(*q));
212
224k
  q->chunk_size = chunk_size;
213
224k
  q->max_chunks = max_chunks;
214
224k
  q->pool = pool;
215
224k
  q->opts = opts;
216
224k
}
217
218
void Curl_bufq_init2(struct bufq *q, size_t chunk_size, size_t max_chunks,
219
                     int opts)
220
164k
{
221
164k
  bufq_init(q, NULL, chunk_size, max_chunks, opts);
222
164k
}
223
224
void Curl_bufq_init(struct bufq *q, size_t chunk_size, size_t max_chunks)
225
608
{
226
608
  bufq_init(q, NULL, chunk_size, max_chunks, BUFQ_OPT_NONE);
227
608
}
228
229
void Curl_bufq_initp(struct bufq *q, struct bufc_pool *pool,
230
                     size_t max_chunks, int opts)
231
59.0k
{
232
59.0k
  bufq_init(q, pool, pool->chunk_size, max_chunks, opts);
233
59.0k
}
234
235
void Curl_bufq_free(struct bufq *q)
236
224k
{
237
224k
  chunk_list_free(&q->head);
238
224k
  chunk_list_free(&q->spare);
239
224k
  q->tail = NULL;
240
224k
  q->chunk_count = 0;
241
224k
}
242
243
void Curl_bufq_reset(struct bufq *q)
244
2.48M
{
245
2.48M
  struct buf_chunk *chunk;
246
9.53M
  while(q->head) {
247
7.04M
    chunk = q->head;
248
7.04M
    q->head = chunk->next;
249
7.04M
    chunk->next = q->spare;
250
7.04M
    q->spare = chunk;
251
7.04M
  }
252
2.48M
  q->tail = NULL;
253
2.48M
}
254
255
size_t Curl_bufq_len(const struct bufq *q)
256
10.4M
{
257
10.4M
  const struct buf_chunk *chunk = q->head;
258
10.4M
  size_t len = 0;
259
786M
  while(chunk) {
260
775M
    len += chunk_len(chunk);
261
775M
    chunk = chunk->next;
262
775M
  }
263
10.4M
  return len;
264
10.4M
}
265
266
bool Curl_bufq_is_empty(const struct bufq *q)
267
144M
{
268
144M
  return !q->head || chunk_is_empty(q->head);
269
144M
}
270
271
bool Curl_bufq_is_full(const struct bufq *q)
272
370k
{
273
370k
  if(!q->tail || q->spare)
274
50.4k
    return FALSE;
275
319k
  if(q->chunk_count < q->max_chunks)
276
0
    return FALSE;
277
319k
  if(q->chunk_count > q->max_chunks)
278
590
    return TRUE;
279
  /* we have no spares and cannot make more, is the tail full? */
280
319k
  return chunk_is_full(q->tail);
281
319k
}
282
283
static struct buf_chunk *get_spare(struct bufq *q)
284
15.7M
{
285
15.7M
  struct buf_chunk *chunk = NULL;
286
287
15.7M
  if(q->spare) {
288
10.1M
    chunk = q->spare;
289
10.1M
    q->spare = chunk->next;
290
10.1M
    chunk_reset(chunk);
291
10.1M
    return chunk;
292
10.1M
  }
293
294
5.54M
  if(q->chunk_count >= q->max_chunks && (!(q->opts & BUFQ_OPT_SOFT_LIMIT)))
295
1.79M
    return NULL;
296
297
3.74M
  if(q->pool) {
298
3.60M
    if(bufcp_take(q->pool, &chunk))
299
0
      return NULL;
300
3.60M
    ++q->chunk_count;
301
3.60M
    return chunk;
302
3.60M
  }
303
141k
  else {
304
    /* Check for integer overflow before allocation */
305
141k
    if(q->chunk_size > SIZE_MAX - sizeof(*chunk)) {
306
0
      return NULL;
307
0
    }
308
309
141k
    chunk = curlx_calloc(1, sizeof(*chunk) + q->chunk_size);
310
141k
    if(!chunk)
311
0
      return NULL;
312
141k
    chunk->dlen = q->chunk_size;
313
141k
    ++q->chunk_count;
314
141k
    return chunk;
315
141k
  }
316
3.74M
}
317
318
static void prune_head(struct bufq *q)
319
6.70M
{
320
6.70M
  struct buf_chunk *chunk;
321
322
13.3M
  while(q->head && chunk_is_empty(q->head)) {
323
6.69M
    chunk = q->head;
324
6.69M
    q->head = chunk->next;
325
6.69M
    if(q->tail == chunk)
326
174k
      q->tail = q->head;
327
6.69M
    if(q->pool) {
328
3.48M
      bufcp_put(q->pool, chunk);
329
3.48M
      --q->chunk_count;
330
3.48M
    }
331
3.20M
    else if((q->chunk_count > q->max_chunks) ||
332
3.19M
            (q->opts & BUFQ_OPT_NO_SPARES)) {
333
      /* SOFT_LIMIT allowed us more than max. free spares until
334
       * we are at max again. Or free them if we are configured
335
       * to not use spares. */
336
16.0k
      curlx_free(chunk);
337
16.0k
      --q->chunk_count;
338
16.0k
    }
339
3.19M
    else {
340
3.19M
      chunk->next = q->spare;
341
3.19M
      q->spare = chunk;
342
3.19M
    }
343
6.69M
  }
344
6.70M
}
345
346
static struct buf_chunk *get_non_full_tail(struct bufq *q)
347
24.8M
{
348
24.8M
  struct buf_chunk *chunk;
349
350
24.8M
  if(q->tail && !chunk_is_full(q->tail))
351
9.09M
    return q->tail;
352
15.7M
  chunk = get_spare(q);
353
15.7M
  if(chunk) {
354
    /* new tail, and possibly new head */
355
13.9M
    if(q->tail) {
356
13.7M
      q->tail->next = chunk;
357
13.7M
      q->tail = chunk;
358
13.7M
    }
359
217k
    else {
360
217k
      DEBUGASSERT(!q->head);
361
217k
      q->head = q->tail = chunk;
362
217k
    }
363
13.9M
  }
364
15.7M
  return chunk;
365
15.7M
}
366
367
CURLcode Curl_bufq_write(struct bufq *q,
368
                         const uint8_t *buf, size_t len,
369
                         size_t *pnwritten)
370
9.84M
{
371
9.84M
  struct buf_chunk *tail;
372
9.84M
  size_t n;
373
374
9.84M
  DEBUGASSERT(q->max_chunks > 0);
375
9.84M
  *pnwritten = 0;
376
22.2M
  while(len) {
377
13.4M
    tail = get_non_full_tail(q);
378
13.4M
    if(!tail) {
379
1.04M
      if((q->chunk_count < q->max_chunks) || (q->opts & BUFQ_OPT_SOFT_LIMIT))
380
        /* should have gotten a tail, but did not */
381
0
        return CURLE_OUT_OF_MEMORY;
382
1.04M
      break;
383
1.04M
    }
384
12.3M
    n = chunk_append(tail, buf, len);
385
12.3M
    if(!n)
386
0
      break;
387
12.3M
    *pnwritten += n;
388
12.3M
    buf += n;
389
12.3M
    len -= n;
390
12.3M
  }
391
9.84M
  return (!*pnwritten && len) ? CURLE_AGAIN : CURLE_OK;
392
9.84M
}
393
394
CURLcode Curl_bufq_cwrite(struct bufq *q,
395
                          const char *buf, size_t len,
396
                          size_t *pnwritten)
397
4.56M
{
398
4.56M
  return Curl_bufq_write(q, (const uint8_t *)buf, len, pnwritten);
399
4.56M
}
400
401
CURLcode Curl_bufq_read(struct bufq *q, uint8_t *buf, size_t len,
402
                        size_t *pnread)
403
252k
{
404
252k
  *pnread = 0;
405
2.25M
  while(len && q->head) {
406
2.00M
    size_t n = chunk_read(q->head, buf, len);
407
2.00M
    if(n) {
408
2.00M
      *pnread += n;
409
2.00M
      buf += n;
410
2.00M
      len -= n;
411
2.00M
    }
412
2.00M
    prune_head(q);
413
2.00M
  }
414
252k
  return (!*pnread) ? CURLE_AGAIN : CURLE_OK;
415
252k
}
416
417
CURLcode Curl_bufq_cread(struct bufq *q, char *buf, size_t len,
418
                         size_t *pnread)
419
4.47k
{
420
4.47k
  return Curl_bufq_read(q, (uint8_t *)buf, len, pnread);
421
4.47k
}
422
423
bool Curl_bufq_peek(struct bufq *q,
424
                    const uint8_t **pbuf, size_t *plen)
425
4.33M
{
426
4.33M
  if(q->head && chunk_is_empty(q->head)) {
427
4.02k
    prune_head(q);
428
4.02k
  }
429
4.33M
  if(q->head && !chunk_is_empty(q->head)) {
430
3.45M
    chunk_peek(q->head, pbuf, plen);
431
3.45M
    return TRUE;
432
3.45M
  }
433
885k
  *pbuf = NULL;
434
885k
  *plen = 0;
435
885k
  return FALSE;
436
4.33M
}
437
438
bool Curl_bufq_peek_at(struct bufq *q, size_t offset,
439
                       const uint8_t **pbuf, size_t *plen)
440
228k
{
441
228k
  struct buf_chunk *c = q->head;
442
228k
  size_t clen;
443
444
2.47M
  while(c) {
445
2.25M
    clen = chunk_len(c);
446
2.25M
    if(!clen)
447
275
      break;
448
2.25M
    if(offset >= clen) {
449
2.25M
      offset -= clen;
450
2.25M
      c = c->next;
451
2.25M
      continue;
452
2.25M
    }
453
4.02k
    chunk_peek_at(c, offset, pbuf, plen);
454
4.02k
    return TRUE;
455
2.25M
  }
456
224k
  *pbuf = NULL;
457
224k
  *plen = 0;
458
224k
  return FALSE;
459
228k
}
460
461
void Curl_bufq_skip(struct bufq *q, size_t amount)
462
4.07M
{
463
4.07M
  size_t n;
464
465
8.76M
  while(amount && q->head) {
466
4.69M
    n = chunk_skip(q->head, amount);
467
4.69M
    amount -= n;
468
4.69M
    prune_head(q);
469
4.69M
  }
470
4.07M
}
471
472
CURLcode Curl_bufq_pass(struct bufq *q, Curl_bufq_writer *writer,
473
                        void *writer_ctx, size_t *pwritten)
474
843k
{
475
843k
  const uint8_t *buf;
476
843k
  size_t blen;
477
843k
  CURLcode result = CURLE_OK;
478
479
843k
  *pwritten = 0;
480
3.30M
  while(Curl_bufq_peek(q, &buf, &blen)) {
481
2.46M
    size_t chunk_written;
482
483
2.46M
    result = writer(writer_ctx, buf, blen, &chunk_written);
484
2.46M
    if(result) {
485
5.33k
      if((result == CURLE_AGAIN) && *pwritten) {
486
        /* blocked on subsequent write, report success */
487
4.11k
        result = CURLE_OK;
488
4.11k
      }
489
5.33k
      break;
490
5.33k
    }
491
2.45M
    if(!chunk_written) {
492
0
      if(!*pwritten) {
493
        /* treat as blocked */
494
0
        result = CURLE_AGAIN;
495
0
      }
496
0
      break;
497
0
    }
498
2.45M
    *pwritten += chunk_written;
499
2.45M
    Curl_bufq_skip(q, chunk_written);
500
2.45M
  }
501
843k
  return result;
502
843k
}
503
504
CURLcode Curl_bufq_write_pass(struct bufq *q,
505
                              const uint8_t *buf, size_t len,
506
                              Curl_bufq_writer *writer, void *writer_ctx,
507
                              size_t *pwritten)
508
83.0k
{
509
83.0k
  CURLcode result = CURLE_OK;
510
83.0k
  size_t n;
511
512
83.0k
  *pwritten = 0;
513
168k
  while(len) {
514
85.9k
    if(Curl_bufq_is_full(q)) {
515
      /* try to make room in case we are full */
516
2.90k
      result = Curl_bufq_pass(q, writer, writer_ctx, &n);
517
2.90k
      if(result) {
518
0
        if(result != CURLE_AGAIN) {
519
          /* real error, fail */
520
0
          return result;
521
0
        }
522
        /* would block, bufq is full, give up */
523
0
        break;
524
0
      }
525
2.90k
    }
526
527
    /* Add to bufq as much as there is room for */
528
85.9k
    result = Curl_bufq_write(q, buf, len, &n);
529
85.9k
    if(result) {
530
0
      if(result != CURLE_AGAIN)
531
        /* real error, fail */
532
0
        return result;
533
      /* result == CURLE_AGAIN */
534
0
      if(*pwritten)
535
        /* we did write successfully before */
536
0
        result = CURLE_OK;
537
0
      return result;
538
0
    }
539
85.9k
    else if(n == 0)
540
      /* edge case of writer returning 0 (and len is >0)
541
       * break or we might enter an infinite loop here */
542
0
      break;
543
544
    /* Track what we added to bufq */
545
85.9k
    buf += n;
546
85.9k
    len -= n;
547
85.9k
    *pwritten += n;
548
85.9k
  }
549
550
83.0k
  return (!*pwritten && len) ? CURLE_AGAIN : CURLE_OK;
551
83.0k
}
552
553
CURLcode Curl_bufq_sipn(struct bufq *q, size_t max_len,
554
                        Curl_bufq_reader *reader, void *reader_ctx,
555
                        size_t *pnread)
556
11.3M
{
557
11.3M
  struct buf_chunk *tail = NULL;
558
559
11.3M
  *pnread = 0;
560
11.3M
  tail = get_non_full_tail(q);
561
11.3M
  if(!tail) {
562
749k
    if(q->chunk_count < q->max_chunks)
563
0
      return CURLE_OUT_OF_MEMORY;
564
    /* full, blocked */
565
749k
    return CURLE_AGAIN;
566
749k
  }
567
568
10.6M
  return chunk_slurpn(tail, max_len, reader, reader_ctx, pnread);
569
11.3M
}
570
571
/**
572
 * Read up to `max_len` bytes and append it to the end of the buffer queue.
573
 * if `max_len` is 0, no limit is imposed and the call behaves exactly
574
 * the same as `Curl_bufq_slurp()`.
575
 * Returns the total amount of buf read (may be 0) in `pnread` or error
576
 * Note that even in case of an error chunks may have been read and
577
 * the buffer queue will have different length than before.
578
 */
579
static CURLcode bufq_slurpn(struct bufq *q, size_t max_len,
580
                            Curl_bufq_reader *reader, void *reader_ctx,
581
                            size_t *pnread)
582
397k
{
583
397k
  CURLcode result;
584
585
397k
  *pnread = 0;
586
3.86M
  while(1) {
587
3.86M
    size_t n;
588
3.86M
    result = Curl_bufq_sipn(q, max_len, reader, reader_ctx, &n);
589
3.86M
    if(result) {
590
393k
      if(!*pnread || result != CURLE_AGAIN) {
591
        /* blocked on first read or real error, fail */
592
384k
        return result;
593
384k
      }
594
8.35k
      result = CURLE_OK;
595
8.35k
      break;
596
393k
    }
597
3.47M
    else if(n == 0) {
598
      /* eof, result remains CURLE_OK */
599
0
      break;
600
0
    }
601
3.47M
    *pnread += n;
602
3.47M
    if(max_len) {
603
0
      DEBUGASSERT(n <= max_len);
604
0
      max_len -= n;
605
0
      if(!max_len)
606
0
        break;
607
0
    }
608
    /* give up slurping when we get less bytes than we asked for */
609
3.47M
    if(q->tail && !chunk_is_full(q->tail))
610
4.60k
      break;
611
3.47M
  }
612
12.9k
  return result;
613
397k
}
614
615
CURLcode Curl_bufq_slurp(struct bufq *q, Curl_bufq_reader *reader,
616
                         void *reader_ctx, size_t *pnread)
617
397k
{
618
397k
  return bufq_slurpn(q, 0, reader, reader_ctx, pnread);
619
397k
}