ip_packet.rs 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. use crate::phy::DeviceCapabilities;
  2. use crate::wire::*;
  3. #[allow(clippy::large_enum_variant)]
  4. #[derive(Debug, PartialEq)]
  5. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  6. #[cfg(feature = "medium-ethernet")]
  7. pub(crate) enum EthernetPacket<'a> {
  8. #[cfg(feature = "proto-ipv4")]
  9. Arp(ArpRepr),
  10. Ip(IpPacket<'a>),
  11. }
  12. #[derive(Debug, PartialEq)]
  13. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  14. pub(crate) enum IpPacket<'p> {
  15. #[cfg(feature = "proto-ipv4")]
  16. Ipv4(Ipv4Packet<'p>),
  17. #[cfg(feature = "proto-ipv6")]
  18. Ipv6(Ipv6Packet<'p>),
  19. }
  20. impl<'p> IpPacket<'p> {
  21. pub(crate) fn new(ip_repr: IpRepr, payload: IpPayload<'p>) -> Self {
  22. match ip_repr {
  23. #[cfg(feature = "proto-ipv4")]
  24. IpRepr::Ipv4(header) => Self::new_ipv4(header, payload),
  25. #[cfg(feature = "proto-ipv6")]
  26. IpRepr::Ipv6(header) => Self::new_ipv6(header, payload),
  27. }
  28. }
  29. #[cfg(feature = "proto-ipv4")]
  30. pub(crate) fn new_ipv4(ip_repr: Ipv4Repr, payload: IpPayload<'p>) -> Self {
  31. Self::Ipv4(Ipv4Packet {
  32. header: ip_repr,
  33. payload,
  34. })
  35. }
  36. #[cfg(feature = "proto-ipv6")]
  37. pub(crate) fn new_ipv6(ip_repr: Ipv6Repr, payload: IpPayload<'p>) -> Self {
  38. Self::Ipv6(Ipv6Packet {
  39. header: ip_repr,
  40. #[cfg(feature = "proto-ipv6-hbh")]
  41. hop_by_hop: None,
  42. #[cfg(feature = "proto-ipv6-fragmentation")]
  43. fragment: None,
  44. #[cfg(feature = "proto-ipv6-routing")]
  45. routing: None,
  46. payload,
  47. })
  48. }
  49. pub(crate) fn ip_repr(&self) -> IpRepr {
  50. match self {
  51. #[cfg(feature = "proto-ipv4")]
  52. IpPacket::Ipv4(p) => IpRepr::Ipv4(p.header),
  53. #[cfg(feature = "proto-ipv6")]
  54. IpPacket::Ipv6(p) => IpRepr::Ipv6(p.header),
  55. }
  56. }
  57. pub(crate) fn payload(&self) -> &IpPayload<'p> {
  58. match self {
  59. #[cfg(feature = "proto-ipv4")]
  60. IpPacket::Ipv4(p) => &p.payload,
  61. #[cfg(feature = "proto-ipv6")]
  62. IpPacket::Ipv6(p) => &p.payload,
  63. }
  64. }
  65. pub(crate) fn emit_payload(
  66. &self,
  67. _ip_repr: &IpRepr,
  68. payload: &mut [u8],
  69. caps: &DeviceCapabilities,
  70. ) {
  71. match self.payload() {
  72. #[cfg(feature = "proto-ipv4")]
  73. IpPayload::Icmpv4(icmpv4_repr) => {
  74. icmpv4_repr.emit(&mut Icmpv4Packet::new_unchecked(payload), &caps.checksum)
  75. }
  76. #[cfg(feature = "proto-igmp")]
  77. IpPayload::Igmp(igmp_repr) => igmp_repr.emit(&mut IgmpPacket::new_unchecked(payload)),
  78. #[cfg(feature = "proto-ipv6")]
  79. IpPayload::Icmpv6(icmpv6_repr) => icmpv6_repr.emit(
  80. &_ip_repr.src_addr(),
  81. &_ip_repr.dst_addr(),
  82. &mut Icmpv6Packet::new_unchecked(payload),
  83. &caps.checksum,
  84. ),
  85. #[cfg(feature = "socket-raw")]
  86. IpPayload::Raw(raw_packet) => payload.copy_from_slice(raw_packet),
  87. #[cfg(any(feature = "socket-udp", feature = "socket-dns"))]
  88. IpPayload::Udp(udp_repr, inner_payload) => udp_repr.emit(
  89. &mut UdpPacket::new_unchecked(payload),
  90. &_ip_repr.src_addr(),
  91. &_ip_repr.dst_addr(),
  92. inner_payload.len(),
  93. |buf| buf.copy_from_slice(inner_payload),
  94. &caps.checksum,
  95. ),
  96. #[cfg(feature = "socket-tcp")]
  97. IpPayload::Tcp(mut tcp_repr) => {
  98. // This is a terrible hack to make TCP performance more acceptable on systems
  99. // where the TCP buffers are significantly larger than network buffers,
  100. // e.g. a 64 kB TCP receive buffer (and so, when empty, a 64k window)
  101. // together with four 1500 B Ethernet receive buffers. If left untreated,
  102. // this would result in our peer pushing our window and sever packet loss.
  103. //
  104. // I'm really not happy about this "solution" but I don't know what else to do.
  105. if let Some(max_burst_size) = caps.max_burst_size {
  106. let mut max_segment_size = caps.max_transmission_unit;
  107. max_segment_size -= _ip_repr.header_len();
  108. max_segment_size -= tcp_repr.header_len();
  109. let max_window_size = max_burst_size * max_segment_size;
  110. if tcp_repr.window_len as usize > max_window_size {
  111. tcp_repr.window_len = max_window_size as u16;
  112. }
  113. }
  114. tcp_repr.emit(
  115. &mut TcpPacket::new_unchecked(payload),
  116. &_ip_repr.src_addr(),
  117. &_ip_repr.dst_addr(),
  118. &caps.checksum,
  119. );
  120. }
  121. #[cfg(feature = "socket-dhcpv4")]
  122. IpPayload::Dhcpv4(udp_repr, dhcp_repr) => udp_repr.emit(
  123. &mut UdpPacket::new_unchecked(payload),
  124. &_ip_repr.src_addr(),
  125. &_ip_repr.dst_addr(),
  126. dhcp_repr.buffer_len(),
  127. |buf| dhcp_repr.emit(&mut DhcpPacket::new_unchecked(buf)).unwrap(),
  128. &caps.checksum,
  129. ),
  130. }
  131. }
  132. }
  133. #[derive(Debug, PartialEq)]
  134. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  135. #[cfg(feature = "proto-ipv4")]
  136. pub(crate) struct Ipv4Packet<'p> {
  137. header: Ipv4Repr,
  138. payload: IpPayload<'p>,
  139. }
  140. #[derive(Debug, PartialEq)]
  141. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  142. #[cfg(feature = "proto-ipv6")]
  143. pub(crate) struct Ipv6Packet<'p> {
  144. header: Ipv6Repr,
  145. #[cfg(feature = "proto-ipv6-hbh")]
  146. hop_by_hop: Option<Ipv6HopByHopRepr<'p>>,
  147. #[cfg(feature = "proto-ipv6-fragmentation")]
  148. fragment: Option<Ipv6FragmentRepr>,
  149. #[cfg(feature = "proto-ipv6-routing")]
  150. routing: Option<Ipv6RoutingRepr<'p>>,
  151. payload: IpPayload<'p>,
  152. }
  153. #[derive(Debug, PartialEq)]
  154. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  155. pub(crate) enum IpPayload<'p> {
  156. #[cfg(feature = "proto-ipv4")]
  157. Icmpv4(Icmpv4Repr<'p>),
  158. #[cfg(feature = "proto-igmp")]
  159. Igmp(IgmpRepr),
  160. #[cfg(feature = "proto-ipv6")]
  161. Icmpv6(Icmpv6Repr<'p>),
  162. #[cfg(feature = "socket-raw")]
  163. Raw(&'p [u8]),
  164. #[cfg(any(feature = "socket-udp", feature = "socket-dns"))]
  165. Udp(UdpRepr, &'p [u8]),
  166. #[cfg(feature = "socket-tcp")]
  167. Tcp(TcpRepr<'p>),
  168. #[cfg(feature = "socket-dhcpv4")]
  169. Dhcpv4(UdpRepr, DhcpRepr<'p>),
  170. }
  171. #[cfg(any(feature = "proto-ipv4", feature = "proto-ipv6"))]
  172. pub(crate) fn icmp_reply_payload_len(len: usize, mtu: usize, header_len: usize) -> usize {
  173. // Send back as much of the original payload as will fit within
  174. // the minimum MTU required by IPv4. See RFC 1812 § 4.3.2.3 for
  175. // more details.
  176. //
  177. // Since the entire network layer packet must fit within the minimum
  178. // MTU supported, the payload must not exceed the following:
  179. //
  180. // <min mtu> - IP Header Size * 2 - ICMPv4 DstUnreachable hdr size
  181. len.min(mtu - header_len * 2 - 8)
  182. }
  183. #[cfg(feature = "proto-igmp")]
  184. pub(crate) enum IgmpReportState {
  185. Inactive,
  186. ToGeneralQuery {
  187. version: IgmpVersion,
  188. timeout: crate::time::Instant,
  189. interval: crate::time::Duration,
  190. next_index: usize,
  191. },
  192. ToSpecificQuery {
  193. version: IgmpVersion,
  194. timeout: crate::time::Instant,
  195. group: Ipv4Address,
  196. },
  197. }