123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986 |
- use super::*;
- fn parse_ipv6(data: &[u8]) -> crate::wire::Result<Packet<'_>> {
- let ipv6_header = Ipv6Packet::new_checked(data)?;
- let ipv6 = Ipv6Repr::parse(&ipv6_header)?;
- match ipv6.next_header {
- IpProtocol::HopByHop => todo!(),
- IpProtocol::Icmp => todo!(),
- IpProtocol::Igmp => todo!(),
- IpProtocol::Tcp => todo!(),
- IpProtocol::Udp => todo!(),
- IpProtocol::Ipv6Route => todo!(),
- IpProtocol::Ipv6Frag => todo!(),
- IpProtocol::IpSecEsp => todo!(),
- IpProtocol::IpSecAh => todo!(),
- IpProtocol::Icmpv6 => {
- let icmp = Icmpv6Repr::parse(
- &ipv6.src_addr.into(),
- &ipv6.dst_addr.into(),
- &Icmpv6Packet::new_checked(ipv6_header.payload())?,
- &Default::default(),
- )?;
- Ok(Packet::new_ipv6(ipv6, IpPayload::Icmpv6(icmp)))
- }
- IpProtocol::Ipv6NoNxt => todo!(),
- IpProtocol::Ipv6Opts => todo!(),
- IpProtocol::Unknown(_) => todo!(),
- }
- }
- #[rstest]
- #[case::ip(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- #[case::ethernet(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- #[case::ieee802154(Medium::Ieee802154)]
- #[cfg(feature = "medium-ieee802154")]
- fn multicast_source_address(#[case] medium: Medium) {
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc, 0x40, 0xff, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1,
- ];
- let response = None;
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- }
- #[rstest]
- #[case::ip(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- #[case::ethernet(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- #[case::ieee802154(Medium::Ieee802154)]
- #[cfg(feature = "medium-ieee802154")]
- fn hop_by_hop_skip_with_icmp(#[case] medium: Medium) {
- // The following contains:
- // - IPv6 header
- // - Hop-by-hop, with options:
- // - PADN (skipped)
- // - Unknown option (skipped)
- // - ICMP echo request
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0x1, 0x0, 0xf, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
- 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
- ];
- let response = Some(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- hop_limit: 64,
- next_header: IpProtocol::Icmpv6,
- payload_len: 19,
- },
- IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
- ident: 42,
- seq_no: 420,
- data: b"Lorem Ipsum",
- }),
- ));
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- }
- #[rstest]
- #[case::ip(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- #[case::ethernet(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- #[case::ieee802154(Medium::Ieee802154)]
- #[cfg(feature = "medium-ieee802154")]
- fn hop_by_hop_discard_with_icmp(#[case] medium: Medium) {
- // The following contains:
- // - IPv6 header
- // - Hop-by-hop, with options:
- // - PADN (skipped)
- // - Unknown option (discard)
- // - ICMP echo request
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0x1, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
- 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
- ];
- let response = None;
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- }
- #[rstest]
- #[case::ip(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- fn hop_by_hop_discard_param_problem(#[case] medium: Medium) {
- // The following contains:
- // - IPv6 header
- // - Hop-by-hop, with options:
- // - PADN (skipped)
- // - Unknown option (discard + ParamProblem)
- // - ICMP echo request
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0xC0, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
- 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
- ];
- let response = Some(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1),
- dst_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
- next_header: IpProtocol::Icmpv6,
- payload_len: 75,
- hop_limit: 64,
- },
- IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
- reason: Icmpv6ParamProblem::UnrecognizedOption,
- pointer: 40,
- header: Ipv6Repr {
- src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
- dst_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1),
- next_header: IpProtocol::HopByHop,
- payload_len: 27,
- hop_limit: 64,
- },
- data: &[
- 0x3a, 0x0, 0xC0, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88, 0x0, 0x2a, 0x1,
- 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
- ],
- }),
- ));
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- }
- #[rstest]
- #[case::ip(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- fn hop_by_hop_discard_with_multicast(#[case] medium: Medium) {
- // The following contains:
- // - IPv6 header
- // - Hop-by-hop, with options:
- // - PADN (skipped)
- // - Unknown option (discard (0b11) + ParamProblem)
- // - ICMP echo request
- //
- // In this case, even if the destination address is a multicast address, an ICMPv6 ParamProblem
- // should be transmitted.
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x1b, 0x0, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xff, 0x02, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1, 0x3a, 0x0, 0x80, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88,
- 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
- ];
- let response = Some(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1),
- dst_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
- next_header: IpProtocol::Icmpv6,
- payload_len: 75,
- hop_limit: 64,
- },
- IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
- reason: Icmpv6ParamProblem::UnrecognizedOption,
- pointer: 40,
- header: Ipv6Repr {
- src_addr: Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2),
- dst_addr: Ipv6Address::new(0xff02, 0, 0, 0, 0, 0, 0, 1),
- next_header: IpProtocol::HopByHop,
- payload_len: 27,
- hop_limit: 64,
- },
- data: &[
- 0x3a, 0x0, 0x80, 0x0, 0x40, 0x0, 0x1, 0x0, 0x80, 0x0, 0x2c, 0x88, 0x0, 0x2a, 0x1,
- 0xa4, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
- ],
- }),
- ));
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- }
- #[rstest]
- #[case::ip(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- #[case::ethernet(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- #[case::ieee802154(Medium::Ieee802154)]
- #[cfg(feature = "medium-ieee802154")]
- fn imcp_empty_echo_request(#[case] medium: Medium) {
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x8, 0x3a, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1, 0x80, 0x0, 0x84, 0x3c, 0x0, 0x0, 0x0, 0x0,
- ];
- assert_eq!(
- parse_ipv6(&data),
- Ok(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- hop_limit: 64,
- next_header: IpProtocol::Icmpv6,
- payload_len: 8,
- },
- IpPayload::Icmpv6(Icmpv6Repr::EchoRequest {
- ident: 0,
- seq_no: 0,
- data: b"",
- })
- ))
- );
- let response = Some(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- hop_limit: 64,
- next_header: IpProtocol::Icmpv6,
- payload_len: 8,
- },
- IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
- ident: 0,
- seq_no: 0,
- data: b"",
- }),
- ));
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- }
- #[rstest]
- #[case::ip(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- #[case::ethernet(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- #[case::ieee802154(Medium::Ieee802154)]
- #[cfg(feature = "medium-ieee802154")]
- fn icmp_echo_request(#[case] medium: Medium) {
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x13, 0x3a, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1, 0x80, 0x0, 0x2c, 0x88, 0x0, 0x2a, 0x1, 0xa4, 0x4c, 0x6f, 0x72,
- 0x65, 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
- ];
- assert_eq!(
- parse_ipv6(&data),
- Ok(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- hop_limit: 64,
- next_header: IpProtocol::Icmpv6,
- payload_len: 19,
- },
- IpPayload::Icmpv6(Icmpv6Repr::EchoRequest {
- ident: 42,
- seq_no: 420,
- data: b"Lorem Ipsum",
- })
- ))
- );
- let response = Some(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- hop_limit: 64,
- next_header: IpProtocol::Icmpv6,
- payload_len: 19,
- },
- IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
- ident: 42,
- seq_no: 420,
- data: b"Lorem Ipsum",
- }),
- ));
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- }
- #[rstest]
- #[case::ip(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- #[case::ethernet(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- #[case::ieee802154(Medium::Ieee802154)]
- #[cfg(feature = "medium-ieee802154")]
- fn icmp_echo_reply_as_input(#[case] medium: Medium) {
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x13, 0x3a, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1, 0x81, 0x0, 0x2d, 0x56, 0x0, 0x0, 0x0, 0x0, 0x4c, 0x6f, 0x72, 0x65,
- 0x6d, 0x20, 0x49, 0x70, 0x73, 0x75, 0x6d,
- ];
- assert_eq!(
- parse_ipv6(&data),
- Ok(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- hop_limit: 64,
- next_header: IpProtocol::Icmpv6,
- payload_len: 19,
- },
- IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
- ident: 0,
- seq_no: 0,
- data: b"Lorem Ipsum",
- })
- ))
- );
- let response = None;
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- }
- #[rstest]
- #[case::ip(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- #[case::ethernet(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- #[case::ieee802154(Medium::Ieee802154)]
- #[cfg(feature = "medium-ieee802154")]
- fn unknown_proto_with_multicast_dst_address(#[case] medium: Medium) {
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xff, 0x2, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1,
- ];
- let response = Some(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- hop_limit: 64,
- next_header: IpProtocol::Icmpv6,
- payload_len: 48,
- },
- IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
- reason: Icmpv6ParamProblem::UnrecognizedNxtHdr,
- pointer: 40,
- header: Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- dst_addr: Ipv6Address::from_parts(&[0xff02, 0, 0, 0, 0, 0, 0, 0x0001]),
- hop_limit: 64,
- next_header: IpProtocol::Unknown(0x0c),
- payload_len: 0,
- },
- data: &[],
- }),
- ));
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- }
- #[rstest]
- #[case::ip(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- #[case::ethernet(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- #[case::ieee802154(Medium::Ieee802154)]
- #[cfg(feature = "medium-ieee802154")]
- fn unknown_proto(#[case] medium: Medium) {
- // Since the destination address is multicast, we should answer with an ICMPv6 message.
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc, 0x40, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1,
- ];
- let response = Some(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- hop_limit: 64,
- next_header: IpProtocol::Icmpv6,
- payload_len: 48,
- },
- IpPayload::Icmpv6(Icmpv6Repr::ParamProblem {
- reason: Icmpv6ParamProblem::UnrecognizedNxtHdr,
- pointer: 40,
- header: Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- hop_limit: 64,
- next_header: IpProtocol::Unknown(0x0c),
- payload_len: 0,
- },
- data: &[],
- }),
- ));
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- }
- #[rstest]
- #[case::ethernet(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- fn ndsic_neighbor_advertisement_ethernet(#[case] medium: Medium) {
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x20, 0x3a, 0xff, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1, 0x88, 0x0, 0x3b, 0x9f, 0x40, 0x0, 0x0, 0x0, 0xfe, 0x80, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0x2, 0x1, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x1,
- ];
- assert_eq!(
- parse_ipv6(&data),
- Ok(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- hop_limit: 255,
- next_header: IpProtocol::Icmpv6,
- payload_len: 32,
- },
- IpPayload::Icmpv6(Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
- flags: NdiscNeighborFlags::SOLICITED,
- target_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0, 0, 0, 0x0002]),
- lladdr: Some(RawHardwareAddress::from_bytes(&[0, 0, 0, 0, 0, 1])),
- }))
- ))
- );
- let response = None;
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- assert_eq!(
- iface.inner.neighbor_cache.lookup(
- &IpAddress::Ipv6(Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002])),
- iface.inner.now,
- ),
- NeighborAnswer::Found(HardwareAddress::Ethernet(EthernetAddress::from_bytes(&[
- 0, 0, 0, 0, 0, 1
- ]))),
- );
- }
- #[rstest]
- #[case::ethernet(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- fn ndsic_neighbor_advertisement_ethernet_multicast_addr(#[case] medium: Medium) {
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x20, 0x3a, 0xff, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1, 0x88, 0x0, 0x3b, 0xa0, 0x40, 0x0, 0x0, 0x0, 0xfe, 0x80, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0x2, 0x1, 0xff, 0xff, 0xff,
- 0xff, 0xff, 0xff,
- ];
- assert_eq!(
- parse_ipv6(&data),
- Ok(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- hop_limit: 255,
- next_header: IpProtocol::Icmpv6,
- payload_len: 32,
- },
- IpPayload::Icmpv6(Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
- flags: NdiscNeighborFlags::SOLICITED,
- target_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0, 0, 0, 0x0002]),
- lladdr: Some(RawHardwareAddress::from_bytes(&[
- 0xff, 0xff, 0xff, 0xff, 0xff, 0xff
- ])),
- }))
- ))
- );
- let response = None;
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- assert_eq!(
- iface.inner.neighbor_cache.lookup(
- &IpAddress::Ipv6(Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002])),
- iface.inner.now,
- ),
- NeighborAnswer::NotFound,
- );
- }
- #[rstest]
- #[case::ieee802154(Medium::Ieee802154)]
- #[cfg(feature = "medium-ieee802154")]
- fn ndsic_neighbor_advertisement_ieee802154(#[case] medium: Medium) {
- let data = [
- 0x60, 0x0, 0x0, 0x0, 0x0, 0x28, 0x3a, 0xff, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0xfd, 0xbe, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x1, 0x88, 0x0, 0x3b, 0x96, 0x40, 0x0, 0x0, 0x0, 0xfe, 0x80, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0x2, 0x2, 0x0, 0x0, 0x0, 0x0,
- 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
- ];
- assert_eq!(
- parse_ipv6(&data),
- Ok(Packet::new_ipv6(
- Ipv6Repr {
- src_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002]),
- dst_addr: Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001]),
- hop_limit: 255,
- next_header: IpProtocol::Icmpv6,
- payload_len: 40,
- },
- IpPayload::Icmpv6(Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
- flags: NdiscNeighborFlags::SOLICITED,
- target_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0, 0, 0, 0x0002]),
- lladdr: Some(RawHardwareAddress::from_bytes(&[0, 0, 0, 0, 0, 0, 0, 1])),
- }))
- ))
- );
- let response = None;
- let (mut iface, mut sockets, _device) = setup(medium);
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- response
- );
- assert_eq!(
- iface.inner.neighbor_cache.lookup(
- &IpAddress::Ipv6(Ipv6Address::from_parts(&[0xfdbe, 0, 0, 0, 0, 0, 0, 0x0002])),
- iface.inner.now,
- ),
- NeighborAnswer::Found(HardwareAddress::Ieee802154(Ieee802154Address::from_bytes(
- &[0, 0, 0, 0, 0, 0, 0, 1]
- ))),
- );
- }
- #[rstest]
- #[case(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- fn test_handle_valid_ndisc_request(#[case] medium: Medium) {
- let (mut iface, mut sockets, _device) = setup(medium);
- let mut eth_bytes = vec![0u8; 86];
- let local_ip_addr = Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 1);
- let remote_ip_addr = Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 2);
- let local_hw_addr = EthernetAddress([0x02, 0x02, 0x02, 0x02, 0x02, 0x02]);
- let remote_hw_addr = EthernetAddress([0x52, 0x54, 0x00, 0x00, 0x00, 0x00]);
- let solicit = Icmpv6Repr::Ndisc(NdiscRepr::NeighborSolicit {
- target_addr: local_ip_addr,
- lladdr: Some(remote_hw_addr.into()),
- });
- let ip_repr = IpRepr::Ipv6(Ipv6Repr {
- src_addr: remote_ip_addr,
- dst_addr: local_ip_addr.solicited_node(),
- next_header: IpProtocol::Icmpv6,
- hop_limit: 0xff,
- payload_len: solicit.buffer_len(),
- });
- let mut frame = EthernetFrame::new_unchecked(&mut eth_bytes);
- frame.set_dst_addr(EthernetAddress([0x33, 0x33, 0x00, 0x00, 0x00, 0x00]));
- frame.set_src_addr(remote_hw_addr);
- frame.set_ethertype(EthernetProtocol::Ipv6);
- ip_repr.emit(frame.payload_mut(), &ChecksumCapabilities::default());
- solicit.emit(
- &remote_ip_addr.into(),
- &local_ip_addr.solicited_node().into(),
- &mut Icmpv6Packet::new_unchecked(&mut frame.payload_mut()[ip_repr.header_len()..]),
- &ChecksumCapabilities::default(),
- );
- let icmpv6_expected = Icmpv6Repr::Ndisc(NdiscRepr::NeighborAdvert {
- flags: NdiscNeighborFlags::SOLICITED,
- target_addr: local_ip_addr,
- lladdr: Some(local_hw_addr.into()),
- });
- let ipv6_expected = Ipv6Repr {
- src_addr: local_ip_addr,
- dst_addr: remote_ip_addr,
- next_header: IpProtocol::Icmpv6,
- hop_limit: 0xff,
- payload_len: icmpv6_expected.buffer_len(),
- };
- // Ensure an Neighbor Solicitation triggers a Neighbor Advertisement
- assert_eq!(
- iface.inner.process_ethernet(
- &mut sockets,
- PacketMeta::default(),
- frame.into_inner(),
- &mut iface.fragments
- ),
- Some(EthernetPacket::Ip(Packet::new_ipv6(
- ipv6_expected,
- IpPayload::Icmpv6(icmpv6_expected)
- )))
- );
- // Ensure the address of the requester was entered in the cache
- assert_eq!(
- iface.inner.lookup_hardware_addr(
- MockTxToken,
- &IpAddress::Ipv6(local_ip_addr),
- &IpAddress::Ipv6(remote_ip_addr),
- &mut iface.fragmenter,
- ),
- Ok((HardwareAddress::Ethernet(remote_hw_addr), MockTxToken))
- );
- }
- #[rstest]
- #[case(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- #[case(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- #[case(Medium::Ieee802154)]
- #[cfg(feature = "medium-ieee802154")]
- fn test_solicited_node_addrs(#[case] medium: Medium) {
- let (mut iface, _, _) = setup(medium);
- let mut new_addrs = heapless::Vec::<IpCidr, IFACE_MAX_ADDR_COUNT>::new();
- new_addrs
- .push(IpCidr::new(IpAddress::v6(0xfe80, 0, 0, 0, 1, 2, 0, 2), 64))
- .unwrap();
- new_addrs
- .push(IpCidr::new(
- IpAddress::v6(0xfe80, 0, 0, 0, 3, 4, 0, 0xffff),
- 64,
- ))
- .unwrap();
- iface.update_ip_addrs(|addrs| {
- new_addrs.extend(addrs.to_vec());
- *addrs = new_addrs;
- });
- assert!(iface
- .inner
- .has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0x0002)));
- assert!(iface
- .inner
- .has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0xffff)));
- assert!(!iface
- .inner
- .has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0x0003)));
- }
- #[rstest]
- #[case(Medium::Ip)]
- #[cfg(all(feature = "socket-udp", feature = "medium-ip"))]
- #[case(Medium::Ethernet)]
- #[cfg(all(feature = "socket-udp", feature = "medium-ethernet"))]
- #[case(Medium::Ieee802154)]
- #[cfg(all(feature = "socket-udp", feature = "medium-ieee802154"))]
- fn test_icmp_reply_size(#[case] medium: Medium) {
- use crate::wire::Icmpv6DstUnreachable;
- use crate::wire::IPV6_MIN_MTU as MIN_MTU;
- const MAX_PAYLOAD_LEN: usize = 1192;
- let (mut iface, mut sockets, _device) = setup(medium);
- let src_addr = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1);
- let dst_addr = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 2);
- // UDP packet that if not tructated will cause a icmp port unreachable reply
- // to exceed the minimum mtu bytes in length.
- let udp_repr = UdpRepr {
- src_port: 67,
- dst_port: 68,
- };
- let mut bytes = vec![0xff; udp_repr.header_len() + MAX_PAYLOAD_LEN];
- let mut packet = UdpPacket::new_unchecked(&mut bytes[..]);
- udp_repr.emit(
- &mut packet,
- &src_addr.into(),
- &dst_addr.into(),
- MAX_PAYLOAD_LEN,
- |buf| fill_slice(buf, 0x2a),
- &ChecksumCapabilities::default(),
- );
- let ip_repr = Ipv6Repr {
- src_addr,
- dst_addr,
- next_header: IpProtocol::Udp,
- hop_limit: 64,
- payload_len: udp_repr.header_len() + MAX_PAYLOAD_LEN,
- };
- let payload = packet.into_inner();
- let expected_icmp_repr = Icmpv6Repr::DstUnreachable {
- reason: Icmpv6DstUnreachable::PortUnreachable,
- header: ip_repr,
- data: &payload[..MAX_PAYLOAD_LEN],
- };
- let expected_ip_repr = Ipv6Repr {
- src_addr: dst_addr,
- dst_addr: src_addr,
- next_header: IpProtocol::Icmpv6,
- hop_limit: 64,
- payload_len: expected_icmp_repr.buffer_len(),
- };
- assert_eq!(
- expected_ip_repr.buffer_len() + expected_icmp_repr.buffer_len(),
- MIN_MTU
- );
- assert_eq!(
- iface.inner.process_udp(
- &mut sockets,
- PacketMeta::default(),
- false,
- ip_repr.into(),
- payload,
- ),
- Some(Packet::new_ipv6(
- expected_ip_repr,
- IpPayload::Icmpv6(expected_icmp_repr)
- ))
- );
- }
- #[cfg(feature = "medium-ip")]
- #[test]
- fn get_source_address() {
- let (mut iface, _, _) = setup(Medium::Ip);
- const OWN_LINK_LOCAL_ADDR: Ipv6Address = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1);
- const OWN_UNIQUE_LOCAL_ADDR1: Ipv6Address = Ipv6Address::new(0xfd00, 0, 0, 201, 1, 1, 1, 2);
- const OWN_UNIQUE_LOCAL_ADDR2: Ipv6Address = Ipv6Address::new(0xfd01, 0, 0, 201, 1, 1, 1, 2);
- const OWN_GLOBAL_UNICAST_ADDR1: Ipv6Address =
- Ipv6Address::new(0x2001, 0x0db8, 0x0003, 0, 0, 0, 0, 1);
- // List of addresses of the interface:
- // fe80::1/64
- // fd00::201:1:1:1:2/64
- // fd01::201:1:1:1:2/64
- // 2001:db8:3::1/64
- // ::1/128
- // ::/128
- iface.update_ip_addrs(|addrs| {
- addrs.clear();
- addrs
- .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_LINK_LOCAL_ADDR, 64)))
- .unwrap();
- addrs
- .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_UNIQUE_LOCAL_ADDR1, 64)))
- .unwrap();
- addrs
- .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_UNIQUE_LOCAL_ADDR2, 64)))
- .unwrap();
- addrs
- .push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_GLOBAL_UNICAST_ADDR1, 64)))
- .unwrap();
- // These should never be used:
- addrs
- .push(IpCidr::Ipv6(Ipv6Cidr::new(Ipv6Address::LOOPBACK, 128)))
- .unwrap();
- addrs
- .push(IpCidr::Ipv6(Ipv6Cidr::new(Ipv6Address::UNSPECIFIED, 128)))
- .unwrap();
- });
- // List of addresses we test:
- // fe80::42 -> fe80::1
- // fd00::201:1:1:1:1 -> fd00::201:1:1:1:2
- // fd01::201:1:1:1:1 -> fd01::201:1:1:1:2
- // fd02::201:1:1:1:1 -> fd00::201:1:1:1:2 (because first added in the list)
- // ff02::1 -> fe80::1 (same scope)
- // 2001:db8:3::2 -> 2001:db8:3::1
- // 2001:db9:3::2 -> 2001:db8:3::1
- const LINK_LOCAL_ADDR: Ipv6Address = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 42);
- const UNIQUE_LOCAL_ADDR1: Ipv6Address = Ipv6Address::new(0xfd00, 0, 0, 201, 1, 1, 1, 1);
- const UNIQUE_LOCAL_ADDR2: Ipv6Address = Ipv6Address::new(0xfd01, 0, 0, 201, 1, 1, 1, 1);
- const UNIQUE_LOCAL_ADDR3: Ipv6Address = Ipv6Address::new(0xfd02, 0, 0, 201, 1, 1, 1, 1);
- const GLOBAL_UNICAST_ADDR1: Ipv6Address =
- Ipv6Address::new(0x2001, 0x0db8, 0x0003, 0, 0, 0, 0, 2);
- const GLOBAL_UNICAST_ADDR2: Ipv6Address =
- Ipv6Address::new(0x2001, 0x0db9, 0x0003, 0, 0, 0, 0, 2);
- assert_eq!(
- iface.inner.get_source_address_ipv6(&LINK_LOCAL_ADDR),
- Some(OWN_LINK_LOCAL_ADDR)
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
- Some(OWN_UNIQUE_LOCAL_ADDR1)
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
- Some(OWN_UNIQUE_LOCAL_ADDR2)
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
- Some(OWN_UNIQUE_LOCAL_ADDR1)
- );
- assert_eq!(
- iface
- .inner
- .get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
- Some(OWN_LINK_LOCAL_ADDR)
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
- Some(OWN_GLOBAL_UNICAST_ADDR1)
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
- Some(OWN_GLOBAL_UNICAST_ADDR1)
- );
- assert_eq!(
- iface.get_source_address_ipv6(&LINK_LOCAL_ADDR),
- Some(OWN_LINK_LOCAL_ADDR)
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
- Some(OWN_UNIQUE_LOCAL_ADDR1)
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
- Some(OWN_UNIQUE_LOCAL_ADDR2)
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
- Some(OWN_UNIQUE_LOCAL_ADDR1)
- );
- assert_eq!(
- iface.get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
- Some(OWN_LINK_LOCAL_ADDR)
- );
- assert_eq!(
- iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
- Some(OWN_GLOBAL_UNICAST_ADDR1)
- );
- assert_eq!(
- iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
- Some(OWN_GLOBAL_UNICAST_ADDR1)
- );
- }
|