ipv6.rs 46 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382
  1. use super::*;
  2. fn parse_ipv6(data: &[u8]) -> crate::wire::Result<Packet<'_>> {
  3. let ipv6_header = Ipv6Packet::new_checked(data)?;
  4. let ipv6 = Ipv6Repr::parse(&ipv6_header)?;
  5. match ipv6.next_header {
  6. IpProtocol::HopByHop => todo!(),
  7. IpProtocol::Icmp => todo!(),
  8. IpProtocol::Igmp => todo!(),
  9. IpProtocol::Tcp => todo!(),
  10. IpProtocol::Udp => todo!(),
  11. IpProtocol::Ipv6Route => todo!(),
  12. IpProtocol::Ipv6Frag => todo!(),
  13. IpProtocol::IpSecEsp => todo!(),
  14. IpProtocol::IpSecAh => todo!(),
  15. IpProtocol::Icmpv6 => {
  16. let icmp = Icmpv6Repr::parse(
  17. &ipv6.src_addr,
  18. &ipv6.dst_addr,
  19. &Icmpv6Packet::new_checked(ipv6_header.payload())?,
  20. &Default::default(),
  21. )?;
  22. Ok(Packet::new_ipv6(ipv6, IpPayload::Icmpv6(icmp)))
  23. }
  24. IpProtocol::Ipv6NoNxt => todo!(),
  25. IpProtocol::Ipv6Opts => todo!(),
  26. IpProtocol::Unknown(_) => todo!(),
  27. }
  28. }
  29. #[rstest]
  30. #[case::ip(Medium::Ip)]
  31. #[cfg(feature = "medium-ip")]
  32. #[case::ethernet(Medium::Ethernet)]
  33. #[cfg(feature = "medium-ethernet")]
  34. #[case::ieee802154(Medium::Ieee802154)]
  35. #[cfg(feature = "medium-ieee802154")]
  36. fn any_ip(#[case] medium: Medium) {
  37. // An empty echo request with destination address fdbe::3, which is not part of the interface
  38. // address list.
  39. let data = [
  40. 0x60, 0x0, 0x0, 0x0, 0x0, 0x8, 0x3a, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  41. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  42. 0x0, 0x0, 0x0, 0x0, 0x3, 0x80, 0x0, 0x84, 0x3a, 0x0, 0x0, 0x0, 0x0,
  43. ];
  44. assert_eq!(
  45. parse_ipv6(&data),
  46. Ok(Packet::new_ipv6(
  47. Ipv6Repr {
  48. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  49. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0003]),
  50. hop_limit: 64,
  51. next_header: IpProtocol::Icmpv6,
  52. payload_len: 8,
  53. },
  54. IpPayload::Icmpv6(Icmpv6Repr::EchoRequest {
  55. ident: 0,
  56. seq_no: 0,
  57. data: b"",
  58. })
  59. ))
  60. );
  61. let (mut iface, mut sockets, _device) = setup(medium);
  62. // Add a route to the interface, otherwise, we don't know if the packet is routed localy.
  63. iface.routes_mut().update(|routes| {
  64. routes
  65. .push(crate::iface::Route {
  66. cidr: IpCidr::Ipv6(Ipv6Cidr::new(
  67. Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 0),
  68. 64,
  69. )),
  70. via_router: IpAddress::Ipv6(Ipv6Address::from_parts(&[
  71. 0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001,
  72. ])),
  73. preferred_until: None,
  74. expires_at: None,
  75. })
  76. .unwrap();
  77. });
  78. assert_eq!(
  79. iface.inner.process_ipv6(
  80. &mut sockets,
  81. PacketMeta::default(),
  82. HardwareAddress::default(),
  83. &Ipv6Packet::new_checked(&data[..]).unwrap()
  84. ),
  85. None
  86. );
  87. // Accept any IP:
  88. iface.set_any_ip(true);
  89. assert!(iface
  90. .inner
  91. .process_ipv6(
  92. &mut sockets,
  93. PacketMeta::default(),
  94. HardwareAddress::default(),
  95. &Ipv6Packet::new_checked(&data[..]).unwrap()
  96. )
  97. .is_some());
  98. }
  99. #[rstest]
  100. #[case::ip(Medium::Ip)]
  101. #[cfg(feature = "medium-ip")]
  102. #[case::ethernet(Medium::Ethernet)]
  103. #[cfg(feature = "medium-ethernet")]
  104. #[case::ieee802154(Medium::Ieee802154)]
  105. #[cfg(feature = "medium-ieee802154")]
  106. fn multicast_source_address(#[case] medium: Medium) {
  107. let data = [
  108. 0x60, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc, 0x40, 0xff, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  109. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  110. 0x0, 0x0, 0x0, 0x0, 0x1,
  111. ];
  112. let response = None;
  113. let (mut iface, mut sockets, _device) = setup(medium);
  114. assert_eq!(
  115. iface.inner.process_ipv6(
  116. &mut sockets,
  117. PacketMeta::default(),
  118. HardwareAddress::default(),
  119. &Ipv6Packet::new_checked(&data[..]).unwrap()
  120. ),
  121. response
  122. );
  123. }
  124. #[rstest]
  125. #[case::ip(Medium::Ip)]
  126. #[cfg(feature = "medium-ip")]
  127. #[case::ethernet(Medium::Ethernet)]
  128. #[cfg(feature = "medium-ethernet")]
  129. #[case::ieee802154(Medium::Ieee802154)]
  130. #[cfg(feature = "medium-ieee802154")]
  131. fn hop_by_hop_skip_with_icmp(#[case] medium: Medium) {
  132. // The following contains:
  133. // - IPv6 header
  134. // - Hop-by-hop, with options:
  135. // - PADN (skipped)
  136. // - Unknown option (skipped)
  137. // - ICMP echo request
  138. let data = [
  139. 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  140. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  141. 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0x1, 0x0, 0xf, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
  142. 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  143. ];
  144. let response = Some(Packet::new_ipv6(
  145. Ipv6Repr {
  146. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  147. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  148. hop_limit: 64,
  149. next_header: IpProtocol::Icmpv6,
  150. payload_len: 19,
  151. },
  152. IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
  153. ident: 42,
  154. seq_no: 420,
  155. data: b"Lorem Ipsum",
  156. }),
  157. ));
  158. let (mut iface, mut sockets, _device) = setup(medium);
  159. assert_eq!(
  160. iface.inner.process_ipv6(
  161. &mut sockets,
  162. PacketMeta::default(),
  163. HardwareAddress::default(),
  164. &Ipv6Packet::new_checked(&data[..]).unwrap()
  165. ),
  166. response
  167. );
  168. }
  169. #[rstest]
  170. #[case::ip(Medium::Ip)]
  171. #[cfg(feature = "medium-ip")]
  172. #[case::ethernet(Medium::Ethernet)]
  173. #[cfg(feature = "medium-ethernet")]
  174. #[case::ieee802154(Medium::Ieee802154)]
  175. #[cfg(feature = "medium-ieee802154")]
  176. fn hop_by_hop_discard_with_icmp(#[case] medium: Medium) {
  177. // The following contains:
  178. // - IPv6 header
  179. // - Hop-by-hop, with options:
  180. // - PADN (skipped)
  181. // - Unknown option (discard)
  182. // - ICMP echo request
  183. let data = [
  184. 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  185. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  186. 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0x1, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
  187. 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  188. ];
  189. let response = None;
  190. let (mut iface, mut sockets, _device) = setup(medium);
  191. assert_eq!(
  192. iface.inner.process_ipv6(
  193. &mut sockets,
  194. PacketMeta::default(),
  195. HardwareAddress::default(),
  196. &Ipv6Packet::new_checked(&data[..]).unwrap()
  197. ),
  198. response
  199. );
  200. }
  201. #[rstest]
  202. #[case::ip(Medium::Ip)]
  203. #[cfg(feature = "medium-ip")]
  204. fn hop_by_hop_discard_param_problem(#[case] medium: Medium) {
  205. // The following contains:
  206. // - IPv6 header
  207. // - Hop-by-hop, with options:
  208. // - PADN (skipped)
  209. // - Unknown option (discard + ParamProblem)
  210. // - ICMP echo request
  211. let data = [
  212. 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  213. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  214. 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0xC0, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
  215. 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  216. ];
  217. let response = Some(Packet::new_ipv6(
  218. Ipv6Repr {
  219. src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1),
  220. dst_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
  221. next_header: IpProtocol::Icmpv6,
  222. payload_len: 75,
  223. hop_limit: 64,
  224. },
  225. IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
  226. reason: Icmpv6ParamProblem::UnrecognizedOption,
  227. pointer: 40,
  228. header: Ipv6Repr {
  229. src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
  230. dst_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1),
  231. next_header: IpProtocol::HopByHop,
  232. payload_len: 27,
  233. hop_limit: 64,
  234. },
  235. data: &[
  236. 0x3a, 0x0, 0xC0, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88, 0x0, 0x2a, 0x1,
  237. 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  238. ],
  239. }),
  240. ));
  241. let (mut iface, mut sockets, _device) = setup(medium);
  242. assert_eq!(
  243. iface.inner.process_ipv6(
  244. &mut sockets,
  245. PacketMeta::default(),
  246. HardwareAddress::default(),
  247. &Ipv6Packet::new_checked(&data[..]).unwrap()
  248. ),
  249. response
  250. );
  251. }
  252. #[rstest]
  253. #[case::ip(Medium::Ip)]
  254. #[cfg(feature = "medium-ip")]
  255. fn hop_by_hop_discard_with_multicast(#[case] medium: Medium) {
  256. // The following contains:
  257. // - IPv6 header
  258. // - Hop-by-hop, with options:
  259. // - PADN (skipped)
  260. // - Unknown option (discard (0b11) + ParamProblem)
  261. // - ICMP echo request
  262. //
  263. // In this case, even if the destination address is a multicast address, an ICMPv6 ParamProblem
  264. // should be transmitted.
  265. let data = [
  266. 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  267. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xff, 0x02, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  268. 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0x80, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
  269. 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  270. ];
  271. let response = Some(Packet::new_ipv6(
  272. Ipv6Repr {
  273. src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1),
  274. dst_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
  275. next_header: IpProtocol::Icmpv6,
  276. payload_len: 75,
  277. hop_limit: 64,
  278. },
  279. IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
  280. reason: Icmpv6ParamProblem::UnrecognizedOption,
  281. pointer: 40,
  282. header: Ipv6Repr {
  283. src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
  284. dst_addr: Ipv6Address::new(0xff02, 0, 0, 0, 0, 0, 0, 1),
  285. next_header: IpProtocol::HopByHop,
  286. payload_len: 27,
  287. hop_limit: 64,
  288. },
  289. data: &[
  290. 0x3a, 0x0, 0x80, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88, 0x0, 0x2a, 0x1,
  291. 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  292. ],
  293. }),
  294. ));
  295. let (mut iface, mut sockets, _device) = setup(medium);
  296. assert_eq!(
  297. iface.inner.process_ipv6(
  298. &mut sockets,
  299. PacketMeta::default(),
  300. HardwareAddress::default(),
  301. &Ipv6Packet::new_checked(&data[..]).unwrap()
  302. ),
  303. response
  304. );
  305. }
  306. #[rstest]
  307. #[case::ip(Medium::Ip)]
  308. #[cfg(feature = "medium-ip")]
  309. #[case::ethernet(Medium::Ethernet)]
  310. #[cfg(feature = "medium-ethernet")]
  311. #[case::ieee802154(Medium::Ieee802154)]
  312. #[cfg(feature = "medium-ieee802154")]
  313. fn imcp_empty_echo_request(#[case] medium: Medium) {
  314. let data = [
  315. 0x60, 0x0, 0x0, 0x0, 0x0, 0x8, 0x3a, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  316. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  317. 0x0, 0x0, 0x0, 0x0, 0x1, 0x80, 0x0, 0x84, 0x3c, 0x0, 0x0, 0x0, 0x0,
  318. ];
  319. assert_eq!(
  320. parse_ipv6(&data),
  321. Ok(Packet::new_ipv6(
  322. Ipv6Repr {
  323. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  324. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  325. hop_limit: 64,
  326. next_header: IpProtocol::Icmpv6,
  327. payload_len: 8,
  328. },
  329. IpPayload::Icmpv6(Icmpv6Repr::EchoRequest {
  330. ident: 0,
  331. seq_no: 0,
  332. data: b"",
  333. })
  334. ))
  335. );
  336. let response = Some(Packet::new_ipv6(
  337. Ipv6Repr {
  338. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  339. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  340. hop_limit: 64,
  341. next_header: IpProtocol::Icmpv6,
  342. payload_len: 8,
  343. },
  344. IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
  345. ident: 0,
  346. seq_no: 0,
  347. data: b"",
  348. }),
  349. ));
  350. let (mut iface, mut sockets, _device) = setup(medium);
  351. assert_eq!(
  352. iface.inner.process_ipv6(
  353. &mut sockets,
  354. PacketMeta::default(),
  355. HardwareAddress::default(),
  356. &Ipv6Packet::new_checked(&data[..]).unwrap()
  357. ),
  358. response
  359. );
  360. }
  361. #[rstest]
  362. #[case::ip(Medium::Ip)]
  363. #[cfg(feature = "medium-ip")]
  364. #[case::ethernet(Medium::Ethernet)]
  365. #[cfg(feature = "medium-ethernet")]
  366. #[case::ieee802154(Medium::Ieee802154)]
  367. #[cfg(feature = "medium-ieee802154")]
  368. fn icmp_echo_request(#[case] medium: Medium) {
  369. let data = [
  370. 0x60, 0x0, 0x0, 0x0, 0x0, 0x13, 0x3a, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  371. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  372. 0x0, 0x0, 0x0, 0x0, 0x1, 0x80, 0x0, 0x2c, 0x88, 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72,
  373. 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  374. ];
  375. assert_eq!(
  376. parse_ipv6(&data),
  377. Ok(Packet::new_ipv6(
  378. Ipv6Repr {
  379. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  380. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  381. hop_limit: 64,
  382. next_header: IpProtocol::Icmpv6,
  383. payload_len: 19,
  384. },
  385. IpPayload::Icmpv6(Icmpv6Repr::EchoRequest {
  386. ident: 42,
  387. seq_no: 420,
  388. data: b"Lorem Ipsum",
  389. })
  390. ))
  391. );
  392. let response = Some(Packet::new_ipv6(
  393. Ipv6Repr {
  394. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  395. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  396. hop_limit: 64,
  397. next_header: IpProtocol::Icmpv6,
  398. payload_len: 19,
  399. },
  400. IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
  401. ident: 42,
  402. seq_no: 420,
  403. data: b"Lorem Ipsum",
  404. }),
  405. ));
  406. let (mut iface, mut sockets, _device) = setup(medium);
  407. assert_eq!(
  408. iface.inner.process_ipv6(
  409. &mut sockets,
  410. PacketMeta::default(),
  411. HardwareAddress::default(),
  412. &Ipv6Packet::new_checked(&data[..]).unwrap()
  413. ),
  414. response
  415. );
  416. }
  417. #[rstest]
  418. #[case::ip(Medium::Ip)]
  419. #[cfg(feature = "medium-ip")]
  420. #[case::ethernet(Medium::Ethernet)]
  421. #[cfg(feature = "medium-ethernet")]
  422. #[case::ieee802154(Medium::Ieee802154)]
  423. #[cfg(feature = "medium-ieee802154")]
  424. fn icmp_echo_reply_as_input(#[case] medium: Medium) {
  425. let data = [
  426. 0x60, 0x0, 0x0, 0x0, 0x0, 0x13, 0x3a, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  427. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  428. 0x0, 0x0, 0x0, 0x0, 0x1, 0x81, 0x0, 0x2d, 0x56, 0x0, 0x0, 0x0, 0x0, 0x4c, 0x6f, 0x72, 0x65,
  429. 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  430. ];
  431. assert_eq!(
  432. parse_ipv6(&data),
  433. Ok(Packet::new_ipv6(
  434. Ipv6Repr {
  435. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  436. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  437. hop_limit: 64,
  438. next_header: IpProtocol::Icmpv6,
  439. payload_len: 19,
  440. },
  441. IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
  442. ident: 0,
  443. seq_no: 0,
  444. data: b"Lorem Ipsum",
  445. })
  446. ))
  447. );
  448. let response = None;
  449. let (mut iface, mut sockets, _device) = setup(medium);
  450. assert_eq!(
  451. iface.inner.process_ipv6(
  452. &mut sockets,
  453. PacketMeta::default(),
  454. HardwareAddress::default(),
  455. &Ipv6Packet::new_checked(&data[..]).unwrap()
  456. ),
  457. response
  458. );
  459. }
  460. #[rstest]
  461. #[case::ip(Medium::Ip)]
  462. #[cfg(feature = "medium-ip")]
  463. #[case::ethernet(Medium::Ethernet)]
  464. #[cfg(feature = "medium-ethernet")]
  465. #[case::ieee802154(Medium::Ieee802154)]
  466. #[cfg(feature = "medium-ieee802154")]
  467. fn unknown_proto_with_multicast_dst_address(#[case] medium: Medium) {
  468. let data = [
  469. 0x60, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  470. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xff, 0x2, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  471. 0x0, 0x0, 0x0, 0x0, 0x1,
  472. ];
  473. let response = Some(Packet::new_ipv6(
  474. Ipv6Repr {
  475. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  476. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  477. hop_limit: 64,
  478. next_header: IpProtocol::Icmpv6,
  479. payload_len: 48,
  480. },
  481. IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
  482. reason: Icmpv6ParamProblem::UnrecognizedNxtHdr,
  483. pointer: 40,
  484. header: Ipv6Repr {
  485. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  486. dst_addr: Ipv6Address::from_parts(&[0xff02, 0, 0, 0, 0, 0, 0, 0x0001]),
  487. hop_limit: 64,
  488. next_header: IpProtocol::Unknown(0x0c),
  489. payload_len: 0,
  490. },
  491. data: &[],
  492. }),
  493. ));
  494. let (mut iface, mut sockets, _device) = setup(medium);
  495. assert_eq!(
  496. iface.inner.process_ipv6(
  497. &mut sockets,
  498. PacketMeta::default(),
  499. HardwareAddress::default(),
  500. &Ipv6Packet::new_checked(&data[..]).unwrap()
  501. ),
  502. response
  503. );
  504. }
  505. #[rstest]
  506. #[case::ip(Medium::Ip)]
  507. #[cfg(feature = "medium-ip")]
  508. #[case::ethernet(Medium::Ethernet)]
  509. #[cfg(feature = "medium-ethernet")]
  510. #[case::ieee802154(Medium::Ieee802154)]
  511. #[cfg(feature = "medium-ieee802154")]
  512. fn unknown_proto(#[case] medium: Medium) {
  513. // Since the destination address is multicast, we should answer with an ICMPv6 message.
  514. let data = [
  515. 0x60, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  516. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  517. 0x0, 0x0, 0x0, 0x0, 0x1,
  518. ];
  519. let response = Some(Packet::new_ipv6(
  520. Ipv6Repr {
  521. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  522. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  523. hop_limit: 64,
  524. next_header: IpProtocol::Icmpv6,
  525. payload_len: 48,
  526. },
  527. IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
  528. reason: Icmpv6ParamProblem::UnrecognizedNxtHdr,
  529. pointer: 40,
  530. header: Ipv6Repr {
  531. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  532. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  533. hop_limit: 64,
  534. next_header: IpProtocol::Unknown(0x0c),
  535. payload_len: 0,
  536. },
  537. data: &[],
  538. }),
  539. ));
  540. let (mut iface, mut sockets, _device) = setup(medium);
  541. assert_eq!(
  542. iface.inner.process_ipv6(
  543. &mut sockets,
  544. PacketMeta::default(),
  545. HardwareAddress::default(),
  546. &Ipv6Packet::new_checked(&data[..]).unwrap()
  547. ),
  548. response
  549. );
  550. }
  551. #[rstest]
  552. #[case::ethernet(Medium::Ethernet)]
  553. #[cfg(feature = "medium-ethernet")]
  554. fn ndsic_neighbor_advertisement_ethernet(#[case] medium: Medium) {
  555. let data = [
  556. 0x60, 0x0, 0x0, 0x0, 0x0, 0x20, 0x3a, 0xff, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  557. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  558. 0x0, 0x0, 0x0, 0x0, 0x1, 0x88, 0x0, 0x3b, 0x9f, 0x40, 0x0, 0x0, 0x0, 0xfe, 0x80, 0x0, 0x0,
  559. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0x2, 0x1, 0x0, 0x0, 0x0, 0x0,
  560. 0x0, 0x1,
  561. ];
  562. assert_eq!(
  563. parse_ipv6(&data),
  564. Ok(Packet::new_ipv6(
  565. Ipv6Repr {
  566. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  567. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  568. hop_limit: 255,
  569. next_header: IpProtocol::Icmpv6,
  570. payload_len: 32,
  571. },
  572. IpPayload::Icmpv6(Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
  573. flags: NdiscNeighborFlags::SOLICITED,
  574. target_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0, 0, 0, 0x0002]),
  575. lladdr: Some(RawHardwareAddress::from_bytes(&[0, 0, 0, 0, 0, 1])),
  576. }))
  577. ))
  578. );
  579. let response = None;
  580. let (mut iface, mut sockets, _device) = setup(medium);
  581. assert_eq!(
  582. iface.inner.process_ipv6(
  583. &mut sockets,
  584. PacketMeta::default(),
  585. HardwareAddress::default(),
  586. &Ipv6Packet::new_checked(&data[..]).unwrap()
  587. ),
  588. response
  589. );
  590. assert_eq!(
  591. iface.inner.neighbor_cache.lookup(
  592. &IpAddress::Ipv6(Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002])),
  593. iface.inner.now,
  594. ),
  595. NeighborAnswer::Found(HardwareAddress::Ethernet(EthernetAddress::from_bytes(&[
  596. 0, 0, 0, 0, 0, 1
  597. ]))),
  598. );
  599. }
  600. #[rstest]
  601. #[case::ethernet(Medium::Ethernet)]
  602. #[cfg(feature = "medium-ethernet")]
  603. fn ndsic_neighbor_advertisement_ethernet_multicast_addr(#[case] medium: Medium) {
  604. let data = [
  605. 0x60, 0x0, 0x0, 0x0, 0x0, 0x20, 0x3a, 0xff, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  606. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  607. 0x0, 0x0, 0x0, 0x0, 0x1, 0x88, 0x0, 0x3b, 0xa0, 0x40, 0x0, 0x0, 0x0, 0xfe, 0x80, 0x0, 0x0,
  608. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0x2, 0x1, 0xff, 0xff, 0xff,
  609. 0xff, 0xff, 0xff,
  610. ];
  611. assert_eq!(
  612. parse_ipv6(&data),
  613. Ok(Packet::new_ipv6(
  614. Ipv6Repr {
  615. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  616. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  617. hop_limit: 255,
  618. next_header: IpProtocol::Icmpv6,
  619. payload_len: 32,
  620. },
  621. IpPayload::Icmpv6(Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
  622. flags: NdiscNeighborFlags::SOLICITED,
  623. target_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0, 0, 0, 0x0002]),
  624. lladdr: Some(RawHardwareAddress::from_bytes(&[
  625. 0xff, 0xff, 0xff, 0xff, 0xff, 0xff
  626. ])),
  627. }))
  628. ))
  629. );
  630. let response = None;
  631. let (mut iface, mut sockets, _device) = setup(medium);
  632. assert_eq!(
  633. iface.inner.process_ipv6(
  634. &mut sockets,
  635. PacketMeta::default(),
  636. HardwareAddress::default(),
  637. &Ipv6Packet::new_checked(&data[..]).unwrap()
  638. ),
  639. response
  640. );
  641. assert_eq!(
  642. iface.inner.neighbor_cache.lookup(
  643. &IpAddress::Ipv6(Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002])),
  644. iface.inner.now,
  645. ),
  646. NeighborAnswer::NotFound,
  647. );
  648. }
  649. #[rstest]
  650. #[case::ieee802154(Medium::Ieee802154)]
  651. #[cfg(feature = "medium-ieee802154")]
  652. fn ndsic_neighbor_advertisement_ieee802154(#[case] medium: Medium) {
  653. let data = [
  654. 0x60, 0x0, 0x0, 0x0, 0x0, 0x28, 0x3a, 0xff, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  655. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  656. 0x0, 0x0, 0x0, 0x0, 0x1, 0x88, 0x0, 0x3b, 0x96, 0x40, 0x0, 0x0, 0x0, 0xfe, 0x80, 0x0, 0x0,
  657. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0x2, 0x2, 0x0, 0x0, 0x0, 0x0,
  658. 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  659. ];
  660. assert_eq!(
  661. parse_ipv6(&data),
  662. Ok(Packet::new_ipv6(
  663. Ipv6Repr {
  664. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  665. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  666. hop_limit: 255,
  667. next_header: IpProtocol::Icmpv6,
  668. payload_len: 40,
  669. },
  670. IpPayload::Icmpv6(Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
  671. flags: NdiscNeighborFlags::SOLICITED,
  672. target_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0, 0, 0, 0x0002]),
  673. lladdr: Some(RawHardwareAddress::from_bytes(&[0, 0, 0, 0, 0, 0, 0, 1])),
  674. }))
  675. ))
  676. );
  677. let response = None;
  678. let (mut iface, mut sockets, _device) = setup(medium);
  679. assert_eq!(
  680. iface.inner.process_ipv6(
  681. &mut sockets,
  682. PacketMeta::default(),
  683. HardwareAddress::default(),
  684. &Ipv6Packet::new_checked(&data[..]).unwrap()
  685. ),
  686. response
  687. );
  688. assert_eq!(
  689. iface.inner.neighbor_cache.lookup(
  690. &IpAddress::Ipv6(Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002])),
  691. iface.inner.now,
  692. ),
  693. NeighborAnswer::Found(HardwareAddress::Ieee802154(Ieee802154Address::from_bytes(
  694. &[0, 0, 0, 0, 0, 0, 0, 1]
  695. ))),
  696. );
  697. }
  698. #[rstest]
  699. #[case(Medium::Ethernet)]
  700. #[cfg(feature = "medium-ethernet")]
  701. fn test_handle_valid_ndisc_request(#[case] medium: Medium) {
  702. let (mut iface, mut sockets, _device) = setup(medium);
  703. let mut eth_bytes = vec![0u8; 86];
  704. let local_ip_addr = Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1);
  705. let remote_ip_addr = Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2);
  706. let local_hw_addr = EthernetAddress([0x02, 0x02, 0x02, 0x02, 0x02, 0x02]);
  707. let remote_hw_addr = EthernetAddress([0x52, 0x54, 0x00, 0x00, 0x00, 0x00]);
  708. let solicit = Icmpv6Repr::Ndisc(NdiscRepr::NeighborSolicit {
  709. target_addr: local_ip_addr,
  710. lladdr: Some(remote_hw_addr.into()),
  711. });
  712. let ip_repr = IpRepr::Ipv6(Ipv6Repr {
  713. src_addr: remote_ip_addr,
  714. dst_addr: local_ip_addr.solicited_node(),
  715. next_header: IpProtocol::Icmpv6,
  716. hop_limit: 0xff,
  717. payload_len: solicit.buffer_len(),
  718. });
  719. let mut frame = EthernetFrame::new_unchecked(&mut eth_bytes);
  720. frame.set_dst_addr(EthernetAddress([0x33, 0x33, 0x00, 0x00, 0x00, 0x00]));
  721. frame.set_src_addr(remote_hw_addr);
  722. frame.set_ethertype(EthernetProtocol::Ipv6);
  723. ip_repr.emit(frame.payload_mut(), &ChecksumCapabilities::default());
  724. solicit.emit(
  725. &remote_ip_addr,
  726. &local_ip_addr.solicited_node(),
  727. &mut Icmpv6Packet::new_unchecked(&mut frame.payload_mut()[ip_repr.header_len()..]),
  728. &ChecksumCapabilities::default(),
  729. );
  730. let icmpv6_expected = Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
  731. flags: NdiscNeighborFlags::SOLICITED,
  732. target_addr: local_ip_addr,
  733. lladdr: Some(local_hw_addr.into()),
  734. });
  735. let ipv6_expected = Ipv6Repr {
  736. src_addr: local_ip_addr,
  737. dst_addr: remote_ip_addr,
  738. next_header: IpProtocol::Icmpv6,
  739. hop_limit: 0xff,
  740. payload_len: icmpv6_expected.buffer_len(),
  741. };
  742. // Ensure an Neighbor Solicitation triggers a Neighbor Advertisement
  743. assert_eq!(
  744. iface.inner.process_ethernet(
  745. &mut sockets,
  746. PacketMeta::default(),
  747. frame.into_inner(),
  748. &mut iface.fragments
  749. ),
  750. Some(EthernetPacket::Ip(Packet::new_ipv6(
  751. ipv6_expected,
  752. IpPayload::Icmpv6(icmpv6_expected)
  753. )))
  754. );
  755. // Ensure the address of the requester was entered in the cache
  756. assert_eq!(
  757. iface.inner.lookup_hardware_addr(
  758. MockTxToken,
  759. &IpAddress::Ipv6(remote_ip_addr),
  760. &mut iface.fragmenter,
  761. ),
  762. Ok((HardwareAddress::Ethernet(remote_hw_addr), MockTxToken))
  763. );
  764. }
  765. #[rstest]
  766. #[case(Medium::Ip)]
  767. #[cfg(feature = "medium-ip")]
  768. #[case(Medium::Ethernet)]
  769. #[cfg(feature = "medium-ethernet")]
  770. #[case(Medium::Ieee802154)]
  771. #[cfg(feature = "medium-ieee802154")]
  772. fn test_solicited_node_addrs(#[case] medium: Medium) {
  773. let (mut iface, _, _) = setup(medium);
  774. let mut new_addrs = heapless::Vec::<IpCidr, IFACE_MAX_ADDR_COUNT>::new();
  775. new_addrs
  776. .push(IpCidr::new(IpAddress::v6(0xfe80, 0, 0, 0, 1, 2, 0, 2), 64))
  777. .unwrap();
  778. new_addrs
  779. .push(IpCidr::new(
  780. IpAddress::v6(0xfe80, 0, 0, 0, 3, 4, 0, 0xffff),
  781. 64,
  782. ))
  783. .unwrap();
  784. iface.update_ip_addrs(|addrs| {
  785. new_addrs.extend(addrs.to_vec());
  786. *addrs = new_addrs;
  787. });
  788. assert!(iface
  789. .inner
  790. .has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0x0002)));
  791. assert!(iface
  792. .inner
  793. .has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0xffff)));
  794. assert!(!iface
  795. .inner
  796. .has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0x0003)));
  797. }
  798. #[rstest]
  799. #[case(Medium::Ip)]
  800. #[cfg(all(feature = "socket-udp", feature = "medium-ip"))]
  801. #[case(Medium::Ethernet)]
  802. #[cfg(all(feature = "socket-udp", feature = "medium-ethernet"))]
  803. #[case(Medium::Ieee802154)]
  804. #[cfg(all(feature = "socket-udp", feature = "medium-ieee802154"))]
  805. fn test_icmp_reply_size(#[case] medium: Medium) {
  806. use crate::wire::Icmpv6DstUnreachable;
  807. use crate::wire::IPV6_MIN_MTU as MIN_MTU;
  808. const MAX_PAYLOAD_LEN: usize = 1192;
  809. let (mut iface, mut sockets, _device) = setup(medium);
  810. let src_addr = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1);
  811. let dst_addr = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 2);
  812. // UDP packet that if not tructated will cause a icmp port unreachable reply
  813. // to exceed the minimum mtu bytes in length.
  814. let udp_repr = UdpRepr {
  815. src_port: 67,
  816. dst_port: 68,
  817. };
  818. let mut bytes = vec![0xff; udp_repr.header_len() + MAX_PAYLOAD_LEN];
  819. let mut packet = UdpPacket::new_unchecked(&mut bytes[..]);
  820. udp_repr.emit(
  821. &mut packet,
  822. &src_addr.into(),
  823. &dst_addr.into(),
  824. MAX_PAYLOAD_LEN,
  825. |buf| fill_slice(buf, 0x2a),
  826. &ChecksumCapabilities::default(),
  827. );
  828. let ip_repr = Ipv6Repr {
  829. src_addr,
  830. dst_addr,
  831. next_header: IpProtocol::Udp,
  832. hop_limit: 64,
  833. payload_len: udp_repr.header_len() + MAX_PAYLOAD_LEN,
  834. };
  835. let payload = packet.into_inner();
  836. let expected_icmp_repr = Icmpv6Repr::DstUnreachable {
  837. reason: Icmpv6DstUnreachable::PortUnreachable,
  838. header: ip_repr,
  839. data: &payload[..MAX_PAYLOAD_LEN],
  840. };
  841. let expected_ip_repr = Ipv6Repr {
  842. src_addr: dst_addr,
  843. dst_addr: src_addr,
  844. next_header: IpProtocol::Icmpv6,
  845. hop_limit: 64,
  846. payload_len: expected_icmp_repr.buffer_len(),
  847. };
  848. assert_eq!(
  849. expected_ip_repr.buffer_len() + expected_icmp_repr.buffer_len(),
  850. MIN_MTU
  851. );
  852. assert_eq!(
  853. iface.inner.process_udp(
  854. &mut sockets,
  855. PacketMeta::default(),
  856. false,
  857. ip_repr.into(),
  858. payload,
  859. ),
  860. Some(Packet::new_ipv6(
  861. expected_ip_repr,
  862. IpPayload::Icmpv6(expected_icmp_repr)
  863. ))
  864. );
  865. }
  866. #[cfg(feature = "medium-ip")]
  867. #[test]
  868. fn get_source_address() {
  869. let (mut iface, _, _) = setup(Medium::Ip);
  870. const OWN_LINK_LOCAL_ADDR: Ipv6Address = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1);
  871. const OWN_UNIQUE_LOCAL_ADDR1: Ipv6Address = Ipv6Address::new(0xfd00, 0, 0, 201, 1, 1, 1, 2);
  872. const OWN_UNIQUE_LOCAL_ADDR2: Ipv6Address = Ipv6Address::new(0xfd01, 0, 0, 201, 1, 1, 1, 2);
  873. const OWN_GLOBAL_UNICAST_ADDR1: Ipv6Address =
  874. Ipv6Address::new(0x2001, 0x0db8, 0x0003, 0, 0, 0, 0, 1);
  875. // List of addresses of the interface:
  876. // fe80::1/64
  877. // fd00::201:1:1:1:2/64
  878. // fd01::201:1:1:1:2/64
  879. // 2001:db8:3::1/64
  880. iface.update_ip_addrs(|addrs| {
  881. addrs.clear();
  882. addrs
  883. .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_LINK_LOCAL_ADDR, 64)))
  884. .unwrap();
  885. addrs
  886. .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_UNIQUE_LOCAL_ADDR1, 64)))
  887. .unwrap();
  888. addrs
  889. .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_UNIQUE_LOCAL_ADDR2, 64)))
  890. .unwrap();
  891. addrs
  892. .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_GLOBAL_UNICAST_ADDR1, 64)))
  893. .unwrap();
  894. });
  895. // List of addresses we test:
  896. // ::1 -> ::1
  897. // fe80::42 -> fe80::1
  898. // fd00::201:1:1:1:1 -> fd00::201:1:1:1:2
  899. // fd01::201:1:1:1:1 -> fd01::201:1:1:1:2
  900. // fd02::201:1:1:1:1 -> fd00::201:1:1:1:2 (because first added in the list)
  901. // ff02::1 -> fe80::1 (same scope)
  902. // 2001:db8:3::2 -> 2001:db8:3::1
  903. // 2001:db9:3::2 -> 2001:db8:3::1
  904. const LINK_LOCAL_ADDR: Ipv6Address = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 42);
  905. const UNIQUE_LOCAL_ADDR1: Ipv6Address = Ipv6Address::new(0xfd00, 0, 0, 201, 1, 1, 1, 1);
  906. const UNIQUE_LOCAL_ADDR2: Ipv6Address = Ipv6Address::new(0xfd01, 0, 0, 201, 1, 1, 1, 1);
  907. const UNIQUE_LOCAL_ADDR3: Ipv6Address = Ipv6Address::new(0xfd02, 0, 0, 201, 1, 1, 1, 1);
  908. const GLOBAL_UNICAST_ADDR1: Ipv6Address =
  909. Ipv6Address::new(0x2001, 0x0db8, 0x0003, 0, 0, 0, 0, 2);
  910. const GLOBAL_UNICAST_ADDR2: Ipv6Address =
  911. Ipv6Address::new(0x2001, 0x0db9, 0x0003, 0, 0, 0, 0, 2);
  912. assert_eq!(
  913. iface.inner.get_source_address_ipv6(&Ipv6Address::LOOPBACK),
  914. Ipv6Address::LOOPBACK
  915. );
  916. assert_eq!(
  917. iface.inner.get_source_address_ipv6(&LINK_LOCAL_ADDR),
  918. OWN_LINK_LOCAL_ADDR
  919. );
  920. assert_eq!(
  921. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
  922. OWN_UNIQUE_LOCAL_ADDR1
  923. );
  924. assert_eq!(
  925. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
  926. OWN_UNIQUE_LOCAL_ADDR2
  927. );
  928. assert_eq!(
  929. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
  930. OWN_UNIQUE_LOCAL_ADDR1
  931. );
  932. assert_eq!(
  933. iface
  934. .inner
  935. .get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
  936. OWN_LINK_LOCAL_ADDR
  937. );
  938. assert_eq!(
  939. iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
  940. OWN_GLOBAL_UNICAST_ADDR1
  941. );
  942. assert_eq!(
  943. iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
  944. OWN_GLOBAL_UNICAST_ADDR1
  945. );
  946. assert_eq!(
  947. iface.get_source_address_ipv6(&LINK_LOCAL_ADDR),
  948. OWN_LINK_LOCAL_ADDR
  949. );
  950. assert_eq!(
  951. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
  952. OWN_UNIQUE_LOCAL_ADDR1
  953. );
  954. assert_eq!(
  955. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
  956. OWN_UNIQUE_LOCAL_ADDR2
  957. );
  958. assert_eq!(
  959. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
  960. OWN_UNIQUE_LOCAL_ADDR1
  961. );
  962. assert_eq!(
  963. iface.get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
  964. OWN_LINK_LOCAL_ADDR
  965. );
  966. assert_eq!(
  967. iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
  968. OWN_GLOBAL_UNICAST_ADDR1
  969. );
  970. assert_eq!(
  971. iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
  972. OWN_GLOBAL_UNICAST_ADDR1
  973. );
  974. }
  975. #[cfg(feature = "medium-ip")]
  976. #[test]
  977. fn get_source_address_only_link_local() {
  978. let (mut iface, _, _) = setup(Medium::Ip);
  979. // List of addresses in the interface:
  980. // fe80::1/64
  981. const OWN_LINK_LOCAL_ADDR: Ipv6Address = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1);
  982. iface.update_ip_addrs(|ips| {
  983. ips.clear();
  984. ips.push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_LINK_LOCAL_ADDR, 64)))
  985. .unwrap();
  986. });
  987. // List of addresses we test:
  988. // ::1 -> ::1
  989. // fe80::42 -> fe80::1
  990. // fd00::201:1:1:1:1 -> fe80::1
  991. // fd01::201:1:1:1:1 -> fe80::1
  992. // fd02::201:1:1:1:1 -> fe80::1
  993. // ff02::1 -> fe80::1
  994. // 2001:db8:3::2 -> fe80::1
  995. // 2001:db9:3::2 -> fe80::1
  996. const LINK_LOCAL_ADDR: Ipv6Address = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 42);
  997. const UNIQUE_LOCAL_ADDR1: Ipv6Address = Ipv6Address::new(0xfd00, 0, 0, 201, 1, 1, 1, 1);
  998. const UNIQUE_LOCAL_ADDR2: Ipv6Address = Ipv6Address::new(0xfd01, 0, 0, 201, 1, 1, 1, 1);
  999. const UNIQUE_LOCAL_ADDR3: Ipv6Address = Ipv6Address::new(0xfd02, 0, 0, 201, 1, 1, 1, 1);
  1000. const GLOBAL_UNICAST_ADDR1: Ipv6Address =
  1001. Ipv6Address::new(0x2001, 0x0db8, 0x0003, 0, 0, 0, 0, 2);
  1002. const GLOBAL_UNICAST_ADDR2: Ipv6Address =
  1003. Ipv6Address::new(0x2001, 0x0db9, 0x0003, 0, 0, 0, 0, 2);
  1004. assert_eq!(
  1005. iface.inner.get_source_address_ipv6(&Ipv6Address::LOOPBACK),
  1006. Ipv6Address::LOOPBACK
  1007. );
  1008. assert_eq!(
  1009. iface.inner.get_source_address_ipv6(&LINK_LOCAL_ADDR),
  1010. OWN_LINK_LOCAL_ADDR
  1011. );
  1012. assert_eq!(
  1013. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
  1014. OWN_LINK_LOCAL_ADDR
  1015. );
  1016. assert_eq!(
  1017. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
  1018. OWN_LINK_LOCAL_ADDR
  1019. );
  1020. assert_eq!(
  1021. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
  1022. OWN_LINK_LOCAL_ADDR
  1023. );
  1024. assert_eq!(
  1025. iface
  1026. .inner
  1027. .get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
  1028. OWN_LINK_LOCAL_ADDR
  1029. );
  1030. assert_eq!(
  1031. iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
  1032. OWN_LINK_LOCAL_ADDR
  1033. );
  1034. assert_eq!(
  1035. iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
  1036. OWN_LINK_LOCAL_ADDR
  1037. );
  1038. assert_eq!(
  1039. iface.get_source_address_ipv6(&LINK_LOCAL_ADDR),
  1040. OWN_LINK_LOCAL_ADDR
  1041. );
  1042. assert_eq!(
  1043. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
  1044. OWN_LINK_LOCAL_ADDR
  1045. );
  1046. assert_eq!(
  1047. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
  1048. OWN_LINK_LOCAL_ADDR
  1049. );
  1050. assert_eq!(
  1051. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
  1052. OWN_LINK_LOCAL_ADDR
  1053. );
  1054. assert_eq!(
  1055. iface.get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
  1056. OWN_LINK_LOCAL_ADDR
  1057. );
  1058. assert_eq!(
  1059. iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
  1060. OWN_LINK_LOCAL_ADDR
  1061. );
  1062. assert_eq!(
  1063. iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
  1064. OWN_LINK_LOCAL_ADDR
  1065. );
  1066. }
  1067. #[cfg(feature = "medium-ip")]
  1068. #[test]
  1069. fn get_source_address_empty_interface() {
  1070. let (mut iface, _, _) = setup(Medium::Ip);
  1071. iface.update_ip_addrs(|ips| ips.clear());
  1072. // List of addresses we test:
  1073. // ::1 -> ::1
  1074. // fe80::42 -> ::1
  1075. // fd00::201:1:1:1:1 -> ::1
  1076. // fd01::201:1:1:1:1 -> ::1
  1077. // fd02::201:1:1:1:1 -> ::1
  1078. // ff02::1 -> ::1
  1079. // 2001:db8:3::2 -> ::1
  1080. // 2001:db9:3::2 -> ::1
  1081. const LINK_LOCAL_ADDR: Ipv6Address = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 42);
  1082. const UNIQUE_LOCAL_ADDR1: Ipv6Address = Ipv6Address::new(0xfd00, 0, 0, 201, 1, 1, 1, 1);
  1083. const UNIQUE_LOCAL_ADDR2: Ipv6Address = Ipv6Address::new(0xfd01, 0, 0, 201, 1, 1, 1, 1);
  1084. const UNIQUE_LOCAL_ADDR3: Ipv6Address = Ipv6Address::new(0xfd02, 0, 0, 201, 1, 1, 1, 1);
  1085. const GLOBAL_UNICAST_ADDR1: Ipv6Address =
  1086. Ipv6Address::new(0x2001, 0x0db8, 0x0003, 0, 0, 0, 0, 2);
  1087. const GLOBAL_UNICAST_ADDR2: Ipv6Address =
  1088. Ipv6Address::new(0x2001, 0x0db9, 0x0003, 0, 0, 0, 0, 2);
  1089. assert_eq!(
  1090. iface.inner.get_source_address_ipv6(&Ipv6Address::LOOPBACK),
  1091. Ipv6Address::LOOPBACK
  1092. );
  1093. assert_eq!(
  1094. iface.inner.get_source_address_ipv6(&LINK_LOCAL_ADDR),
  1095. Ipv6Address::LOOPBACK
  1096. );
  1097. assert_eq!(
  1098. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
  1099. Ipv6Address::LOOPBACK
  1100. );
  1101. assert_eq!(
  1102. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
  1103. Ipv6Address::LOOPBACK
  1104. );
  1105. assert_eq!(
  1106. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
  1107. Ipv6Address::LOOPBACK
  1108. );
  1109. assert_eq!(
  1110. iface
  1111. .inner
  1112. .get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
  1113. Ipv6Address::LOOPBACK
  1114. );
  1115. assert_eq!(
  1116. iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
  1117. Ipv6Address::LOOPBACK
  1118. );
  1119. assert_eq!(
  1120. iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
  1121. Ipv6Address::LOOPBACK
  1122. );
  1123. assert_eq!(
  1124. iface.get_source_address_ipv6(&LINK_LOCAL_ADDR),
  1125. Ipv6Address::LOOPBACK
  1126. );
  1127. assert_eq!(
  1128. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
  1129. Ipv6Address::LOOPBACK
  1130. );
  1131. assert_eq!(
  1132. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
  1133. Ipv6Address::LOOPBACK
  1134. );
  1135. assert_eq!(
  1136. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
  1137. Ipv6Address::LOOPBACK
  1138. );
  1139. assert_eq!(
  1140. iface.get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
  1141. Ipv6Address::LOOPBACK
  1142. );
  1143. assert_eq!(
  1144. iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
  1145. Ipv6Address::LOOPBACK
  1146. );
  1147. assert_eq!(
  1148. iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
  1149. Ipv6Address::LOOPBACK
  1150. );
  1151. }
  1152. #[rstest]
  1153. #[case(Medium::Ip)]
  1154. #[cfg(feature = "medium-ip")]
  1155. #[case(Medium::Ethernet)]
  1156. #[cfg(feature = "medium-ethernet")]
  1157. fn test_join_ipv6_multicast_group(#[case] medium: Medium) {
  1158. fn recv_icmpv6(
  1159. device: &mut crate::tests::TestingDevice,
  1160. timestamp: Instant,
  1161. ) -> std::vec::Vec<Ipv6Packet<std::vec::Vec<u8>>> {
  1162. let caps = device.capabilities();
  1163. recv_all(device, timestamp)
  1164. .iter()
  1165. .filter_map(|frame| {
  1166. let ipv6_packet = match caps.medium {
  1167. #[cfg(feature = "medium-ethernet")]
  1168. Medium::Ethernet => {
  1169. let eth_frame = EthernetFrame::new_checked(frame).ok()?;
  1170. Ipv6Packet::new_checked(eth_frame.payload()).ok()?
  1171. }
  1172. #[cfg(feature = "medium-ip")]
  1173. Medium::Ip => Ipv6Packet::new_checked(&frame[..]).ok()?,
  1174. #[cfg(feature = "medium-ieee802154")]
  1175. Medium::Ieee802154 => todo!(),
  1176. };
  1177. let buf = ipv6_packet.into_inner().to_vec();
  1178. Some(Ipv6Packet::new_unchecked(buf))
  1179. })
  1180. .collect::<std::vec::Vec<_>>()
  1181. }
  1182. let (mut iface, _sockets, mut device) = setup(medium);
  1183. let groups = [
  1184. Ipv6Address::from_parts(&[0xff05, 0, 0, 0, 0, 0, 0, 0x00fb]),
  1185. Ipv6Address::from_parts(&[0xff0e, 0, 0, 0, 0, 0, 0, 0x0017]),
  1186. ];
  1187. let timestamp = Instant::from_millis(0);
  1188. for &group in &groups {
  1189. iface
  1190. .join_multicast_group(&mut device, group, timestamp)
  1191. .unwrap();
  1192. assert!(iface.has_multicast_group(group));
  1193. }
  1194. assert!(iface.has_multicast_group(Ipv6Address::LINK_LOCAL_ALL_NODES));
  1195. let reports = recv_icmpv6(&mut device, timestamp);
  1196. assert_eq!(reports.len(), 2);
  1197. let caps = device.capabilities();
  1198. let checksum_caps = &caps.checksum;
  1199. for (&group_addr, ipv6_packet) in groups.iter().zip(reports) {
  1200. let buf = ipv6_packet.into_inner();
  1201. let ipv6_packet = Ipv6Packet::new_unchecked(buf.as_slice());
  1202. let _ipv6_repr = Ipv6Repr::parse(&ipv6_packet).unwrap();
  1203. let ip_payload = ipv6_packet.payload();
  1204. // The first 2 octets of this payload hold the next-header indicator and the
  1205. // Hop-by-Hop header length (in 8-octet words, minus 1). The remaining 6 octets
  1206. // hold the Hop-by-Hop PadN and Router Alert options.
  1207. let hbh_header = Ipv6HopByHopHeader::new_checked(&ip_payload[..8]).unwrap();
  1208. let hbh_repr = Ipv6HopByHopRepr::parse(&hbh_header).unwrap();
  1209. assert_eq!(hbh_repr.options.len(), 3);
  1210. assert_eq!(
  1211. hbh_repr.options[0],
  1212. Ipv6OptionRepr::Unknown {
  1213. type_: Ipv6OptionType::Unknown(IpProtocol::Icmpv6.into()),
  1214. length: 0,
  1215. data: &[],
  1216. }
  1217. );
  1218. assert_eq!(
  1219. hbh_repr.options[1],
  1220. Ipv6OptionRepr::RouterAlert(Ipv6OptionRouterAlert::MulticastListenerDiscovery)
  1221. );
  1222. assert_eq!(hbh_repr.options[2], Ipv6OptionRepr::PadN(0));
  1223. let icmpv6_packet =
  1224. Icmpv6Packet::new_checked(&ip_payload[hbh_repr.buffer_len()..]).unwrap();
  1225. let icmpv6_repr = Icmpv6Repr::parse(
  1226. &ipv6_packet.src_addr(),
  1227. &ipv6_packet.dst_addr(),
  1228. &icmpv6_packet,
  1229. checksum_caps,
  1230. )
  1231. .unwrap();
  1232. let record_data = match icmpv6_repr {
  1233. Icmpv6Repr::Mld(MldRepr::Report {
  1234. nr_mcast_addr_rcrds,
  1235. data,
  1236. }) => {
  1237. assert_eq!(nr_mcast_addr_rcrds, 1);
  1238. data
  1239. }
  1240. other => panic!("unexpected icmpv6_repr: {:?}", other),
  1241. };
  1242. let record = MldAddressRecord::new_checked(record_data).unwrap();
  1243. let record_repr = MldAddressRecordRepr::parse(&record).unwrap();
  1244. assert_eq!(
  1245. record_repr,
  1246. MldAddressRecordRepr {
  1247. num_srcs: 0,
  1248. mcast_addr: group_addr,
  1249. record_type: MldRecordType::ChangeToInclude,
  1250. aux_data_len: 0,
  1251. payload: &[],
  1252. }
  1253. );
  1254. iface
  1255. .leave_multicast_group(&mut device, group_addr, timestamp)
  1256. .unwrap();
  1257. assert!(!iface.has_multicast_group(group_addr));
  1258. }
  1259. }