tokio_quiche/socket/
listener.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 std::io;
28#[cfg(unix)]
29use std::os::fd::AsFd;
30#[cfg(unix)]
31use std::os::fd::AsRawFd;
32#[cfg(unix)]
33use std::os::fd::BorrowedFd;
34#[cfg(unix)]
35use std::os::fd::RawFd;
36use tokio::net::UdpSocket;
37
38use super::SocketCapabilities;
39
40/// Wrapper around a [`UdpSocket`] for server-side QUIC connections.
41///
42/// The wrapper carries socket-specific parameters, in contrast to the
43/// [`settings`](crate::settings) structs which apply to _all_ sockets
44/// for a given QUIC server.
45///
46/// To create a [`QuicListener`], you may either instantiate the struct yourself
47/// or use one of the `TryFrom` implementations.
48#[derive(Debug)]
49pub struct QuicListener {
50    /// The wrapped [tokio] socket.
51    pub socket: UdpSocket,
52    /// An opaque value that is later passed to the
53    /// [`ConnectionIdGenerator`](crate::ConnectionIdGenerator).
54    pub socket_cookie: u64,
55    /// The [`SocketCapabilities`] to use for this socket.
56    ///
57    /// By default, [`QuicListener`]s are constructed with all capabilities
58    /// disabled. On Linux, you can use `apply_max_capabilities()` to (try
59    /// to) enable all supported capabilities.
60    pub capabilities: SocketCapabilities,
61}
62
63impl QuicListener {
64    /// Tries to enable all sockopts supported by the crate for this socket.
65    /// See `SocketCapabilities::apply_all_and_get_compatibility` for details.
66    #[cfg(target_os = "linux")]
67    pub fn apply_max_capabilities(&mut self, max_send_udp_payload_size: usize) {
68        let capabilities = SocketCapabilities::apply_all_and_get_compatibility(
69            &self.socket,
70            max_send_udp_payload_size,
71        );
72        self.capabilities = capabilities;
73    }
74}
75
76impl TryFrom<UdpSocket> for QuicListener {
77    type Error = io::Error;
78
79    fn try_from(socket: UdpSocket) -> Result<Self, Self::Error> {
80        Ok(Self {
81            socket,
82            socket_cookie: 0,
83            capabilities: SocketCapabilities::default(),
84        })
85    }
86}
87
88impl TryFrom<std::net::UdpSocket> for QuicListener {
89    type Error = io::Error;
90
91    fn try_from(socket: std::net::UdpSocket) -> Result<Self, Self::Error> {
92        socket.set_nonblocking(true)?;
93        let socket = UdpSocket::from_std(socket)?;
94        Self::try_from(socket)
95    }
96}
97
98#[cfg(unix)]
99impl AsFd for QuicListener {
100    fn as_fd(&self) -> BorrowedFd {
101        self.socket.as_fd()
102    }
103}
104
105#[cfg(unix)]
106impl AsRawFd for QuicListener {
107    fn as_raw_fd(&self) -> RawFd {
108        self.socket.as_raw_fd()
109    }
110}