mod.rs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. /*! Access to networking hardware.
  2. The `phy` module deals with the *network devices*. It provides a trait
  3. for transmitting and receiving frames, [Device](trait.Device.html)
  4. and implementations of it:
  5. * the [_loopback_](struct.Loopback.html), for zero dependency testing;
  6. * _middleware_ [Tracer](struct.Tracer.html) and
  7. [FaultInjector](struct.FaultInjector.html), to facilitate debugging;
  8. * _adapters_ [RawSocket](struct.RawSocket.html) and
  9. [TunTapInterface](struct.TunTapInterface.html), to transmit and receive frames
  10. on the host OS.
  11. */
  12. #![cfg_attr(
  13. feature = "medium-ethernet",
  14. doc = r##"
  15. # Examples
  16. An implementation of the [Device](trait.Device.html) trait for a simple hardware
  17. Ethernet controller could look as follows:
  18. ```rust
  19. use smoltcp::phy::{self, DeviceCapabilities, Device, Medium};
  20. use smoltcp::time::Instant;
  21. struct StmPhy {
  22. rx_buffer: [u8; 1536],
  23. tx_buffer: [u8; 1536],
  24. }
  25. impl<'a> StmPhy {
  26. fn new() -> StmPhy {
  27. StmPhy {
  28. rx_buffer: [0; 1536],
  29. tx_buffer: [0; 1536],
  30. }
  31. }
  32. }
  33. impl phy::Device for StmPhy {
  34. type RxToken<'a> = StmPhyRxToken<'a> where Self: 'a;
  35. type TxToken<'a> = StmPhyTxToken<'a> where Self: 'a;
  36. fn receive(&mut self, _timestamp: Instant) -> Option<(Self::RxToken<'_>, Self::TxToken<'_>)> {
  37. Some((StmPhyRxToken(&mut self.rx_buffer[..]),
  38. StmPhyTxToken(&mut self.tx_buffer[..])))
  39. }
  40. fn transmit(&mut self, _timestamp: Instant) -> Option<Self::TxToken<'_>> {
  41. Some(StmPhyTxToken(&mut self.tx_buffer[..]))
  42. }
  43. fn capabilities(&self) -> DeviceCapabilities {
  44. let mut caps = DeviceCapabilities::default();
  45. caps.max_transmission_unit = 1536;
  46. caps.max_burst_size = Some(1);
  47. caps.medium = Medium::Ethernet;
  48. caps
  49. }
  50. }
  51. struct StmPhyRxToken<'a>(&'a mut [u8]);
  52. impl<'a> phy::RxToken for StmPhyRxToken<'a> {
  53. fn consume<R, F>(self, f: F) -> R
  54. where F: FnOnce(& [u8]) -> R
  55. {
  56. // TODO: receive packet into buffer
  57. let result = f(&self.0);
  58. println!("rx called");
  59. result
  60. }
  61. }
  62. struct StmPhyTxToken<'a>(&'a mut [u8]);
  63. impl<'a> phy::TxToken for StmPhyTxToken<'a> {
  64. fn consume<R, F>(self, len: usize, f: F) -> R
  65. where F: FnOnce(&mut [u8]) -> R
  66. {
  67. let result = f(&mut self.0[..len]);
  68. println!("tx called {}", len);
  69. // TODO: send packet out
  70. result
  71. }
  72. }
  73. ```
  74. "##
  75. )]
  76. use crate::time::Instant;
  77. #[cfg(all(
  78. any(feature = "phy-raw_socket", feature = "phy-tuntap_interface"),
  79. unix
  80. ))]
  81. mod sys;
  82. mod fault_injector;
  83. #[cfg(feature = "alloc")]
  84. mod fuzz_injector;
  85. #[cfg(feature = "alloc")]
  86. mod loopback;
  87. mod pcap_writer;
  88. #[cfg(all(feature = "phy-raw_socket", unix))]
  89. mod raw_socket;
  90. mod tracer;
  91. #[cfg(all(
  92. feature = "phy-tuntap_interface",
  93. any(target_os = "linux", target_os = "android")
  94. ))]
  95. mod tuntap_interface;
  96. #[cfg(all(
  97. any(feature = "phy-raw_socket", feature = "phy-tuntap_interface"),
  98. unix
  99. ))]
  100. pub use self::sys::wait;
  101. pub use self::fault_injector::FaultInjector;
  102. #[cfg(feature = "alloc")]
  103. pub use self::fuzz_injector::{FuzzInjector, Fuzzer};
  104. #[cfg(feature = "alloc")]
  105. pub use self::loopback::Loopback;
  106. pub use self::pcap_writer::{PcapLinkType, PcapMode, PcapSink, PcapWriter};
  107. #[cfg(all(feature = "phy-raw_socket", unix))]
  108. pub use self::raw_socket::RawSocket;
  109. pub use self::tracer::Tracer;
  110. #[cfg(all(
  111. feature = "phy-tuntap_interface",
  112. any(target_os = "linux", target_os = "android")
  113. ))]
  114. pub use self::tuntap_interface::TunTapInterface;
  115. /// Metadata associated to a packet.
  116. ///
  117. /// The packet metadata is a set of attributes associated to network packets
  118. /// as they travel up or down the stack. The metadata is get/set by the
  119. /// [`Device`] implementations or by the user when sending/receiving packets from a
  120. /// socket.
  121. ///
  122. /// Metadata fields are enabled via Cargo features. If no field is enabled, this
  123. /// struct becomes zero-sized, which allows the compiler to optimize it out as if
  124. /// the packet metadata mechanism didn't exist at all.
  125. ///
  126. /// Currently only UDP sockets allow setting/retrieving packet metadata. The metadata
  127. /// for packets emitted with other sockets will be all default values.
  128. ///
  129. /// This struct is marked as `#[non_exhaustive]`. This means it is not possible to
  130. /// create it directly by specifying all fields. You have to instead create it with
  131. /// default values and then set the fields you want. This makes adding metadata
  132. /// fields a non-breaking change.
  133. ///
  134. /// ```rust
  135. /// let mut meta = smoltcp::phy::PacketMeta::default();
  136. /// #[cfg(feature = "packetmeta-id")]
  137. /// {
  138. /// meta.id = 15;
  139. /// }
  140. /// ```
  141. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  142. #[derive(Debug, PartialEq, Eq, Hash, Clone, Copy, Default)]
  143. #[non_exhaustive]
  144. pub struct PacketMeta {
  145. #[cfg(feature = "packetmeta-id")]
  146. pub id: u32,
  147. }
  148. /// A description of checksum behavior for a particular protocol.
  149. #[derive(Debug, Clone, Copy, Default)]
  150. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  151. pub enum Checksum {
  152. /// Verify checksum when receiving and compute checksum when sending.
  153. #[default]
  154. Both,
  155. /// Verify checksum when receiving.
  156. Rx,
  157. /// Compute checksum before sending.
  158. Tx,
  159. /// Ignore checksum completely.
  160. None,
  161. }
  162. impl Checksum {
  163. /// Returns whether checksum should be verified when receiving.
  164. pub fn rx(&self) -> bool {
  165. match *self {
  166. Checksum::Both | Checksum::Rx => true,
  167. _ => false,
  168. }
  169. }
  170. /// Returns whether checksum should be verified when sending.
  171. pub fn tx(&self) -> bool {
  172. match *self {
  173. Checksum::Both | Checksum::Tx => true,
  174. _ => false,
  175. }
  176. }
  177. }
  178. /// A description of checksum behavior for every supported protocol.
  179. #[derive(Debug, Clone, Default)]
  180. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  181. #[non_exhaustive]
  182. pub struct ChecksumCapabilities {
  183. pub ipv4: Checksum,
  184. pub udp: Checksum,
  185. pub tcp: Checksum,
  186. #[cfg(feature = "proto-ipv4")]
  187. pub icmpv4: Checksum,
  188. #[cfg(feature = "proto-ipv6")]
  189. pub icmpv6: Checksum,
  190. }
  191. impl ChecksumCapabilities {
  192. /// Checksum behavior that results in not computing or verifying checksums
  193. /// for any of the supported protocols.
  194. pub fn ignored() -> Self {
  195. ChecksumCapabilities {
  196. ipv4: Checksum::None,
  197. udp: Checksum::None,
  198. tcp: Checksum::None,
  199. #[cfg(feature = "proto-ipv4")]
  200. icmpv4: Checksum::None,
  201. #[cfg(feature = "proto-ipv6")]
  202. icmpv6: Checksum::None,
  203. }
  204. }
  205. }
  206. /// A description of device capabilities.
  207. ///
  208. /// Higher-level protocols may achieve higher throughput or lower latency if they consider
  209. /// the bandwidth or packet size limitations.
  210. #[derive(Debug, Clone, Default)]
  211. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  212. #[non_exhaustive]
  213. pub struct DeviceCapabilities {
  214. /// Medium of the device.
  215. ///
  216. /// This indicates what kind of packet the sent/received bytes are, and determines
  217. /// some behaviors of Interface. For example, ARP/NDISC address resolution is only done
  218. /// for Ethernet mediums.
  219. pub medium: Medium,
  220. /// Maximum transmission unit.
  221. ///
  222. /// The network device is unable to send or receive frames larger than the value returned
  223. /// by this function.
  224. ///
  225. /// For Ethernet devices, this is the maximum Ethernet frame size, including the Ethernet header (14 octets), but
  226. /// *not* including the Ethernet FCS (4 octets). Therefore, Ethernet MTU = IP MTU + 14.
  227. ///
  228. /// Note that in Linux and other OSes, "MTU" is the IP MTU, not the Ethernet MTU, even for Ethernet
  229. /// devices. This is a common source of confusion.
  230. ///
  231. /// Most common IP MTU is 1500. Minimum is 576 (for IPv4) or 1280 (for IPv6). Maximum is 9216 octets.
  232. pub max_transmission_unit: usize,
  233. /// Maximum burst size, in terms of MTU.
  234. ///
  235. /// The network device is unable to send or receive bursts large than the value returned
  236. /// by this function.
  237. ///
  238. /// If `None`, there is no fixed limit on burst size, e.g. if network buffers are
  239. /// dynamically allocated.
  240. pub max_burst_size: Option<usize>,
  241. /// Checksum behavior.
  242. ///
  243. /// If the network device is capable of verifying or computing checksums for some protocols,
  244. /// it can request that the stack not do so in software to improve performance.
  245. pub checksum: ChecksumCapabilities,
  246. }
  247. impl DeviceCapabilities {
  248. pub fn ip_mtu(&self) -> usize {
  249. match self.medium {
  250. #[cfg(feature = "medium-ethernet")]
  251. Medium::Ethernet => {
  252. self.max_transmission_unit - crate::wire::EthernetFrame::<&[u8]>::header_len()
  253. }
  254. #[cfg(feature = "medium-ip")]
  255. Medium::Ip => self.max_transmission_unit,
  256. #[cfg(feature = "medium-ieee802154")]
  257. Medium::Ieee802154 => self.max_transmission_unit, // TODO(thvdveld): what is the MTU for Medium::IEEE802
  258. }
  259. }
  260. }
  261. /// Type of medium of a device.
  262. #[derive(Debug, Eq, PartialEq, Copy, Clone)]
  263. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  264. pub enum Medium {
  265. /// Ethernet medium. Devices of this type send and receive Ethernet frames,
  266. /// and interfaces using it must do neighbor discovery via ARP or NDISC.
  267. ///
  268. /// Examples of devices of this type are Ethernet, WiFi (802.11), Linux `tap`, and VPNs in tap (layer 2) mode.
  269. #[cfg(feature = "medium-ethernet")]
  270. Ethernet,
  271. /// IP medium. Devices of this type send and receive IP frames, without an
  272. /// Ethernet header. MAC addresses are not used, and no neighbor discovery (ARP, NDISC) is done.
  273. ///
  274. /// Examples of devices of this type are the Linux `tun`, PPP interfaces, VPNs in tun (layer 3) mode.
  275. #[cfg(feature = "medium-ip")]
  276. Ip,
  277. #[cfg(feature = "medium-ieee802154")]
  278. Ieee802154,
  279. }
  280. impl Default for Medium {
  281. fn default() -> Medium {
  282. #[cfg(feature = "medium-ethernet")]
  283. return Medium::Ethernet;
  284. #[cfg(all(feature = "medium-ip", not(feature = "medium-ethernet")))]
  285. return Medium::Ip;
  286. #[cfg(all(
  287. feature = "medium-ieee802154",
  288. not(feature = "medium-ip"),
  289. not(feature = "medium-ethernet")
  290. ))]
  291. return Medium::Ieee802154;
  292. #[cfg(all(
  293. not(feature = "medium-ip"),
  294. not(feature = "medium-ethernet"),
  295. not(feature = "medium-ieee802154")
  296. ))]
  297. return panic!("No medium enabled");
  298. }
  299. }
  300. /// An interface for sending and receiving raw network frames.
  301. ///
  302. /// The interface is based on _tokens_, which are types that allow to receive/transmit a
  303. /// single packet. The `receive` and `transmit` functions only construct such tokens, the
  304. /// real sending/receiving operation are performed when the tokens are consumed.
  305. pub trait Device {
  306. type RxToken<'a>: RxToken
  307. where
  308. Self: 'a;
  309. type TxToken<'a>: TxToken
  310. where
  311. Self: 'a;
  312. /// Construct a token pair consisting of one receive token and one transmit token.
  313. ///
  314. /// The additional transmit token makes it possible to generate a reply packet based
  315. /// on the contents of the received packet. For example, this makes it possible to
  316. /// handle arbitrarily large ICMP echo ("ping") requests, where the all received bytes
  317. /// need to be sent back, without heap allocation.
  318. ///
  319. /// The timestamp must be a number of milliseconds, monotonically increasing since an
  320. /// arbitrary moment in time, such as system startup.
  321. fn receive(&mut self, timestamp: Instant) -> Option<(Self::RxToken<'_>, Self::TxToken<'_>)>;
  322. /// Construct a transmit token.
  323. ///
  324. /// The timestamp must be a number of milliseconds, monotonically increasing since an
  325. /// arbitrary moment in time, such as system startup.
  326. fn transmit(&mut self, timestamp: Instant) -> Option<Self::TxToken<'_>>;
  327. /// Get a description of device capabilities.
  328. fn capabilities(&self) -> DeviceCapabilities;
  329. }
  330. /// A token to receive a single network packet.
  331. pub trait RxToken {
  332. /// Consumes the token to receive a single network packet.
  333. ///
  334. /// This method receives a packet and then calls the given closure `f` with the raw
  335. /// packet bytes as argument.
  336. fn consume<R, F>(self, f: F) -> R
  337. where
  338. F: FnOnce(&[u8]) -> R;
  339. /// The Packet ID associated with the frame received by this [`RxToken`]
  340. fn meta(&self) -> PacketMeta {
  341. PacketMeta::default()
  342. }
  343. }
  344. /// A token to transmit a single network packet.
  345. pub trait TxToken {
  346. /// Consumes the token to send a single network packet.
  347. ///
  348. /// This method constructs a transmit buffer of size `len` and calls the passed
  349. /// closure `f` with a mutable reference to that buffer. The closure should construct
  350. /// a valid network packet (e.g. an ethernet packet) in the buffer. When the closure
  351. /// returns, the transmit buffer is sent out.
  352. fn consume<R, F>(self, len: usize, f: F) -> R
  353. where
  354. F: FnOnce(&mut [u8]) -> R;
  355. /// The Packet ID to be associated with the frame to be transmitted by this [`TxToken`].
  356. #[allow(unused_variables)]
  357. fn set_meta(&mut self, meta: PacketMeta) {}
  358. }