ref_.rs 2.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  1. use core::ops::{Deref, DerefMut};
  2. #[cfg(feature = "socket-raw")]
  3. use socket::RawSocket;
  4. #[cfg(all(feature = "socket-icmp", any(feature = "proto-ipv4", feature = "proto-ipv6")))]
  5. use socket::IcmpSocket;
  6. #[cfg(feature = "socket-udp")]
  7. use socket::UdpSocket;
  8. #[cfg(feature = "socket-tcp")]
  9. use socket::TcpSocket;
  10. /// A trait for tracking a socket usage session.
  11. ///
  12. /// Allows implementation of custom drop logic that runs only if the socket was changed
  13. /// in specific ways. For example, drop logic for UDP would check if the local endpoint
  14. /// has changed, and if yes, notify the socket set.
  15. #[doc(hidden)]
  16. pub trait Session {
  17. fn finish(&mut self) {}
  18. }
  19. #[cfg(feature = "socket-raw")]
  20. impl<'a, 'b> Session for RawSocket<'a, 'b> {}
  21. #[cfg(all(feature = "socket-icmp", any(feature = "proto-ipv4", feature = "proto-ipv6")))]
  22. impl<'a, 'b> Session for IcmpSocket<'a, 'b> {}
  23. #[cfg(feature = "socket-udp")]
  24. impl<'a, 'b> Session for UdpSocket<'a, 'b> {}
  25. #[cfg(feature = "socket-tcp")]
  26. impl<'a> Session for TcpSocket<'a> {}
  27. /// A smart pointer to a socket.
  28. ///
  29. /// Allows the network stack to efficiently determine if the socket state was changed in any way.
  30. pub struct Ref<'a, T: Session + 'a> {
  31. socket: &'a mut T,
  32. consumed: bool,
  33. }
  34. impl<'a, T: Session + 'a> Ref<'a, T> {
  35. /// Wrap a pointer to a socket to make a smart pointer.
  36. ///
  37. /// Calling this function is only necessary if your code is using [into_inner].
  38. ///
  39. /// [into_inner]: #method.into_inner
  40. pub fn new(socket: &'a mut T) -> Self {
  41. Ref { socket, consumed: false }
  42. }
  43. /// Unwrap a smart pointer to a socket.
  44. ///
  45. /// The finalization code is not run. Prompt operation of the network stack depends
  46. /// on wrapping the returned pointer back and dropping it.
  47. ///
  48. /// Calling this function is only necessary to achieve composability if you *must*
  49. /// map a `&mut SocketRef<'a, XSocket>` to a `&'a mut XSocket` (note the lifetimes);
  50. /// be sure to call [new] afterwards.
  51. ///
  52. /// [new_unchecked]: #method.new_unchecked
  53. pub fn into_inner(mut ref_: Self) -> &'a mut T {
  54. ref_.consumed = true;
  55. ref_.socket
  56. }
  57. }
  58. impl<'a, T: Session> Deref for Ref<'a, T> {
  59. type Target = T;
  60. fn deref(&self) -> &Self::Target {
  61. self.socket
  62. }
  63. }
  64. impl<'a, T: Session> DerefMut for Ref<'a, T> {
  65. fn deref_mut(&mut self) -> &mut Self::Target {
  66. self.socket
  67. }
  68. }
  69. impl<'a, T: Session> Drop for Ref<'a, T> {
  70. fn drop(&mut self) {
  71. if !self.consumed {
  72. Session::finish(self.socket);
  73. }
  74. }
  75. }