raw_socket.rs 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. use std::cell::RefCell;
  2. use std::io;
  3. use std::os::unix::io::{AsRawFd, RawFd};
  4. use std::rc::Rc;
  5. use std::vec::Vec;
  6. use crate::phy::{self, sys, Device, DeviceCapabilities, Medium};
  7. use crate::time::Instant;
  8. /// A socket that captures or transmits the complete frame.
  9. #[derive(Debug)]
  10. pub struct RawSocket {
  11. medium: Medium,
  12. lower: Rc<RefCell<sys::RawSocketDesc>>,
  13. mtu: usize,
  14. }
  15. impl AsRawFd for RawSocket {
  16. fn as_raw_fd(&self) -> RawFd {
  17. self.lower.borrow().as_raw_fd()
  18. }
  19. }
  20. impl RawSocket {
  21. /// Creates a raw socket, bound to the interface called `name`.
  22. ///
  23. /// This requires superuser privileges or a corresponding capability bit
  24. /// set on the executable.
  25. pub fn new(name: &str, medium: Medium) -> io::Result<RawSocket> {
  26. let mut lower = sys::RawSocketDesc::new(name, medium)?;
  27. lower.bind_interface()?;
  28. let mut mtu = lower.interface_mtu()?;
  29. #[cfg(feature = "medium-ieee802154")]
  30. if medium == Medium::Ieee802154 {
  31. // SIOCGIFMTU returns 127 - (ACK_PSDU - FCS - 1) - FCS.
  32. // 127 - (5 - 2 - 1) - 2 = 123
  33. // For IEEE802154, we want to add (ACK_PSDU - FCS - 1), since that is what SIOCGIFMTU
  34. // uses as the size of the link layer header.
  35. //
  36. // https://github.com/torvalds/linux/blob/7475e51b87969e01a6812eac713a1c8310372e8a/net/mac802154/iface.c#L541
  37. mtu += 2;
  38. }
  39. #[cfg(feature = "medium-ethernet")]
  40. if medium == Medium::Ethernet {
  41. // SIOCGIFMTU returns the IP MTU (typically 1500 bytes.)
  42. // smoltcp counts the entire Ethernet packet in the MTU, so add the Ethernet header size to it.
  43. mtu += crate::wire::EthernetFrame::<&[u8]>::header_len()
  44. }
  45. Ok(RawSocket {
  46. medium,
  47. lower: Rc::new(RefCell::new(lower)),
  48. mtu,
  49. })
  50. }
  51. }
  52. impl Device for RawSocket {
  53. type RxToken<'a>
  54. = RxToken
  55. where
  56. Self: 'a;
  57. type TxToken<'a>
  58. = TxToken
  59. where
  60. Self: 'a;
  61. fn capabilities(&self) -> DeviceCapabilities {
  62. DeviceCapabilities {
  63. max_transmission_unit: self.mtu,
  64. medium: self.medium,
  65. ..DeviceCapabilities::default()
  66. }
  67. }
  68. fn receive(&mut self, _timestamp: Instant) -> Option<(Self::RxToken<'_>, Self::TxToken<'_>)> {
  69. let mut lower = self.lower.borrow_mut();
  70. let mut buffer = vec![0; self.mtu];
  71. match lower.recv(&mut buffer[..]) {
  72. Ok(size) => {
  73. buffer.resize(size, 0);
  74. let rx = RxToken { buffer };
  75. let tx = TxToken {
  76. lower: self.lower.clone(),
  77. };
  78. Some((rx, tx))
  79. }
  80. Err(err) if err.kind() == io::ErrorKind::WouldBlock => None,
  81. Err(err) => panic!("{}", err),
  82. }
  83. }
  84. fn transmit(&mut self, _timestamp: Instant) -> Option<Self::TxToken<'_>> {
  85. Some(TxToken {
  86. lower: self.lower.clone(),
  87. })
  88. }
  89. }
  90. #[doc(hidden)]
  91. pub struct RxToken {
  92. buffer: Vec<u8>,
  93. }
  94. impl phy::RxToken for RxToken {
  95. fn consume<R, F>(self, f: F) -> R
  96. where
  97. F: FnOnce(&[u8]) -> R,
  98. {
  99. f(&self.buffer[..])
  100. }
  101. }
  102. #[doc(hidden)]
  103. pub struct TxToken {
  104. lower: Rc<RefCell<sys::RawSocketDesc>>,
  105. }
  106. impl phy::TxToken for TxToken {
  107. fn consume<R, F>(self, len: usize, f: F) -> R
  108. where
  109. F: FnOnce(&mut [u8]) -> R,
  110. {
  111. let mut lower = self.lower.borrow_mut();
  112. let mut buffer = vec![0; len];
  113. let result = f(&mut buffer);
  114. match lower.send(&buffer[..]) {
  115. Ok(_) => {}
  116. Err(err) if err.kind() == io::ErrorKind::WouldBlock => {
  117. net_debug!("phy: tx failed due to WouldBlock")
  118. }
  119. Err(err) => panic!("{}", err),
  120. }
  121. result
  122. }
  123. }