mod.rs 4.1 KB

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