Annotation of embedaddon/curl/lib/vquic/quiche.c, revision 1.1.1.1
1.1 misho 1: /***************************************************************************
2: * _ _ ____ _
3: * Project ___| | | | _ \| |
4: * / __| | | | |_) | |
5: * | (__| |_| | _ <| |___
6: * \___|\___/|_| \_\_____|
7: *
8: * Copyright (C) 1998 - 2020, 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.haxx.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: ***************************************************************************/
22:
23: #include "curl_setup.h"
24:
25: #ifdef USE_QUICHE
26: #include <quiche.h>
27: #include <openssl/err.h>
28: #include "urldata.h"
29: #include "sendf.h"
30: #include "strdup.h"
31: #include "rand.h"
32: #include "quic.h"
33: #include "strcase.h"
34: #include "multiif.h"
35: #include "connect.h"
36: #include "strerror.h"
37:
38: /* The last 3 #include files should be in this order */
39: #include "curl_printf.h"
40: #include "curl_memory.h"
41: #include "memdebug.h"
42:
43: #define DEBUG_HTTP3
44: /* #define DEBUG_QUICHE */
45: #ifdef DEBUG_HTTP3
46: #define H3BUGF(x) x
47: #else
48: #define H3BUGF(x) do { } while(0)
49: #endif
50:
51: #define QUIC_MAX_STREAMS (256*1024)
52: #define QUIC_MAX_DATA (1*1024*1024)
53: #define QUIC_IDLE_TIMEOUT (60 * 1000) /* milliseconds */
54:
55: static CURLcode process_ingress(struct connectdata *conn,
56: curl_socket_t sockfd,
57: struct quicsocket *qs);
58:
59: static CURLcode flush_egress(struct connectdata *conn, curl_socket_t sockfd,
60: struct quicsocket *qs);
61:
62: static CURLcode http_request(struct connectdata *conn, const void *mem,
63: size_t len);
64: static Curl_recv h3_stream_recv;
65: static Curl_send h3_stream_send;
66:
67:
68: static int quiche_getsock(struct connectdata *conn, curl_socket_t *socks)
69: {
70: struct SingleRequest *k = &conn->data->req;
71: int bitmap = GETSOCK_BLANK;
72:
73: socks[0] = conn->sock[FIRSTSOCKET];
74:
75: /* in a HTTP/2 connection we can basically always get a frame so we should
76: always be ready for one */
77: bitmap |= GETSOCK_READSOCK(FIRSTSOCKET);
78:
79: /* we're still uploading or the HTTP/2 layer wants to send data */
80: if((k->keepon & (KEEP_SEND|KEEP_SEND_PAUSE)) == KEEP_SEND)
81: bitmap |= GETSOCK_WRITESOCK(FIRSTSOCKET);
82:
83: return bitmap;
84: }
85:
86: static int quiche_perform_getsock(const struct connectdata *conn,
87: curl_socket_t *socks)
88: {
89: return quiche_getsock((struct connectdata *)conn, socks);
90: }
91:
92: static CURLcode quiche_disconnect(struct connectdata *conn,
93: bool dead_connection)
94: {
95: struct quicsocket *qs = conn->quic;
96: (void)dead_connection;
97: quiche_h3_config_free(qs->h3config);
98: quiche_h3_conn_free(qs->h3c);
99: quiche_config_free(qs->cfg);
100: quiche_conn_free(qs->conn);
101: return CURLE_OK;
102: }
103:
104: static unsigned int quiche_conncheck(struct connectdata *conn,
105: unsigned int checks_to_perform)
106: {
107: (void)conn;
108: (void)checks_to_perform;
109: return CONNRESULT_NONE;
110: }
111:
112: static CURLcode quiche_do(struct connectdata *conn, bool *done)
113: {
114: struct HTTP *stream = conn->data->req.protop;
115: stream->h3req = FALSE; /* not sent */
116: return Curl_http(conn, done);
117: }
118:
119: static const struct Curl_handler Curl_handler_http3 = {
120: "HTTPS", /* scheme */
121: ZERO_NULL, /* setup_connection */
122: quiche_do, /* do_it */
123: Curl_http_done, /* done */
124: ZERO_NULL, /* do_more */
125: ZERO_NULL, /* connect_it */
126: ZERO_NULL, /* connecting */
127: ZERO_NULL, /* doing */
128: quiche_getsock, /* proto_getsock */
129: quiche_getsock, /* doing_getsock */
130: ZERO_NULL, /* domore_getsock */
131: quiche_perform_getsock, /* perform_getsock */
132: quiche_disconnect, /* disconnect */
133: ZERO_NULL, /* readwrite */
134: quiche_conncheck, /* connection_check */
135: PORT_HTTP, /* defport */
136: CURLPROTO_HTTPS, /* protocol */
137: PROTOPT_SSL | PROTOPT_STREAM /* flags */
138: };
139:
140: #ifdef DEBUG_QUICHE
141: static void quiche_debug_log(const char *line, void *argp)
142: {
143: (void)argp;
144: fprintf(stderr, "%s\n", line);
145: }
146: #endif
147:
148: CURLcode Curl_quic_connect(struct connectdata *conn, curl_socket_t sockfd,
149: int sockindex,
150: const struct sockaddr *addr, socklen_t addrlen)
151: {
152: CURLcode result;
153: struct quicsocket *qs = &conn->hequic[sockindex];
154: struct Curl_easy *data = conn->data;
155:
156: #ifdef DEBUG_QUICHE
157: /* initialize debug log callback only once */
158: static int debug_log_init = 0;
159: if(!debug_log_init) {
160: quiche_enable_debug_logging(quiche_debug_log, NULL);
161: debug_log_init = 1;
162: }
163: #endif
164:
165: (void)addr;
166: (void)addrlen;
167:
168: qs->cfg = quiche_config_new(QUICHE_PROTOCOL_VERSION);
169: if(!qs->cfg) {
170: failf(data, "can't create quiche config");
171: return CURLE_FAILED_INIT;
172: }
173:
174: quiche_config_set_max_idle_timeout(qs->cfg, QUIC_IDLE_TIMEOUT);
175: quiche_config_set_initial_max_data(qs->cfg, QUIC_MAX_DATA);
176: quiche_config_set_initial_max_stream_data_bidi_local(qs->cfg, QUIC_MAX_DATA);
177: quiche_config_set_initial_max_stream_data_bidi_remote(qs->cfg,
178: QUIC_MAX_DATA);
179: quiche_config_set_initial_max_stream_data_uni(qs->cfg, QUIC_MAX_DATA);
180: quiche_config_set_initial_max_streams_bidi(qs->cfg, QUIC_MAX_STREAMS);
181: quiche_config_set_initial_max_streams_uni(qs->cfg, QUIC_MAX_STREAMS);
182: quiche_config_set_application_protos(qs->cfg,
183: (uint8_t *)
184: QUICHE_H3_APPLICATION_PROTOCOL,
185: sizeof(QUICHE_H3_APPLICATION_PROTOCOL)
186: - 1);
187:
188: result = Curl_rand(data, qs->scid, sizeof(qs->scid));
189: if(result)
190: return result;
191:
192: if(getenv("SSLKEYLOGFILE"))
193: quiche_config_log_keys(qs->cfg);
194:
195: qs->conn = quiche_connect(conn->host.name, (const uint8_t *) qs->scid,
196: sizeof(qs->scid), qs->cfg);
197: if(!qs->conn) {
198: failf(data, "can't create quiche connection");
199: return CURLE_OUT_OF_MEMORY;
200: }
201:
202: result = flush_egress(conn, sockfd, qs);
203: if(result)
204: return result;
205:
206: /* store the used address as a string */
207: if(!Curl_addr2string((struct sockaddr*)addr, addrlen,
208: conn->primary_ip, &conn->primary_port)) {
209: char buffer[STRERROR_LEN];
210: failf(data, "ssrem inet_ntop() failed with errno %d: %s",
211: SOCKERRNO, Curl_strerror(SOCKERRNO, buffer, sizeof(buffer)));
212: return CURLE_BAD_FUNCTION_ARGUMENT;
213: }
214: memcpy(conn->ip_addr_str, conn->primary_ip, MAX_IPADR_LEN);
215: Curl_persistconninfo(conn);
216:
217: /* for connection reuse purposes: */
218: conn->ssl[FIRSTSOCKET].state = ssl_connection_complete;
219:
220: infof(data, "Sent QUIC client Initial, ALPN: %s\n",
221: QUICHE_H3_APPLICATION_PROTOCOL + 1);
222:
223: return CURLE_OK;
224: }
225:
226: static CURLcode quiche_has_connected(struct connectdata *conn,
227: int sockindex,
228: int tempindex)
229: {
230: CURLcode result;
231: struct quicsocket *qs = conn->quic = &conn->hequic[tempindex];
232:
233: conn->recv[sockindex] = h3_stream_recv;
234: conn->send[sockindex] = h3_stream_send;
235: conn->handler = &Curl_handler_http3;
236: conn->bits.multiplex = TRUE; /* at least potentially multiplexed */
237: conn->httpversion = 30;
238: conn->bundle->multiuse = BUNDLE_MULTIPLEX;
239:
240: qs->h3config = quiche_h3_config_new();
241: if(!qs->h3config)
242: return CURLE_OUT_OF_MEMORY;
243:
244: /* Create a new HTTP/3 connection on the QUIC connection. */
245: qs->h3c = quiche_h3_conn_new_with_transport(qs->conn, qs->h3config);
246: if(!qs->h3c) {
247: result = CURLE_OUT_OF_MEMORY;
248: goto fail;
249: }
250: if(conn->hequic[1-tempindex].cfg) {
251: qs = &conn->hequic[1-tempindex];
252: quiche_config_free(qs->cfg);
253: quiche_conn_free(qs->conn);
254: qs->cfg = NULL;
255: qs->conn = NULL;
256: }
257: return CURLE_OK;
258: fail:
259: quiche_h3_config_free(qs->h3config);
260: quiche_h3_conn_free(qs->h3c);
261: return result;
262: }
263:
264: /*
265: * This function gets polled to check if this QUIC connection has connected.
266: */
267: CURLcode Curl_quic_is_connected(struct connectdata *conn, int sockindex,
268: bool *done)
269: {
270: CURLcode result;
271: struct quicsocket *qs = &conn->hequic[sockindex];
272: curl_socket_t sockfd = conn->tempsock[sockindex];
273:
274: result = process_ingress(conn, sockfd, qs);
275: if(result)
276: return result;
277:
278: result = flush_egress(conn, sockfd, qs);
279: if(result)
280: return result;
281:
282: if(quiche_conn_is_established(qs->conn)) {
283: *done = TRUE;
284: result = quiche_has_connected(conn, 0, sockindex);
285: DEBUGF(infof(conn->data, "quiche established connection!\n"));
286: }
287:
288: return result;
289: }
290:
291: static CURLcode process_ingress(struct connectdata *conn, int sockfd,
292: struct quicsocket *qs)
293: {
294: ssize_t recvd;
295: struct Curl_easy *data = conn->data;
296: uint8_t *buf = (uint8_t *)data->state.buffer;
297: size_t bufsize = data->set.buffer_size;
298:
299: /* in case the timeout expired */
300: quiche_conn_on_timeout(qs->conn);
301:
302: do {
303: recvd = recv(sockfd, buf, bufsize, 0);
304: if((recvd < 0) && ((SOCKERRNO == EAGAIN) || (SOCKERRNO == EWOULDBLOCK)))
305: break;
306:
307: if(recvd < 0) {
308: failf(conn->data, "quiche: recv() unexpectedly returned %d "
309: "(errno: %d, socket %d)", recvd, SOCKERRNO, sockfd);
310: return CURLE_RECV_ERROR;
311: }
312:
313: recvd = quiche_conn_recv(qs->conn, buf, recvd);
314: if(recvd == QUICHE_ERR_DONE)
315: break;
316:
317: if(recvd < 0) {
318: failf(conn->data, "quiche_conn_recv() == %d", recvd);
319: return CURLE_RECV_ERROR;
320: }
321: } while(1);
322:
323: return CURLE_OK;
324: }
325:
326: /*
327: * flush_egress drains the buffers and sends off data.
328: * Calls failf() on errors.
329: */
330: static CURLcode flush_egress(struct connectdata *conn, int sockfd,
331: struct quicsocket *qs)
332: {
333: ssize_t sent;
334: static uint8_t out[1200];
335: int64_t timeout_ns;
336:
337: do {
338: sent = quiche_conn_send(qs->conn, out, sizeof(out));
339: if(sent == QUICHE_ERR_DONE)
340: break;
341:
342: if(sent < 0) {
343: failf(conn->data, "quiche_conn_send returned %zd\n",
344: sent);
345: return CURLE_SEND_ERROR;
346: }
347:
348: sent = send(sockfd, out, sent, 0);
349: if(sent < 0) {
350: failf(conn->data, "send() returned %zd\n", sent);
351: return CURLE_SEND_ERROR;
352: }
353: } while(1);
354:
355: /* time until the next timeout event, as nanoseconds. */
356: timeout_ns = quiche_conn_timeout_as_nanos(qs->conn);
357: if(timeout_ns)
358: /* expire uses milliseconds */
359: Curl_expire(conn->data, (timeout_ns + 999999) / 1000000, EXPIRE_QUIC);
360:
361: return CURLE_OK;
362: }
363:
364: struct h3h1header {
365: char *dest;
366: size_t destlen; /* left to use */
367: size_t nlen; /* used */
368: };
369:
370: static int cb_each_header(uint8_t *name, size_t name_len,
371: uint8_t *value, size_t value_len,
372: void *argp)
373: {
374: struct h3h1header *headers = (struct h3h1header *)argp;
375: size_t olen = 0;
376:
377: if((name_len == 7) && !strncmp(":status", (char *)name, 7)) {
378: msnprintf(headers->dest,
379: headers->destlen, "HTTP/3 %.*s\n",
380: (int) value_len, value);
381: }
382: else if(!headers->nlen) {
383: return CURLE_HTTP3;
384: }
385: else {
386: msnprintf(headers->dest,
387: headers->destlen, "%.*s: %.*s\n",
388: (int)name_len, name, (int) value_len, value);
389: }
390: olen = strlen(headers->dest);
391: headers->destlen -= olen;
392: headers->nlen += olen;
393: headers->dest += olen;
394: return 0;
395: }
396:
397: static ssize_t h3_stream_recv(struct connectdata *conn,
398: int sockindex,
399: char *buf,
400: size_t buffersize,
401: CURLcode *curlcode)
402: {
403: ssize_t recvd = -1;
404: ssize_t rcode;
405: struct quicsocket *qs = conn->quic;
406: curl_socket_t sockfd = conn->sock[sockindex];
407: quiche_h3_event *ev;
408: int rc;
409: struct h3h1header headers;
410: struct Curl_easy *data = conn->data;
411: struct HTTP *stream = data->req.protop;
412: headers.dest = buf;
413: headers.destlen = buffersize;
414: headers.nlen = 0;
415:
416: if(process_ingress(conn, sockfd, qs)) {
417: infof(data, "h3_stream_recv returns on ingress\n");
418: *curlcode = CURLE_RECV_ERROR;
419: return -1;
420: }
421:
422: while(recvd < 0) {
423: int64_t s = quiche_h3_conn_poll(qs->h3c, qs->conn, &ev);
424: if(s < 0)
425: /* nothing more to do */
426: break;
427:
428: if(s != stream->stream3_id) {
429: /* another transfer, ignore for now */
430: infof(data, "Got h3 for stream %u, expects %u\n",
431: s, stream->stream3_id);
432: continue;
433: }
434:
435: switch(quiche_h3_event_type(ev)) {
436: case QUICHE_H3_EVENT_HEADERS:
437: rc = quiche_h3_event_for_each_header(ev, cb_each_header, &headers);
438: if(rc) {
439: *curlcode = rc;
440: failf(data, "Error in HTTP/3 response header");
441: break;
442: }
443: recvd = headers.nlen;
444: break;
445: case QUICHE_H3_EVENT_DATA:
446: if(!stream->firstbody) {
447: /* add a header-body separator CRLF */
448: buf[0] = '\r';
449: buf[1] = '\n';
450: buf += 2;
451: buffersize -= 2;
452: stream->firstbody = TRUE;
453: recvd = 2; /* two bytes already */
454: }
455: else
456: recvd = 0;
457: rcode = quiche_h3_recv_body(qs->h3c, qs->conn, s, (unsigned char *)buf,
458: buffersize);
459: if(rcode <= 0) {
460: recvd = -1;
461: break;
462: }
463: recvd += rcode;
464: break;
465:
466: case QUICHE_H3_EVENT_FINISHED:
467: streamclose(conn, "End of stream");
468: recvd = 0; /* end of stream */
469: break;
470: default:
471: break;
472: }
473:
474: quiche_h3_event_free(ev);
475: }
476: if(flush_egress(conn, sockfd, qs)) {
477: *curlcode = CURLE_SEND_ERROR;
478: return -1;
479: }
480:
481: *curlcode = (-1 == recvd)? CURLE_AGAIN : CURLE_OK;
482: if(recvd >= 0)
483: /* Get this called again to drain the event queue */
484: Curl_expire(data, 0, EXPIRE_QUIC);
485:
486: data->state.drain = (recvd >= 0) ? 1 : 0;
487: return recvd;
488: }
489:
490: static ssize_t h3_stream_send(struct connectdata *conn,
491: int sockindex,
492: const void *mem,
493: size_t len,
494: CURLcode *curlcode)
495: {
496: ssize_t sent;
497: struct quicsocket *qs = conn->quic;
498: curl_socket_t sockfd = conn->sock[sockindex];
499: struct HTTP *stream = conn->data->req.protop;
500:
501: if(!stream->h3req) {
502: CURLcode result = http_request(conn, mem, len);
503: if(result) {
504: *curlcode = CURLE_SEND_ERROR;
505: return -1;
506: }
507: sent = len;
508: }
509: else {
510: H3BUGF(infof(conn->data, "Pass on %zd body bytes to quiche\n",
511: len));
512: sent = quiche_h3_send_body(qs->h3c, qs->conn, stream->stream3_id,
513: (uint8_t *)mem, len, FALSE);
514: if(sent < 0) {
515: *curlcode = CURLE_SEND_ERROR;
516: return -1;
517: }
518: }
519:
520: if(flush_egress(conn, sockfd, qs)) {
521: *curlcode = CURLE_SEND_ERROR;
522: return -1;
523: }
524:
525: *curlcode = CURLE_OK;
526: return sent;
527: }
528:
529: /*
530: * Store quiche version info in this buffer, Prefix with a space. Return total
531: * length written.
532: */
533: int Curl_quic_ver(char *p, size_t len)
534: {
535: return msnprintf(p, len, "quiche/%s", quiche_version());
536: }
537:
538: /* Index where :authority header field will appear in request header
539: field list. */
540: #define AUTHORITY_DST_IDX 3
541:
542: static CURLcode http_request(struct connectdata *conn, const void *mem,
543: size_t len)
544: {
545: /*
546: */
547: struct HTTP *stream = conn->data->req.protop;
548: size_t nheader;
549: size_t i;
550: size_t authority_idx;
551: char *hdbuf = (char *)mem;
552: char *end, *line_end;
553: int64_t stream3_id;
554: quiche_h3_header *nva = NULL;
555: struct quicsocket *qs = conn->quic;
556: CURLcode result = CURLE_OK;
557: struct Curl_easy *data = conn->data;
558:
559: stream->h3req = TRUE; /* senf off! */
560:
561: /* Calculate number of headers contained in [mem, mem + len). Assumes a
562: correctly generated HTTP header field block. */
563: nheader = 0;
564: for(i = 1; i < len; ++i) {
565: if(hdbuf[i] == '\n' && hdbuf[i - 1] == '\r') {
566: ++nheader;
567: ++i;
568: }
569: }
570: if(nheader < 2)
571: goto fail;
572:
573: /* We counted additional 2 \r\n in the first and last line. We need 3
574: new headers: :method, :path and :scheme. Therefore we need one
575: more space. */
576: nheader += 1;
577: nva = malloc(sizeof(quiche_h3_header) * nheader);
578: if(!nva) {
579: result = CURLE_OUT_OF_MEMORY;
580: goto fail;
581: }
582:
583: /* Extract :method, :path from request line
584: We do line endings with CRLF so checking for CR is enough */
585: line_end = memchr(hdbuf, '\r', len);
586: if(!line_end) {
587: result = CURLE_BAD_FUNCTION_ARGUMENT; /* internal error */
588: goto fail;
589: }
590:
591: /* Method does not contain spaces */
592: end = memchr(hdbuf, ' ', line_end - hdbuf);
593: if(!end || end == hdbuf)
594: goto fail;
595: nva[0].name = (unsigned char *)":method";
596: nva[0].name_len = strlen((char *)nva[0].name);
597: nva[0].value = (unsigned char *)hdbuf;
598: nva[0].value_len = (size_t)(end - hdbuf);
599:
600: hdbuf = end + 1;
601:
602: /* Path may contain spaces so scan backwards */
603: end = NULL;
604: for(i = (size_t)(line_end - hdbuf); i; --i) {
605: if(hdbuf[i - 1] == ' ') {
606: end = &hdbuf[i - 1];
607: break;
608: }
609: }
610: if(!end || end == hdbuf)
611: goto fail;
612: nva[1].name = (unsigned char *)":path";
613: nva[1].name_len = strlen((char *)nva[1].name);
614: nva[1].value = (unsigned char *)hdbuf;
615: nva[1].value_len = (size_t)(end - hdbuf);
616:
617: nva[2].name = (unsigned char *)":scheme";
618: nva[2].name_len = strlen((char *)nva[2].name);
619: if(conn->handler->flags & PROTOPT_SSL)
620: nva[2].value = (unsigned char *)"https";
621: else
622: nva[2].value = (unsigned char *)"http";
623: nva[2].value_len = strlen((char *)nva[2].value);
624:
625:
626: authority_idx = 0;
627: i = 3;
628: while(i < nheader) {
629: size_t hlen;
630:
631: hdbuf = line_end + 2;
632:
633: /* check for next CR, but only within the piece of data left in the given
634: buffer */
635: line_end = memchr(hdbuf, '\r', len - (hdbuf - (char *)mem));
636: if(!line_end || (line_end == hdbuf))
637: goto fail;
638:
639: /* header continuation lines are not supported */
640: if(*hdbuf == ' ' || *hdbuf == '\t')
641: goto fail;
642:
643: for(end = hdbuf; end < line_end && *end != ':'; ++end)
644: ;
645: if(end == hdbuf || end == line_end)
646: goto fail;
647: hlen = end - hdbuf;
648:
649: if(hlen == 4 && strncasecompare("host", hdbuf, 4)) {
650: authority_idx = i;
651: nva[i].name = (unsigned char *)":authority";
652: nva[i].name_len = strlen((char *)nva[i].name);
653: }
654: else {
655: nva[i].name_len = (size_t)(end - hdbuf);
656: /* Lower case the header name for HTTP/3 */
657: Curl_strntolower((char *)hdbuf, hdbuf, nva[i].name_len);
658: nva[i].name = (unsigned char *)hdbuf;
659: }
660: hdbuf = end + 1;
661: while(*hdbuf == ' ' || *hdbuf == '\t')
662: ++hdbuf;
663: end = line_end;
664:
665: #if 0 /* This should probably go in more or less like this */
666: switch(inspect_header((const char *)nva[i].name, nva[i].namelen, hdbuf,
667: end - hdbuf)) {
668: case HEADERINST_IGNORE:
669: /* skip header fields prohibited by HTTP/2 specification. */
670: --nheader;
671: continue;
672: case HEADERINST_TE_TRAILERS:
673: nva[i].value = (uint8_t*)"trailers";
674: nva[i].value_len = sizeof("trailers") - 1;
675: break;
676: default:
677: nva[i].value = (unsigned char *)hdbuf;
678: nva[i].value_len = (size_t)(end - hdbuf);
679: }
680: #endif
681: nva[i].value = (unsigned char *)hdbuf;
682: nva[i].value_len = (size_t)(end - hdbuf);
683:
684: ++i;
685: }
686:
687: /* :authority must come before non-pseudo header fields */
688: if(authority_idx != 0 && authority_idx != AUTHORITY_DST_IDX) {
689: quiche_h3_header authority = nva[authority_idx];
690: for(i = authority_idx; i > AUTHORITY_DST_IDX; --i) {
691: nva[i] = nva[i - 1];
692: }
693: nva[i] = authority;
694: }
695:
696: /* Warn stream may be rejected if cumulative length of headers is too
697: large. */
698: #define MAX_ACC 60000 /* <64KB to account for some overhead */
699: {
700: size_t acc = 0;
701:
702: for(i = 0; i < nheader; ++i) {
703: acc += nva[i].name_len + nva[i].value_len;
704:
705: H3BUGF(infof(data, "h3 [%.*s: %.*s]\n",
706: nva[i].name_len, nva[i].name,
707: nva[i].value_len, nva[i].value));
708: }
709:
710: if(acc > MAX_ACC) {
711: infof(data, "http_request: Warning: The cumulative length of all "
712: "headers exceeds %zu bytes and that could cause the "
713: "stream to be rejected.\n", MAX_ACC);
714: }
715: }
716:
717: switch(data->set.httpreq) {
718: case HTTPREQ_POST:
719: case HTTPREQ_POST_FORM:
720: case HTTPREQ_POST_MIME:
721: case HTTPREQ_PUT:
722: if(data->state.infilesize != -1)
723: stream->upload_left = data->state.infilesize;
724: else
725: /* data sending without specifying the data amount up front */
726: stream->upload_left = -1; /* unknown, but not zero */
727:
728: stream3_id = quiche_h3_send_request(qs->h3c, qs->conn, nva, nheader,
729: stream->upload_left ? FALSE: TRUE);
730: if((stream3_id >= 0) && data->set.postfields) {
731: ssize_t sent = quiche_h3_send_body(qs->h3c, qs->conn, stream3_id,
732: (uint8_t *)data->set.postfields,
733: stream->upload_left, TRUE);
734: if(sent <= 0) {
735: failf(data, "quiche_h3_send_body failed!");
736: result = CURLE_SEND_ERROR;
737: }
738: stream->upload_left = 0; /* nothing left to send */
739: }
740: break;
741: default:
742: stream3_id = quiche_h3_send_request(qs->h3c, qs->conn, nva, nheader,
743: TRUE);
744: break;
745: }
746:
747: Curl_safefree(nva);
748:
749: if(stream3_id < 0) {
750: H3BUGF(infof(data, "quiche_h3_send_request returned %d\n",
751: stream3_id));
752: result = CURLE_SEND_ERROR;
753: goto fail;
754: }
755:
756: infof(data, "Using HTTP/3 Stream ID: %x (easy handle %p)\n",
757: stream3_id, (void *)data);
758: stream->stream3_id = stream3_id;
759:
760: return CURLE_OK;
761:
762: fail:
763: free(nva);
764: return result;
765: }
766:
767: /*
768: * Called from transfer.c:done_sending when we stop HTTP/3 uploading.
769: */
770: CURLcode Curl_quic_done_sending(struct connectdata *conn)
771: {
772: if(conn->handler == &Curl_handler_http3) {
773: /* only for HTTP/3 transfers */
774: ssize_t sent;
775: struct HTTP *stream = conn->data->req.protop;
776: struct quicsocket *qs = conn->quic;
777: fprintf(stderr, "!!! Curl_quic_done_sending\n");
778: stream->upload_done = TRUE;
779: sent = quiche_h3_send_body(qs->h3c, qs->conn, stream->stream3_id,
780: NULL, 0, TRUE);
781: if(sent < 0)
782: return CURLE_SEND_ERROR;
783: }
784:
785: return CURLE_OK;
786: }
787:
788: /*
789: * Called from http.c:Curl_http_done when a request completes.
790: */
791: void Curl_quic_done(struct Curl_easy *data, bool premature)
792: {
793: (void)data;
794: (void)premature;
795: }
796:
797: /*
798: * Called from transfer.c:data_pending to know if we should keep looping
799: * to receive more data from the connection.
800: */
801: bool Curl_quic_data_pending(const struct Curl_easy *data)
802: {
803: (void)data;
804: return FALSE;
805: }
806:
807: #endif
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>