tokio_quiche/http3/driver/datagram.rs
1// Copyright (C) 2025, Cloudflare, Inc.
2// All rights reserved.
3//
4// Redistribution and use in source and binary forms, with or without
5// modification, are permitted provided that the following conditions are
6// met:
7//
8// * Redistributions of source code must retain the above copyright notice,
9// this list of conditions and the following disclaimer.
10//
11// * Redistributions in binary form must reproduce the above copyright
12// notice, this list of conditions and the following disclaimer in the
13// documentation and/or other materials provided with the distribution.
14//
15// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
16// IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
17// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
18// PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
19// CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
20// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
22// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
23// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
24// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26
27use super::InboundFrame;
28use crate::buf_factory::BufFactory;
29use crate::buf_factory::PooledDgram;
30use crate::quic::QuicheConnection;
31use quiche::h3::NameValue;
32use quiche::h3::{
33 self,
34};
35
36/// Extracts the DATAGRAM flow ID proxied over the given `stream_id`,
37/// or `None` if this is not a proxy request.
38pub(crate) fn extract_flow_id(
39 stream_id: u64, headers: &[h3::Header],
40) -> Option<u64> {
41 let mut method = None;
42 let mut datagram_flow_id: Option<u64> = None;
43 let mut protocol = None;
44
45 for header in headers {
46 match header.name() {
47 b":method" => method = Some(header.value()),
48 b":protocol" => protocol = Some(header.value()),
49 b"datagram-flow-id" =>
50 datagram_flow_id = std::str::from_utf8(header.value())
51 .ok()
52 .and_then(|v| v.parse().ok()),
53 _ => {},
54 };
55
56 // We have all of the information needed to get a flow_id or
57 // quarter_stream_id
58 if method.is_some() && (datagram_flow_id.is_some() || protocol.is_some())
59 {
60 break;
61 }
62 }
63
64 // draft-ietf-masque-connect-udp-03 CONNECT-UDP
65 if method == Some(b"CONNECT-UDP") && datagram_flow_id.is_some() {
66 datagram_flow_id
67 // RFC 9298 CONNECT-UDP
68 } else if method == Some(b"CONNECT") && protocol.is_some() {
69 // we use the quarter_stream_id for RFC 9297
70 // https://www.rfc-editor.org/rfc/rfc9297.html#name-http-3-datagrams
71 Some(stream_id / 4)
72 } else {
73 None
74 }
75}
76
77/// Sends an HTTP/3 datagram over the QUIC connection with the given `flow_id`.
78pub(crate) fn send_h3_dgram(
79 conn: &mut QuicheConnection, flow_id: u64, mut dgram: PooledDgram,
80) -> quiche::Result<()> {
81 let mut prefix = [0u8; 8];
82 let mut buf = octets::OctetsMut::with_slice(&mut prefix);
83 let flow_id = buf.put_varint(flow_id)?;
84
85 if dgram.add_prefix(flow_id) {
86 conn.dgram_send(&dgram)
87 } else {
88 let mut inner = dgram.into_inner().into_vec();
89 inner.splice(..0, flow_id.iter().copied());
90 conn.dgram_send_vec(inner)
91 }
92}
93
94/// Reads the next HTTP/3 datagram from the QUIC connection.
95///
96/// [`quiche::Error::Done`] is returned if there is no datagram to read.
97pub(crate) fn receive_h3_dgram(
98 conn: &mut QuicheConnection,
99) -> quiche::Result<(u64, InboundFrame)> {
100 let dgram = conn.dgram_recv_vec()?;
101 let mut buf = octets::Octets::with_slice(&dgram);
102 let flow_id = buf.get_varint()?;
103 let advance = buf.off();
104 let datagram =
105 InboundFrame::Datagram(BufFactory::dgram_from_slice(&dgram[advance..]));
106
107 Ok((flow_id, datagram))
108}