ipv6.rs 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986
  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.into(),
  18. &ipv6.dst_addr.into(),
  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 multicast_source_address(#[case] medium: Medium) {
  37. let data = [
  38. 0x60, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc, 0x40, 0xff, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  39. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  40. 0x0, 0x0, 0x0, 0x0, 0x1,
  41. ];
  42. let response = None;
  43. let (mut iface, mut sockets, _device) = setup(medium);
  44. assert_eq!(
  45. iface.inner.process_ipv6(
  46. &mut sockets,
  47. PacketMeta::default(),
  48. &Ipv6Packet::new_checked(&data[..]).unwrap()
  49. ),
  50. response
  51. );
  52. }
  53. #[rstest]
  54. #[case::ip(Medium::Ip)]
  55. #[cfg(feature = "medium-ip")]
  56. #[case::ethernet(Medium::Ethernet)]
  57. #[cfg(feature = "medium-ethernet")]
  58. #[case::ieee802154(Medium::Ieee802154)]
  59. #[cfg(feature = "medium-ieee802154")]
  60. fn hop_by_hop_skip_with_icmp(#[case] medium: Medium) {
  61. // The following contains:
  62. // - IPv6 header
  63. // - Hop-by-hop, with options:
  64. // - PADN (skipped)
  65. // - Unknown option (skipped)
  66. // - ICMP echo request
  67. let data = [
  68. 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  69. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  70. 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0x1, 0x0, 0xf, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
  71. 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  72. ];
  73. let response = Some(Packet::new_ipv6(
  74. Ipv6Repr {
  75. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  76. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  77. hop_limit: 64,
  78. next_header: IpProtocol::Icmpv6,
  79. payload_len: 19,
  80. },
  81. IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
  82. ident: 42,
  83. seq_no: 420,
  84. data: b"Lorem Ipsum",
  85. }),
  86. ));
  87. let (mut iface, mut sockets, _device) = setup(medium);
  88. assert_eq!(
  89. iface.inner.process_ipv6(
  90. &mut sockets,
  91. PacketMeta::default(),
  92. &Ipv6Packet::new_checked(&data[..]).unwrap()
  93. ),
  94. response
  95. );
  96. }
  97. #[rstest]
  98. #[case::ip(Medium::Ip)]
  99. #[cfg(feature = "medium-ip")]
  100. #[case::ethernet(Medium::Ethernet)]
  101. #[cfg(feature = "medium-ethernet")]
  102. #[case::ieee802154(Medium::Ieee802154)]
  103. #[cfg(feature = "medium-ieee802154")]
  104. fn hop_by_hop_discard_with_icmp(#[case] medium: Medium) {
  105. // The following contains:
  106. // - IPv6 header
  107. // - Hop-by-hop, with options:
  108. // - PADN (skipped)
  109. // - Unknown option (discard)
  110. // - ICMP echo request
  111. let data = [
  112. 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  113. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  114. 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0x1, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
  115. 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  116. ];
  117. let response = None;
  118. let (mut iface, mut sockets, _device) = setup(medium);
  119. assert_eq!(
  120. iface.inner.process_ipv6(
  121. &mut sockets,
  122. PacketMeta::default(),
  123. &Ipv6Packet::new_checked(&data[..]).unwrap()
  124. ),
  125. response
  126. );
  127. }
  128. #[rstest]
  129. #[case::ip(Medium::Ip)]
  130. #[cfg(feature = "medium-ip")]
  131. fn hop_by_hop_discard_param_problem(#[case] medium: Medium) {
  132. // The following contains:
  133. // - IPv6 header
  134. // - Hop-by-hop, with options:
  135. // - PADN (skipped)
  136. // - Unknown option (discard + ParamProblem)
  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, 0xC0, 0x0, 0x40, 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::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1),
  147. dst_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
  148. next_header: IpProtocol::Icmpv6,
  149. payload_len: 75,
  150. hop_limit: 64,
  151. },
  152. IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
  153. reason: Icmpv6ParamProblem::UnrecognizedOption,
  154. pointer: 40,
  155. header: Ipv6Repr {
  156. src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
  157. dst_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1),
  158. next_header: IpProtocol::HopByHop,
  159. payload_len: 27,
  160. hop_limit: 64,
  161. },
  162. data: &[
  163. 0x3a, 0x0, 0xC0, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88, 0x0, 0x2a, 0x1,
  164. 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  165. ],
  166. }),
  167. ));
  168. let (mut iface, mut sockets, _device) = setup(medium);
  169. assert_eq!(
  170. iface.inner.process_ipv6(
  171. &mut sockets,
  172. PacketMeta::default(),
  173. &Ipv6Packet::new_checked(&data[..]).unwrap()
  174. ),
  175. response
  176. );
  177. }
  178. #[rstest]
  179. #[case::ip(Medium::Ip)]
  180. #[cfg(feature = "medium-ip")]
  181. fn hop_by_hop_discard_with_multicast(#[case] medium: Medium) {
  182. // The following contains:
  183. // - IPv6 header
  184. // - Hop-by-hop, with options:
  185. // - PADN (skipped)
  186. // - Unknown option (discard (0b11) + ParamProblem)
  187. // - ICMP echo request
  188. //
  189. // In this case, even if the destination address is a multicast address, an ICMPv6 ParamProblem
  190. // should be transmitted.
  191. let data = [
  192. 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  193. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xff, 0x02, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  194. 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0x80, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
  195. 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  196. ];
  197. let response = Some(Packet::new_ipv6(
  198. Ipv6Repr {
  199. src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1),
  200. dst_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
  201. next_header: IpProtocol::Icmpv6,
  202. payload_len: 75,
  203. hop_limit: 64,
  204. },
  205. IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
  206. reason: Icmpv6ParamProblem::UnrecognizedOption,
  207. pointer: 40,
  208. header: Ipv6Repr {
  209. src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
  210. dst_addr: Ipv6Address::new(0xff02, 0, 0, 0, 0, 0, 0, 1),
  211. next_header: IpProtocol::HopByHop,
  212. payload_len: 27,
  213. hop_limit: 64,
  214. },
  215. data: &[
  216. 0x3a, 0x0, 0x80, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88, 0x0, 0x2a, 0x1,
  217. 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  218. ],
  219. }),
  220. ));
  221. let (mut iface, mut sockets, _device) = setup(medium);
  222. assert_eq!(
  223. iface.inner.process_ipv6(
  224. &mut sockets,
  225. PacketMeta::default(),
  226. &Ipv6Packet::new_checked(&data[..]).unwrap()
  227. ),
  228. response
  229. );
  230. }
  231. #[rstest]
  232. #[case::ip(Medium::Ip)]
  233. #[cfg(feature = "medium-ip")]
  234. #[case::ethernet(Medium::Ethernet)]
  235. #[cfg(feature = "medium-ethernet")]
  236. #[case::ieee802154(Medium::Ieee802154)]
  237. #[cfg(feature = "medium-ieee802154")]
  238. fn imcp_empty_echo_request(#[case] medium: Medium) {
  239. let data = [
  240. 0x60, 0x0, 0x0, 0x0, 0x0, 0x8, 0x3a, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  241. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  242. 0x0, 0x0, 0x0, 0x0, 0x1, 0x80, 0x0, 0x84, 0x3c, 0x0, 0x0, 0x0, 0x0,
  243. ];
  244. assert_eq!(
  245. parse_ipv6(&data),
  246. Ok(Packet::new_ipv6(
  247. Ipv6Repr {
  248. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  249. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  250. hop_limit: 64,
  251. next_header: IpProtocol::Icmpv6,
  252. payload_len: 8,
  253. },
  254. IpPayload::Icmpv6(Icmpv6Repr::EchoRequest {
  255. ident: 0,
  256. seq_no: 0,
  257. data: b"",
  258. })
  259. ))
  260. );
  261. let response = Some(Packet::new_ipv6(
  262. Ipv6Repr {
  263. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  264. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  265. hop_limit: 64,
  266. next_header: IpProtocol::Icmpv6,
  267. payload_len: 8,
  268. },
  269. IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
  270. ident: 0,
  271. seq_no: 0,
  272. data: b"",
  273. }),
  274. ));
  275. let (mut iface, mut sockets, _device) = setup(medium);
  276. assert_eq!(
  277. iface.inner.process_ipv6(
  278. &mut sockets,
  279. PacketMeta::default(),
  280. &Ipv6Packet::new_checked(&data[..]).unwrap()
  281. ),
  282. response
  283. );
  284. }
  285. #[rstest]
  286. #[case::ip(Medium::Ip)]
  287. #[cfg(feature = "medium-ip")]
  288. #[case::ethernet(Medium::Ethernet)]
  289. #[cfg(feature = "medium-ethernet")]
  290. #[case::ieee802154(Medium::Ieee802154)]
  291. #[cfg(feature = "medium-ieee802154")]
  292. fn icmp_echo_request(#[case] medium: Medium) {
  293. let data = [
  294. 0x60, 0x0, 0x0, 0x0, 0x0, 0x13, 0x3a, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  295. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  296. 0x0, 0x0, 0x0, 0x0, 0x1, 0x80, 0x0, 0x2c, 0x88, 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72,
  297. 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  298. ];
  299. assert_eq!(
  300. parse_ipv6(&data),
  301. Ok(Packet::new_ipv6(
  302. Ipv6Repr {
  303. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  304. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  305. hop_limit: 64,
  306. next_header: IpProtocol::Icmpv6,
  307. payload_len: 19,
  308. },
  309. IpPayload::Icmpv6(Icmpv6Repr::EchoRequest {
  310. ident: 42,
  311. seq_no: 420,
  312. data: b"Lorem Ipsum",
  313. })
  314. ))
  315. );
  316. let response = Some(Packet::new_ipv6(
  317. Ipv6Repr {
  318. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  319. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  320. hop_limit: 64,
  321. next_header: IpProtocol::Icmpv6,
  322. payload_len: 19,
  323. },
  324. IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
  325. ident: 42,
  326. seq_no: 420,
  327. data: b"Lorem Ipsum",
  328. }),
  329. ));
  330. let (mut iface, mut sockets, _device) = setup(medium);
  331. assert_eq!(
  332. iface.inner.process_ipv6(
  333. &mut sockets,
  334. PacketMeta::default(),
  335. &Ipv6Packet::new_checked(&data[..]).unwrap()
  336. ),
  337. response
  338. );
  339. }
  340. #[rstest]
  341. #[case::ip(Medium::Ip)]
  342. #[cfg(feature = "medium-ip")]
  343. #[case::ethernet(Medium::Ethernet)]
  344. #[cfg(feature = "medium-ethernet")]
  345. #[case::ieee802154(Medium::Ieee802154)]
  346. #[cfg(feature = "medium-ieee802154")]
  347. fn icmp_echo_reply_as_input(#[case] medium: Medium) {
  348. let data = [
  349. 0x60, 0x0, 0x0, 0x0, 0x0, 0x13, 0x3a, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  350. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  351. 0x0, 0x0, 0x0, 0x0, 0x1, 0x81, 0x0, 0x2d, 0x56, 0x0, 0x0, 0x0, 0x0, 0x4c, 0x6f, 0x72, 0x65,
  352. 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
  353. ];
  354. assert_eq!(
  355. parse_ipv6(&data),
  356. Ok(Packet::new_ipv6(
  357. Ipv6Repr {
  358. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  359. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  360. hop_limit: 64,
  361. next_header: IpProtocol::Icmpv6,
  362. payload_len: 19,
  363. },
  364. IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
  365. ident: 0,
  366. seq_no: 0,
  367. data: b"Lorem Ipsum",
  368. })
  369. ))
  370. );
  371. let response = None;
  372. let (mut iface, mut sockets, _device) = setup(medium);
  373. assert_eq!(
  374. iface.inner.process_ipv6(
  375. &mut sockets,
  376. PacketMeta::default(),
  377. &Ipv6Packet::new_checked(&data[..]).unwrap()
  378. ),
  379. response
  380. );
  381. }
  382. #[rstest]
  383. #[case::ip(Medium::Ip)]
  384. #[cfg(feature = "medium-ip")]
  385. #[case::ethernet(Medium::Ethernet)]
  386. #[cfg(feature = "medium-ethernet")]
  387. #[case::ieee802154(Medium::Ieee802154)]
  388. #[cfg(feature = "medium-ieee802154")]
  389. fn unknown_proto_with_multicast_dst_address(#[case] medium: Medium) {
  390. let data = [
  391. 0x60, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  392. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xff, 0x2, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  393. 0x0, 0x0, 0x0, 0x0, 0x1,
  394. ];
  395. let response = Some(Packet::new_ipv6(
  396. Ipv6Repr {
  397. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  398. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  399. hop_limit: 64,
  400. next_header: IpProtocol::Icmpv6,
  401. payload_len: 48,
  402. },
  403. IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
  404. reason: Icmpv6ParamProblem::UnrecognizedNxtHdr,
  405. pointer: 40,
  406. header: Ipv6Repr {
  407. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  408. dst_addr: Ipv6Address::from_parts(&[0xff02, 0, 0, 0, 0, 0, 0, 0x0001]),
  409. hop_limit: 64,
  410. next_header: IpProtocol::Unknown(0x0c),
  411. payload_len: 0,
  412. },
  413. data: &[],
  414. }),
  415. ));
  416. let (mut iface, mut sockets, _device) = setup(medium);
  417. assert_eq!(
  418. iface.inner.process_ipv6(
  419. &mut sockets,
  420. PacketMeta::default(),
  421. &Ipv6Packet::new_checked(&data[..]).unwrap()
  422. ),
  423. response
  424. );
  425. }
  426. #[rstest]
  427. #[case::ip(Medium::Ip)]
  428. #[cfg(feature = "medium-ip")]
  429. #[case::ethernet(Medium::Ethernet)]
  430. #[cfg(feature = "medium-ethernet")]
  431. #[case::ieee802154(Medium::Ieee802154)]
  432. #[cfg(feature = "medium-ieee802154")]
  433. fn unknown_proto(#[case] medium: Medium) {
  434. // Since the destination address is multicast, we should answer with an ICMPv6 message.
  435. let data = [
  436. 0x60, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  437. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  438. 0x0, 0x0, 0x0, 0x0, 0x1,
  439. ];
  440. let response = Some(Packet::new_ipv6(
  441. Ipv6Repr {
  442. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  443. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  444. hop_limit: 64,
  445. next_header: IpProtocol::Icmpv6,
  446. payload_len: 48,
  447. },
  448. IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
  449. reason: Icmpv6ParamProblem::UnrecognizedNxtHdr,
  450. pointer: 40,
  451. header: Ipv6Repr {
  452. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  453. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  454. hop_limit: 64,
  455. next_header: IpProtocol::Unknown(0x0c),
  456. payload_len: 0,
  457. },
  458. data: &[],
  459. }),
  460. ));
  461. let (mut iface, mut sockets, _device) = setup(medium);
  462. assert_eq!(
  463. iface.inner.process_ipv6(
  464. &mut sockets,
  465. PacketMeta::default(),
  466. &Ipv6Packet::new_checked(&data[..]).unwrap()
  467. ),
  468. response
  469. );
  470. }
  471. #[rstest]
  472. #[case::ethernet(Medium::Ethernet)]
  473. #[cfg(feature = "medium-ethernet")]
  474. fn ndsic_neighbor_advertisement_ethernet(#[case] medium: Medium) {
  475. let data = [
  476. 0x60, 0x0, 0x0, 0x0, 0x0, 0x20, 0x3a, 0xff, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  477. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  478. 0x0, 0x0, 0x0, 0x0, 0x1, 0x88, 0x0, 0x3b, 0x9f, 0x40, 0x0, 0x0, 0x0, 0xfe, 0x80, 0x0, 0x0,
  479. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0x2, 0x1, 0x0, 0x0, 0x0, 0x0,
  480. 0x0, 0x1,
  481. ];
  482. assert_eq!(
  483. parse_ipv6(&data),
  484. Ok(Packet::new_ipv6(
  485. Ipv6Repr {
  486. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  487. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  488. hop_limit: 255,
  489. next_header: IpProtocol::Icmpv6,
  490. payload_len: 32,
  491. },
  492. IpPayload::Icmpv6(Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
  493. flags: NdiscNeighborFlags::SOLICITED,
  494. target_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0, 0, 0, 0x0002]),
  495. lladdr: Some(RawHardwareAddress::from_bytes(&[0, 0, 0, 0, 0, 1])),
  496. }))
  497. ))
  498. );
  499. let response = None;
  500. let (mut iface, mut sockets, _device) = setup(medium);
  501. assert_eq!(
  502. iface.inner.process_ipv6(
  503. &mut sockets,
  504. PacketMeta::default(),
  505. &Ipv6Packet::new_checked(&data[..]).unwrap()
  506. ),
  507. response
  508. );
  509. assert_eq!(
  510. iface.inner.neighbor_cache.lookup(
  511. &IpAddress::Ipv6(Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002])),
  512. iface.inner.now,
  513. ),
  514. NeighborAnswer::Found(HardwareAddress::Ethernet(EthernetAddress::from_bytes(&[
  515. 0, 0, 0, 0, 0, 1
  516. ]))),
  517. );
  518. }
  519. #[rstest]
  520. #[case::ethernet(Medium::Ethernet)]
  521. #[cfg(feature = "medium-ethernet")]
  522. fn ndsic_neighbor_advertisement_ethernet_multicast_addr(#[case] medium: Medium) {
  523. let data = [
  524. 0x60, 0x0, 0x0, 0x0, 0x0, 0x20, 0x3a, 0xff, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  525. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  526. 0x0, 0x0, 0x0, 0x0, 0x1, 0x88, 0x0, 0x3b, 0xa0, 0x40, 0x0, 0x0, 0x0, 0xfe, 0x80, 0x0, 0x0,
  527. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0x2, 0x1, 0xff, 0xff, 0xff,
  528. 0xff, 0xff, 0xff,
  529. ];
  530. assert_eq!(
  531. parse_ipv6(&data),
  532. Ok(Packet::new_ipv6(
  533. Ipv6Repr {
  534. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  535. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  536. hop_limit: 255,
  537. next_header: IpProtocol::Icmpv6,
  538. payload_len: 32,
  539. },
  540. IpPayload::Icmpv6(Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
  541. flags: NdiscNeighborFlags::SOLICITED,
  542. target_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0, 0, 0, 0x0002]),
  543. lladdr: Some(RawHardwareAddress::from_bytes(&[
  544. 0xff, 0xff, 0xff, 0xff, 0xff, 0xff
  545. ])),
  546. }))
  547. ))
  548. );
  549. let response = None;
  550. let (mut iface, mut sockets, _device) = setup(medium);
  551. assert_eq!(
  552. iface.inner.process_ipv6(
  553. &mut sockets,
  554. PacketMeta::default(),
  555. &Ipv6Packet::new_checked(&data[..]).unwrap()
  556. ),
  557. response
  558. );
  559. assert_eq!(
  560. iface.inner.neighbor_cache.lookup(
  561. &IpAddress::Ipv6(Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002])),
  562. iface.inner.now,
  563. ),
  564. NeighborAnswer::NotFound,
  565. );
  566. }
  567. #[rstest]
  568. #[case::ieee802154(Medium::Ieee802154)]
  569. #[cfg(feature = "medium-ieee802154")]
  570. fn ndsic_neighbor_advertisement_ieee802154(#[case] medium: Medium) {
  571. let data = [
  572. 0x60, 0x0, 0x0, 0x0, 0x0, 0x28, 0x3a, 0xff, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  573. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  574. 0x0, 0x0, 0x0, 0x0, 0x1, 0x88, 0x0, 0x3b, 0x96, 0x40, 0x0, 0x0, 0x0, 0xfe, 0x80, 0x0, 0x0,
  575. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0x2, 0x2, 0x0, 0x0, 0x0, 0x0,
  576. 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  577. ];
  578. assert_eq!(
  579. parse_ipv6(&data),
  580. Ok(Packet::new_ipv6(
  581. Ipv6Repr {
  582. src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
  583. dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
  584. hop_limit: 255,
  585. next_header: IpProtocol::Icmpv6,
  586. payload_len: 40,
  587. },
  588. IpPayload::Icmpv6(Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
  589. flags: NdiscNeighborFlags::SOLICITED,
  590. target_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0, 0, 0, 0x0002]),
  591. lladdr: Some(RawHardwareAddress::from_bytes(&[0, 0, 0, 0, 0, 0, 0, 1])),
  592. }))
  593. ))
  594. );
  595. let response = None;
  596. let (mut iface, mut sockets, _device) = setup(medium);
  597. assert_eq!(
  598. iface.inner.process_ipv6(
  599. &mut sockets,
  600. PacketMeta::default(),
  601. &Ipv6Packet::new_checked(&data[..]).unwrap()
  602. ),
  603. response
  604. );
  605. assert_eq!(
  606. iface.inner.neighbor_cache.lookup(
  607. &IpAddress::Ipv6(Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002])),
  608. iface.inner.now,
  609. ),
  610. NeighborAnswer::Found(HardwareAddress::Ieee802154(Ieee802154Address::from_bytes(
  611. &[0, 0, 0, 0, 0, 0, 0, 1]
  612. ))),
  613. );
  614. }
  615. #[rstest]
  616. #[case(Medium::Ethernet)]
  617. #[cfg(feature = "medium-ethernet")]
  618. fn test_handle_valid_ndisc_request(#[case] medium: Medium) {
  619. let (mut iface, mut sockets, _device) = setup(medium);
  620. let mut eth_bytes = vec![0u8; 86];
  621. let local_ip_addr = Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1);
  622. let remote_ip_addr = Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2);
  623. let local_hw_addr = EthernetAddress([0x02, 0x02, 0x02, 0x02, 0x02, 0x02]);
  624. let remote_hw_addr = EthernetAddress([0x52, 0x54, 0x00, 0x00, 0x00, 0x00]);
  625. let solicit = Icmpv6Repr::Ndisc(NdiscRepr::NeighborSolicit {
  626. target_addr: local_ip_addr,
  627. lladdr: Some(remote_hw_addr.into()),
  628. });
  629. let ip_repr = IpRepr::Ipv6(Ipv6Repr {
  630. src_addr: remote_ip_addr,
  631. dst_addr: local_ip_addr.solicited_node(),
  632. next_header: IpProtocol::Icmpv6,
  633. hop_limit: 0xff,
  634. payload_len: solicit.buffer_len(),
  635. });
  636. let mut frame = EthernetFrame::new_unchecked(&mut eth_bytes);
  637. frame.set_dst_addr(EthernetAddress([0x33, 0x33, 0x00, 0x00, 0x00, 0x00]));
  638. frame.set_src_addr(remote_hw_addr);
  639. frame.set_ethertype(EthernetProtocol::Ipv6);
  640. ip_repr.emit(frame.payload_mut(), &ChecksumCapabilities::default());
  641. solicit.emit(
  642. &remote_ip_addr.into(),
  643. &local_ip_addr.solicited_node().into(),
  644. &mut Icmpv6Packet::new_unchecked(&mut frame.payload_mut()[ip_repr.header_len()..]),
  645. &ChecksumCapabilities::default(),
  646. );
  647. let icmpv6_expected = Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
  648. flags: NdiscNeighborFlags::SOLICITED,
  649. target_addr: local_ip_addr,
  650. lladdr: Some(local_hw_addr.into()),
  651. });
  652. let ipv6_expected = Ipv6Repr {
  653. src_addr: local_ip_addr,
  654. dst_addr: remote_ip_addr,
  655. next_header: IpProtocol::Icmpv6,
  656. hop_limit: 0xff,
  657. payload_len: icmpv6_expected.buffer_len(),
  658. };
  659. // Ensure an Neighbor Solicitation triggers a Neighbor Advertisement
  660. assert_eq!(
  661. iface.inner.process_ethernet(
  662. &mut sockets,
  663. PacketMeta::default(),
  664. frame.into_inner(),
  665. &mut iface.fragments
  666. ),
  667. Some(EthernetPacket::Ip(Packet::new_ipv6(
  668. ipv6_expected,
  669. IpPayload::Icmpv6(icmpv6_expected)
  670. )))
  671. );
  672. // Ensure the address of the requester was entered in the cache
  673. assert_eq!(
  674. iface.inner.lookup_hardware_addr(
  675. MockTxToken,
  676. &IpAddress::Ipv6(local_ip_addr),
  677. &IpAddress::Ipv6(remote_ip_addr),
  678. &mut iface.fragmenter,
  679. ),
  680. Ok((HardwareAddress::Ethernet(remote_hw_addr), MockTxToken))
  681. );
  682. }
  683. #[rstest]
  684. #[case(Medium::Ip)]
  685. #[cfg(feature = "medium-ip")]
  686. #[case(Medium::Ethernet)]
  687. #[cfg(feature = "medium-ethernet")]
  688. #[case(Medium::Ieee802154)]
  689. #[cfg(feature = "medium-ieee802154")]
  690. fn test_solicited_node_addrs(#[case] medium: Medium) {
  691. let (mut iface, _, _) = setup(medium);
  692. let mut new_addrs = heapless::Vec::<IpCidr, IFACE_MAX_ADDR_COUNT>::new();
  693. new_addrs
  694. .push(IpCidr::new(IpAddress::v6(0xfe80, 0, 0, 0, 1, 2, 0, 2), 64))
  695. .unwrap();
  696. new_addrs
  697. .push(IpCidr::new(
  698. IpAddress::v6(0xfe80, 0, 0, 0, 3, 4, 0, 0xffff),
  699. 64,
  700. ))
  701. .unwrap();
  702. iface.update_ip_addrs(|addrs| {
  703. new_addrs.extend(addrs.to_vec());
  704. *addrs = new_addrs;
  705. });
  706. assert!(iface
  707. .inner
  708. .has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0x0002)));
  709. assert!(iface
  710. .inner
  711. .has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0xffff)));
  712. assert!(!iface
  713. .inner
  714. .has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0x0003)));
  715. }
  716. #[rstest]
  717. #[case(Medium::Ip)]
  718. #[cfg(all(feature = "socket-udp", feature = "medium-ip"))]
  719. #[case(Medium::Ethernet)]
  720. #[cfg(all(feature = "socket-udp", feature = "medium-ethernet"))]
  721. #[case(Medium::Ieee802154)]
  722. #[cfg(all(feature = "socket-udp", feature = "medium-ieee802154"))]
  723. fn test_icmp_reply_size(#[case] medium: Medium) {
  724. use crate::wire::Icmpv6DstUnreachable;
  725. use crate::wire::IPV6_MIN_MTU as MIN_MTU;
  726. const MAX_PAYLOAD_LEN: usize = 1192;
  727. let (mut iface, mut sockets, _device) = setup(medium);
  728. let src_addr = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1);
  729. let dst_addr = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 2);
  730. // UDP packet that if not tructated will cause a icmp port unreachable reply
  731. // to exceed the minimum mtu bytes in length.
  732. let udp_repr = UdpRepr {
  733. src_port: 67,
  734. dst_port: 68,
  735. };
  736. let mut bytes = vec![0xff; udp_repr.header_len() + MAX_PAYLOAD_LEN];
  737. let mut packet = UdpPacket::new_unchecked(&mut bytes[..]);
  738. udp_repr.emit(
  739. &mut packet,
  740. &src_addr.into(),
  741. &dst_addr.into(),
  742. MAX_PAYLOAD_LEN,
  743. |buf| fill_slice(buf, 0x2a),
  744. &ChecksumCapabilities::default(),
  745. );
  746. let ip_repr = Ipv6Repr {
  747. src_addr,
  748. dst_addr,
  749. next_header: IpProtocol::Udp,
  750. hop_limit: 64,
  751. payload_len: udp_repr.header_len() + MAX_PAYLOAD_LEN,
  752. };
  753. let payload = packet.into_inner();
  754. let expected_icmp_repr = Icmpv6Repr::DstUnreachable {
  755. reason: Icmpv6DstUnreachable::PortUnreachable,
  756. header: ip_repr,
  757. data: &payload[..MAX_PAYLOAD_LEN],
  758. };
  759. let expected_ip_repr = Ipv6Repr {
  760. src_addr: dst_addr,
  761. dst_addr: src_addr,
  762. next_header: IpProtocol::Icmpv6,
  763. hop_limit: 64,
  764. payload_len: expected_icmp_repr.buffer_len(),
  765. };
  766. assert_eq!(
  767. expected_ip_repr.buffer_len() + expected_icmp_repr.buffer_len(),
  768. MIN_MTU
  769. );
  770. assert_eq!(
  771. iface.inner.process_udp(
  772. &mut sockets,
  773. PacketMeta::default(),
  774. false,
  775. ip_repr.into(),
  776. payload,
  777. ),
  778. Some(Packet::new_ipv6(
  779. expected_ip_repr,
  780. IpPayload::Icmpv6(expected_icmp_repr)
  781. ))
  782. );
  783. }
  784. #[cfg(feature = "medium-ip")]
  785. #[test]
  786. fn get_source_address() {
  787. let (mut iface, _, _) = setup(Medium::Ip);
  788. const OWN_LINK_LOCAL_ADDR: Ipv6Address = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1);
  789. const OWN_UNIQUE_LOCAL_ADDR1: Ipv6Address = Ipv6Address::new(0xfd00, 0, 0, 201, 1, 1, 1, 2);
  790. const OWN_UNIQUE_LOCAL_ADDR2: Ipv6Address = Ipv6Address::new(0xfd01, 0, 0, 201, 1, 1, 1, 2);
  791. const OWN_GLOBAL_UNICAST_ADDR1: Ipv6Address =
  792. Ipv6Address::new(0x2001, 0x0db8, 0x0003, 0, 0, 0, 0, 1);
  793. // List of addresses of the interface:
  794. // fe80::1/64
  795. // fd00::201:1:1:1:2/64
  796. // fd01::201:1:1:1:2/64
  797. // 2001:db8:3::1/64
  798. // ::1/128
  799. // ::/128
  800. iface.update_ip_addrs(|addrs| {
  801. addrs.clear();
  802. addrs
  803. .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_LINK_LOCAL_ADDR, 64)))
  804. .unwrap();
  805. addrs
  806. .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_UNIQUE_LOCAL_ADDR1, 64)))
  807. .unwrap();
  808. addrs
  809. .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_UNIQUE_LOCAL_ADDR2, 64)))
  810. .unwrap();
  811. addrs
  812. .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_GLOBAL_UNICAST_ADDR1, 64)))
  813. .unwrap();
  814. // These should never be used:
  815. addrs
  816. .push(IpCidr::Ipv6(Ipv6Cidr::new(Ipv6Address::LOOPBACK, 128)))
  817. .unwrap();
  818. addrs
  819. .push(IpCidr::Ipv6(Ipv6Cidr::new(Ipv6Address::UNSPECIFIED, 128)))
  820. .unwrap();
  821. });
  822. // List of addresses we test:
  823. // fe80::42 -> fe80::1
  824. // fd00::201:1:1:1:1 -> fd00::201:1:1:1:2
  825. // fd01::201:1:1:1:1 -> fd01::201:1:1:1:2
  826. // fd02::201:1:1:1:1 -> fd00::201:1:1:1:2 (because first added in the list)
  827. // ff02::1 -> fe80::1 (same scope)
  828. // 2001:db8:3::2 -> 2001:db8:3::1
  829. // 2001:db9:3::2 -> 2001:db8:3::1
  830. const LINK_LOCAL_ADDR: Ipv6Address = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 42);
  831. const UNIQUE_LOCAL_ADDR1: Ipv6Address = Ipv6Address::new(0xfd00, 0, 0, 201, 1, 1, 1, 1);
  832. const UNIQUE_LOCAL_ADDR2: Ipv6Address = Ipv6Address::new(0xfd01, 0, 0, 201, 1, 1, 1, 1);
  833. const UNIQUE_LOCAL_ADDR3: Ipv6Address = Ipv6Address::new(0xfd02, 0, 0, 201, 1, 1, 1, 1);
  834. const GLOBAL_UNICAST_ADDR1: Ipv6Address =
  835. Ipv6Address::new(0x2001, 0x0db8, 0x0003, 0, 0, 0, 0, 2);
  836. const GLOBAL_UNICAST_ADDR2: Ipv6Address =
  837. Ipv6Address::new(0x2001, 0x0db9, 0x0003, 0, 0, 0, 0, 2);
  838. assert_eq!(
  839. iface.inner.get_source_address_ipv6(&LINK_LOCAL_ADDR),
  840. Some(OWN_LINK_LOCAL_ADDR)
  841. );
  842. assert_eq!(
  843. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
  844. Some(OWN_UNIQUE_LOCAL_ADDR1)
  845. );
  846. assert_eq!(
  847. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
  848. Some(OWN_UNIQUE_LOCAL_ADDR2)
  849. );
  850. assert_eq!(
  851. iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
  852. Some(OWN_UNIQUE_LOCAL_ADDR1)
  853. );
  854. assert_eq!(
  855. iface
  856. .inner
  857. .get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
  858. Some(OWN_LINK_LOCAL_ADDR)
  859. );
  860. assert_eq!(
  861. iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
  862. Some(OWN_GLOBAL_UNICAST_ADDR1)
  863. );
  864. assert_eq!(
  865. iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
  866. Some(OWN_GLOBAL_UNICAST_ADDR1)
  867. );
  868. assert_eq!(
  869. iface.get_source_address_ipv6(&LINK_LOCAL_ADDR),
  870. Some(OWN_LINK_LOCAL_ADDR)
  871. );
  872. assert_eq!(
  873. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
  874. Some(OWN_UNIQUE_LOCAL_ADDR1)
  875. );
  876. assert_eq!(
  877. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
  878. Some(OWN_UNIQUE_LOCAL_ADDR2)
  879. );
  880. assert_eq!(
  881. iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
  882. Some(OWN_UNIQUE_LOCAL_ADDR1)
  883. );
  884. assert_eq!(
  885. iface.get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
  886. Some(OWN_LINK_LOCAL_ADDR)
  887. );
  888. assert_eq!(
  889. iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
  890. Some(OWN_GLOBAL_UNICAST_ADDR1)
  891. );
  892. assert_eq!(
  893. iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
  894. Some(OWN_GLOBAL_UNICAST_ADDR1)
  895. );
  896. }