tokio_quiche/quic/connection/
map.rs1use super::Incoming;
28use super::InitialQuicConnection;
29use crate::metrics::Metrics;
30
31use datagram_socket::DatagramSocketSend;
32use quiche::ConnectionId;
33use quiche::MAX_CONN_ID_LEN;
34use std::collections::BTreeMap;
35use std::collections::HashMap;
36use tokio::sync::mpsc;
37
38const U64_SZ: usize = std::mem::size_of::<u64>();
39const MAX_CONN_ID_QUADS: usize = MAX_CONN_ID_LEN.div_ceil(U64_SZ);
40const CONN_ID_USABLE_LEN: usize = min_usize(
41 MAX_CONN_ID_QUADS * U64_SZ - 1,
43 min_usize(MAX_CONN_ID_LEN, u8::MAX as _),
45);
46
47const fn min_usize(v1: usize, v2: usize) -> usize {
48 if v1 < v2 {
49 v1
50 } else {
51 v2
52 }
53}
54
55#[derive(PartialEq, Eq, PartialOrd, Ord)]
58enum CidOwned {
59 Generic(Box<[u8]>),
62 Optimized([u64; MAX_CONN_ID_QUADS]),
67}
68
69impl From<&ConnectionId<'_>> for CidOwned {
70 #[inline(always)]
71 fn from(value: &ConnectionId<'_>) -> Self {
72 if value.len() > CONN_ID_USABLE_LEN {
73 return CidOwned::Generic(value.as_ref().into());
74 }
75
76 let mut cid = [0; MAX_CONN_ID_QUADS];
77
78 value
79 .chunks(U64_SZ)
80 .map(|c| match c.try_into() {
81 Ok(v) => u64::from_le_bytes(v),
82 Err(_) => {
83 let mut remainder = [0u8; U64_SZ];
84 remainder[..c.len()].copy_from_slice(c);
85 u64::from_le_bytes(remainder)
86 },
87 })
88 .enumerate()
89 .for_each(|(i, v)| cid[i] = v);
90
91 *cid.last_mut().unwrap() |= (value.len() as u64) << 56;
94
95 CidOwned::Optimized(cid)
96 }
97}
98
99type QuicheId = u64;
101
102#[derive(Default)]
108pub(crate) struct ConnectionMap {
109 quic_id_map: BTreeMap<CidOwned, (QuicheId, mpsc::Sender<Incoming>)>,
110 conn_map: HashMap<QuicheId, mpsc::Sender<Incoming>>,
111}
112
113impl ConnectionMap {
114 pub(crate) fn insert<Tx, M>(
115 &mut self, cid: ConnectionId<'_>, conn: &InitialQuicConnection<Tx, M>,
116 ) where
117 Tx: DatagramSocketSend + Send + 'static,
118 M: Metrics,
119 {
120 let id = conn.id;
121 let ev_sender = conn.incoming_ev_sender.clone();
122
123 self.conn_map.insert(id, ev_sender.clone());
124 self.quic_id_map.insert((&cid).into(), (id, ev_sender));
125 }
126
127 pub(crate) fn remove(&mut self, cid: &ConnectionId<'_>) {
128 if let Some((id, _)) = self.quic_id_map.remove(&cid.into()) {
129 self.conn_map.remove(&id);
130 }
131 }
132
133 pub(crate) fn map_cid<Tx, M>(
134 &mut self, cid: ConnectionId<'_>, conn: &InitialQuicConnection<Tx, M>,
135 ) where
136 Tx: DatagramSocketSend + Send + 'static,
137 M: Metrics,
138 {
139 let id = conn.id;
140
141 if let Some(ev_sender) = self.conn_map.get(&id) {
142 self.quic_id_map
143 .insert((&cid).into(), (id, ev_sender.clone()));
144 }
145 }
146
147 pub(crate) fn unmap_cid(&mut self, cid: &ConnectionId<'_>) {
148 self.quic_id_map.remove(&cid.into());
149 }
150
151 pub(crate) fn get(
152 &self, id: &ConnectionId,
153 ) -> Option<&mpsc::Sender<Incoming>> {
154 if id.len() == MAX_CONN_ID_LEN {
155 self.quic_id_map.get(&id.into()).map(|(_id, sender)| sender)
159 } else {
160 self.quic_id_map.get(&id.into()).map(|(_id, sender)| sender)
161 }
162 }
163}
164
165#[cfg(test)]
166mod tests {
167 use super::*;
168 use quiche::ConnectionId;
169
170 #[test]
171 fn cid_storage() {
172 let max_v1_cid = ConnectionId::from_ref(&[0xfa; MAX_CONN_ID_LEN]);
173 let optimized = CidOwned::from(&max_v1_cid);
174 assert!(
175 matches!(optimized, CidOwned::Optimized(_)),
176 "QUIC v1 CID is not stored inline"
177 );
178
179 let oversize_cid = ConnectionId::from_ref(&[0x1b; MAX_CONN_ID_LEN + 20]);
180 let boxed = CidOwned::from(&oversize_cid);
181 assert!(
182 matches!(boxed, CidOwned::Generic(_)),
183 "Oversized CID is not boxed"
184 );
185 }
186}