Coverage Report

Created: 2026-03-11 07:11

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/curl/lib/sendf.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
#ifdef HAVE_NETINET_IN_H
27
#include <netinet/in.h>
28
#endif
29
30
#ifdef HAVE_LINUX_TCP_H
31
#include <linux/tcp.h>
32
#elif defined(HAVE_NETINET_TCP_H)
33
#include <netinet/tcp.h>
34
#endif
35
36
#include "urldata.h"
37
#include "sendf.h"
38
#include "curl_trc.h"
39
#include "transfer.h"
40
#include "cfilters.h"
41
#include "connect.h"
42
#include "cw-out.h"
43
#include "cw-pause.h"
44
#include "multiif.h"
45
#include "progress.h"
46
47
static void cl_reset_writer(struct Curl_easy *data)
48
0
{
49
0
  struct Curl_cwriter *writer = data->req.writer_stack;
50
0
  while(writer) {
51
0
    data->req.writer_stack = writer->next;
52
0
    writer->cwt->do_close(data, writer);
53
0
    curlx_free(writer);
54
0
    writer = data->req.writer_stack;
55
0
  }
56
0
}
57
58
static void cl_reset_reader(struct Curl_easy *data)
59
0
{
60
0
  struct Curl_creader *reader = data->req.reader_stack;
61
0
  data->req.reader_started = FALSE;
62
0
  while(reader) {
63
0
    data->req.reader_stack = reader->next;
64
0
    reader->crt->do_close(data, reader);
65
0
    curlx_free(reader);
66
0
    reader = data->req.reader_stack;
67
0
  }
68
0
}
69
70
void Curl_client_cleanup(struct Curl_easy *data)
71
0
{
72
0
  cl_reset_reader(data);
73
0
  cl_reset_writer(data);
74
75
0
  data->req.bytecount = 0;
76
0
  data->req.headerline = 0;
77
0
}
78
79
void Curl_client_reset(struct Curl_easy *data)
80
0
{
81
0
  if(data->req.rewind_read) {
82
    /* already requested */
83
0
    CURL_TRC_READ(data, "client_reset, will rewind reader");
84
0
  }
85
0
  else {
86
0
    CURL_TRC_READ(data, "client_reset, clear readers");
87
0
    cl_reset_reader(data);
88
0
  }
89
0
  cl_reset_writer(data);
90
91
0
  data->req.bytecount = 0;
92
0
  data->req.headerline = 0;
93
0
}
94
95
CURLcode Curl_client_start(struct Curl_easy *data)
96
0
{
97
0
  if(data->req.rewind_read) {
98
0
    struct Curl_creader *r = data->req.reader_stack;
99
0
    CURLcode result = CURLE_OK;
100
101
0
    CURL_TRC_READ(data, "client start, rewind readers");
102
0
    while(r) {
103
0
      result = r->crt->cntrl(data, r, CURL_CRCNTRL_REWIND);
104
0
      if(result) {
105
0
        failf(data, "rewind of client reader '%s' failed: %d",
106
0
              r->crt->name, result);
107
0
        return result;
108
0
      }
109
0
      r = r->next;
110
0
    }
111
0
    data->req.rewind_read = FALSE;
112
0
    cl_reset_reader(data);
113
0
  }
114
0
  return CURLE_OK;
115
0
}
116
117
bool Curl_creader_will_rewind(struct Curl_easy *data)
118
0
{
119
0
  return (bool)data->req.rewind_read;
120
0
}
121
122
void Curl_creader_set_rewind(struct Curl_easy *data, bool enable)
123
0
{
124
0
  data->req.rewind_read = !!enable;
125
0
}
126
127
/* Write data using an unencoding writer stack. */
128
CURLcode Curl_cwriter_write(struct Curl_easy *data,
129
                            struct Curl_cwriter *writer, int type,
130
                            const char *buf, size_t nbytes)
131
0
{
132
0
  if(!writer)
133
0
    return CURLE_WRITE_ERROR;
134
0
  return writer->cwt->do_write(data, writer, type, buf, nbytes);
135
0
}
136
137
CURLcode Curl_cwriter_def_init(struct Curl_easy *data,
138
                               struct Curl_cwriter *writer)
139
0
{
140
0
  (void)data;
141
0
  (void)writer;
142
0
  return CURLE_OK;
143
0
}
144
145
CURLcode Curl_cwriter_def_write(struct Curl_easy *data,
146
                                struct Curl_cwriter *writer, int type,
147
                                const char *buf, size_t nbytes)
148
0
{
149
0
  return Curl_cwriter_write(data, writer->next, type, buf, nbytes);
150
0
}
151
152
void Curl_cwriter_def_close(struct Curl_easy *data,
153
                            struct Curl_cwriter *writer)
154
0
{
155
0
  (void)data;
156
0
  (void)writer;
157
0
}
158
159
static size_t get_max_body_write_len(struct Curl_easy *data, curl_off_t limit)
160
0
{
161
0
  if(limit != -1) {
162
    /* How much more are we allowed to write? */
163
0
    return curlx_sotouz_range(limit - data->req.bytecount, 0, SIZE_MAX);
164
0
  }
165
0
  return SIZE_MAX;
166
0
}
167
168
struct cw_download_ctx {
169
  struct Curl_cwriter super;
170
  BIT(started_response);
171
  BIT(started_body);
172
};
173
174
/* Download client writer in phase CURL_CW_PROTOCOL that
175
 * sees the "real" download body data. */
176
static CURLcode cw_download_write(struct Curl_easy *data,
177
                                  struct Curl_cwriter *writer, int type,
178
                                  const char *buf, size_t nbytes)
179
0
{
180
0
  struct cw_download_ctx *ctx = writer->ctx;
181
0
  CURLcode result;
182
0
  size_t nwrite, excess_len = 0;
183
0
  bool is_connect = !!(type & CLIENTWRITE_CONNECT);
184
185
0
  if(!ctx->started_response &&
186
0
     !(type & (CLIENTWRITE_INFO | CLIENTWRITE_CONNECT))) {
187
0
    Curl_pgrsTime(data, TIMER_STARTTRANSFER);
188
0
    ctx->started_response = TRUE;
189
0
  }
190
191
0
  if(!(type & CLIENTWRITE_BODY)) {
192
0
    if(is_connect && data->set.suppress_connect_headers)
193
0
      return CURLE_OK;
194
0
    result = Curl_cwriter_write(data, writer->next, type, buf, nbytes);
195
0
    CURL_TRC_WRITE(data, "download_write header(type=%x, blen=%zu) -> %d",
196
0
                   type, nbytes, result);
197
0
    return result;
198
0
  }
199
200
0
  if(!ctx->started_body &&
201
0
     !(type & (CLIENTWRITE_INFO | CLIENTWRITE_CONNECT))) {
202
0
    Curl_rlimit_start(&data->progress.dl.rlimit, Curl_pgrs_now(data),
203
0
                      data->req.size);
204
0
    ctx->started_body = TRUE;
205
0
  }
206
207
  /* Here, we deal with REAL BODY bytes. All filtering and transfer
208
   * encodings have been applied and only the true content, e.g. BODY,
209
   * bytes are passed here.
210
   * This allows us to check sizes, update stats, etc. independent
211
   * from the protocol in play. */
212
213
0
  if(data->req.no_body && nbytes > 0) {
214
    /* BODY arrives although we want none, bail out */
215
0
    streamclose(data->conn, "ignoring body");
216
0
    CURL_TRC_WRITE(data, "download_write body(type=%x, blen=%zu), "
217
0
                   "did not want a BODY", type, nbytes);
218
0
    data->req.download_done = TRUE;
219
0
    if(data->info.header_size)
220
      /* if headers have been received, this is fine */
221
0
      return CURLE_OK;
222
0
    return CURLE_WEIRD_SERVER_REPLY;
223
0
  }
224
225
  /* Determine if we see any bytes in excess to what is allowed.
226
   * We write the allowed bytes and handle excess further below.
227
   * This gives deterministic BODY writes on varying buffer receive
228
   * lengths. */
229
0
  nwrite = nbytes;
230
0
  if(data->req.maxdownload != -1) {
231
0
    size_t wmax = get_max_body_write_len(data, data->req.maxdownload);
232
0
    if(nwrite > wmax) {
233
0
      excess_len = nbytes - wmax;
234
0
      nwrite = wmax;
235
0
    }
236
237
0
    if(nwrite == wmax) {
238
0
      data->req.download_done = TRUE;
239
0
    }
240
241
0
    if((type & CLIENTWRITE_EOS) && !data->req.no_body &&
242
0
       (data->req.size > data->req.bytecount)) {
243
0
      failf(data, "end of response with %" FMT_OFF_T " bytes missing",
244
0
            data->req.size - data->req.bytecount);
245
0
      return CURLE_PARTIAL_FILE;
246
0
    }
247
0
  }
248
249
  /* Error on too large filesize is handled below, after writing
250
   * the permitted bytes */
251
0
  if(data->set.max_filesize && !data->req.ignorebody) {
252
0
    size_t wmax = get_max_body_write_len(data, data->set.max_filesize);
253
0
    if(nwrite > wmax) {
254
0
      nwrite = wmax;
255
0
    }
256
0
  }
257
258
0
  if(!data->req.ignorebody && (nwrite || (type & CLIENTWRITE_EOS))) {
259
0
    result = Curl_cwriter_write(data, writer->next, type, buf, nwrite);
260
0
    CURL_TRC_WRITE(data, "download_write body(type=%x, blen=%zu) -> %d",
261
0
                   type, nbytes, result);
262
0
    if(result)
263
0
      return result;
264
0
  }
265
266
  /* Update stats, write and report progress */
267
0
  if(nwrite) {
268
0
    data->req.bytecount += nwrite;
269
0
    Curl_pgrs_download_inc(data, nwrite);
270
0
  }
271
272
0
  if(excess_len) {
273
0
    if(!data->req.ignorebody) {
274
0
      infof(data,
275
0
            "Excess found writing body:"
276
0
            " excess = %zu"
277
0
            ", size = %" FMT_OFF_T
278
0
            ", maxdownload = %" FMT_OFF_T
279
0
            ", bytecount = %" FMT_OFF_T,
280
0
            excess_len, data->req.size, data->req.maxdownload,
281
0
            data->req.bytecount);
282
0
      connclose(data->conn, "excess found in a read");
283
0
    }
284
0
  }
285
0
  else if((nwrite < nbytes) && !data->req.ignorebody) {
286
0
    failf(data, "Exceeded the maximum allowed file size "
287
0
          "(%" FMT_OFF_T ") with %" FMT_OFF_T " bytes",
288
0
          data->set.max_filesize, data->req.bytecount);
289
0
    return CURLE_FILESIZE_EXCEEDED;
290
0
  }
291
292
0
  return CURLE_OK;
293
0
}
294
295
static const struct Curl_cwtype cw_download = {
296
  "protocol",
297
  NULL,
298
  Curl_cwriter_def_init,
299
  cw_download_write,
300
  Curl_cwriter_def_close,
301
  sizeof(struct cw_download_ctx)
302
};
303
304
/* RAW client writer in phase CURL_CW_RAW that
305
 * enabled tracing of raw data. */
306
static CURLcode cw_raw_write(struct Curl_easy *data,
307
                             struct Curl_cwriter *writer, int type,
308
                             const char *buf, size_t nbytes)
309
0
{
310
0
  if(type & CLIENTWRITE_BODY && data->set.verbose && !data->req.ignorebody) {
311
0
    Curl_debug(data, CURLINFO_DATA_IN, buf, nbytes);
312
0
  }
313
0
  return Curl_cwriter_write(data, writer->next, type, buf, nbytes);
314
0
}
315
316
static const struct Curl_cwtype cw_raw = {
317
  "raw",
318
  NULL,
319
  Curl_cwriter_def_init,
320
  cw_raw_write,
321
  Curl_cwriter_def_close,
322
  sizeof(struct Curl_cwriter)
323
};
324
325
static CURLcode do_init_writer_stack(struct Curl_easy *data)
326
0
{
327
0
  struct Curl_cwriter *writer;
328
0
  CURLcode result;
329
330
0
  DEBUGASSERT(!data->req.writer_stack);
331
0
  result = Curl_cwriter_create(&data->req.writer_stack,
332
0
                               data, &Curl_cwt_out, CURL_CW_CLIENT);
333
0
  if(result)
334
0
    return result;
335
336
  /* This places the "pause" writer behind the "download" writer that
337
   * is added below. Meaning the "download" can do checks on content length
338
   * and other things *before* write outs are buffered for paused transfers. */
339
0
  result = Curl_cwriter_create(&writer, data, &Curl_cwt_pause,
340
0
                               CURL_CW_PROTOCOL);
341
0
  if(!result) {
342
0
    result = Curl_cwriter_add(data, writer);
343
0
    if(result)
344
0
      Curl_cwriter_free(data, writer);
345
0
  }
346
0
  if(result)
347
0
    return result;
348
349
0
  result = Curl_cwriter_create(&writer, data, &cw_download, CURL_CW_PROTOCOL);
350
0
  if(!result) {
351
0
    result = Curl_cwriter_add(data, writer);
352
0
    if(result)
353
0
      Curl_cwriter_free(data, writer);
354
0
  }
355
0
  if(result)
356
0
    return result;
357
358
0
  result = Curl_cwriter_create(&writer, data, &cw_raw, CURL_CW_RAW);
359
0
  if(!result) {
360
0
    result = Curl_cwriter_add(data, writer);
361
0
    if(result)
362
0
      Curl_cwriter_free(data, writer);
363
0
  }
364
0
  if(result)
365
0
    return result;
366
367
0
  return result;
368
0
}
369
370
/* Curl_client_write() sends data to the write callback(s)
371
372
   The bit pattern defines to what "streams" to write to. Body and/or header.
373
   The defines are in sendf.h of course.
374
 */
375
CURLcode Curl_client_write(struct Curl_easy *data,
376
                           int type, const char *buf, size_t len)
377
0
{
378
0
  CURLcode result;
379
380
  /* it is one of those, at least */
381
0
  DEBUGASSERT(type &
382
0
              (CLIENTWRITE_BODY | CLIENTWRITE_HEADER | CLIENTWRITE_INFO));
383
  /* BODY is only BODY (with optional EOS) */
384
0
  DEBUGASSERT(!(type & CLIENTWRITE_BODY) ||
385
0
              ((type & ~(CLIENTWRITE_BODY | CLIENTWRITE_EOS)) == 0));
386
  /* INFO is only INFO (with optional EOS) */
387
0
  DEBUGASSERT(!(type & CLIENTWRITE_INFO) ||
388
0
              ((type & ~(CLIENTWRITE_INFO | CLIENTWRITE_EOS)) == 0));
389
390
0
  if(!data->req.writer_stack) {
391
0
    result = do_init_writer_stack(data);
392
0
    if(result)
393
0
      return result;
394
0
    DEBUGASSERT(data->req.writer_stack);
395
0
  }
396
397
0
  result = Curl_cwriter_write(data, data->req.writer_stack, type, buf, len);
398
0
  CURL_TRC_WRITE(data, "client_write(type=%x, len=%zu) -> %d",
399
0
                 type, len, result);
400
0
  return result;
401
0
}
402
403
/* Create an unencoding writer stage using the given handler. */
404
CURLcode Curl_cwriter_create(struct Curl_cwriter **pwriter,
405
                             struct Curl_easy *data,
406
                             const struct Curl_cwtype *cwt,
407
                             Curl_cwriter_phase phase)
408
0
{
409
0
  struct Curl_cwriter *writer = NULL;
410
0
  CURLcode result = CURLE_OUT_OF_MEMORY;
411
0
  void *p;
412
413
0
  DEBUGASSERT(cwt->cwriter_size >= sizeof(struct Curl_cwriter));
414
0
  p = curlx_calloc(1, cwt->cwriter_size);
415
0
  if(!p)
416
0
    goto out;
417
418
0
  writer = (struct Curl_cwriter *)p;
419
0
  writer->cwt = cwt;
420
0
  writer->ctx = p;
421
0
  writer->phase = phase;
422
0
  result = cwt->do_init(data, writer);
423
424
0
out:
425
0
  *pwriter = result ? NULL : writer;
426
0
  if(result)
427
0
    curlx_free(writer);
428
0
  return result;
429
0
}
430
431
void Curl_cwriter_free(struct Curl_easy *data,
432
                       struct Curl_cwriter *writer)
433
0
{
434
0
  if(writer) {
435
0
    writer->cwt->do_close(data, writer);
436
0
    curlx_free(writer);
437
0
  }
438
0
}
439
440
size_t Curl_cwriter_count(struct Curl_easy *data, Curl_cwriter_phase phase)
441
0
{
442
0
  struct Curl_cwriter *w;
443
0
  size_t n = 0;
444
445
0
  for(w = data->req.writer_stack; w; w = w->next) {
446
0
    if(w->phase == phase)
447
0
      ++n;
448
0
  }
449
0
  return n;
450
0
}
451
452
CURLcode Curl_cwriter_add(struct Curl_easy *data,
453
                          struct Curl_cwriter *writer)
454
0
{
455
0
  CURLcode result;
456
0
  struct Curl_cwriter **anchor = &data->req.writer_stack;
457
458
0
  if(!*anchor) {
459
0
    result = do_init_writer_stack(data);
460
0
    if(result)
461
0
      return result;
462
0
  }
463
464
  /* Insert the writer as first in its phase.
465
   * Skip existing writers of lower phases. */
466
0
  while(*anchor && (*anchor)->phase < writer->phase)
467
0
    anchor = &((*anchor)->next);
468
0
  writer->next = *anchor;
469
0
  *anchor = writer;
470
0
  return CURLE_OK;
471
0
}
472
473
struct Curl_cwriter *Curl_cwriter_get_by_name(struct Curl_easy *data,
474
                                              const char *name)
475
0
{
476
0
  struct Curl_cwriter *writer;
477
0
  for(writer = data->req.writer_stack; writer; writer = writer->next) {
478
0
    if(!strcmp(name, writer->cwt->name))
479
0
      return writer;
480
0
  }
481
0
  return NULL;
482
0
}
483
484
struct Curl_cwriter *Curl_cwriter_get_by_type(struct Curl_easy *data,
485
                                              const struct Curl_cwtype *cwt)
486
0
{
487
0
  struct Curl_cwriter *writer;
488
0
  for(writer = data->req.writer_stack; writer; writer = writer->next) {
489
0
    if(writer->cwt == cwt)
490
0
      return writer;
491
0
  }
492
0
  return NULL;
493
0
}
494
495
bool Curl_cwriter_is_content_decoding(struct Curl_easy *data)
496
0
{
497
0
  struct Curl_cwriter *writer;
498
0
  for(writer = data->req.writer_stack; writer; writer = writer->next) {
499
0
    if(writer->phase == CURL_CW_CONTENT_DECODE)
500
0
      return TRUE;
501
0
  }
502
0
  return FALSE;
503
0
}
504
505
bool Curl_cwriter_is_paused(struct Curl_easy *data)
506
0
{
507
0
  return Curl_cw_out_is_paused(data);
508
0
}
509
510
CURLcode Curl_cwriter_unpause(struct Curl_easy *data)
511
0
{
512
0
  return Curl_cw_out_unpause(data);
513
0
}
514
515
CURLcode Curl_creader_read(struct Curl_easy *data,
516
                           struct Curl_creader *reader,
517
                           char *buf, size_t blen, size_t *nread, bool *eos)
518
0
{
519
0
  *nread = 0;
520
0
  *eos = FALSE;
521
0
  if(!reader)
522
0
    return CURLE_READ_ERROR;
523
0
  return reader->crt->do_read(data, reader, buf, blen, nread, eos);
524
0
}
525
526
void Curl_creader_clear_eos(struct Curl_easy *data,
527
                            struct Curl_creader *reader)
528
0
{
529
0
  while(reader) {
530
0
    (void)reader->crt->cntrl(data, reader, CURL_CRCNTRL_CLEAR_EOS);
531
0
    reader = reader->next;
532
0
  }
533
0
}
534
535
CURLcode Curl_creader_def_init(struct Curl_easy *data,
536
                               struct Curl_creader *reader)
537
0
{
538
0
  (void)data;
539
0
  (void)reader;
540
0
  return CURLE_OK;
541
0
}
542
543
void Curl_creader_def_close(struct Curl_easy *data,
544
                            struct Curl_creader *reader)
545
0
{
546
0
  (void)data;
547
0
  (void)reader;
548
0
}
549
550
CURLcode Curl_creader_def_read(struct Curl_easy *data,
551
                               struct Curl_creader *reader,
552
                               char *buf, size_t blen,
553
                               size_t *nread, bool *eos)
554
0
{
555
0
  if(reader->next)
556
0
    return reader->next->crt->do_read(data, reader->next, buf, blen,
557
0
                                      nread, eos);
558
0
  else {
559
0
    *nread = 0;
560
0
    *eos = FALSE;
561
0
    return CURLE_READ_ERROR;
562
0
  }
563
0
}
564
565
bool Curl_creader_def_needs_rewind(struct Curl_easy *data,
566
                                   struct Curl_creader *reader)
567
0
{
568
0
  (void)data;
569
0
  (void)reader;
570
0
  return FALSE;
571
0
}
572
573
curl_off_t Curl_creader_def_total_length(struct Curl_easy *data,
574
                                         struct Curl_creader *reader)
575
0
{
576
0
  return reader->next ?
577
0
         reader->next->crt->total_length(data, reader->next) : -1;
578
0
}
579
580
CURLcode Curl_creader_def_resume_from(struct Curl_easy *data,
581
                                      struct Curl_creader *reader,
582
                                      curl_off_t offset)
583
0
{
584
0
  (void)data;
585
0
  (void)reader;
586
0
  (void)offset;
587
0
  return CURLE_READ_ERROR;
588
0
}
589
590
CURLcode Curl_creader_def_cntrl(struct Curl_easy *data,
591
                                struct Curl_creader *reader,
592
                                Curl_creader_cntrl opcode)
593
0
{
594
0
  (void)data;
595
0
  (void)reader;
596
0
  (void)opcode;
597
0
  return CURLE_OK;
598
0
}
599
600
bool Curl_creader_def_is_paused(struct Curl_easy *data,
601
                                struct Curl_creader *reader)
602
0
{
603
0
  (void)data;
604
0
  (void)reader;
605
0
  return FALSE;
606
0
}
607
608
void Curl_creader_def_done(struct Curl_easy *data,
609
                           struct Curl_creader *reader, int premature)
610
0
{
611
0
  (void)data;
612
0
  (void)reader;
613
0
  (void)premature;
614
0
}
615
616
struct cr_in_ctx {
617
  struct Curl_creader super;
618
  curl_read_callback read_cb;
619
  void *cb_user_data;
620
  curl_off_t total_len;
621
  curl_off_t read_len;
622
  CURLcode error_result;
623
  BIT(seen_eos);
624
  BIT(errored);
625
  BIT(has_used_cb);
626
  BIT(is_paused);
627
};
628
629
static CURLcode cr_in_init(struct Curl_easy *data, struct Curl_creader *reader)
630
0
{
631
0
  struct cr_in_ctx *ctx = reader->ctx;
632
0
  ctx->read_cb = data->state.fread_func;
633
0
  ctx->cb_user_data = data->state.in;
634
0
  ctx->total_len = -1;
635
0
  ctx->read_len = 0;
636
0
  return CURLE_OK;
637
0
}
638
639
/* Real client reader to installed client callbacks. */
640
static CURLcode cr_in_read(struct Curl_easy *data,
641
                           struct Curl_creader *reader,
642
                           char *buf, size_t blen,
643
                           size_t *pnread, bool *peos)
644
0
{
645
0
  struct cr_in_ctx *ctx = reader->ctx;
646
0
  CURLcode result = CURLE_OK;
647
0
  size_t nread;
648
649
0
  ctx->is_paused = FALSE;
650
651
  /* Once we have errored, we will return the same error forever */
652
0
  if(ctx->errored) {
653
0
    *pnread = 0;
654
0
    *peos = FALSE;
655
0
    return ctx->error_result;
656
0
  }
657
0
  if(ctx->seen_eos) {
658
0
    *pnread = 0;
659
0
    *peos = TRUE;
660
0
    return CURLE_OK;
661
0
  }
662
  /* respect length limitations */
663
0
  if(ctx->total_len >= 0) {
664
0
    blen = curlx_sotouz_range(ctx->total_len - ctx->read_len, 0, blen);
665
0
  }
666
0
  nread = 0;
667
0
  if(ctx->read_cb && blen) {
668
0
    Curl_set_in_callback(data, TRUE);
669
0
    nread = ctx->read_cb(buf, 1, blen, ctx->cb_user_data);
670
0
    Curl_set_in_callback(data, FALSE);
671
0
    ctx->has_used_cb = TRUE;
672
0
  }
673
674
0
  switch(nread) {
675
0
  case 0:
676
0
    if((ctx->total_len >= 0) && (ctx->read_len < ctx->total_len)) {
677
0
      failf(data, "client read function EOF fail, "
678
0
            "only %"FMT_OFF_T"/%"FMT_OFF_T " of needed bytes read",
679
0
            ctx->read_len, ctx->total_len);
680
0
      result = CURLE_READ_ERROR;
681
0
      break;
682
0
    }
683
0
    *pnread = 0;
684
0
    *peos = TRUE;
685
0
    ctx->seen_eos = TRUE;
686
0
    break;
687
688
0
  case CURL_READFUNC_ABORT:
689
0
    failf(data, "operation aborted by callback");
690
0
    *pnread = 0;
691
0
    *peos = FALSE;
692
0
    ctx->errored = TRUE;
693
0
    ctx->error_result = CURLE_ABORTED_BY_CALLBACK;
694
0
    result = CURLE_ABORTED_BY_CALLBACK;
695
0
    break;
696
697
0
  case CURL_READFUNC_PAUSE:
698
0
    if(data->conn->scheme->flags & PROTOPT_NONETWORK) {
699
      /* protocols that work without network cannot be paused. This is
700
         actually only FILE:// now, and it cannot pause since the transfer
701
         is not done using the "normal" procedure. */
702
0
      failf(data, "Read callback asked for PAUSE when not supported");
703
0
      result = CURLE_READ_ERROR;
704
0
      break;
705
0
    }
706
    /* CURL_READFUNC_PAUSE pauses read callbacks that feed socket writes */
707
0
    CURL_TRC_READ(data, "cr_in_read, callback returned CURL_READFUNC_PAUSE");
708
0
    ctx->is_paused = TRUE;
709
0
    *pnread = 0;
710
0
    *peos = FALSE;
711
0
    result = Curl_xfer_pause_send(data, TRUE);
712
0
    break; /* nothing was read */
713
714
0
  default:
715
0
    if(nread > blen) {
716
      /* the read function returned a too large value */
717
0
      failf(data, "read function returned funny value");
718
0
      *pnread = 0;
719
0
      *peos = FALSE;
720
0
      ctx->errored = TRUE;
721
0
      ctx->error_result = CURLE_READ_ERROR;
722
0
      result = CURLE_READ_ERROR;
723
0
      break;
724
0
    }
725
0
    ctx->read_len += nread;
726
0
    if(ctx->total_len >= 0)
727
0
      ctx->seen_eos = (ctx->read_len >= ctx->total_len);
728
0
    *pnread = nread;
729
0
    *peos = (bool)ctx->seen_eos;
730
0
    break;
731
0
  }
732
0
  CURL_TRC_READ(data, "cr_in_read(len=%zu, total=%"FMT_OFF_T
733
0
                ", read=%"FMT_OFF_T") -> %d, nread=%zu, eos=%d",
734
0
                blen, ctx->total_len, ctx->read_len, result,
735
0
                *pnread, *peos);
736
0
  return result;
737
0
}
738
739
static bool cr_in_needs_rewind(struct Curl_easy *data,
740
                               struct Curl_creader *reader)
741
0
{
742
0
  struct cr_in_ctx *ctx = reader->ctx;
743
0
  (void)data;
744
0
  return (bool)ctx->has_used_cb;
745
0
}
746
747
static curl_off_t cr_in_total_length(struct Curl_easy *data,
748
                                     struct Curl_creader *reader)
749
0
{
750
0
  struct cr_in_ctx *ctx = reader->ctx;
751
0
  (void)data;
752
0
  return ctx->total_len;
753
0
}
754
755
static CURLcode cr_in_resume_from(struct Curl_easy *data,
756
                                  struct Curl_creader *reader,
757
                                  curl_off_t offset)
758
0
{
759
0
  struct cr_in_ctx *ctx = reader->ctx;
760
0
  int seekerr = CURL_SEEKFUNC_CANTSEEK;
761
762
0
  DEBUGASSERT(data->conn);
763
  /* already started reading? */
764
0
  if(ctx->read_len)
765
0
    return CURLE_READ_ERROR;
766
767
0
  if(data->set.seek_func) {
768
0
    Curl_set_in_callback(data, TRUE);
769
0
    seekerr = data->set.seek_func(data->set.seek_client, offset, SEEK_SET);
770
0
    Curl_set_in_callback(data, FALSE);
771
0
  }
772
773
0
  if(seekerr != CURL_SEEKFUNC_OK) {
774
0
    curl_off_t passed = 0;
775
776
0
    if(seekerr != CURL_SEEKFUNC_CANTSEEK) {
777
0
      failf(data, "Could not seek stream");
778
0
      return CURLE_READ_ERROR;
779
0
    }
780
    /* when seekerr == CURL_SEEKFUNC_CANTSEEK (cannot seek to offset) */
781
0
    do {
782
0
      char scratch[4 * 1024];
783
0
      size_t readthisamountnow =
784
0
        (offset - passed > (curl_off_t)sizeof(scratch)) ?
785
0
        sizeof(scratch) :
786
0
        curlx_sotouz(offset - passed);
787
0
      size_t actuallyread;
788
789
0
      Curl_set_in_callback(data, TRUE);
790
0
      actuallyread = ctx->read_cb(scratch, 1, readthisamountnow,
791
0
                                  ctx->cb_user_data);
792
0
      Curl_set_in_callback(data, FALSE);
793
794
0
      passed += actuallyread;
795
0
      if((actuallyread == 0) || (actuallyread > readthisamountnow)) {
796
        /* this checks for greater-than only to make sure that the
797
           CURL_READFUNC_ABORT return code still aborts */
798
0
        failf(data, "Could only read %" FMT_OFF_T " bytes from the input",
799
0
              passed);
800
0
        return CURLE_READ_ERROR;
801
0
      }
802
0
    } while(passed < offset);
803
0
  }
804
805
  /* now, decrease the size of the read */
806
0
  if(ctx->total_len > 0) {
807
0
    ctx->total_len -= offset;
808
809
0
    if(ctx->total_len <= 0) {
810
0
      failf(data, "File already completely uploaded");
811
0
      return CURLE_PARTIAL_FILE;
812
0
    }
813
0
  }
814
  /* we have passed, proceed as normal */
815
0
  return CURLE_OK;
816
0
}
817
818
static CURLcode cr_in_rewind(struct Curl_easy *data,
819
                             struct Curl_creader *reader)
820
0
{
821
0
  struct cr_in_ctx *ctx = reader->ctx;
822
823
  /* If we never invoked the callback, there is noting to rewind */
824
0
  if(!ctx->has_used_cb)
825
0
    return CURLE_OK;
826
827
0
  if(data->set.seek_func) {
828
0
    int err;
829
830
0
    Curl_set_in_callback(data, TRUE);
831
0
    err = (data->set.seek_func)(data->set.seek_client, 0, SEEK_SET);
832
0
    Curl_set_in_callback(data, FALSE);
833
0
    CURL_TRC_READ(data, "cr_in, rewind via set.seek_func -> %d", err);
834
0
    if(err) {
835
0
      failf(data, "seek callback returned error %d", err);
836
0
      return CURLE_SEND_FAIL_REWIND;
837
0
    }
838
0
  }
839
0
  else if(data->set.ioctl_func) {
840
0
    curlioerr err;
841
842
0
    Curl_set_in_callback(data, TRUE);
843
0
    err = (data->set.ioctl_func)(data, CURLIOCMD_RESTARTREAD,
844
0
                                 data->set.ioctl_client);
845
0
    Curl_set_in_callback(data, FALSE);
846
0
    CURL_TRC_READ(data, "cr_in, rewind via set.ioctl_func -> %d", (int)err);
847
0
    if(err) {
848
0
      failf(data, "ioctl callback returned error %d", (int)err);
849
0
      return CURLE_SEND_FAIL_REWIND;
850
0
    }
851
0
  }
852
0
  else {
853
    /* If no CURLOPT_READFUNCTION is used, we know that we operate on a
854
       given FILE * stream and we can actually attempt to rewind that
855
       ourselves with fseek() */
856
0
#if defined(__clang__) && __clang_major__ >= 16
857
0
#pragma clang diagnostic push
858
0
#pragma clang diagnostic ignored "-Wcast-function-type-strict"
859
0
#endif
860
0
    if(data->state.fread_func == (curl_read_callback)fread) {
861
0
#if defined(__clang__) && __clang_major__ >= 16
862
0
#pragma clang diagnostic pop
863
0
#endif
864
0
      int err = fseek(data->state.in, 0, SEEK_SET);
865
0
      CURL_TRC_READ(data, "cr_in, rewind via fseek -> %d(%d)",
866
0
                    (int)err, (int)errno);
867
0
      if(err != -1)
868
        /* successful rewind */
869
0
        return CURLE_OK;
870
0
    }
871
872
    /* no callback set or failure above, makes us fail at once */
873
0
    failf(data, "necessary data rewind was not possible");
874
0
    return CURLE_SEND_FAIL_REWIND;
875
0
  }
876
0
  return CURLE_OK;
877
0
}
878
879
static CURLcode cr_in_cntrl(struct Curl_easy *data,
880
                            struct Curl_creader *reader,
881
                            Curl_creader_cntrl opcode)
882
0
{
883
0
  struct cr_in_ctx *ctx = reader->ctx;
884
885
0
  switch(opcode) {
886
0
  case CURL_CRCNTRL_REWIND:
887
0
    return cr_in_rewind(data, reader);
888
0
  case CURL_CRCNTRL_UNPAUSE:
889
0
    ctx->is_paused = FALSE;
890
0
    break;
891
0
  case CURL_CRCNTRL_CLEAR_EOS:
892
0
    ctx->seen_eos = FALSE;
893
0
    break;
894
0
  default:
895
0
    break;
896
0
  }
897
0
  return CURLE_OK;
898
0
}
899
900
static bool cr_in_is_paused(struct Curl_easy *data,
901
                            struct Curl_creader *reader)
902
0
{
903
0
  struct cr_in_ctx *ctx = reader->ctx;
904
0
  (void)data;
905
0
  return (bool)ctx->is_paused;
906
0
}
907
908
static const struct Curl_crtype cr_in = {
909
  "cr-in",
910
  cr_in_init,
911
  cr_in_read,
912
  Curl_creader_def_close,
913
  cr_in_needs_rewind,
914
  cr_in_total_length,
915
  cr_in_resume_from,
916
  cr_in_cntrl,
917
  cr_in_is_paused,
918
  Curl_creader_def_done,
919
  sizeof(struct cr_in_ctx)
920
};
921
922
CURLcode Curl_creader_create(struct Curl_creader **preader,
923
                             struct Curl_easy *data,
924
                             const struct Curl_crtype *crt,
925
                             Curl_creader_phase phase)
926
0
{
927
0
  struct Curl_creader *reader = NULL;
928
0
  CURLcode result = CURLE_OUT_OF_MEMORY;
929
0
  void *p;
930
931
0
  DEBUGASSERT(crt->creader_size >= sizeof(struct Curl_creader));
932
0
  p = curlx_calloc(1, crt->creader_size);
933
0
  if(!p)
934
0
    goto out;
935
936
0
  reader = (struct Curl_creader *)p;
937
0
  reader->crt = crt;
938
0
  reader->ctx = p;
939
0
  reader->phase = phase;
940
0
  result = crt->do_init(data, reader);
941
942
0
out:
943
0
  *preader = result ? NULL : reader;
944
0
  if(result)
945
0
    curlx_free(reader);
946
0
  return result;
947
0
}
948
949
void Curl_creader_free(struct Curl_easy *data, struct Curl_creader *reader)
950
0
{
951
0
  if(reader) {
952
0
    reader->crt->do_close(data, reader);
953
0
    curlx_free(reader);
954
0
  }
955
0
}
956
957
struct cr_lc_ctx {
958
  struct Curl_creader super;
959
  struct bufq buf;
960
  BIT(read_eos);  /* we read an EOS from the next reader */
961
  BIT(eos);       /* we have returned an EOS */
962
  BIT(prev_cr);   /* the last byte was a CR */
963
};
964
965
static CURLcode cr_lc_init(struct Curl_easy *data, struct Curl_creader *reader)
966
0
{
967
0
  struct cr_lc_ctx *ctx = reader->ctx;
968
0
  (void)data;
969
0
  Curl_bufq_init2(&ctx->buf, (16 * 1024), 1, BUFQ_OPT_SOFT_LIMIT);
970
0
  return CURLE_OK;
971
0
}
972
973
static void cr_lc_close(struct Curl_easy *data, struct Curl_creader *reader)
974
0
{
975
0
  struct cr_lc_ctx *ctx = reader->ctx;
976
0
  (void)data;
977
0
  Curl_bufq_free(&ctx->buf);
978
0
}
979
980
/* client reader doing line end conversions. */
981
static CURLcode cr_lc_read(struct Curl_easy *data,
982
                           struct Curl_creader *reader,
983
                           char *buf, size_t blen,
984
                           size_t *pnread, bool *peos)
985
0
{
986
0
  struct cr_lc_ctx *ctx = reader->ctx;
987
0
  CURLcode result;
988
0
  size_t nread, i, start, n;
989
0
  bool eos;
990
991
0
  if(ctx->eos) {
992
0
    *pnread = 0;
993
0
    *peos = TRUE;
994
0
    return CURLE_OK;
995
0
  }
996
997
0
  if(Curl_bufq_is_empty(&ctx->buf)) {
998
0
    if(ctx->read_eos) {
999
0
      ctx->eos = TRUE;
1000
0
      *pnread = 0;
1001
0
      *peos = TRUE;
1002
0
      return CURLE_OK;
1003
0
    }
1004
    /* Still getting data form the next reader, ctx->buf is empty */
1005
0
    result = Curl_creader_read(data, reader->next, buf, blen, &nread, &eos);
1006
0
    if(result)
1007
0
      return result;
1008
0
    ctx->read_eos = eos;
1009
1010
0
    if(!nread || !memchr(buf, '\n', nread)) {
1011
      /* nothing to convert, return this right away */
1012
0
      if(ctx->read_eos)
1013
0
        ctx->eos = TRUE;
1014
0
      *pnread = nread;
1015
0
      *peos = (bool)ctx->eos;
1016
0
      goto out;
1017
0
    }
1018
1019
    /* at least one \n might need conversion to '\r\n', place into ctx->buf */
1020
0
    for(i = start = 0; i < nread; ++i) {
1021
      /* if this byte is not an LF character, or if the preceding character is
1022
         a CR (meaning this already is a CRLF pair), go to next */
1023
0
      if((buf[i] != '\n') || ctx->prev_cr) {
1024
0
        ctx->prev_cr = (buf[i] == '\r');
1025
0
        continue;
1026
0
      }
1027
0
      ctx->prev_cr = FALSE;
1028
      /* on a soft limit bufq, we do not need to check length */
1029
0
      result = Curl_bufq_cwrite(&ctx->buf, buf + start, i - start, &n);
1030
0
      if(!result)
1031
0
        result = Curl_bufq_cwrite(&ctx->buf, STRCONST("\r\n"), &n);
1032
0
      if(result)
1033
0
        return result;
1034
0
      start = i + 1;
1035
0
    }
1036
1037
0
    if(start < i) { /* leftover */
1038
0
      result = Curl_bufq_cwrite(&ctx->buf, buf + start, i - start, &n);
1039
0
      if(result)
1040
0
        return result;
1041
0
    }
1042
0
  }
1043
1044
0
  DEBUGASSERT(!Curl_bufq_is_empty(&ctx->buf));
1045
0
  *peos = FALSE;
1046
0
  result = Curl_bufq_cread(&ctx->buf, buf, blen, pnread);
1047
0
  if(!result && ctx->read_eos && Curl_bufq_is_empty(&ctx->buf)) {
1048
    /* no more data, read all, done. */
1049
0
    ctx->eos = TRUE;
1050
0
    *peos = TRUE;
1051
0
  }
1052
1053
0
out:
1054
0
  CURL_TRC_READ(data, "cr_lc_read(len=%zu) -> %d, nread=%zu, eos=%d",
1055
0
                blen, result, *pnread, *peos);
1056
0
  return result;
1057
0
}
1058
1059
static curl_off_t cr_lc_total_length(struct Curl_easy *data,
1060
                                     struct Curl_creader *reader)
1061
0
{
1062
  /* this reader changes length depending on input */
1063
0
  (void)data;
1064
0
  (void)reader;
1065
0
  return -1;
1066
0
}
1067
1068
static const struct Curl_crtype cr_lc = {
1069
  "cr-lineconv",
1070
  cr_lc_init,
1071
  cr_lc_read,
1072
  cr_lc_close,
1073
  Curl_creader_def_needs_rewind,
1074
  cr_lc_total_length,
1075
  Curl_creader_def_resume_from,
1076
  Curl_creader_def_cntrl,
1077
  Curl_creader_def_is_paused,
1078
  Curl_creader_def_done,
1079
  sizeof(struct cr_lc_ctx)
1080
};
1081
1082
static CURLcode cr_lc_add(struct Curl_easy *data)
1083
0
{
1084
0
  struct Curl_creader *reader = NULL;
1085
0
  CURLcode result;
1086
1087
0
  result = Curl_creader_create(&reader, data, &cr_lc, CURL_CR_CONTENT_ENCODE);
1088
0
  if(!result)
1089
0
    result = Curl_creader_add(data, reader);
1090
1091
0
  if(result && reader)
1092
0
    Curl_creader_free(data, reader);
1093
0
  return result;
1094
0
}
1095
1096
static CURLcode do_init_reader_stack(struct Curl_easy *data,
1097
                                     struct Curl_creader *r)
1098
0
{
1099
0
  CURLcode result = CURLE_OK;
1100
0
  curl_off_t clen;
1101
1102
0
  DEBUGASSERT(r);
1103
0
  DEBUGASSERT(r->crt);
1104
0
  DEBUGASSERT(r->phase == CURL_CR_CLIENT);
1105
0
  DEBUGASSERT(!data->req.reader_stack);
1106
1107
0
  data->req.reader_stack = r;
1108
0
  clen = r->crt->total_length(data, r);
1109
  /* if we do not have 0 length init, and crlf conversion is wanted,
1110
   * add the reader for it */
1111
0
  if(clen && (data->set.crlf
1112
0
#ifdef CURL_PREFER_LF_LINEENDS
1113
0
     || data->state.prefer_ascii
1114
0
#endif
1115
0
    )) {
1116
0
    result = cr_lc_add(data);
1117
0
    if(result)
1118
0
      return result;
1119
0
  }
1120
1121
0
  return result;
1122
0
}
1123
1124
CURLcode Curl_creader_set_fread(struct Curl_easy *data, curl_off_t len)
1125
0
{
1126
0
  CURLcode result;
1127
0
  struct Curl_creader *r;
1128
0
  struct cr_in_ctx *ctx;
1129
1130
0
  result = Curl_creader_create(&r, data, &cr_in, CURL_CR_CLIENT);
1131
0
  if(result)
1132
0
    goto out;
1133
0
  ctx = r->ctx;
1134
0
  ctx->total_len = len;
1135
1136
0
  cl_reset_reader(data);
1137
0
  result = do_init_reader_stack(data, r);
1138
0
out:
1139
0
  CURL_TRC_READ(data, "add fread reader, len=%"FMT_OFF_T " -> %d",
1140
0
                len, result);
1141
0
  return result;
1142
0
}
1143
1144
CURLcode Curl_creader_add(struct Curl_easy *data,
1145
                          struct Curl_creader *reader)
1146
0
{
1147
0
  CURLcode result;
1148
0
  struct Curl_creader **anchor = &data->req.reader_stack;
1149
1150
0
  if(!*anchor) {
1151
0
    result = Curl_creader_set_fread(data, data->state.infilesize);
1152
0
    if(result)
1153
0
      return result;
1154
0
  }
1155
1156
  /* Insert the writer as first in its phase.
1157
   * Skip existing readers of lower phases. */
1158
0
  while(*anchor && (*anchor)->phase < reader->phase)
1159
0
    anchor = &((*anchor)->next);
1160
0
  reader->next = *anchor;
1161
0
  *anchor = reader;
1162
0
  return CURLE_OK;
1163
0
}
1164
1165
CURLcode Curl_creader_set(struct Curl_easy *data, struct Curl_creader *r)
1166
0
{
1167
0
  CURLcode result;
1168
1169
0
  DEBUGASSERT(r);
1170
0
  DEBUGASSERT(r->crt);
1171
0
  DEBUGASSERT(r->phase == CURL_CR_CLIENT);
1172
1173
0
  cl_reset_reader(data);
1174
0
  result = do_init_reader_stack(data, r);
1175
0
  if(result)
1176
0
    Curl_creader_free(data, r);
1177
0
  return result;
1178
0
}
1179
1180
CURLcode Curl_client_read(struct Curl_easy *data, char *buf, size_t blen,
1181
                          size_t *nread, bool *eos)
1182
0
{
1183
0
  CURLcode result;
1184
1185
0
  DEBUGASSERT(buf);
1186
0
  DEBUGASSERT(blen);
1187
0
  DEBUGASSERT(nread);
1188
0
  DEBUGASSERT(eos);
1189
0
  *nread = 0;
1190
1191
0
  if(!data->req.reader_stack) {
1192
0
    result = Curl_creader_set_fread(data, data->state.infilesize);
1193
0
    if(result)
1194
0
      return result;
1195
0
    DEBUGASSERT(data->req.reader_stack);
1196
0
  }
1197
0
  if(!data->req.reader_started) {
1198
0
    Curl_rlimit_start(&data->progress.ul.rlimit, Curl_pgrs_now(data), -1);
1199
0
    data->req.reader_started = TRUE;
1200
0
  }
1201
1202
0
  if(Curl_rlimit_active(&data->progress.ul.rlimit)) {
1203
0
    curl_off_t ul_avail = Curl_rlimit_avail(&data->progress.ul.rlimit,
1204
0
                                            Curl_pgrs_now(data));
1205
0
    if(ul_avail <= 0) {
1206
0
      result = CURLE_OK;
1207
0
      *eos = FALSE;
1208
0
      goto out;
1209
0
    }
1210
0
    if(ul_avail < (curl_off_t)blen)
1211
0
      blen = (size_t)ul_avail;
1212
0
  }
1213
0
  result = Curl_creader_read(data, data->req.reader_stack, buf, blen,
1214
0
                             nread, eos);
1215
1216
0
out:
1217
0
  CURL_TRC_READ(data, "client_read(len=%zu) -> %d, nread=%zu, eos=%d",
1218
0
                blen, result, *nread, *eos);
1219
0
  return result;
1220
0
}
1221
1222
bool Curl_creader_needs_rewind(struct Curl_easy *data)
1223
0
{
1224
0
  struct Curl_creader *reader = data->req.reader_stack;
1225
0
  while(reader) {
1226
0
    if(reader->crt->needs_rewind(data, reader)) {
1227
0
      CURL_TRC_READ(data, "client reader needs rewind before next request");
1228
0
      return TRUE;
1229
0
    }
1230
0
    reader = reader->next;
1231
0
  }
1232
0
  return FALSE;
1233
0
}
1234
1235
static CURLcode cr_null_read(struct Curl_easy *data,
1236
                             struct Curl_creader *reader,
1237
                             char *buf, size_t blen,
1238
                             size_t *pnread, bool *peos)
1239
0
{
1240
0
  (void)data;
1241
0
  (void)reader;
1242
0
  (void)buf;
1243
0
  (void)blen;
1244
0
  *pnread = 0;
1245
0
  *peos = TRUE;
1246
0
  return CURLE_OK;
1247
0
}
1248
1249
static curl_off_t cr_null_total_length(struct Curl_easy *data,
1250
                                       struct Curl_creader *reader)
1251
0
{
1252
  /* this reader changes length depending on input */
1253
0
  (void)data;
1254
0
  (void)reader;
1255
0
  return 0;
1256
0
}
1257
1258
static const struct Curl_crtype cr_null = {
1259
  "cr-null",
1260
  Curl_creader_def_init,
1261
  cr_null_read,
1262
  Curl_creader_def_close,
1263
  Curl_creader_def_needs_rewind,
1264
  cr_null_total_length,
1265
  Curl_creader_def_resume_from,
1266
  Curl_creader_def_cntrl,
1267
  Curl_creader_def_is_paused,
1268
  Curl_creader_def_done,
1269
  sizeof(struct Curl_creader)
1270
};
1271
1272
CURLcode Curl_creader_set_null(struct Curl_easy *data)
1273
0
{
1274
0
  struct Curl_creader *r;
1275
0
  CURLcode result;
1276
1277
0
  result = Curl_creader_create(&r, data, &cr_null, CURL_CR_CLIENT);
1278
0
  if(result)
1279
0
    return result;
1280
1281
0
  cl_reset_reader(data);
1282
0
  return do_init_reader_stack(data, r);
1283
0
}
1284
1285
struct cr_buf_ctx {
1286
  struct Curl_creader super;
1287
  const char *buf;
1288
  size_t blen;
1289
  size_t index;
1290
};
1291
1292
static CURLcode cr_buf_read(struct Curl_easy *data,
1293
                            struct Curl_creader *reader,
1294
                            char *buf, size_t blen,
1295
                            size_t *pnread, bool *peos)
1296
0
{
1297
0
  struct cr_buf_ctx *ctx = reader->ctx;
1298
0
  size_t nread = ctx->blen - ctx->index;
1299
1300
0
  if(!nread || !ctx->buf) {
1301
0
    *pnread = 0;
1302
0
    *peos = TRUE;
1303
0
  }
1304
0
  else {
1305
0
    if(nread > blen)
1306
0
      nread = blen;
1307
0
    memcpy(buf, ctx->buf + ctx->index, nread);
1308
0
    *pnread = nread;
1309
0
    ctx->index += nread;
1310
0
    *peos = (ctx->index == ctx->blen);
1311
0
  }
1312
0
  CURL_TRC_READ(data, "cr_buf_read(len=%zu) -> 0, nread=%zu, eos=%d",
1313
0
                blen, *pnread, *peos);
1314
0
  return CURLE_OK;
1315
0
}
1316
1317
static bool cr_buf_needs_rewind(struct Curl_easy *data,
1318
                                struct Curl_creader *reader)
1319
0
{
1320
0
  struct cr_buf_ctx *ctx = reader->ctx;
1321
0
  (void)data;
1322
0
  return ctx->index > 0;
1323
0
}
1324
1325
static CURLcode cr_buf_cntrl(struct Curl_easy *data,
1326
                             struct Curl_creader *reader,
1327
                             Curl_creader_cntrl opcode)
1328
0
{
1329
0
  struct cr_buf_ctx *ctx = reader->ctx;
1330
0
  (void)data;
1331
0
  switch(opcode) {
1332
0
  case CURL_CRCNTRL_REWIND:
1333
0
    ctx->index = 0;
1334
0
    break;
1335
0
  default:
1336
0
    break;
1337
0
  }
1338
0
  return CURLE_OK;
1339
0
}
1340
1341
static curl_off_t cr_buf_total_length(struct Curl_easy *data,
1342
                                      struct Curl_creader *reader)
1343
0
{
1344
0
  struct cr_buf_ctx *ctx = reader->ctx;
1345
0
  (void)data;
1346
0
  return (curl_off_t)ctx->blen;
1347
0
}
1348
1349
static CURLcode cr_buf_resume_from(struct Curl_easy *data,
1350
                                   struct Curl_creader *reader,
1351
                                   curl_off_t offset)
1352
0
{
1353
0
  struct cr_buf_ctx *ctx = reader->ctx;
1354
0
  size_t boffset;
1355
1356
0
  (void)data;
1357
0
  DEBUGASSERT(data->conn);
1358
  /* already started reading? */
1359
0
  if(ctx->index)
1360
0
    return CURLE_READ_ERROR;
1361
0
  boffset = curlx_sotouz_range(offset, 0, SIZE_MAX);
1362
0
  if(!boffset)
1363
0
    return CURLE_OK;
1364
0
  if(boffset > ctx->blen)
1365
0
    return CURLE_READ_ERROR;
1366
1367
0
  ctx->buf += boffset;
1368
0
  ctx->blen -= boffset;
1369
0
  return CURLE_OK;
1370
0
}
1371
1372
static const struct Curl_crtype cr_buf = {
1373
  "cr-buf",
1374
  Curl_creader_def_init,
1375
  cr_buf_read,
1376
  Curl_creader_def_close,
1377
  cr_buf_needs_rewind,
1378
  cr_buf_total_length,
1379
  cr_buf_resume_from,
1380
  cr_buf_cntrl,
1381
  Curl_creader_def_is_paused,
1382
  Curl_creader_def_done,
1383
  sizeof(struct cr_buf_ctx)
1384
};
1385
1386
CURLcode Curl_creader_set_buf(struct Curl_easy *data,
1387
                              const char *buf, size_t blen)
1388
0
{
1389
0
  CURLcode result;
1390
0
  struct Curl_creader *r;
1391
0
  struct cr_buf_ctx *ctx;
1392
1393
0
  result = Curl_creader_create(&r, data, &cr_buf, CURL_CR_CLIENT);
1394
0
  if(result)
1395
0
    goto out;
1396
0
  ctx = r->ctx;
1397
0
  ctx->buf = buf;
1398
0
  ctx->blen = blen;
1399
0
  ctx->index = 0;
1400
1401
0
  cl_reset_reader(data);
1402
0
  result = do_init_reader_stack(data, r);
1403
0
out:
1404
0
  CURL_TRC_READ(data, "add buf reader, len=%zu -> %d", blen, result);
1405
0
  return result;
1406
0
}
1407
1408
curl_off_t Curl_creader_total_length(struct Curl_easy *data)
1409
0
{
1410
0
  struct Curl_creader *r = data->req.reader_stack;
1411
0
  return r ? r->crt->total_length(data, r) : -1;
1412
0
}
1413
1414
curl_off_t Curl_creader_client_length(struct Curl_easy *data)
1415
0
{
1416
0
  struct Curl_creader *r = data->req.reader_stack;
1417
0
  while(r && r->phase != CURL_CR_CLIENT)
1418
0
    r = r->next;
1419
0
  return r ? r->crt->total_length(data, r) : -1;
1420
0
}
1421
1422
CURLcode Curl_creader_resume_from(struct Curl_easy *data, curl_off_t offset)
1423
0
{
1424
0
  struct Curl_creader *r = data->req.reader_stack;
1425
0
  while(r && r->phase != CURL_CR_CLIENT)
1426
0
    r = r->next;
1427
0
  return r ? r->crt->resume_from(data, r, offset) : CURLE_READ_ERROR;
1428
0
}
1429
1430
CURLcode Curl_creader_unpause(struct Curl_easy *data)
1431
0
{
1432
0
  struct Curl_creader *reader = data->req.reader_stack;
1433
0
  CURLcode result = CURLE_OK;
1434
1435
0
  while(reader) {
1436
0
    result = reader->crt->cntrl(data, reader, CURL_CRCNTRL_UNPAUSE);
1437
0
    CURL_TRC_READ(data, "unpausing %s -> %d", reader->crt->name, result);
1438
0
    if(result)
1439
0
      break;
1440
0
    reader = reader->next;
1441
0
  }
1442
0
  return result;
1443
0
}
1444
1445
bool Curl_creader_is_paused(struct Curl_easy *data)
1446
0
{
1447
0
  struct Curl_creader *reader = data->req.reader_stack;
1448
1449
0
  while(reader) {
1450
0
    if(reader->crt->is_paused(data, reader))
1451
0
      return TRUE;
1452
0
    reader = reader->next;
1453
0
  }
1454
0
  return FALSE;
1455
0
}
1456
1457
void Curl_creader_done(struct Curl_easy *data, int premature)
1458
0
{
1459
0
  struct Curl_creader *reader = data->req.reader_stack;
1460
0
  while(reader) {
1461
0
    reader->crt->done(data, reader, premature);
1462
0
    reader = reader->next;
1463
0
  }
1464
0
}
1465
1466
struct Curl_creader *Curl_creader_get_by_type(struct Curl_easy *data,
1467
                                              const struct Curl_crtype *crt)
1468
0
{
1469
0
  struct Curl_creader *r;
1470
0
  for(r = data->req.reader_stack; r; r = r->next) {
1471
0
    if(r->crt == crt)
1472
0
      return r;
1473
0
  }
1474
0
  return NULL;
1475
0
}