mod.rs 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243
  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::socket::udp;
  59. use crate::wire::IpEndpoint;
  60. static UDP_PAYLOAD: [u8; 5] = [0x48, 0x65, 0x6c, 0x6c, 0x6f];
  61. let (mut iface, mut sockets, _device) = setup(medium);
  62. let rx_buffer = udp::PacketBuffer::new(vec![udp::PacketMetadata::EMPTY], vec![0; 15]);
  63. let tx_buffer = udp::PacketBuffer::new(vec![udp::PacketMetadata::EMPTY], vec![0; 15]);
  64. let udp_socket = udp::Socket::new(rx_buffer, tx_buffer);
  65. let mut udp_bytes = vec![0u8; 13];
  66. let mut packet = UdpPacket::new_unchecked(&mut udp_bytes);
  67. let socket_handle = sockets.add(udp_socket);
  68. #[cfg(feature = "proto-ipv6")]
  69. let src_ip = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1);
  70. #[cfg(all(not(feature = "proto-ipv6"), feature = "proto-ipv4"))]
  71. let src_ip = Ipv4Address::new(0x7f, 0x00, 0x00, 0x02);
  72. let udp_repr = UdpRepr {
  73. src_port: 67,
  74. dst_port: 68,
  75. };
  76. #[cfg(feature = "proto-ipv6")]
  77. let ip_repr = IpRepr::Ipv6(Ipv6Repr {
  78. src_addr: src_ip,
  79. dst_addr: Ipv6Address::LINK_LOCAL_ALL_NODES,
  80. next_header: IpProtocol::Udp,
  81. payload_len: udp_repr.header_len() + UDP_PAYLOAD.len(),
  82. hop_limit: 0x40,
  83. });
  84. #[cfg(all(not(feature = "proto-ipv6"), feature = "proto-ipv4"))]
  85. let ip_repr = IpRepr::Ipv4(Ipv4Repr {
  86. src_addr: src_ip,
  87. dst_addr: Ipv4Address::BROADCAST,
  88. next_header: IpProtocol::Udp,
  89. payload_len: udp_repr.header_len() + UDP_PAYLOAD.len(),
  90. hop_limit: 0x40,
  91. });
  92. let dst_addr = ip_repr.dst_addr();
  93. // Bind the socket to port 68
  94. let socket = sockets.get_mut::<udp::Socket>(socket_handle);
  95. assert_eq!(socket.bind(68), Ok(()));
  96. assert!(!socket.can_recv());
  97. assert!(socket.can_send());
  98. udp_repr.emit(
  99. &mut packet,
  100. &ip_repr.src_addr(),
  101. &ip_repr.dst_addr(),
  102. UDP_PAYLOAD.len(),
  103. |buf| buf.copy_from_slice(&UDP_PAYLOAD),
  104. &ChecksumCapabilities::default(),
  105. );
  106. // Packet should be handled by bound UDP socket
  107. assert_eq!(
  108. iface.inner.process_udp(
  109. &mut sockets,
  110. PacketMeta::default(),
  111. false,
  112. ip_repr,
  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((
  124. &UDP_PAYLOAD[..],
  125. udp::UdpMetadata {
  126. local_address: Some(dst_addr),
  127. ..IpEndpoint::new(src_ip.into(), 67).into()
  128. }
  129. ))
  130. );
  131. }
  132. #[test]
  133. #[cfg(all(feature = "medium-ip", feature = "socket-tcp", feature = "proto-ipv6"))]
  134. pub fn tcp_not_accepted() {
  135. let (mut iface, mut sockets, _) = setup(Medium::Ip);
  136. let tcp = TcpRepr {
  137. src_port: 4242,
  138. dst_port: 4243,
  139. control: TcpControl::Syn,
  140. seq_number: TcpSeqNumber(-10001),
  141. ack_number: None,
  142. window_len: 256,
  143. window_scale: None,
  144. max_seg_size: None,
  145. sack_permitted: false,
  146. sack_ranges: [None, None, None],
  147. timestamp: None,
  148. payload: &[],
  149. };
  150. let mut tcp_bytes = vec![0u8; tcp.buffer_len()];
  151. tcp.emit(
  152. &mut TcpPacket::new_unchecked(&mut tcp_bytes),
  153. &Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 2).into(),
  154. &Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1).into(),
  155. &ChecksumCapabilities::default(),
  156. );
  157. assert_eq!(
  158. iface.inner.process_tcp(
  159. &mut sockets,
  160. IpRepr::Ipv6(Ipv6Repr {
  161. src_addr: Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 2),
  162. dst_addr: Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1),
  163. next_header: IpProtocol::Tcp,
  164. payload_len: tcp.buffer_len(),
  165. hop_limit: 64,
  166. }),
  167. &tcp_bytes,
  168. ),
  169. Some(Packet::new_ipv6(
  170. Ipv6Repr {
  171. src_addr: Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1),
  172. dst_addr: Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 2),
  173. next_header: IpProtocol::Tcp,
  174. payload_len: tcp.buffer_len(),
  175. hop_limit: 64,
  176. },
  177. IpPayload::Tcp(TcpRepr {
  178. src_port: 4243,
  179. dst_port: 4242,
  180. control: TcpControl::Rst,
  181. seq_number: TcpSeqNumber(0),
  182. ack_number: Some(TcpSeqNumber(-10000)),
  183. window_len: 0,
  184. window_scale: None,
  185. max_seg_size: None,
  186. sack_permitted: false,
  187. sack_ranges: [None, None, None],
  188. timestamp: None,
  189. payload: &[],
  190. })
  191. ))
  192. );
  193. // Unspecified destination address.
  194. tcp.emit(
  195. &mut TcpPacket::new_unchecked(&mut tcp_bytes),
  196. &Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 2).into(),
  197. &Ipv6Address::UNSPECIFIED.into(),
  198. &ChecksumCapabilities::default(),
  199. );
  200. assert_eq!(
  201. iface.inner.process_tcp(
  202. &mut sockets,
  203. IpRepr::Ipv6(Ipv6Repr {
  204. src_addr: Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 2),
  205. dst_addr: Ipv6Address::UNSPECIFIED,
  206. next_header: IpProtocol::Tcp,
  207. payload_len: tcp.buffer_len(),
  208. hop_limit: 64,
  209. }),
  210. &tcp_bytes,
  211. ),
  212. None,
  213. );
  214. }