1*6236dae4SAndroid Build Coastguard Worker /***************************************************************************
2*6236dae4SAndroid Build Coastguard Worker * _ _ ____ _
3*6236dae4SAndroid Build Coastguard Worker * Project ___| | | | _ \| |
4*6236dae4SAndroid Build Coastguard Worker * / __| | | | |_) | |
5*6236dae4SAndroid Build Coastguard Worker * | (__| |_| | _ <| |___
6*6236dae4SAndroid Build Coastguard Worker * \___|\___/|_| \_\_____|
7*6236dae4SAndroid Build Coastguard Worker *
8*6236dae4SAndroid Build Coastguard Worker * Copyright (C) Daniel Stenberg, <[email protected]>, et al.
9*6236dae4SAndroid Build Coastguard Worker *
10*6236dae4SAndroid Build Coastguard Worker * This software is licensed as described in the file COPYING, which
11*6236dae4SAndroid Build Coastguard Worker * you should have received as part of this distribution. The terms
12*6236dae4SAndroid Build Coastguard Worker * are also available at https://curl.se/docs/copyright.html.
13*6236dae4SAndroid Build Coastguard Worker *
14*6236dae4SAndroid Build Coastguard Worker * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15*6236dae4SAndroid Build Coastguard Worker * copies of the Software, and permit persons to whom the Software is
16*6236dae4SAndroid Build Coastguard Worker * furnished to do so, under the terms of the COPYING file.
17*6236dae4SAndroid Build Coastguard Worker *
18*6236dae4SAndroid Build Coastguard Worker * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19*6236dae4SAndroid Build Coastguard Worker * KIND, either express or implied.
20*6236dae4SAndroid Build Coastguard Worker *
21*6236dae4SAndroid Build Coastguard Worker * SPDX-License-Identifier: curl
22*6236dae4SAndroid Build Coastguard Worker *
23*6236dae4SAndroid Build Coastguard Worker ***************************************************************************/
24*6236dae4SAndroid Build Coastguard Worker
25*6236dae4SAndroid Build Coastguard Worker #include "curl_setup.h"
26*6236dae4SAndroid Build Coastguard Worker
27*6236dae4SAndroid Build Coastguard Worker #ifdef HAVE_NETINET_UDP_H
28*6236dae4SAndroid Build Coastguard Worker #include <netinet/udp.h>
29*6236dae4SAndroid Build Coastguard Worker #endif
30*6236dae4SAndroid Build Coastguard Worker #ifdef HAVE_FCNTL_H
31*6236dae4SAndroid Build Coastguard Worker #include <fcntl.h>
32*6236dae4SAndroid Build Coastguard Worker #endif
33*6236dae4SAndroid Build Coastguard Worker #include "urldata.h"
34*6236dae4SAndroid Build Coastguard Worker #include "bufq.h"
35*6236dae4SAndroid Build Coastguard Worker #include "dynbuf.h"
36*6236dae4SAndroid Build Coastguard Worker #include "cfilters.h"
37*6236dae4SAndroid Build Coastguard Worker #include "curl_trc.h"
38*6236dae4SAndroid Build Coastguard Worker #include "curl_msh3.h"
39*6236dae4SAndroid Build Coastguard Worker #include "curl_ngtcp2.h"
40*6236dae4SAndroid Build Coastguard Worker #include "curl_osslq.h"
41*6236dae4SAndroid Build Coastguard Worker #include "curl_quiche.h"
42*6236dae4SAndroid Build Coastguard Worker #include "multiif.h"
43*6236dae4SAndroid Build Coastguard Worker #include "rand.h"
44*6236dae4SAndroid Build Coastguard Worker #include "vquic.h"
45*6236dae4SAndroid Build Coastguard Worker #include "vquic_int.h"
46*6236dae4SAndroid Build Coastguard Worker #include "strerror.h"
47*6236dae4SAndroid Build Coastguard Worker
48*6236dae4SAndroid Build Coastguard Worker /* The last 3 #include files should be in this order */
49*6236dae4SAndroid Build Coastguard Worker #include "curl_printf.h"
50*6236dae4SAndroid Build Coastguard Worker #include "curl_memory.h"
51*6236dae4SAndroid Build Coastguard Worker #include "memdebug.h"
52*6236dae4SAndroid Build Coastguard Worker
53*6236dae4SAndroid Build Coastguard Worker
54*6236dae4SAndroid Build Coastguard Worker #ifdef USE_HTTP3
55*6236dae4SAndroid Build Coastguard Worker
56*6236dae4SAndroid Build Coastguard Worker #ifdef O_BINARY
57*6236dae4SAndroid Build Coastguard Worker #define QLOGMODE O_WRONLY|O_CREAT|O_BINARY
58*6236dae4SAndroid Build Coastguard Worker #else
59*6236dae4SAndroid Build Coastguard Worker #define QLOGMODE O_WRONLY|O_CREAT
60*6236dae4SAndroid Build Coastguard Worker #endif
61*6236dae4SAndroid Build Coastguard Worker
62*6236dae4SAndroid Build Coastguard Worker #define NW_CHUNK_SIZE (64 * 1024)
63*6236dae4SAndroid Build Coastguard Worker #define NW_SEND_CHUNKS 2
64*6236dae4SAndroid Build Coastguard Worker
65*6236dae4SAndroid Build Coastguard Worker
Curl_quic_ver(char * p,size_t len)66*6236dae4SAndroid Build Coastguard Worker void Curl_quic_ver(char *p, size_t len)
67*6236dae4SAndroid Build Coastguard Worker {
68*6236dae4SAndroid Build Coastguard Worker #if defined(USE_NGTCP2) && defined(USE_NGHTTP3)
69*6236dae4SAndroid Build Coastguard Worker Curl_ngtcp2_ver(p, len);
70*6236dae4SAndroid Build Coastguard Worker #elif defined(USE_OPENSSL_QUIC) && defined(USE_NGHTTP3)
71*6236dae4SAndroid Build Coastguard Worker Curl_osslq_ver(p, len);
72*6236dae4SAndroid Build Coastguard Worker #elif defined(USE_QUICHE)
73*6236dae4SAndroid Build Coastguard Worker Curl_quiche_ver(p, len);
74*6236dae4SAndroid Build Coastguard Worker #elif defined(USE_MSH3)
75*6236dae4SAndroid Build Coastguard Worker Curl_msh3_ver(p, len);
76*6236dae4SAndroid Build Coastguard Worker #endif
77*6236dae4SAndroid Build Coastguard Worker }
78*6236dae4SAndroid Build Coastguard Worker
vquic_ctx_init(struct cf_quic_ctx * qctx)79*6236dae4SAndroid Build Coastguard Worker CURLcode vquic_ctx_init(struct cf_quic_ctx *qctx)
80*6236dae4SAndroid Build Coastguard Worker {
81*6236dae4SAndroid Build Coastguard Worker Curl_bufq_init2(&qctx->sendbuf, NW_CHUNK_SIZE, NW_SEND_CHUNKS,
82*6236dae4SAndroid Build Coastguard Worker BUFQ_OPT_SOFT_LIMIT);
83*6236dae4SAndroid Build Coastguard Worker #if defined(__linux__) && defined(UDP_SEGMENT) && defined(HAVE_SENDMSG)
84*6236dae4SAndroid Build Coastguard Worker qctx->no_gso = FALSE;
85*6236dae4SAndroid Build Coastguard Worker #else
86*6236dae4SAndroid Build Coastguard Worker qctx->no_gso = TRUE;
87*6236dae4SAndroid Build Coastguard Worker #endif
88*6236dae4SAndroid Build Coastguard Worker #ifdef DEBUGBUILD
89*6236dae4SAndroid Build Coastguard Worker {
90*6236dae4SAndroid Build Coastguard Worker char *p = getenv("CURL_DBG_QUIC_WBLOCK");
91*6236dae4SAndroid Build Coastguard Worker if(p) {
92*6236dae4SAndroid Build Coastguard Worker long l = strtol(p, NULL, 10);
93*6236dae4SAndroid Build Coastguard Worker if(l >= 0 && l <= 100)
94*6236dae4SAndroid Build Coastguard Worker qctx->wblock_percent = (int)l;
95*6236dae4SAndroid Build Coastguard Worker }
96*6236dae4SAndroid Build Coastguard Worker }
97*6236dae4SAndroid Build Coastguard Worker #endif
98*6236dae4SAndroid Build Coastguard Worker vquic_ctx_update_time(qctx);
99*6236dae4SAndroid Build Coastguard Worker
100*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
101*6236dae4SAndroid Build Coastguard Worker }
102*6236dae4SAndroid Build Coastguard Worker
vquic_ctx_free(struct cf_quic_ctx * qctx)103*6236dae4SAndroid Build Coastguard Worker void vquic_ctx_free(struct cf_quic_ctx *qctx)
104*6236dae4SAndroid Build Coastguard Worker {
105*6236dae4SAndroid Build Coastguard Worker Curl_bufq_free(&qctx->sendbuf);
106*6236dae4SAndroid Build Coastguard Worker }
107*6236dae4SAndroid Build Coastguard Worker
vquic_ctx_update_time(struct cf_quic_ctx * qctx)108*6236dae4SAndroid Build Coastguard Worker void vquic_ctx_update_time(struct cf_quic_ctx *qctx)
109*6236dae4SAndroid Build Coastguard Worker {
110*6236dae4SAndroid Build Coastguard Worker qctx->last_op = Curl_now();
111*6236dae4SAndroid Build Coastguard Worker }
112*6236dae4SAndroid Build Coastguard Worker
113*6236dae4SAndroid Build Coastguard Worker static CURLcode send_packet_no_gso(struct Curl_cfilter *cf,
114*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
115*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx,
116*6236dae4SAndroid Build Coastguard Worker const uint8_t *pkt, size_t pktlen,
117*6236dae4SAndroid Build Coastguard Worker size_t gsolen, size_t *psent);
118*6236dae4SAndroid Build Coastguard Worker
do_sendmsg(struct Curl_cfilter * cf,struct Curl_easy * data,struct cf_quic_ctx * qctx,const uint8_t * pkt,size_t pktlen,size_t gsolen,size_t * psent)119*6236dae4SAndroid Build Coastguard Worker static CURLcode do_sendmsg(struct Curl_cfilter *cf,
120*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
121*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx,
122*6236dae4SAndroid Build Coastguard Worker const uint8_t *pkt, size_t pktlen, size_t gsolen,
123*6236dae4SAndroid Build Coastguard Worker size_t *psent)
124*6236dae4SAndroid Build Coastguard Worker {
125*6236dae4SAndroid Build Coastguard Worker #ifdef HAVE_SENDMSG
126*6236dae4SAndroid Build Coastguard Worker struct iovec msg_iov;
127*6236dae4SAndroid Build Coastguard Worker struct msghdr msg = {0};
128*6236dae4SAndroid Build Coastguard Worker ssize_t sent;
129*6236dae4SAndroid Build Coastguard Worker #if defined(__linux__) && defined(UDP_SEGMENT)
130*6236dae4SAndroid Build Coastguard Worker uint8_t msg_ctrl[32];
131*6236dae4SAndroid Build Coastguard Worker struct cmsghdr *cm;
132*6236dae4SAndroid Build Coastguard Worker #endif
133*6236dae4SAndroid Build Coastguard Worker
134*6236dae4SAndroid Build Coastguard Worker *psent = 0;
135*6236dae4SAndroid Build Coastguard Worker msg_iov.iov_base = (uint8_t *)pkt;
136*6236dae4SAndroid Build Coastguard Worker msg_iov.iov_len = pktlen;
137*6236dae4SAndroid Build Coastguard Worker msg.msg_iov = &msg_iov;
138*6236dae4SAndroid Build Coastguard Worker msg.msg_iovlen = 1;
139*6236dae4SAndroid Build Coastguard Worker
140*6236dae4SAndroid Build Coastguard Worker #if defined(__linux__) && defined(UDP_SEGMENT)
141*6236dae4SAndroid Build Coastguard Worker if(pktlen > gsolen) {
142*6236dae4SAndroid Build Coastguard Worker /* Only set this, when we need it. macOS, for example,
143*6236dae4SAndroid Build Coastguard Worker * does not seem to like a msg_control of length 0. */
144*6236dae4SAndroid Build Coastguard Worker msg.msg_control = msg_ctrl;
145*6236dae4SAndroid Build Coastguard Worker assert(sizeof(msg_ctrl) >= CMSG_SPACE(sizeof(int)));
146*6236dae4SAndroid Build Coastguard Worker msg.msg_controllen = CMSG_SPACE(sizeof(int));
147*6236dae4SAndroid Build Coastguard Worker cm = CMSG_FIRSTHDR(&msg);
148*6236dae4SAndroid Build Coastguard Worker cm->cmsg_level = SOL_UDP;
149*6236dae4SAndroid Build Coastguard Worker cm->cmsg_type = UDP_SEGMENT;
150*6236dae4SAndroid Build Coastguard Worker cm->cmsg_len = CMSG_LEN(sizeof(uint16_t));
151*6236dae4SAndroid Build Coastguard Worker *(uint16_t *)(void *)CMSG_DATA(cm) = gsolen & 0xffff;
152*6236dae4SAndroid Build Coastguard Worker }
153*6236dae4SAndroid Build Coastguard Worker #endif
154*6236dae4SAndroid Build Coastguard Worker
155*6236dae4SAndroid Build Coastguard Worker
156*6236dae4SAndroid Build Coastguard Worker while((sent = sendmsg(qctx->sockfd, &msg, 0)) == -1 && SOCKERRNO == EINTR)
157*6236dae4SAndroid Build Coastguard Worker ;
158*6236dae4SAndroid Build Coastguard Worker
159*6236dae4SAndroid Build Coastguard Worker if(sent == -1) {
160*6236dae4SAndroid Build Coastguard Worker switch(SOCKERRNO) {
161*6236dae4SAndroid Build Coastguard Worker case EAGAIN:
162*6236dae4SAndroid Build Coastguard Worker #if EAGAIN != EWOULDBLOCK
163*6236dae4SAndroid Build Coastguard Worker case EWOULDBLOCK:
164*6236dae4SAndroid Build Coastguard Worker #endif
165*6236dae4SAndroid Build Coastguard Worker return CURLE_AGAIN;
166*6236dae4SAndroid Build Coastguard Worker case EMSGSIZE:
167*6236dae4SAndroid Build Coastguard Worker /* UDP datagram is too large; caused by PMTUD. Just let it be lost. */
168*6236dae4SAndroid Build Coastguard Worker break;
169*6236dae4SAndroid Build Coastguard Worker case EIO:
170*6236dae4SAndroid Build Coastguard Worker if(pktlen > gsolen) {
171*6236dae4SAndroid Build Coastguard Worker /* GSO failure */
172*6236dae4SAndroid Build Coastguard Worker failf(data, "sendmsg() returned %zd (errno %d); disable GSO", sent,
173*6236dae4SAndroid Build Coastguard Worker SOCKERRNO);
174*6236dae4SAndroid Build Coastguard Worker qctx->no_gso = TRUE;
175*6236dae4SAndroid Build Coastguard Worker return send_packet_no_gso(cf, data, qctx, pkt, pktlen, gsolen, psent);
176*6236dae4SAndroid Build Coastguard Worker }
177*6236dae4SAndroid Build Coastguard Worker FALLTHROUGH();
178*6236dae4SAndroid Build Coastguard Worker default:
179*6236dae4SAndroid Build Coastguard Worker failf(data, "sendmsg() returned %zd (errno %d)", sent, SOCKERRNO);
180*6236dae4SAndroid Build Coastguard Worker return CURLE_SEND_ERROR;
181*6236dae4SAndroid Build Coastguard Worker }
182*6236dae4SAndroid Build Coastguard Worker }
183*6236dae4SAndroid Build Coastguard Worker else {
184*6236dae4SAndroid Build Coastguard Worker assert(pktlen == (size_t)sent);
185*6236dae4SAndroid Build Coastguard Worker }
186*6236dae4SAndroid Build Coastguard Worker #else
187*6236dae4SAndroid Build Coastguard Worker ssize_t sent;
188*6236dae4SAndroid Build Coastguard Worker (void)gsolen;
189*6236dae4SAndroid Build Coastguard Worker
190*6236dae4SAndroid Build Coastguard Worker *psent = 0;
191*6236dae4SAndroid Build Coastguard Worker
192*6236dae4SAndroid Build Coastguard Worker while((sent = send(qctx->sockfd,
193*6236dae4SAndroid Build Coastguard Worker (const char *)pkt, (SEND_TYPE_ARG3)pktlen, 0)) == -1 &&
194*6236dae4SAndroid Build Coastguard Worker SOCKERRNO == EINTR)
195*6236dae4SAndroid Build Coastguard Worker ;
196*6236dae4SAndroid Build Coastguard Worker
197*6236dae4SAndroid Build Coastguard Worker if(sent == -1) {
198*6236dae4SAndroid Build Coastguard Worker if(SOCKERRNO == EAGAIN || SOCKERRNO == EWOULDBLOCK) {
199*6236dae4SAndroid Build Coastguard Worker return CURLE_AGAIN;
200*6236dae4SAndroid Build Coastguard Worker }
201*6236dae4SAndroid Build Coastguard Worker else {
202*6236dae4SAndroid Build Coastguard Worker failf(data, "send() returned %zd (errno %d)", sent, SOCKERRNO);
203*6236dae4SAndroid Build Coastguard Worker if(SOCKERRNO != EMSGSIZE) {
204*6236dae4SAndroid Build Coastguard Worker return CURLE_SEND_ERROR;
205*6236dae4SAndroid Build Coastguard Worker }
206*6236dae4SAndroid Build Coastguard Worker /* UDP datagram is too large; caused by PMTUD. Just let it be
207*6236dae4SAndroid Build Coastguard Worker lost. */
208*6236dae4SAndroid Build Coastguard Worker }
209*6236dae4SAndroid Build Coastguard Worker }
210*6236dae4SAndroid Build Coastguard Worker #endif
211*6236dae4SAndroid Build Coastguard Worker (void)cf;
212*6236dae4SAndroid Build Coastguard Worker *psent = pktlen;
213*6236dae4SAndroid Build Coastguard Worker
214*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
215*6236dae4SAndroid Build Coastguard Worker }
216*6236dae4SAndroid Build Coastguard Worker
send_packet_no_gso(struct Curl_cfilter * cf,struct Curl_easy * data,struct cf_quic_ctx * qctx,const uint8_t * pkt,size_t pktlen,size_t gsolen,size_t * psent)217*6236dae4SAndroid Build Coastguard Worker static CURLcode send_packet_no_gso(struct Curl_cfilter *cf,
218*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
219*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx,
220*6236dae4SAndroid Build Coastguard Worker const uint8_t *pkt, size_t pktlen,
221*6236dae4SAndroid Build Coastguard Worker size_t gsolen, size_t *psent)
222*6236dae4SAndroid Build Coastguard Worker {
223*6236dae4SAndroid Build Coastguard Worker const uint8_t *p, *end = pkt + pktlen;
224*6236dae4SAndroid Build Coastguard Worker size_t sent;
225*6236dae4SAndroid Build Coastguard Worker
226*6236dae4SAndroid Build Coastguard Worker *psent = 0;
227*6236dae4SAndroid Build Coastguard Worker
228*6236dae4SAndroid Build Coastguard Worker for(p = pkt; p < end; p += gsolen) {
229*6236dae4SAndroid Build Coastguard Worker size_t len = CURLMIN(gsolen, (size_t)(end - p));
230*6236dae4SAndroid Build Coastguard Worker CURLcode curlcode = do_sendmsg(cf, data, qctx, p, len, len, &sent);
231*6236dae4SAndroid Build Coastguard Worker if(curlcode != CURLE_OK) {
232*6236dae4SAndroid Build Coastguard Worker return curlcode;
233*6236dae4SAndroid Build Coastguard Worker }
234*6236dae4SAndroid Build Coastguard Worker *psent += sent;
235*6236dae4SAndroid Build Coastguard Worker }
236*6236dae4SAndroid Build Coastguard Worker
237*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
238*6236dae4SAndroid Build Coastguard Worker }
239*6236dae4SAndroid Build Coastguard Worker
vquic_send_packets(struct Curl_cfilter * cf,struct Curl_easy * data,struct cf_quic_ctx * qctx,const uint8_t * pkt,size_t pktlen,size_t gsolen,size_t * psent)240*6236dae4SAndroid Build Coastguard Worker static CURLcode vquic_send_packets(struct Curl_cfilter *cf,
241*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
242*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx,
243*6236dae4SAndroid Build Coastguard Worker const uint8_t *pkt, size_t pktlen,
244*6236dae4SAndroid Build Coastguard Worker size_t gsolen, size_t *psent)
245*6236dae4SAndroid Build Coastguard Worker {
246*6236dae4SAndroid Build Coastguard Worker CURLcode result;
247*6236dae4SAndroid Build Coastguard Worker #ifdef DEBUGBUILD
248*6236dae4SAndroid Build Coastguard Worker /* simulate network blocking/partial writes */
249*6236dae4SAndroid Build Coastguard Worker if(qctx->wblock_percent > 0) {
250*6236dae4SAndroid Build Coastguard Worker unsigned char c;
251*6236dae4SAndroid Build Coastguard Worker Curl_rand(data, &c, 1);
252*6236dae4SAndroid Build Coastguard Worker if(c >= ((100-qctx->wblock_percent)*256/100)) {
253*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "vquic_flush() simulate EWOULDBLOCK");
254*6236dae4SAndroid Build Coastguard Worker return CURLE_AGAIN;
255*6236dae4SAndroid Build Coastguard Worker }
256*6236dae4SAndroid Build Coastguard Worker }
257*6236dae4SAndroid Build Coastguard Worker #endif
258*6236dae4SAndroid Build Coastguard Worker if(qctx->no_gso && pktlen > gsolen) {
259*6236dae4SAndroid Build Coastguard Worker result = send_packet_no_gso(cf, data, qctx, pkt, pktlen, gsolen, psent);
260*6236dae4SAndroid Build Coastguard Worker }
261*6236dae4SAndroid Build Coastguard Worker else {
262*6236dae4SAndroid Build Coastguard Worker result = do_sendmsg(cf, data, qctx, pkt, pktlen, gsolen, psent);
263*6236dae4SAndroid Build Coastguard Worker }
264*6236dae4SAndroid Build Coastguard Worker if(!result)
265*6236dae4SAndroid Build Coastguard Worker qctx->last_io = qctx->last_op;
266*6236dae4SAndroid Build Coastguard Worker return result;
267*6236dae4SAndroid Build Coastguard Worker }
268*6236dae4SAndroid Build Coastguard Worker
vquic_flush(struct Curl_cfilter * cf,struct Curl_easy * data,struct cf_quic_ctx * qctx)269*6236dae4SAndroid Build Coastguard Worker CURLcode vquic_flush(struct Curl_cfilter *cf, struct Curl_easy *data,
270*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx)
271*6236dae4SAndroid Build Coastguard Worker {
272*6236dae4SAndroid Build Coastguard Worker const unsigned char *buf;
273*6236dae4SAndroid Build Coastguard Worker size_t blen, sent;
274*6236dae4SAndroid Build Coastguard Worker CURLcode result;
275*6236dae4SAndroid Build Coastguard Worker size_t gsolen;
276*6236dae4SAndroid Build Coastguard Worker
277*6236dae4SAndroid Build Coastguard Worker while(Curl_bufq_peek(&qctx->sendbuf, &buf, &blen)) {
278*6236dae4SAndroid Build Coastguard Worker gsolen = qctx->gsolen;
279*6236dae4SAndroid Build Coastguard Worker if(qctx->split_len) {
280*6236dae4SAndroid Build Coastguard Worker gsolen = qctx->split_gsolen;
281*6236dae4SAndroid Build Coastguard Worker if(blen > qctx->split_len)
282*6236dae4SAndroid Build Coastguard Worker blen = qctx->split_len;
283*6236dae4SAndroid Build Coastguard Worker }
284*6236dae4SAndroid Build Coastguard Worker
285*6236dae4SAndroid Build Coastguard Worker result = vquic_send_packets(cf, data, qctx, buf, blen, gsolen, &sent);
286*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "vquic_send(len=%zu, gso=%zu) -> %d, sent=%zu",
287*6236dae4SAndroid Build Coastguard Worker blen, gsolen, result, sent);
288*6236dae4SAndroid Build Coastguard Worker if(result) {
289*6236dae4SAndroid Build Coastguard Worker if(result == CURLE_AGAIN) {
290*6236dae4SAndroid Build Coastguard Worker Curl_bufq_skip(&qctx->sendbuf, sent);
291*6236dae4SAndroid Build Coastguard Worker if(qctx->split_len)
292*6236dae4SAndroid Build Coastguard Worker qctx->split_len -= sent;
293*6236dae4SAndroid Build Coastguard Worker }
294*6236dae4SAndroid Build Coastguard Worker return result;
295*6236dae4SAndroid Build Coastguard Worker }
296*6236dae4SAndroid Build Coastguard Worker Curl_bufq_skip(&qctx->sendbuf, sent);
297*6236dae4SAndroid Build Coastguard Worker if(qctx->split_len)
298*6236dae4SAndroid Build Coastguard Worker qctx->split_len -= sent;
299*6236dae4SAndroid Build Coastguard Worker }
300*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
301*6236dae4SAndroid Build Coastguard Worker }
302*6236dae4SAndroid Build Coastguard Worker
vquic_send(struct Curl_cfilter * cf,struct Curl_easy * data,struct cf_quic_ctx * qctx,size_t gsolen)303*6236dae4SAndroid Build Coastguard Worker CURLcode vquic_send(struct Curl_cfilter *cf, struct Curl_easy *data,
304*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx, size_t gsolen)
305*6236dae4SAndroid Build Coastguard Worker {
306*6236dae4SAndroid Build Coastguard Worker qctx->gsolen = gsolen;
307*6236dae4SAndroid Build Coastguard Worker return vquic_flush(cf, data, qctx);
308*6236dae4SAndroid Build Coastguard Worker }
309*6236dae4SAndroid Build Coastguard Worker
vquic_send_tail_split(struct Curl_cfilter * cf,struct Curl_easy * data,struct cf_quic_ctx * qctx,size_t gsolen,size_t tail_len,size_t tail_gsolen)310*6236dae4SAndroid Build Coastguard Worker CURLcode vquic_send_tail_split(struct Curl_cfilter *cf, struct Curl_easy *data,
311*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx, size_t gsolen,
312*6236dae4SAndroid Build Coastguard Worker size_t tail_len, size_t tail_gsolen)
313*6236dae4SAndroid Build Coastguard Worker {
314*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(Curl_bufq_len(&qctx->sendbuf) > tail_len);
315*6236dae4SAndroid Build Coastguard Worker qctx->split_len = Curl_bufq_len(&qctx->sendbuf) - tail_len;
316*6236dae4SAndroid Build Coastguard Worker qctx->split_gsolen = gsolen;
317*6236dae4SAndroid Build Coastguard Worker qctx->gsolen = tail_gsolen;
318*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "vquic_send_tail_split: [%zu gso=%zu][%zu gso=%zu]",
319*6236dae4SAndroid Build Coastguard Worker qctx->split_len, qctx->split_gsolen,
320*6236dae4SAndroid Build Coastguard Worker tail_len, qctx->gsolen);
321*6236dae4SAndroid Build Coastguard Worker return vquic_flush(cf, data, qctx);
322*6236dae4SAndroid Build Coastguard Worker }
323*6236dae4SAndroid Build Coastguard Worker
324*6236dae4SAndroid Build Coastguard Worker #if defined(HAVE_SENDMMSG) || defined(HAVE_SENDMSG)
vquic_msghdr_get_udp_gro(struct msghdr * msg)325*6236dae4SAndroid Build Coastguard Worker static size_t vquic_msghdr_get_udp_gro(struct msghdr *msg)
326*6236dae4SAndroid Build Coastguard Worker {
327*6236dae4SAndroid Build Coastguard Worker int gso_size = 0;
328*6236dae4SAndroid Build Coastguard Worker #if defined(__linux__) && defined(UDP_GRO)
329*6236dae4SAndroid Build Coastguard Worker struct cmsghdr *cmsg;
330*6236dae4SAndroid Build Coastguard Worker
331*6236dae4SAndroid Build Coastguard Worker /* Workaround musl CMSG_NXTHDR issue */
332*6236dae4SAndroid Build Coastguard Worker #if defined(__clang__) && !defined(__GLIBC__)
333*6236dae4SAndroid Build Coastguard Worker #pragma clang diagnostic push
334*6236dae4SAndroid Build Coastguard Worker #pragma clang diagnostic ignored "-Wsign-compare"
335*6236dae4SAndroid Build Coastguard Worker #pragma clang diagnostic ignored "-Wcast-align"
336*6236dae4SAndroid Build Coastguard Worker #endif
337*6236dae4SAndroid Build Coastguard Worker for(cmsg = CMSG_FIRSTHDR(msg); cmsg; cmsg = CMSG_NXTHDR(msg, cmsg)) {
338*6236dae4SAndroid Build Coastguard Worker #if defined(__clang__) && !defined(__GLIBC__)
339*6236dae4SAndroid Build Coastguard Worker #pragma clang diagnostic pop
340*6236dae4SAndroid Build Coastguard Worker #endif
341*6236dae4SAndroid Build Coastguard Worker if(cmsg->cmsg_level == SOL_UDP && cmsg->cmsg_type == UDP_GRO) {
342*6236dae4SAndroid Build Coastguard Worker memcpy(&gso_size, CMSG_DATA(cmsg), sizeof(gso_size));
343*6236dae4SAndroid Build Coastguard Worker
344*6236dae4SAndroid Build Coastguard Worker break;
345*6236dae4SAndroid Build Coastguard Worker }
346*6236dae4SAndroid Build Coastguard Worker }
347*6236dae4SAndroid Build Coastguard Worker #endif
348*6236dae4SAndroid Build Coastguard Worker (void)msg;
349*6236dae4SAndroid Build Coastguard Worker
350*6236dae4SAndroid Build Coastguard Worker return (size_t)gso_size;
351*6236dae4SAndroid Build Coastguard Worker }
352*6236dae4SAndroid Build Coastguard Worker #endif
353*6236dae4SAndroid Build Coastguard Worker
354*6236dae4SAndroid Build Coastguard Worker #ifdef HAVE_SENDMMSG
recvmmsg_packets(struct Curl_cfilter * cf,struct Curl_easy * data,struct cf_quic_ctx * qctx,size_t max_pkts,vquic_recv_pkt_cb * recv_cb,void * userp)355*6236dae4SAndroid Build Coastguard Worker static CURLcode recvmmsg_packets(struct Curl_cfilter *cf,
356*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
357*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx,
358*6236dae4SAndroid Build Coastguard Worker size_t max_pkts,
359*6236dae4SAndroid Build Coastguard Worker vquic_recv_pkt_cb *recv_cb, void *userp)
360*6236dae4SAndroid Build Coastguard Worker {
361*6236dae4SAndroid Build Coastguard Worker #define MMSG_NUM 16
362*6236dae4SAndroid Build Coastguard Worker struct iovec msg_iov[MMSG_NUM];
363*6236dae4SAndroid Build Coastguard Worker struct mmsghdr mmsg[MMSG_NUM];
364*6236dae4SAndroid Build Coastguard Worker uint8_t msg_ctrl[MMSG_NUM * CMSG_SPACE(sizeof(int))];
365*6236dae4SAndroid Build Coastguard Worker struct sockaddr_storage remote_addr[MMSG_NUM];
366*6236dae4SAndroid Build Coastguard Worker size_t total_nread = 0, pkts;
367*6236dae4SAndroid Build Coastguard Worker int mcount, i, n;
368*6236dae4SAndroid Build Coastguard Worker char errstr[STRERROR_LEN];
369*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
370*6236dae4SAndroid Build Coastguard Worker size_t gso_size;
371*6236dae4SAndroid Build Coastguard Worker size_t pktlen;
372*6236dae4SAndroid Build Coastguard Worker size_t offset, to;
373*6236dae4SAndroid Build Coastguard Worker char *sockbuf = NULL;
374*6236dae4SAndroid Build Coastguard Worker uint8_t (*bufs)[64*1024] = NULL;
375*6236dae4SAndroid Build Coastguard Worker
376*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(max_pkts > 0);
377*6236dae4SAndroid Build Coastguard Worker result = Curl_multi_xfer_sockbuf_borrow(data, MMSG_NUM * sizeof(bufs[0]),
378*6236dae4SAndroid Build Coastguard Worker &sockbuf);
379*6236dae4SAndroid Build Coastguard Worker if(result)
380*6236dae4SAndroid Build Coastguard Worker goto out;
381*6236dae4SAndroid Build Coastguard Worker bufs = (uint8_t (*)[64*1024])sockbuf;
382*6236dae4SAndroid Build Coastguard Worker
383*6236dae4SAndroid Build Coastguard Worker pkts = 0;
384*6236dae4SAndroid Build Coastguard Worker total_nread = 0;
385*6236dae4SAndroid Build Coastguard Worker while(pkts < max_pkts) {
386*6236dae4SAndroid Build Coastguard Worker n = (int)CURLMIN(MMSG_NUM, max_pkts);
387*6236dae4SAndroid Build Coastguard Worker memset(&mmsg, 0, sizeof(mmsg));
388*6236dae4SAndroid Build Coastguard Worker for(i = 0; i < n; ++i) {
389*6236dae4SAndroid Build Coastguard Worker msg_iov[i].iov_base = bufs[i];
390*6236dae4SAndroid Build Coastguard Worker msg_iov[i].iov_len = (int)sizeof(bufs[i]);
391*6236dae4SAndroid Build Coastguard Worker mmsg[i].msg_hdr.msg_iov = &msg_iov[i];
392*6236dae4SAndroid Build Coastguard Worker mmsg[i].msg_hdr.msg_iovlen = 1;
393*6236dae4SAndroid Build Coastguard Worker mmsg[i].msg_hdr.msg_name = &remote_addr[i];
394*6236dae4SAndroid Build Coastguard Worker mmsg[i].msg_hdr.msg_namelen = sizeof(remote_addr[i]);
395*6236dae4SAndroid Build Coastguard Worker mmsg[i].msg_hdr.msg_control = &msg_ctrl[i * CMSG_SPACE(sizeof(int))];
396*6236dae4SAndroid Build Coastguard Worker mmsg[i].msg_hdr.msg_controllen = CMSG_SPACE(sizeof(int));
397*6236dae4SAndroid Build Coastguard Worker }
398*6236dae4SAndroid Build Coastguard Worker
399*6236dae4SAndroid Build Coastguard Worker while((mcount = recvmmsg(qctx->sockfd, mmsg, n, 0, NULL)) == -1 &&
400*6236dae4SAndroid Build Coastguard Worker SOCKERRNO == EINTR)
401*6236dae4SAndroid Build Coastguard Worker ;
402*6236dae4SAndroid Build Coastguard Worker if(mcount == -1) {
403*6236dae4SAndroid Build Coastguard Worker if(SOCKERRNO == EAGAIN || SOCKERRNO == EWOULDBLOCK) {
404*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "ingress, recvmmsg -> EAGAIN");
405*6236dae4SAndroid Build Coastguard Worker goto out;
406*6236dae4SAndroid Build Coastguard Worker }
407*6236dae4SAndroid Build Coastguard Worker if(!cf->connected && SOCKERRNO == ECONNREFUSED) {
408*6236dae4SAndroid Build Coastguard Worker struct ip_quadruple ip;
409*6236dae4SAndroid Build Coastguard Worker Curl_cf_socket_peek(cf->next, data, NULL, NULL, &ip);
410*6236dae4SAndroid Build Coastguard Worker failf(data, "QUIC: connection to %s port %u refused",
411*6236dae4SAndroid Build Coastguard Worker ip.remote_ip, ip.remote_port);
412*6236dae4SAndroid Build Coastguard Worker result = CURLE_COULDNT_CONNECT;
413*6236dae4SAndroid Build Coastguard Worker goto out;
414*6236dae4SAndroid Build Coastguard Worker }
415*6236dae4SAndroid Build Coastguard Worker Curl_strerror(SOCKERRNO, errstr, sizeof(errstr));
416*6236dae4SAndroid Build Coastguard Worker failf(data, "QUIC: recvmsg() unexpectedly returned %d (errno=%d; %s)",
417*6236dae4SAndroid Build Coastguard Worker mcount, SOCKERRNO, errstr);
418*6236dae4SAndroid Build Coastguard Worker result = CURLE_RECV_ERROR;
419*6236dae4SAndroid Build Coastguard Worker goto out;
420*6236dae4SAndroid Build Coastguard Worker }
421*6236dae4SAndroid Build Coastguard Worker
422*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "recvmmsg() -> %d packets", mcount);
423*6236dae4SAndroid Build Coastguard Worker for(i = 0; i < mcount; ++i) {
424*6236dae4SAndroid Build Coastguard Worker total_nread += mmsg[i].msg_len;
425*6236dae4SAndroid Build Coastguard Worker
426*6236dae4SAndroid Build Coastguard Worker gso_size = vquic_msghdr_get_udp_gro(&mmsg[i].msg_hdr);
427*6236dae4SAndroid Build Coastguard Worker if(gso_size == 0) {
428*6236dae4SAndroid Build Coastguard Worker gso_size = mmsg[i].msg_len;
429*6236dae4SAndroid Build Coastguard Worker }
430*6236dae4SAndroid Build Coastguard Worker
431*6236dae4SAndroid Build Coastguard Worker for(offset = 0; offset < mmsg[i].msg_len; offset = to) {
432*6236dae4SAndroid Build Coastguard Worker ++pkts;
433*6236dae4SAndroid Build Coastguard Worker
434*6236dae4SAndroid Build Coastguard Worker to = offset + gso_size;
435*6236dae4SAndroid Build Coastguard Worker if(to > mmsg[i].msg_len) {
436*6236dae4SAndroid Build Coastguard Worker pktlen = mmsg[i].msg_len - offset;
437*6236dae4SAndroid Build Coastguard Worker }
438*6236dae4SAndroid Build Coastguard Worker else {
439*6236dae4SAndroid Build Coastguard Worker pktlen = gso_size;
440*6236dae4SAndroid Build Coastguard Worker }
441*6236dae4SAndroid Build Coastguard Worker
442*6236dae4SAndroid Build Coastguard Worker result = recv_cb(bufs[i] + offset, pktlen, mmsg[i].msg_hdr.msg_name,
443*6236dae4SAndroid Build Coastguard Worker mmsg[i].msg_hdr.msg_namelen, 0, userp);
444*6236dae4SAndroid Build Coastguard Worker if(result)
445*6236dae4SAndroid Build Coastguard Worker goto out;
446*6236dae4SAndroid Build Coastguard Worker }
447*6236dae4SAndroid Build Coastguard Worker }
448*6236dae4SAndroid Build Coastguard Worker }
449*6236dae4SAndroid Build Coastguard Worker
450*6236dae4SAndroid Build Coastguard Worker out:
451*6236dae4SAndroid Build Coastguard Worker if(total_nread || result)
452*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "recvd %zu packets with %zu bytes -> %d",
453*6236dae4SAndroid Build Coastguard Worker pkts, total_nread, result);
454*6236dae4SAndroid Build Coastguard Worker Curl_multi_xfer_sockbuf_release(data, sockbuf);
455*6236dae4SAndroid Build Coastguard Worker return result;
456*6236dae4SAndroid Build Coastguard Worker }
457*6236dae4SAndroid Build Coastguard Worker
458*6236dae4SAndroid Build Coastguard Worker #elif defined(HAVE_SENDMSG)
recvmsg_packets(struct Curl_cfilter * cf,struct Curl_easy * data,struct cf_quic_ctx * qctx,size_t max_pkts,vquic_recv_pkt_cb * recv_cb,void * userp)459*6236dae4SAndroid Build Coastguard Worker static CURLcode recvmsg_packets(struct Curl_cfilter *cf,
460*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
461*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx,
462*6236dae4SAndroid Build Coastguard Worker size_t max_pkts,
463*6236dae4SAndroid Build Coastguard Worker vquic_recv_pkt_cb *recv_cb, void *userp)
464*6236dae4SAndroid Build Coastguard Worker {
465*6236dae4SAndroid Build Coastguard Worker struct iovec msg_iov;
466*6236dae4SAndroid Build Coastguard Worker struct msghdr msg;
467*6236dae4SAndroid Build Coastguard Worker uint8_t buf[64*1024];
468*6236dae4SAndroid Build Coastguard Worker struct sockaddr_storage remote_addr;
469*6236dae4SAndroid Build Coastguard Worker size_t total_nread, pkts;
470*6236dae4SAndroid Build Coastguard Worker ssize_t nread;
471*6236dae4SAndroid Build Coastguard Worker char errstr[STRERROR_LEN];
472*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
473*6236dae4SAndroid Build Coastguard Worker uint8_t msg_ctrl[CMSG_SPACE(sizeof(int))];
474*6236dae4SAndroid Build Coastguard Worker size_t gso_size;
475*6236dae4SAndroid Build Coastguard Worker size_t pktlen;
476*6236dae4SAndroid Build Coastguard Worker size_t offset, to;
477*6236dae4SAndroid Build Coastguard Worker
478*6236dae4SAndroid Build Coastguard Worker msg_iov.iov_base = buf;
479*6236dae4SAndroid Build Coastguard Worker msg_iov.iov_len = (int)sizeof(buf);
480*6236dae4SAndroid Build Coastguard Worker
481*6236dae4SAndroid Build Coastguard Worker memset(&msg, 0, sizeof(msg));
482*6236dae4SAndroid Build Coastguard Worker msg.msg_iov = &msg_iov;
483*6236dae4SAndroid Build Coastguard Worker msg.msg_iovlen = 1;
484*6236dae4SAndroid Build Coastguard Worker msg.msg_control = msg_ctrl;
485*6236dae4SAndroid Build Coastguard Worker
486*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(max_pkts > 0);
487*6236dae4SAndroid Build Coastguard Worker for(pkts = 0, total_nread = 0; pkts < max_pkts;) {
488*6236dae4SAndroid Build Coastguard Worker msg.msg_name = &remote_addr;
489*6236dae4SAndroid Build Coastguard Worker msg.msg_namelen = sizeof(remote_addr);
490*6236dae4SAndroid Build Coastguard Worker msg.msg_controllen = sizeof(msg_ctrl);
491*6236dae4SAndroid Build Coastguard Worker while((nread = recvmsg(qctx->sockfd, &msg, 0)) == -1 &&
492*6236dae4SAndroid Build Coastguard Worker SOCKERRNO == EINTR)
493*6236dae4SAndroid Build Coastguard Worker ;
494*6236dae4SAndroid Build Coastguard Worker if(nread == -1) {
495*6236dae4SAndroid Build Coastguard Worker if(SOCKERRNO == EAGAIN || SOCKERRNO == EWOULDBLOCK) {
496*6236dae4SAndroid Build Coastguard Worker goto out;
497*6236dae4SAndroid Build Coastguard Worker }
498*6236dae4SAndroid Build Coastguard Worker if(!cf->connected && SOCKERRNO == ECONNREFUSED) {
499*6236dae4SAndroid Build Coastguard Worker struct ip_quadruple ip;
500*6236dae4SAndroid Build Coastguard Worker Curl_cf_socket_peek(cf->next, data, NULL, NULL, &ip);
501*6236dae4SAndroid Build Coastguard Worker failf(data, "QUIC: connection to %s port %u refused",
502*6236dae4SAndroid Build Coastguard Worker ip.remote_ip, ip.remote_port);
503*6236dae4SAndroid Build Coastguard Worker result = CURLE_COULDNT_CONNECT;
504*6236dae4SAndroid Build Coastguard Worker goto out;
505*6236dae4SAndroid Build Coastguard Worker }
506*6236dae4SAndroid Build Coastguard Worker Curl_strerror(SOCKERRNO, errstr, sizeof(errstr));
507*6236dae4SAndroid Build Coastguard Worker failf(data, "QUIC: recvmsg() unexpectedly returned %zd (errno=%d; %s)",
508*6236dae4SAndroid Build Coastguard Worker nread, SOCKERRNO, errstr);
509*6236dae4SAndroid Build Coastguard Worker result = CURLE_RECV_ERROR;
510*6236dae4SAndroid Build Coastguard Worker goto out;
511*6236dae4SAndroid Build Coastguard Worker }
512*6236dae4SAndroid Build Coastguard Worker
513*6236dae4SAndroid Build Coastguard Worker total_nread += (size_t)nread;
514*6236dae4SAndroid Build Coastguard Worker
515*6236dae4SAndroid Build Coastguard Worker gso_size = vquic_msghdr_get_udp_gro(&msg);
516*6236dae4SAndroid Build Coastguard Worker if(gso_size == 0) {
517*6236dae4SAndroid Build Coastguard Worker gso_size = (size_t)nread;
518*6236dae4SAndroid Build Coastguard Worker }
519*6236dae4SAndroid Build Coastguard Worker
520*6236dae4SAndroid Build Coastguard Worker for(offset = 0; offset < (size_t)nread; offset = to) {
521*6236dae4SAndroid Build Coastguard Worker ++pkts;
522*6236dae4SAndroid Build Coastguard Worker
523*6236dae4SAndroid Build Coastguard Worker to = offset + gso_size;
524*6236dae4SAndroid Build Coastguard Worker if(to > (size_t)nread) {
525*6236dae4SAndroid Build Coastguard Worker pktlen = (size_t)nread - offset;
526*6236dae4SAndroid Build Coastguard Worker }
527*6236dae4SAndroid Build Coastguard Worker else {
528*6236dae4SAndroid Build Coastguard Worker pktlen = gso_size;
529*6236dae4SAndroid Build Coastguard Worker }
530*6236dae4SAndroid Build Coastguard Worker
531*6236dae4SAndroid Build Coastguard Worker result =
532*6236dae4SAndroid Build Coastguard Worker recv_cb(buf + offset, pktlen, msg.msg_name, msg.msg_namelen, 0, userp);
533*6236dae4SAndroid Build Coastguard Worker if(result)
534*6236dae4SAndroid Build Coastguard Worker goto out;
535*6236dae4SAndroid Build Coastguard Worker }
536*6236dae4SAndroid Build Coastguard Worker }
537*6236dae4SAndroid Build Coastguard Worker
538*6236dae4SAndroid Build Coastguard Worker out:
539*6236dae4SAndroid Build Coastguard Worker if(total_nread || result)
540*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "recvd %zu packets with %zu bytes -> %d",
541*6236dae4SAndroid Build Coastguard Worker pkts, total_nread, result);
542*6236dae4SAndroid Build Coastguard Worker return result;
543*6236dae4SAndroid Build Coastguard Worker }
544*6236dae4SAndroid Build Coastguard Worker
545*6236dae4SAndroid Build Coastguard Worker #else /* HAVE_SENDMMSG || HAVE_SENDMSG */
recvfrom_packets(struct Curl_cfilter * cf,struct Curl_easy * data,struct cf_quic_ctx * qctx,size_t max_pkts,vquic_recv_pkt_cb * recv_cb,void * userp)546*6236dae4SAndroid Build Coastguard Worker static CURLcode recvfrom_packets(struct Curl_cfilter *cf,
547*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
548*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx,
549*6236dae4SAndroid Build Coastguard Worker size_t max_pkts,
550*6236dae4SAndroid Build Coastguard Worker vquic_recv_pkt_cb *recv_cb, void *userp)
551*6236dae4SAndroid Build Coastguard Worker {
552*6236dae4SAndroid Build Coastguard Worker uint8_t buf[64*1024];
553*6236dae4SAndroid Build Coastguard Worker int bufsize = (int)sizeof(buf);
554*6236dae4SAndroid Build Coastguard Worker struct sockaddr_storage remote_addr;
555*6236dae4SAndroid Build Coastguard Worker socklen_t remote_addrlen = sizeof(remote_addr);
556*6236dae4SAndroid Build Coastguard Worker size_t total_nread, pkts;
557*6236dae4SAndroid Build Coastguard Worker ssize_t nread;
558*6236dae4SAndroid Build Coastguard Worker char errstr[STRERROR_LEN];
559*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
560*6236dae4SAndroid Build Coastguard Worker
561*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(max_pkts > 0);
562*6236dae4SAndroid Build Coastguard Worker for(pkts = 0, total_nread = 0; pkts < max_pkts;) {
563*6236dae4SAndroid Build Coastguard Worker while((nread = recvfrom(qctx->sockfd, (char *)buf, bufsize, 0,
564*6236dae4SAndroid Build Coastguard Worker (struct sockaddr *)&remote_addr,
565*6236dae4SAndroid Build Coastguard Worker &remote_addrlen)) == -1 &&
566*6236dae4SAndroid Build Coastguard Worker SOCKERRNO == EINTR)
567*6236dae4SAndroid Build Coastguard Worker ;
568*6236dae4SAndroid Build Coastguard Worker if(nread == -1) {
569*6236dae4SAndroid Build Coastguard Worker if(SOCKERRNO == EAGAIN || SOCKERRNO == EWOULDBLOCK) {
570*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "ingress, recvfrom -> EAGAIN");
571*6236dae4SAndroid Build Coastguard Worker goto out;
572*6236dae4SAndroid Build Coastguard Worker }
573*6236dae4SAndroid Build Coastguard Worker if(!cf->connected && SOCKERRNO == ECONNREFUSED) {
574*6236dae4SAndroid Build Coastguard Worker struct ip_quadruple ip;
575*6236dae4SAndroid Build Coastguard Worker Curl_cf_socket_peek(cf->next, data, NULL, NULL, &ip);
576*6236dae4SAndroid Build Coastguard Worker failf(data, "QUIC: connection to %s port %u refused",
577*6236dae4SAndroid Build Coastguard Worker ip.remote_ip, ip.remote_port);
578*6236dae4SAndroid Build Coastguard Worker result = CURLE_COULDNT_CONNECT;
579*6236dae4SAndroid Build Coastguard Worker goto out;
580*6236dae4SAndroid Build Coastguard Worker }
581*6236dae4SAndroid Build Coastguard Worker Curl_strerror(SOCKERRNO, errstr, sizeof(errstr));
582*6236dae4SAndroid Build Coastguard Worker failf(data, "QUIC: recvfrom() unexpectedly returned %zd (errno=%d; %s)",
583*6236dae4SAndroid Build Coastguard Worker nread, SOCKERRNO, errstr);
584*6236dae4SAndroid Build Coastguard Worker result = CURLE_RECV_ERROR;
585*6236dae4SAndroid Build Coastguard Worker goto out;
586*6236dae4SAndroid Build Coastguard Worker }
587*6236dae4SAndroid Build Coastguard Worker
588*6236dae4SAndroid Build Coastguard Worker ++pkts;
589*6236dae4SAndroid Build Coastguard Worker total_nread += (size_t)nread;
590*6236dae4SAndroid Build Coastguard Worker result = recv_cb(buf, (size_t)nread, &remote_addr, remote_addrlen,
591*6236dae4SAndroid Build Coastguard Worker 0, userp);
592*6236dae4SAndroid Build Coastguard Worker if(result)
593*6236dae4SAndroid Build Coastguard Worker goto out;
594*6236dae4SAndroid Build Coastguard Worker }
595*6236dae4SAndroid Build Coastguard Worker
596*6236dae4SAndroid Build Coastguard Worker out:
597*6236dae4SAndroid Build Coastguard Worker if(total_nread || result)
598*6236dae4SAndroid Build Coastguard Worker CURL_TRC_CF(data, cf, "recvd %zu packets with %zu bytes -> %d",
599*6236dae4SAndroid Build Coastguard Worker pkts, total_nread, result);
600*6236dae4SAndroid Build Coastguard Worker return result;
601*6236dae4SAndroid Build Coastguard Worker }
602*6236dae4SAndroid Build Coastguard Worker #endif /* !HAVE_SENDMMSG && !HAVE_SENDMSG */
603*6236dae4SAndroid Build Coastguard Worker
vquic_recv_packets(struct Curl_cfilter * cf,struct Curl_easy * data,struct cf_quic_ctx * qctx,size_t max_pkts,vquic_recv_pkt_cb * recv_cb,void * userp)604*6236dae4SAndroid Build Coastguard Worker CURLcode vquic_recv_packets(struct Curl_cfilter *cf,
605*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
606*6236dae4SAndroid Build Coastguard Worker struct cf_quic_ctx *qctx,
607*6236dae4SAndroid Build Coastguard Worker size_t max_pkts,
608*6236dae4SAndroid Build Coastguard Worker vquic_recv_pkt_cb *recv_cb, void *userp)
609*6236dae4SAndroid Build Coastguard Worker {
610*6236dae4SAndroid Build Coastguard Worker CURLcode result;
611*6236dae4SAndroid Build Coastguard Worker #if defined(HAVE_SENDMMSG)
612*6236dae4SAndroid Build Coastguard Worker result = recvmmsg_packets(cf, data, qctx, max_pkts, recv_cb, userp);
613*6236dae4SAndroid Build Coastguard Worker #elif defined(HAVE_SENDMSG)
614*6236dae4SAndroid Build Coastguard Worker result = recvmsg_packets(cf, data, qctx, max_pkts, recv_cb, userp);
615*6236dae4SAndroid Build Coastguard Worker #else
616*6236dae4SAndroid Build Coastguard Worker result = recvfrom_packets(cf, data, qctx, max_pkts, recv_cb, userp);
617*6236dae4SAndroid Build Coastguard Worker #endif
618*6236dae4SAndroid Build Coastguard Worker if(!result) {
619*6236dae4SAndroid Build Coastguard Worker if(!qctx->got_first_byte) {
620*6236dae4SAndroid Build Coastguard Worker qctx->got_first_byte = TRUE;
621*6236dae4SAndroid Build Coastguard Worker qctx->first_byte_at = qctx->last_op;
622*6236dae4SAndroid Build Coastguard Worker }
623*6236dae4SAndroid Build Coastguard Worker qctx->last_io = qctx->last_op;
624*6236dae4SAndroid Build Coastguard Worker }
625*6236dae4SAndroid Build Coastguard Worker return result;
626*6236dae4SAndroid Build Coastguard Worker }
627*6236dae4SAndroid Build Coastguard Worker
628*6236dae4SAndroid Build Coastguard Worker /*
629*6236dae4SAndroid Build Coastguard Worker * If the QLOGDIR environment variable is set, open and return a file
630*6236dae4SAndroid Build Coastguard Worker * descriptor to write the log to.
631*6236dae4SAndroid Build Coastguard Worker *
632*6236dae4SAndroid Build Coastguard Worker * This function returns error if something failed outside of failing to
633*6236dae4SAndroid Build Coastguard Worker * create the file. Open file success is deemed by seeing if the returned fd
634*6236dae4SAndroid Build Coastguard Worker * is != -1.
635*6236dae4SAndroid Build Coastguard Worker */
Curl_qlogdir(struct Curl_easy * data,unsigned char * scid,size_t scidlen,int * qlogfdp)636*6236dae4SAndroid Build Coastguard Worker CURLcode Curl_qlogdir(struct Curl_easy *data,
637*6236dae4SAndroid Build Coastguard Worker unsigned char *scid,
638*6236dae4SAndroid Build Coastguard Worker size_t scidlen,
639*6236dae4SAndroid Build Coastguard Worker int *qlogfdp)
640*6236dae4SAndroid Build Coastguard Worker {
641*6236dae4SAndroid Build Coastguard Worker const char *qlog_dir = getenv("QLOGDIR");
642*6236dae4SAndroid Build Coastguard Worker *qlogfdp = -1;
643*6236dae4SAndroid Build Coastguard Worker if(qlog_dir) {
644*6236dae4SAndroid Build Coastguard Worker struct dynbuf fname;
645*6236dae4SAndroid Build Coastguard Worker CURLcode result;
646*6236dae4SAndroid Build Coastguard Worker unsigned int i;
647*6236dae4SAndroid Build Coastguard Worker Curl_dyn_init(&fname, DYN_QLOG_NAME);
648*6236dae4SAndroid Build Coastguard Worker result = Curl_dyn_add(&fname, qlog_dir);
649*6236dae4SAndroid Build Coastguard Worker if(!result)
650*6236dae4SAndroid Build Coastguard Worker result = Curl_dyn_add(&fname, "/");
651*6236dae4SAndroid Build Coastguard Worker for(i = 0; (i < scidlen) && !result; i++) {
652*6236dae4SAndroid Build Coastguard Worker char hex[3];
653*6236dae4SAndroid Build Coastguard Worker msnprintf(hex, 3, "%02x", scid[i]);
654*6236dae4SAndroid Build Coastguard Worker result = Curl_dyn_add(&fname, hex);
655*6236dae4SAndroid Build Coastguard Worker }
656*6236dae4SAndroid Build Coastguard Worker if(!result)
657*6236dae4SAndroid Build Coastguard Worker result = Curl_dyn_add(&fname, ".sqlog");
658*6236dae4SAndroid Build Coastguard Worker
659*6236dae4SAndroid Build Coastguard Worker if(!result) {
660*6236dae4SAndroid Build Coastguard Worker int qlogfd = open(Curl_dyn_ptr(&fname), QLOGMODE,
661*6236dae4SAndroid Build Coastguard Worker data->set.new_file_perms);
662*6236dae4SAndroid Build Coastguard Worker if(qlogfd != -1)
663*6236dae4SAndroid Build Coastguard Worker *qlogfdp = qlogfd;
664*6236dae4SAndroid Build Coastguard Worker }
665*6236dae4SAndroid Build Coastguard Worker Curl_dyn_free(&fname);
666*6236dae4SAndroid Build Coastguard Worker if(result)
667*6236dae4SAndroid Build Coastguard Worker return result;
668*6236dae4SAndroid Build Coastguard Worker }
669*6236dae4SAndroid Build Coastguard Worker
670*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
671*6236dae4SAndroid Build Coastguard Worker }
672*6236dae4SAndroid Build Coastguard Worker
Curl_cf_quic_create(struct Curl_cfilter ** pcf,struct Curl_easy * data,struct connectdata * conn,const struct Curl_addrinfo * ai,int transport)673*6236dae4SAndroid Build Coastguard Worker CURLcode Curl_cf_quic_create(struct Curl_cfilter **pcf,
674*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data,
675*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn,
676*6236dae4SAndroid Build Coastguard Worker const struct Curl_addrinfo *ai,
677*6236dae4SAndroid Build Coastguard Worker int transport)
678*6236dae4SAndroid Build Coastguard Worker {
679*6236dae4SAndroid Build Coastguard Worker (void)transport;
680*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(transport == TRNSPRT_QUIC);
681*6236dae4SAndroid Build Coastguard Worker #if defined(USE_NGTCP2) && defined(USE_NGHTTP3)
682*6236dae4SAndroid Build Coastguard Worker return Curl_cf_ngtcp2_create(pcf, data, conn, ai);
683*6236dae4SAndroid Build Coastguard Worker #elif defined(USE_OPENSSL_QUIC) && defined(USE_NGHTTP3)
684*6236dae4SAndroid Build Coastguard Worker return Curl_cf_osslq_create(pcf, data, conn, ai);
685*6236dae4SAndroid Build Coastguard Worker #elif defined(USE_QUICHE)
686*6236dae4SAndroid Build Coastguard Worker return Curl_cf_quiche_create(pcf, data, conn, ai);
687*6236dae4SAndroid Build Coastguard Worker #elif defined(USE_MSH3)
688*6236dae4SAndroid Build Coastguard Worker return Curl_cf_msh3_create(pcf, data, conn, ai);
689*6236dae4SAndroid Build Coastguard Worker #else
690*6236dae4SAndroid Build Coastguard Worker *pcf = NULL;
691*6236dae4SAndroid Build Coastguard Worker (void)data;
692*6236dae4SAndroid Build Coastguard Worker (void)conn;
693*6236dae4SAndroid Build Coastguard Worker (void)ai;
694*6236dae4SAndroid Build Coastguard Worker return CURLE_NOT_BUILT_IN;
695*6236dae4SAndroid Build Coastguard Worker #endif
696*6236dae4SAndroid Build Coastguard Worker }
697*6236dae4SAndroid Build Coastguard Worker
Curl_conn_is_http3(const struct Curl_easy * data,const struct connectdata * conn,int sockindex)698*6236dae4SAndroid Build Coastguard Worker bool Curl_conn_is_http3(const struct Curl_easy *data,
699*6236dae4SAndroid Build Coastguard Worker const struct connectdata *conn,
700*6236dae4SAndroid Build Coastguard Worker int sockindex)
701*6236dae4SAndroid Build Coastguard Worker {
702*6236dae4SAndroid Build Coastguard Worker #if defined(USE_NGTCP2) && defined(USE_NGHTTP3)
703*6236dae4SAndroid Build Coastguard Worker return Curl_conn_is_ngtcp2(data, conn, sockindex);
704*6236dae4SAndroid Build Coastguard Worker #elif defined(USE_OPENSSL_QUIC) && defined(USE_NGHTTP3)
705*6236dae4SAndroid Build Coastguard Worker return Curl_conn_is_osslq(data, conn, sockindex);
706*6236dae4SAndroid Build Coastguard Worker #elif defined(USE_QUICHE)
707*6236dae4SAndroid Build Coastguard Worker return Curl_conn_is_quiche(data, conn, sockindex);
708*6236dae4SAndroid Build Coastguard Worker #elif defined(USE_MSH3)
709*6236dae4SAndroid Build Coastguard Worker return Curl_conn_is_msh3(data, conn, sockindex);
710*6236dae4SAndroid Build Coastguard Worker #else
711*6236dae4SAndroid Build Coastguard Worker return ((conn->handler->protocol & PROTO_FAMILY_HTTP) &&
712*6236dae4SAndroid Build Coastguard Worker (conn->httpversion == 30));
713*6236dae4SAndroid Build Coastguard Worker #endif
714*6236dae4SAndroid Build Coastguard Worker }
715*6236dae4SAndroid Build Coastguard Worker
Curl_conn_may_http3(struct Curl_easy * data,const struct connectdata * conn)716*6236dae4SAndroid Build Coastguard Worker CURLcode Curl_conn_may_http3(struct Curl_easy *data,
717*6236dae4SAndroid Build Coastguard Worker const struct connectdata *conn)
718*6236dae4SAndroid Build Coastguard Worker {
719*6236dae4SAndroid Build Coastguard Worker if(conn->transport == TRNSPRT_UNIX) {
720*6236dae4SAndroid Build Coastguard Worker /* cannot do QUIC over a Unix domain socket */
721*6236dae4SAndroid Build Coastguard Worker return CURLE_QUIC_CONNECT_ERROR;
722*6236dae4SAndroid Build Coastguard Worker }
723*6236dae4SAndroid Build Coastguard Worker if(!(conn->handler->flags & PROTOPT_SSL)) {
724*6236dae4SAndroid Build Coastguard Worker failf(data, "HTTP/3 requested for non-HTTPS URL");
725*6236dae4SAndroid Build Coastguard Worker return CURLE_URL_MALFORMAT;
726*6236dae4SAndroid Build Coastguard Worker }
727*6236dae4SAndroid Build Coastguard Worker #ifndef CURL_DISABLE_PROXY
728*6236dae4SAndroid Build Coastguard Worker if(conn->bits.socksproxy) {
729*6236dae4SAndroid Build Coastguard Worker failf(data, "HTTP/3 is not supported over a SOCKS proxy");
730*6236dae4SAndroid Build Coastguard Worker return CURLE_URL_MALFORMAT;
731*6236dae4SAndroid Build Coastguard Worker }
732*6236dae4SAndroid Build Coastguard Worker if(conn->bits.httpproxy && conn->bits.tunnel_proxy) {
733*6236dae4SAndroid Build Coastguard Worker failf(data, "HTTP/3 is not supported over an HTTP proxy");
734*6236dae4SAndroid Build Coastguard Worker return CURLE_URL_MALFORMAT;
735*6236dae4SAndroid Build Coastguard Worker }
736*6236dae4SAndroid Build Coastguard Worker #endif
737*6236dae4SAndroid Build Coastguard Worker
738*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
739*6236dae4SAndroid Build Coastguard Worker }
740*6236dae4SAndroid Build Coastguard Worker
741*6236dae4SAndroid Build Coastguard Worker #else /* USE_HTTP3 */
742*6236dae4SAndroid Build Coastguard Worker
Curl_conn_may_http3(struct Curl_easy * data,const struct connectdata * conn)743*6236dae4SAndroid Build Coastguard Worker CURLcode Curl_conn_may_http3(struct Curl_easy *data,
744*6236dae4SAndroid Build Coastguard Worker const struct connectdata *conn)
745*6236dae4SAndroid Build Coastguard Worker {
746*6236dae4SAndroid Build Coastguard Worker (void)conn;
747*6236dae4SAndroid Build Coastguard Worker (void)data;
748*6236dae4SAndroid Build Coastguard Worker DEBUGF(infof(data, "QUIC is not supported in this build"));
749*6236dae4SAndroid Build Coastguard Worker return CURLE_NOT_BUILT_IN;
750*6236dae4SAndroid Build Coastguard Worker }
751*6236dae4SAndroid Build Coastguard Worker
752*6236dae4SAndroid Build Coastguard Worker #endif /* !USE_HTTP3 */
753