mod.rs 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. #[cfg(feature = "proto-ipv4")]
  2. mod ipv4;
  3. #[cfg(feature = "proto-ipv6")]
  4. mod ipv6;
  5. #[cfg(feature = "proto-sixlowpan")]
  6. mod sixlowpan;
  7. #[cfg(feature = "proto-igmp")]
  8. use std::vec::Vec;
  9. use rstest::*;
  10. use super::*;
  11. use crate::iface::Interface;
  12. use crate::phy::{ChecksumCapabilities, Loopback};
  13. use crate::time::Instant;
  14. #[allow(unused)]
  15. fn fill_slice(s: &mut [u8], val: u8) {
  16. for x in s.iter_mut() {
  17. *x = val
  18. }
  19. }
  20. fn setup<'a>(medium: Medium) -> (Interface, SocketSet<'a>, Loopback) {
  21. let mut device = Loopback::new(medium);
  22. let config = Config::new(match medium {
  23. #[cfg(feature = "medium-ethernet")]
  24. Medium::Ethernet => HardwareAddress::Ethernet(Default::default()),
  25. #[cfg(feature = "medium-ip")]
  26. Medium::Ip => HardwareAddress::Ip,
  27. #[cfg(feature = "medium-ieee802154")]
  28. Medium::Ieee802154 => HardwareAddress::Ieee802154(Default::default()),
  29. });
  30. let mut iface = Interface::new(config, &mut device, Instant::ZERO);
  31. #[cfg(feature = "proto-ipv4")]
  32. {
  33. iface.update_ip_addrs(|ip_addrs| {
  34. ip_addrs
  35. .push(IpCidr::new(IpAddress::v4(127, 0, 0, 1), 8))
  36. .unwrap();
  37. });
  38. }
  39. #[cfg(feature = "proto-ipv6")]
  40. {
  41. iface.update_ip_addrs(|ip_addrs| {
  42. ip_addrs
  43. .push(IpCidr::new(IpAddress::v6(0, 0, 0, 0, 0, 0, 0, 1), 128))
  44. .unwrap();
  45. ip_addrs
  46. .push(IpCidr::new(IpAddress::v6(0xfdbe, 0, 0, 0, 0, 0, 0, 1), 64))
  47. .unwrap();
  48. });
  49. }
  50. (iface, SocketSet::new(vec![]), device)
  51. }
  52. #[cfg(feature = "proto-igmp")]
  53. fn recv_all(device: &mut Loopback, timestamp: Instant) -> Vec<Vec<u8>> {
  54. let mut pkts = Vec::new();
  55. while let Some((rx, _tx)) = device.receive(timestamp) {
  56. rx.consume(|pkt| {
  57. pkts.push(pkt.to_vec());
  58. });
  59. }
  60. pkts
  61. }
  62. #[derive(Debug, PartialEq)]
  63. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  64. struct MockTxToken;
  65. impl TxToken for MockTxToken {
  66. fn consume<R, F>(self, len: usize, f: F) -> R
  67. where
  68. F: FnOnce(&mut [u8]) -> R,
  69. {
  70. let mut junk = [0; 1536];
  71. f(&mut junk[..len])
  72. }
  73. }
  74. #[test]
  75. #[should_panic(expected = "The hardware address does not match the medium of the interface.")]
  76. #[cfg(all(feature = "medium-ip", feature = "medium-ethernet"))]
  77. fn test_new_panic() {
  78. let mut device = Loopback::new(Medium::Ethernet);
  79. let config = Config::new(HardwareAddress::Ip);
  80. Interface::new(config, &mut device, Instant::ZERO);
  81. }
  82. #[rstest]
  83. #[cfg(feature = "default")]
  84. fn test_handle_udp_broadcast(
  85. #[values(Medium::Ip, Medium::Ethernet, Medium::Ieee802154)] medium: Medium,
  86. ) {
  87. use crate::wire::IpEndpoint;
  88. static UDP_PAYLOAD: [u8; 5] = [0x48, 0x65, 0x6c, 0x6c, 0x6f];
  89. let (mut iface, mut sockets, _device) = setup(medium);
  90. let rx_buffer = udp::PacketBuffer::new(vec![udp::PacketMetadata::EMPTY], vec![0; 15]);
  91. let tx_buffer = udp::PacketBuffer::new(vec![udp::PacketMetadata::EMPTY], vec![0; 15]);
  92. let udp_socket = udp::Socket::new(rx_buffer, tx_buffer);
  93. let mut udp_bytes = vec![0u8; 13];
  94. let mut packet = UdpPacket::new_unchecked(&mut udp_bytes);
  95. let socket_handle = sockets.add(udp_socket);
  96. #[cfg(feature = "proto-ipv6")]
  97. let src_ip = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1);
  98. #[cfg(all(not(feature = "proto-ipv6"), feature = "proto-ipv4"))]
  99. let src_ip = Ipv4Address::new(0x7f, 0x00, 0x00, 0x02);
  100. let udp_repr = UdpRepr {
  101. src_port: 67,
  102. dst_port: 68,
  103. };
  104. #[cfg(feature = "proto-ipv6")]
  105. let ip_repr = IpRepr::Ipv6(Ipv6Repr {
  106. src_addr: src_ip,
  107. dst_addr: Ipv6Address::LINK_LOCAL_ALL_NODES,
  108. next_header: IpProtocol::Udp,
  109. payload_len: udp_repr.header_len() + UDP_PAYLOAD.len(),
  110. hop_limit: 0x40,
  111. });
  112. #[cfg(all(not(feature = "proto-ipv6"), feature = "proto-ipv4"))]
  113. let ip_repr = IpRepr::Ipv4(Ipv4Repr {
  114. src_addr: src_ip,
  115. dst_addr: Ipv4Address::BROADCAST,
  116. next_header: IpProtocol::Udp,
  117. payload_len: udp_repr.header_len() + UDP_PAYLOAD.len(),
  118. hop_limit: 0x40,
  119. });
  120. // Bind the socket to port 68
  121. let socket = sockets.get_mut::<udp::Socket>(socket_handle);
  122. assert_eq!(socket.bind(68), Ok(()));
  123. assert!(!socket.can_recv());
  124. assert!(socket.can_send());
  125. udp_repr.emit(
  126. &mut packet,
  127. &ip_repr.src_addr(),
  128. &ip_repr.dst_addr(),
  129. UDP_PAYLOAD.len(),
  130. |buf| buf.copy_from_slice(&UDP_PAYLOAD),
  131. &ChecksumCapabilities::default(),
  132. );
  133. // Packet should be handled by bound UDP socket
  134. assert_eq!(
  135. iface.inner.process_udp(
  136. &mut sockets,
  137. PacketMeta::default(),
  138. ip_repr,
  139. udp_repr,
  140. false,
  141. &UDP_PAYLOAD,
  142. packet.into_inner(),
  143. ),
  144. None
  145. );
  146. // Make sure the payload to the UDP packet processed by process_udp is
  147. // appended to the bound sockets rx_buffer
  148. let socket = sockets.get_mut::<udp::Socket>(socket_handle);
  149. assert!(socket.can_recv());
  150. assert_eq!(
  151. socket.recv(),
  152. Ok((&UDP_PAYLOAD[..], IpEndpoint::new(src_ip.into(), 67).into()))
  153. );
  154. }