12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382 |
- 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,
- &ipv6.dst_addr,
- &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 any_ip(#[case] medium: Medium) {
- // An empty echo request with destination address fdbe::3, which is not part of the interface
- // address list.
- 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, 0x3, 0x80, 0x0, 0x84, 0x3a, 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, 0x0003]),
- hop_limit: 64,
- next_header: IpProtocol::Icmpv6,
- payload_len: 8,
- },
- IpPayload::Icmpv6(Icmpv6Repr::EchoRequest {
- ident: 0,
- seq_no: 0,
- data: b"",
- })
- ))
- );
- let (mut iface, mut sockets, _device) = setup(medium);
- // Add a route to the interface, otherwise, we don't know if the packet is routed localy.
- iface.routes_mut().update(|routes| {
- routes
- .push(crate::iface::Route {
- cidr: IpCidr::Ipv6(Ipv6Cidr::new(
- Ipv6Address::new(0xfdbe, 0, 0, 0, 0, 0, 0, 0),
- 64,
- )),
- via_router: IpAddress::Ipv6(Ipv6Address::from_parts(&[
- 0xfdbe, 0, 0, 0, 0, 0, 0, 0x0001,
- ])),
- preferred_until: None,
- expires_at: None,
- })
- .unwrap();
- });
- assert_eq!(
- iface.inner.process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- HardwareAddress::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- ),
- None
- );
- // Accept any IP:
- iface.set_any_ip(true);
- assert!(iface
- .inner
- .process_ipv6(
- &mut sockets,
- PacketMeta::default(),
- HardwareAddress::default(),
- &Ipv6Packet::new_checked(&data[..]).unwrap()
- )
- .is_some());
- }
- #[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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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(),
- HardwareAddress::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,
- &local_ip_addr.solicited_node(),
- &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(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
- 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();
- });
- // List of addresses we test:
- // ::1 -> ::1
- // 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(&Ipv6Address::LOOPBACK),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&LINK_LOCAL_ADDR),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
- OWN_UNIQUE_LOCAL_ADDR1
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
- OWN_UNIQUE_LOCAL_ADDR2
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
- OWN_UNIQUE_LOCAL_ADDR1
- );
- assert_eq!(
- iface
- .inner
- .get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
- OWN_GLOBAL_UNICAST_ADDR1
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
- OWN_GLOBAL_UNICAST_ADDR1
- );
- assert_eq!(
- iface.get_source_address_ipv6(&LINK_LOCAL_ADDR),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
- OWN_UNIQUE_LOCAL_ADDR1
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
- OWN_UNIQUE_LOCAL_ADDR2
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
- OWN_UNIQUE_LOCAL_ADDR1
- );
- assert_eq!(
- iface.get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
- OWN_GLOBAL_UNICAST_ADDR1
- );
- assert_eq!(
- iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
- OWN_GLOBAL_UNICAST_ADDR1
- );
- }
- #[cfg(feature = "medium-ip")]
- #[test]
- fn get_source_address_only_link_local() {
- let (mut iface, _, _) = setup(Medium::Ip);
- // List of addresses in the interface:
- // fe80::1/64
- const OWN_LINK_LOCAL_ADDR: Ipv6Address = Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1);
- iface.update_ip_addrs(|ips| {
- ips.clear();
- ips.push(IpCidr::Ipv6(Ipv6Cidr::new(OWN_LINK_LOCAL_ADDR, 64)))
- .unwrap();
- });
- // List of addresses we test:
- // ::1 -> ::1
- // fe80::42 -> fe80::1
- // fd00::201:1:1:1:1 -> fe80::1
- // fd01::201:1:1:1:1 -> fe80::1
- // fd02::201:1:1:1:1 -> fe80::1
- // ff02::1 -> fe80::1
- // 2001:db8:3::2 -> fe80::1
- // 2001:db9:3::2 -> fe80::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(&Ipv6Address::LOOPBACK),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&LINK_LOCAL_ADDR),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface
- .inner
- .get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.get_source_address_ipv6(&LINK_LOCAL_ADDR),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
- OWN_LINK_LOCAL_ADDR
- );
- assert_eq!(
- iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
- OWN_LINK_LOCAL_ADDR
- );
- }
- #[cfg(feature = "medium-ip")]
- #[test]
- fn get_source_address_empty_interface() {
- let (mut iface, _, _) = setup(Medium::Ip);
- iface.update_ip_addrs(|ips| ips.clear());
- // List of addresses we test:
- // ::1 -> ::1
- // fe80::42 -> ::1
- // fd00::201:1:1:1:1 -> ::1
- // fd01::201:1:1:1:1 -> ::1
- // fd02::201:1:1:1:1 -> ::1
- // ff02::1 -> ::1
- // 2001:db8:3::2 -> ::1
- // 2001:db9:3::2 -> ::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(&Ipv6Address::LOOPBACK),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&LINK_LOCAL_ADDR),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface
- .inner
- .get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.inner.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.get_source_address_ipv6(&LINK_LOCAL_ADDR),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR1),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR2),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.get_source_address_ipv6(&UNIQUE_LOCAL_ADDR3),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.get_source_address_ipv6(&Ipv6Address::LINK_LOCAL_ALL_NODES),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR1),
- Ipv6Address::LOOPBACK
- );
- assert_eq!(
- iface.get_source_address_ipv6(&GLOBAL_UNICAST_ADDR2),
- Ipv6Address::LOOPBACK
- );
- }
- #[rstest]
- #[case(Medium::Ip)]
- #[cfg(feature = "medium-ip")]
- #[case(Medium::Ethernet)]
- #[cfg(feature = "medium-ethernet")]
- fn test_join_ipv6_multicast_group(#[case] medium: Medium) {
- fn recv_icmpv6(
- device: &mut crate::tests::TestingDevice,
- timestamp: Instant,
- ) -> std::vec::Vec<Ipv6Packet<std::vec::Vec<u8>>> {
- let caps = device.capabilities();
- recv_all(device, timestamp)
- .iter()
- .filter_map(|frame| {
- let ipv6_packet = match caps.medium {
- #[cfg(feature = "medium-ethernet")]
- Medium::Ethernet => {
- let eth_frame = EthernetFrame::new_checked(frame).ok()?;
- Ipv6Packet::new_checked(eth_frame.payload()).ok()?
- }
- #[cfg(feature = "medium-ip")]
- Medium::Ip => Ipv6Packet::new_checked(&frame[..]).ok()?,
- #[cfg(feature = "medium-ieee802154")]
- Medium::Ieee802154 => todo!(),
- };
- let buf = ipv6_packet.into_inner().to_vec();
- Some(Ipv6Packet::new_unchecked(buf))
- })
- .collect::<std::vec::Vec<_>>()
- }
- let (mut iface, _sockets, mut device) = setup(medium);
- let groups = [
- Ipv6Address::from_parts(&[0xff05, 0, 0, 0, 0, 0, 0, 0x00fb]),
- Ipv6Address::from_parts(&[0xff0e, 0, 0, 0, 0, 0, 0, 0x0017]),
- ];
- let timestamp = Instant::from_millis(0);
- for &group in &groups {
- iface
- .join_multicast_group(&mut device, group, timestamp)
- .unwrap();
- assert!(iface.has_multicast_group(group));
- }
- assert!(iface.has_multicast_group(Ipv6Address::LINK_LOCAL_ALL_NODES));
- let reports = recv_icmpv6(&mut device, timestamp);
- assert_eq!(reports.len(), 2);
- let caps = device.capabilities();
- let checksum_caps = &caps.checksum;
- for (&group_addr, ipv6_packet) in groups.iter().zip(reports) {
- let buf = ipv6_packet.into_inner();
- let ipv6_packet = Ipv6Packet::new_unchecked(buf.as_slice());
- let _ipv6_repr = Ipv6Repr::parse(&ipv6_packet).unwrap();
- let ip_payload = ipv6_packet.payload();
- // The first 2 octets of this payload hold the next-header indicator and the
- // Hop-by-Hop header length (in 8-octet words, minus 1). The remaining 6 octets
- // hold the Hop-by-Hop PadN and Router Alert options.
- let hbh_header = Ipv6HopByHopHeader::new_checked(&ip_payload[..8]).unwrap();
- let hbh_repr = Ipv6HopByHopRepr::parse(&hbh_header).unwrap();
- assert_eq!(hbh_repr.options.len(), 3);
- assert_eq!(
- hbh_repr.options[0],
- Ipv6OptionRepr::Unknown {
- type_: Ipv6OptionType::Unknown(IpProtocol::Icmpv6.into()),
- length: 0,
- data: &[],
- }
- );
- assert_eq!(
- hbh_repr.options[1],
- Ipv6OptionRepr::RouterAlert(Ipv6OptionRouterAlert::MulticastListenerDiscovery)
- );
- assert_eq!(hbh_repr.options[2], Ipv6OptionRepr::PadN(0));
- let icmpv6_packet =
- Icmpv6Packet::new_checked(&ip_payload[hbh_repr.buffer_len()..]).unwrap();
- let icmpv6_repr = Icmpv6Repr::parse(
- &ipv6_packet.src_addr(),
- &ipv6_packet.dst_addr(),
- &icmpv6_packet,
- checksum_caps,
- )
- .unwrap();
- let record_data = match icmpv6_repr {
- Icmpv6Repr::Mld(MldRepr::Report {
- nr_mcast_addr_rcrds,
- data,
- }) => {
- assert_eq!(nr_mcast_addr_rcrds, 1);
- data
- }
- other => panic!("unexpected icmpv6_repr: {:?}", other),
- };
- let record = MldAddressRecord::new_checked(record_data).unwrap();
- let record_repr = MldAddressRecordRepr::parse(&record).unwrap();
- assert_eq!(
- record_repr,
- MldAddressRecordRepr {
- num_srcs: 0,
- mcast_addr: group_addr,
- record_type: MldRecordType::ChangeToInclude,
- aux_data_len: 0,
- payload: &[],
- }
- );
- iface
- .leave_multicast_group(&mut device, group_addr, timestamp)
- .unwrap();
- assert!(!iface.has_multicast_group(group_addr));
- }
- }
|