305 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			305 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| //! UDP sockets.
 | |
| 
 | |
| use core::future::poll_fn;
 | |
| use core::mem;
 | |
| use core::task::{Context, Poll};
 | |
| 
 | |
| use smoltcp::iface::{Interface, SocketHandle};
 | |
| use smoltcp::socket::udp;
 | |
| pub use smoltcp::socket::udp::{PacketMetadata, UdpMetadata};
 | |
| use smoltcp::wire::IpListenEndpoint;
 | |
| 
 | |
| use crate::Stack;
 | |
| 
 | |
| /// Error returned by [`UdpSocket::bind`].
 | |
| #[derive(PartialEq, Eq, Clone, Copy, Debug)]
 | |
| #[cfg_attr(feature = "defmt", derive(defmt::Format))]
 | |
| pub enum BindError {
 | |
|     /// The socket was already open.
 | |
|     InvalidState,
 | |
|     /// No route to host.
 | |
|     NoRoute,
 | |
| }
 | |
| 
 | |
| /// Error returned by [`UdpSocket::recv_from`] and [`UdpSocket::send_to`].
 | |
| #[derive(PartialEq, Eq, Clone, Copy, Debug)]
 | |
| #[cfg_attr(feature = "defmt", derive(defmt::Format))]
 | |
| pub enum SendError {
 | |
|     /// No route to host.
 | |
|     NoRoute,
 | |
|     /// Socket not bound to an outgoing port.
 | |
|     SocketNotBound,
 | |
| }
 | |
| 
 | |
| /// Error returned by [`UdpSocket::recv_from`] and [`UdpSocket::send_to`].
 | |
| #[derive(PartialEq, Eq, Clone, Copy, Debug)]
 | |
| #[cfg_attr(feature = "defmt", derive(defmt::Format))]
 | |
| pub enum RecvError {
 | |
|     /// Provided buffer was smaller than the received packet.
 | |
|     Truncated,
 | |
| }
 | |
| 
 | |
| /// An UDP socket.
 | |
| pub struct UdpSocket<'a> {
 | |
|     stack: Stack<'a>,
 | |
|     handle: SocketHandle,
 | |
| }
 | |
| 
 | |
| impl<'a> UdpSocket<'a> {
 | |
|     /// Create a new UDP socket using the provided stack and buffers.
 | |
|     pub fn new(
 | |
|         stack: Stack<'a>,
 | |
|         rx_meta: &'a mut [PacketMetadata],
 | |
|         rx_buffer: &'a mut [u8],
 | |
|         tx_meta: &'a mut [PacketMetadata],
 | |
|         tx_buffer: &'a mut [u8],
 | |
|     ) -> Self {
 | |
|         let handle = stack.with_mut(|i| {
 | |
|             let rx_meta: &'static mut [PacketMetadata] = unsafe { mem::transmute(rx_meta) };
 | |
|             let rx_buffer: &'static mut [u8] = unsafe { mem::transmute(rx_buffer) };
 | |
|             let tx_meta: &'static mut [PacketMetadata] = unsafe { mem::transmute(tx_meta) };
 | |
|             let tx_buffer: &'static mut [u8] = unsafe { mem::transmute(tx_buffer) };
 | |
|             i.sockets.add(udp::Socket::new(
 | |
|                 udp::PacketBuffer::new(rx_meta, rx_buffer),
 | |
|                 udp::PacketBuffer::new(tx_meta, tx_buffer),
 | |
|             ))
 | |
|         });
 | |
| 
 | |
|         Self { stack, handle }
 | |
|     }
 | |
| 
 | |
|     /// Bind the socket to a local endpoint.
 | |
|     pub fn bind<T>(&mut self, endpoint: T) -> Result<(), BindError>
 | |
|     where
 | |
|         T: Into<IpListenEndpoint>,
 | |
|     {
 | |
|         let mut endpoint = endpoint.into();
 | |
| 
 | |
|         if endpoint.port == 0 {
 | |
|             // If user didn't specify port allocate a dynamic port.
 | |
|             endpoint.port = self.stack.with_mut(|i| i.get_local_port());
 | |
|         }
 | |
| 
 | |
|         match self.with_mut(|s, _| s.bind(endpoint)) {
 | |
|             Ok(()) => Ok(()),
 | |
|             Err(udp::BindError::InvalidState) => Err(BindError::InvalidState),
 | |
|             Err(udp::BindError::Unaddressable) => Err(BindError::NoRoute),
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     fn with<R>(&self, f: impl FnOnce(&udp::Socket, &Interface) -> R) -> R {
 | |
|         self.stack.with(|i| {
 | |
|             let socket = i.sockets.get::<udp::Socket>(self.handle);
 | |
|             f(socket, &i.iface)
 | |
|         })
 | |
|     }
 | |
| 
 | |
|     fn with_mut<R>(&self, f: impl FnOnce(&mut udp::Socket, &mut Interface) -> R) -> R {
 | |
|         self.stack.with_mut(|i| {
 | |
|             let socket = i.sockets.get_mut::<udp::Socket>(self.handle);
 | |
|             let res = f(socket, &mut i.iface);
 | |
|             i.waker.wake();
 | |
|             res
 | |
|         })
 | |
|     }
 | |
| 
 | |
|     /// Receive a datagram.
 | |
|     ///
 | |
|     /// This method will wait until a datagram is received.
 | |
|     ///
 | |
|     /// Returns the number of bytes received and the remote endpoint.
 | |
|     pub async fn recv_from(&self, buf: &mut [u8]) -> Result<(usize, UdpMetadata), RecvError> {
 | |
|         poll_fn(move |cx| self.poll_recv_from(buf, cx)).await
 | |
|     }
 | |
| 
 | |
|     /// Receive a datagram.
 | |
|     ///
 | |
|     /// When no datagram is available, this method will return `Poll::Pending` and
 | |
|     /// register the current task to be notified when a datagram is received.
 | |
|     ///
 | |
|     /// When a datagram is received, this method will return `Poll::Ready` with the
 | |
|     /// number of bytes received and the remote endpoint.
 | |
|     pub fn poll_recv_from(
 | |
|         &self,
 | |
|         buf: &mut [u8],
 | |
|         cx: &mut Context<'_>,
 | |
|     ) -> Poll<Result<(usize, UdpMetadata), RecvError>> {
 | |
|         self.with_mut(|s, _| match s.recv_slice(buf) {
 | |
|             Ok((n, meta)) => Poll::Ready(Ok((n, meta))),
 | |
|             // No data ready
 | |
|             Err(udp::RecvError::Truncated) => Poll::Ready(Err(RecvError::Truncated)),
 | |
|             Err(udp::RecvError::Exhausted) => {
 | |
|                 s.register_recv_waker(cx.waker());
 | |
|                 Poll::Pending
 | |
|             }
 | |
|         })
 | |
|     }
 | |
| 
 | |
|     /// Receive a datagram with a zero-copy function.
 | |
|     ///
 | |
|     /// When no datagram is available, this method will return `Poll::Pending` and
 | |
|     /// register the current task to be notified when a datagram is received.
 | |
|     ///
 | |
|     /// When a datagram is received, this method will call the provided function
 | |
|     /// with the number of bytes received and the remote endpoint and return
 | |
|     /// `Poll::Ready` with the function's returned value.
 | |
|     pub async fn recv_from_with<F, R>(&mut self, f: F) -> R
 | |
|     where
 | |
|         F: FnOnce(&[u8], UdpMetadata) -> R,
 | |
|     {
 | |
|         let mut f = Some(f);
 | |
|         poll_fn(move |cx| {
 | |
|             self.with_mut(|s, _| {
 | |
|                 match s.recv() {
 | |
|                     Ok((buffer, endpoint)) => Poll::Ready(unwrap!(f.take())(buffer, endpoint)),
 | |
|                     Err(udp::RecvError::Truncated) => unreachable!(),
 | |
|                     Err(udp::RecvError::Exhausted) => {
 | |
|                         // socket buffer is empty wait until at least one byte has arrived
 | |
|                         s.register_recv_waker(cx.waker());
 | |
|                         Poll::Pending
 | |
|                     }
 | |
|                 }
 | |
|             })
 | |
|         })
 | |
|         .await
 | |
|     }
 | |
| 
 | |
|     /// Send a datagram to the specified remote endpoint.
 | |
|     ///
 | |
|     /// This method will wait until the datagram has been sent.
 | |
|     ///
 | |
|     /// When the remote endpoint is not reachable, this method will return `Err(SendError::NoRoute)`
 | |
|     pub async fn send_to<T>(&self, buf: &[u8], remote_endpoint: T) -> Result<(), SendError>
 | |
|     where
 | |
|         T: Into<UdpMetadata>,
 | |
|     {
 | |
|         let remote_endpoint: UdpMetadata = remote_endpoint.into();
 | |
|         poll_fn(move |cx| self.poll_send_to(buf, remote_endpoint, cx)).await
 | |
|     }
 | |
| 
 | |
|     /// Send a datagram to the specified remote endpoint.
 | |
|     ///
 | |
|     /// When the datagram has been sent, this method will return `Poll::Ready(Ok())`.
 | |
|     ///
 | |
|     /// When the socket's send buffer is full, this method will return `Poll::Pending`
 | |
|     /// and register the current task to be notified when the buffer has space available.
 | |
|     ///
 | |
|     /// When the remote endpoint is not reachable, this method will return `Poll::Ready(Err(Error::NoRoute))`.
 | |
|     pub fn poll_send_to<T>(&self, buf: &[u8], remote_endpoint: T, cx: &mut Context<'_>) -> Poll<Result<(), SendError>>
 | |
|     where
 | |
|         T: Into<UdpMetadata>,
 | |
|     {
 | |
|         self.with_mut(|s, _| match s.send_slice(buf, remote_endpoint) {
 | |
|             // Entire datagram has been sent
 | |
|             Ok(()) => Poll::Ready(Ok(())),
 | |
|             Err(udp::SendError::BufferFull) => {
 | |
|                 s.register_send_waker(cx.waker());
 | |
|                 Poll::Pending
 | |
|             }
 | |
|             Err(udp::SendError::Unaddressable) => {
 | |
|                 // If no sender/outgoing port is specified, there is not really "no route"
 | |
|                 if s.endpoint().port == 0 {
 | |
|                     Poll::Ready(Err(SendError::SocketNotBound))
 | |
|                 } else {
 | |
|                     Poll::Ready(Err(SendError::NoRoute))
 | |
|                 }
 | |
|             }
 | |
|         })
 | |
|     }
 | |
| 
 | |
|     /// Send a datagram to the specified remote endpoint with a zero-copy function.
 | |
|     ///
 | |
|     /// This method will wait until the buffer can fit the requested size before
 | |
|     /// calling the function to fill its contents.
 | |
|     ///
 | |
|     /// When the remote endpoint is not reachable, this method will return `Err(SendError::NoRoute)`
 | |
|     pub async fn send_to_with<T, F, R>(&mut self, size: usize, remote_endpoint: T, f: F) -> Result<R, SendError>
 | |
|     where
 | |
|         T: Into<UdpMetadata> + Copy,
 | |
|         F: FnOnce(&mut [u8]) -> R,
 | |
|     {
 | |
|         let mut f = Some(f);
 | |
|         poll_fn(move |cx| {
 | |
|             self.with_mut(|s, _| {
 | |
|                 match s.send(size, remote_endpoint) {
 | |
|                     Ok(buffer) => Poll::Ready(Ok(unwrap!(f.take())(buffer))),
 | |
|                     Err(udp::SendError::BufferFull) => {
 | |
|                         s.register_send_waker(cx.waker());
 | |
|                         Poll::Pending
 | |
|                     }
 | |
|                     Err(udp::SendError::Unaddressable) => {
 | |
|                         // If no sender/outgoing port is specified, there is not really "no route"
 | |
|                         if s.endpoint().port == 0 {
 | |
|                             Poll::Ready(Err(SendError::SocketNotBound))
 | |
|                         } else {
 | |
|                             Poll::Ready(Err(SendError::NoRoute))
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             })
 | |
|         })
 | |
|         .await
 | |
|     }
 | |
| 
 | |
|     /// Returns the local endpoint of the socket.
 | |
|     pub fn endpoint(&self) -> IpListenEndpoint {
 | |
|         self.with(|s, _| s.endpoint())
 | |
|     }
 | |
| 
 | |
|     /// Returns whether the socket is open.
 | |
| 
 | |
|     pub fn is_open(&self) -> bool {
 | |
|         self.with(|s, _| s.is_open())
 | |
|     }
 | |
| 
 | |
|     /// Close the socket.
 | |
|     pub fn close(&mut self) {
 | |
|         self.with_mut(|s, _| s.close())
 | |
|     }
 | |
| 
 | |
|     /// Returns whether the socket is ready to send data, i.e. it has enough buffer space to hold a packet.
 | |
|     pub fn may_send(&self) -> bool {
 | |
|         self.with(|s, _| s.can_send())
 | |
|     }
 | |
| 
 | |
|     /// Returns whether the socket is ready to receive data, i.e. it has received a packet that's now in the buffer.
 | |
|     pub fn may_recv(&self) -> bool {
 | |
|         self.with(|s, _| s.can_recv())
 | |
|     }
 | |
| 
 | |
|     /// Return the maximum number packets the socket can receive.
 | |
|     pub fn packet_recv_capacity(&self) -> usize {
 | |
|         self.with(|s, _| s.packet_recv_capacity())
 | |
|     }
 | |
| 
 | |
|     /// Return the maximum number packets the socket can receive.
 | |
|     pub fn packet_send_capacity(&self) -> usize {
 | |
|         self.with(|s, _| s.packet_send_capacity())
 | |
|     }
 | |
| 
 | |
|     /// Return the maximum number of bytes inside the recv buffer.
 | |
|     pub fn payload_recv_capacity(&self) -> usize {
 | |
|         self.with(|s, _| s.payload_recv_capacity())
 | |
|     }
 | |
| 
 | |
|     /// Return the maximum number of bytes inside the transmit buffer.
 | |
|     pub fn payload_send_capacity(&self) -> usize {
 | |
|         self.with(|s, _| s.payload_send_capacity())
 | |
|     }
 | |
| 
 | |
|     /// Set the hop limit field in the IP header of sent packets.
 | |
|     pub fn set_hop_limit(&mut self, hop_limit: Option<u8>) {
 | |
|         self.with_mut(|s, _| s.set_hop_limit(hop_limit))
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl Drop for UdpSocket<'_> {
 | |
|     fn drop(&mut self) {
 | |
|         self.stack.with_mut(|i| i.sockets.remove(self.handle));
 | |
|     }
 | |
| }
 | |
| 
 | |
| fn _assert_covariant<'a, 'b: 'a>(x: UdpSocket<'b>) -> UdpSocket<'a> {
 | |
|     x
 | |
| }
 |