123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579 |
- // Packet implementation for the Multicast Listener Discovery
- // protocol. See [RFC 3810] and [RFC 2710].
- //
- // [RFC 3810]: https://tools.ietf.org/html/rfc3810
- // [RFC 2710]: https://tools.ietf.org/html/rfc2710
- use byteorder::{ByteOrder, NetworkEndian};
- use super::{Error, Result};
- use crate::wire::icmpv6::{field, Message, Packet};
- use crate::wire::Ipv6Address;
- enum_with_unknown! {
- /// MLDv2 Multicast Listener Report Record Type. See [RFC 3810 § 5.2.12] for
- /// more details.
- ///
- /// [RFC 3810 § 5.2.12]: https://tools.ietf.org/html/rfc3010#section-5.2.12
- pub enum RecordType(u8) {
- /// Interface has a filter mode of INCLUDE for the specified multicast address.
- ModeIsInclude = 0x01,
- /// Interface has a filter mode of EXCLUDE for the specified multicast address.
- ModeIsExclude = 0x02,
- /// Interface has changed to a filter mode of INCLUDE for the specified
- /// multicast address.
- ChangeToInclude = 0x03,
- /// Interface has changed to a filter mode of EXCLUDE for the specified
- /// multicast address.
- ChangeToExclude = 0x04,
- /// Interface wishes to listen to the sources in the specified list.
- AllowNewSources = 0x05,
- /// Interface no longer wishes to listen to the sources in the specified list.
- BlockOldSources = 0x06
- }
- }
- /// Getters for the Multicast Listener Query message header.
- /// See [RFC 3810 § 5.1].
- ///
- /// [RFC 3810 § 5.1]: https://tools.ietf.org/html/rfc3010#section-5.1
- impl<T: AsRef<[u8]>> Packet<T> {
- /// Return the maximum response code field.
- #[inline]
- pub fn max_resp_code(&self) -> u16 {
- let data = self.buffer.as_ref();
- NetworkEndian::read_u16(&data[field::MAX_RESP_CODE])
- }
- /// Return the address being queried.
- #[inline]
- pub fn mcast_addr(&self) -> Ipv6Address {
- let data = self.buffer.as_ref();
- Ipv6Address::from_bytes(&data[field::QUERY_MCAST_ADDR])
- }
- /// Return the Suppress Router-Side Processing flag.
- #[inline]
- pub fn s_flag(&self) -> bool {
- let data = self.buffer.as_ref();
- (data[field::SQRV] & 0x08) != 0
- }
- /// Return the Querier's Robustness Variable.
- #[inline]
- pub fn qrv(&self) -> u8 {
- let data = self.buffer.as_ref();
- data[field::SQRV] & 0x7
- }
- /// Return the Querier's Query Interval Code.
- #[inline]
- pub fn qqic(&self) -> u8 {
- let data = self.buffer.as_ref();
- data[field::QQIC]
- }
- /// Return number of sources.
- #[inline]
- pub fn num_srcs(&self) -> u16 {
- let data = self.buffer.as_ref();
- NetworkEndian::read_u16(&data[field::QUERY_NUM_SRCS])
- }
- }
- /// Getters for the Multicast Listener Report message header.
- /// See [RFC 3810 § 5.2].
- ///
- /// [RFC 3810 § 5.2]: https://tools.ietf.org/html/rfc3010#section-5.2
- impl<T: AsRef<[u8]>> Packet<T> {
- /// Return the number of Multicast Address Records.
- #[inline]
- pub fn nr_mcast_addr_rcrds(&self) -> u16 {
- let data = self.buffer.as_ref();
- NetworkEndian::read_u16(&data[field::NR_MCAST_RCRDS])
- }
- }
- /// Setters for the Multicast Listener Query message header.
- /// See [RFC 3810 § 5.1].
- ///
- /// [RFC 3810 § 5.1]: https://tools.ietf.org/html/rfc3010#section-5.1
- impl<T: AsRef<[u8]> + AsMut<[u8]>> Packet<T> {
- /// Set the maximum response code field.
- #[inline]
- pub fn set_max_resp_code(&mut self, code: u16) {
- let data = self.buffer.as_mut();
- NetworkEndian::write_u16(&mut data[field::MAX_RESP_CODE], code);
- }
- /// Set the address being queried.
- #[inline]
- pub fn set_mcast_addr(&mut self, addr: Ipv6Address) {
- let data = self.buffer.as_mut();
- data[field::QUERY_MCAST_ADDR].copy_from_slice(addr.as_bytes());
- }
- /// Set the Suppress Router-Side Processing flag.
- #[inline]
- pub fn set_s_flag(&mut self) {
- let data = self.buffer.as_mut();
- let current = data[field::SQRV];
- data[field::SQRV] = 0x8 | (current & 0x7);
- }
- /// Clear the Suppress Router-Side Processing flag.
- #[inline]
- pub fn clear_s_flag(&mut self) {
- let data = self.buffer.as_mut();
- data[field::SQRV] &= 0x7;
- }
- /// Set the Querier's Robustness Variable.
- #[inline]
- pub fn set_qrv(&mut self, value: u8) {
- assert!(value < 8);
- let data = self.buffer.as_mut();
- data[field::SQRV] = (data[field::SQRV] & 0x8) | value & 0x7;
- }
- /// Set the Querier's Query Interval Code.
- #[inline]
- pub fn set_qqic(&mut self, value: u8) {
- let data = self.buffer.as_mut();
- data[field::QQIC] = value;
- }
- /// Set number of sources.
- #[inline]
- pub fn set_num_srcs(&mut self, value: u16) {
- let data = self.buffer.as_mut();
- NetworkEndian::write_u16(&mut data[field::QUERY_NUM_SRCS], value);
- }
- }
- /// Setters for the Multicast Listener Report message header.
- /// See [RFC 3810 § 5.2].
- ///
- /// [RFC 3810 § 5.2]: https://tools.ietf.org/html/rfc3010#section-5.2
- impl<T: AsRef<[u8]> + AsMut<[u8]>> Packet<T> {
- /// Set the number of Multicast Address Records.
- #[inline]
- pub fn set_nr_mcast_addr_rcrds(&mut self, value: u16) {
- let data = self.buffer.as_mut();
- NetworkEndian::write_u16(&mut data[field::NR_MCAST_RCRDS], value)
- }
- }
- /// A read/write wrapper around an MLDv2 Listener Report Message Address Record.
- #[derive(Debug, PartialEq, Eq, Clone)]
- #[cfg_attr(feature = "defmt", derive(defmt::Format))]
- pub struct AddressRecord<T: AsRef<[u8]>> {
- buffer: T,
- }
- impl<T: AsRef<[u8]>> AddressRecord<T> {
- /// Imbue a raw octet buffer with a Address Record structure.
- pub const fn new_unchecked(buffer: T) -> Self {
- Self { buffer }
- }
- /// Shorthand for a combination of [new_unchecked] and [check_len].
- ///
- /// [new_unchecked]: #method.new_unchecked
- /// [check_len]: #method.check_len
- pub fn new_checked(buffer: T) -> Result<Self> {
- let packet = Self::new_unchecked(buffer);
- packet.check_len()?;
- Ok(packet)
- }
- /// Ensure that no accessor method will panic if called.
- /// Returns `Err(Error::Truncated)` if the buffer is too short.
- pub fn check_len(&self) -> Result<()> {
- let len = self.buffer.as_ref().len();
- if len < field::RECORD_MCAST_ADDR.end {
- Err(Error)
- } else {
- Ok(())
- }
- }
- /// Consume the packet, returning the underlying buffer.
- pub fn into_inner(self) -> T {
- self.buffer
- }
- }
- /// Getters for a MLDv2 Listener Report Message Address Record.
- /// See [RFC 3810 § 5.2].
- ///
- /// [RFC 3810 § 5.2]: https://tools.ietf.org/html/rfc3010#section-5.2
- impl<T: AsRef<[u8]>> AddressRecord<T> {
- /// Return the record type for the given sources.
- #[inline]
- pub fn record_type(&self) -> RecordType {
- let data = self.buffer.as_ref();
- RecordType::from(data[field::RECORD_TYPE])
- }
- /// Return the length of the auxiliary data.
- #[inline]
- pub fn aux_data_len(&self) -> u8 {
- let data = self.buffer.as_ref();
- data[field::AUX_DATA_LEN]
- }
- /// Return the number of sources field.
- #[inline]
- pub fn num_srcs(&self) -> u16 {
- let data = self.buffer.as_ref();
- NetworkEndian::read_u16(&data[field::RECORD_NUM_SRCS])
- }
- /// Return the multicast address field.
- #[inline]
- pub fn mcast_addr(&self) -> Ipv6Address {
- let data = self.buffer.as_ref();
- Ipv6Address::from_bytes(&data[field::RECORD_MCAST_ADDR])
- }
- }
- impl<'a, T: AsRef<[u8]> + ?Sized> AddressRecord<&'a T> {
- /// Return a pointer to the address records.
- #[inline]
- pub fn payload(&self) -> &'a [u8] {
- let data = self.buffer.as_ref();
- &data[field::RECORD_MCAST_ADDR.end..]
- }
- }
- /// Setters for a MLDv2 Listener Report Message Address Record.
- /// See [RFC 3810 § 5.2].
- ///
- /// [RFC 3810 § 5.2]: https://tools.ietf.org/html/rfc3010#section-5.2
- impl<T: AsMut<[u8]> + AsRef<[u8]>> AddressRecord<T> {
- /// Return the record type for the given sources.
- #[inline]
- pub fn set_record_type(&mut self, rty: RecordType) {
- let data = self.buffer.as_mut();
- data[field::RECORD_TYPE] = rty.into();
- }
- /// Return the length of the auxiliary data.
- #[inline]
- pub fn set_aux_data_len(&mut self, len: u8) {
- let data = self.buffer.as_mut();
- data[field::AUX_DATA_LEN] = len;
- }
- /// Return the number of sources field.
- #[inline]
- pub fn set_num_srcs(&mut self, num_srcs: u16) {
- let data = self.buffer.as_mut();
- NetworkEndian::write_u16(&mut data[field::RECORD_NUM_SRCS], num_srcs);
- }
- /// Return the multicast address field.
- ///
- /// # Panics
- /// This function panics if the given address is not a multicast address.
- #[inline]
- pub fn set_mcast_addr(&mut self, addr: Ipv6Address) {
- assert!(addr.is_multicast());
- let data = self.buffer.as_mut();
- data[field::RECORD_MCAST_ADDR].copy_from_slice(addr.as_bytes());
- }
- }
- impl<T: AsRef<[u8]> + AsMut<[u8]>> AddressRecord<T> {
- /// Return a pointer to the address records.
- #[inline]
- pub fn payload_mut(&mut self) -> &mut [u8] {
- let data = self.buffer.as_mut();
- &mut data[field::RECORD_MCAST_ADDR.end..]
- }
- }
- /// A high-level representation of an MLDv2 packet header.
- #[derive(Debug, PartialEq, Eq, Clone, Copy)]
- #[cfg_attr(feature = "defmt", derive(defmt::Format))]
- pub enum Repr<'a> {
- Query {
- max_resp_code: u16,
- mcast_addr: Ipv6Address,
- s_flag: bool,
- qrv: u8,
- qqic: u8,
- num_srcs: u16,
- data: &'a [u8],
- },
- Report {
- nr_mcast_addr_rcrds: u16,
- data: &'a [u8],
- },
- }
- impl<'a> Repr<'a> {
- /// Parse an MLDv2 packet and return a high-level representation.
- pub fn parse<T>(packet: &Packet<&'a T>) -> Result<Repr<'a>>
- where
- T: AsRef<[u8]> + ?Sized,
- {
- packet.check_len()?;
- match packet.msg_type() {
- Message::MldQuery => Ok(Repr::Query {
- max_resp_code: packet.max_resp_code(),
- mcast_addr: packet.mcast_addr(),
- s_flag: packet.s_flag(),
- qrv: packet.qrv(),
- qqic: packet.qqic(),
- num_srcs: packet.num_srcs(),
- data: packet.payload(),
- }),
- Message::MldReport => Ok(Repr::Report {
- nr_mcast_addr_rcrds: packet.nr_mcast_addr_rcrds(),
- data: packet.payload(),
- }),
- _ => Err(Error),
- }
- }
- /// Return the length of a packet that will be emitted from this high-level representation.
- pub const fn buffer_len(&self) -> usize {
- match self {
- Repr::Query { data, .. } => field::QUERY_NUM_SRCS.end + data.len(),
- Repr::Report { data, .. } => field::NR_MCAST_RCRDS.end + data.len(),
- }
- }
- /// Emit a high-level representation into an MLDv2 packet.
- pub fn emit<T>(&self, packet: &mut Packet<&mut T>)
- where
- T: AsRef<[u8]> + AsMut<[u8]> + ?Sized,
- {
- match self {
- Repr::Query {
- max_resp_code,
- mcast_addr,
- s_flag,
- qrv,
- qqic,
- num_srcs,
- data,
- } => {
- packet.set_msg_type(Message::MldQuery);
- packet.set_msg_code(0);
- packet.clear_reserved();
- packet.set_max_resp_code(*max_resp_code);
- packet.set_mcast_addr(*mcast_addr);
- if *s_flag {
- packet.set_s_flag();
- } else {
- packet.clear_s_flag();
- }
- packet.set_qrv(*qrv);
- packet.set_qqic(*qqic);
- packet.set_num_srcs(*num_srcs);
- packet.payload_mut().copy_from_slice(&data[..]);
- }
- Repr::Report {
- nr_mcast_addr_rcrds,
- data,
- } => {
- packet.set_msg_type(Message::MldReport);
- packet.set_msg_code(0);
- packet.clear_reserved();
- packet.set_nr_mcast_addr_rcrds(*nr_mcast_addr_rcrds);
- packet.payload_mut().copy_from_slice(&data[..]);
- }
- }
- }
- }
- #[cfg(test)]
- mod test {
- use super::*;
- use crate::phy::ChecksumCapabilities;
- use crate::wire::icmpv6::Message;
- use crate::wire::Icmpv6Repr;
- static QUERY_PACKET_BYTES: [u8; 44] = [
- 0x82, 0x00, 0x73, 0x74, 0x04, 0x00, 0x00, 0x00, 0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00,
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x0a, 0x12, 0x00, 0x01, 0xff, 0x02,
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02,
- ];
- static QUERY_PACKET_PAYLOAD: [u8; 16] = [
- 0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
- 0x02,
- ];
- static REPORT_PACKET_BYTES: [u8; 44] = [
- 0x8f, 0x00, 0x73, 0x85, 0x00, 0x00, 0x00, 0x01, 0x01, 0x00, 0x00, 0x01, 0xff, 0x02, 0x00,
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xff, 0x02,
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02,
- ];
- static REPORT_PACKET_PAYLOAD: [u8; 36] = [
- 0x01, 0x00, 0x00, 0x01, 0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
- 0x00, 0x00, 0x00, 0x00, 0x01, 0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x02,
- ];
- fn create_repr<'a>(ty: Message) -> Icmpv6Repr<'a> {
- match ty {
- Message::MldQuery => Icmpv6Repr::Mld(Repr::Query {
- max_resp_code: 0x400,
- mcast_addr: Ipv6Address::LINK_LOCAL_ALL_NODES,
- s_flag: true,
- qrv: 0x02,
- qqic: 0x12,
- num_srcs: 0x01,
- data: &QUERY_PACKET_PAYLOAD,
- }),
- Message::MldReport => Icmpv6Repr::Mld(Repr::Report {
- nr_mcast_addr_rcrds: 1,
- data: &REPORT_PACKET_PAYLOAD,
- }),
- _ => {
- panic!("Message type must be a MLDv2 message type");
- }
- }
- }
- #[test]
- fn test_query_deconstruct() {
- let packet = Packet::new_unchecked(&QUERY_PACKET_BYTES[..]);
- assert_eq!(packet.msg_type(), Message::MldQuery);
- assert_eq!(packet.msg_code(), 0);
- assert_eq!(packet.checksum(), 0x7374);
- assert_eq!(packet.max_resp_code(), 0x0400);
- assert_eq!(packet.mcast_addr(), Ipv6Address::LINK_LOCAL_ALL_NODES);
- assert!(packet.s_flag());
- assert_eq!(packet.qrv(), 0x02);
- assert_eq!(packet.qqic(), 0x12);
- assert_eq!(packet.num_srcs(), 0x01);
- assert_eq!(
- Ipv6Address::from_bytes(packet.payload()),
- Ipv6Address::LINK_LOCAL_ALL_ROUTERS
- );
- }
- #[test]
- fn test_query_construct() {
- let mut bytes = vec![0xff; 44];
- let mut packet = Packet::new_unchecked(&mut bytes[..]);
- packet.set_msg_type(Message::MldQuery);
- packet.set_msg_code(0);
- packet.set_max_resp_code(0x0400);
- packet.set_mcast_addr(Ipv6Address::LINK_LOCAL_ALL_NODES);
- packet.set_s_flag();
- packet.set_qrv(0x02);
- packet.set_qqic(0x12);
- packet.set_num_srcs(0x01);
- packet
- .payload_mut()
- .copy_from_slice(Ipv6Address::LINK_LOCAL_ALL_ROUTERS.as_bytes());
- packet.clear_reserved();
- packet.fill_checksum(
- &Ipv6Address::LINK_LOCAL_ALL_NODES.into(),
- &Ipv6Address::LINK_LOCAL_ALL_ROUTERS.into(),
- );
- assert_eq!(&*packet.into_inner(), &QUERY_PACKET_BYTES[..]);
- }
- #[test]
- fn test_record_deconstruct() {
- let packet = Packet::new_unchecked(&REPORT_PACKET_BYTES[..]);
- assert_eq!(packet.msg_type(), Message::MldReport);
- assert_eq!(packet.msg_code(), 0);
- assert_eq!(packet.checksum(), 0x7385);
- assert_eq!(packet.nr_mcast_addr_rcrds(), 0x01);
- let addr_rcrd = AddressRecord::new_unchecked(packet.payload());
- assert_eq!(addr_rcrd.record_type(), RecordType::ModeIsInclude);
- assert_eq!(addr_rcrd.aux_data_len(), 0x00);
- assert_eq!(addr_rcrd.num_srcs(), 0x01);
- assert_eq!(addr_rcrd.mcast_addr(), Ipv6Address::LINK_LOCAL_ALL_NODES);
- assert_eq!(
- Ipv6Address::from_bytes(addr_rcrd.payload()),
- Ipv6Address::LINK_LOCAL_ALL_ROUTERS
- );
- }
- #[test]
- fn test_record_construct() {
- let mut bytes = vec![0xff; 44];
- let mut packet = Packet::new_unchecked(&mut bytes[..]);
- packet.set_msg_type(Message::MldReport);
- packet.set_msg_code(0);
- packet.clear_reserved();
- packet.set_nr_mcast_addr_rcrds(1);
- {
- let mut addr_rcrd = AddressRecord::new_unchecked(packet.payload_mut());
- addr_rcrd.set_record_type(RecordType::ModeIsInclude);
- addr_rcrd.set_aux_data_len(0);
- addr_rcrd.set_num_srcs(1);
- addr_rcrd.set_mcast_addr(Ipv6Address::LINK_LOCAL_ALL_NODES);
- addr_rcrd
- .payload_mut()
- .copy_from_slice(Ipv6Address::LINK_LOCAL_ALL_ROUTERS.as_bytes());
- }
- packet.fill_checksum(
- &Ipv6Address::LINK_LOCAL_ALL_NODES.into(),
- &Ipv6Address::LINK_LOCAL_ALL_ROUTERS.into(),
- );
- assert_eq!(&*packet.into_inner(), &REPORT_PACKET_BYTES[..]);
- }
- #[test]
- fn test_query_repr_parse() {
- let packet = Packet::new_unchecked(&QUERY_PACKET_BYTES[..]);
- let repr = Icmpv6Repr::parse(
- &Ipv6Address::LINK_LOCAL_ALL_NODES.into(),
- &Ipv6Address::LINK_LOCAL_ALL_ROUTERS.into(),
- &packet,
- &ChecksumCapabilities::default(),
- );
- assert_eq!(repr, Ok(create_repr(Message::MldQuery)));
- }
- #[test]
- fn test_report_repr_parse() {
- let packet = Packet::new_unchecked(&REPORT_PACKET_BYTES[..]);
- let repr = Icmpv6Repr::parse(
- &Ipv6Address::LINK_LOCAL_ALL_NODES.into(),
- &Ipv6Address::LINK_LOCAL_ALL_ROUTERS.into(),
- &packet,
- &ChecksumCapabilities::default(),
- );
- assert_eq!(repr, Ok(create_repr(Message::MldReport)));
- }
- #[test]
- fn test_query_repr_emit() {
- let mut bytes = [0x2a; 44];
- let mut packet = Packet::new_unchecked(&mut bytes[..]);
- let repr = create_repr(Message::MldQuery);
- repr.emit(
- &Ipv6Address::LINK_LOCAL_ALL_NODES.into(),
- &Ipv6Address::LINK_LOCAL_ALL_ROUTERS.into(),
- &mut packet,
- &ChecksumCapabilities::default(),
- );
- assert_eq!(&*packet.into_inner(), &QUERY_PACKET_BYTES[..]);
- }
- #[test]
- fn test_report_repr_emit() {
- let mut bytes = [0x2a; 44];
- let mut packet = Packet::new_unchecked(&mut bytes[..]);
- let repr = create_repr(Message::MldReport);
- repr.emit(
- &Ipv6Address::LINK_LOCAL_ALL_NODES.into(),
- &Ipv6Address::LINK_LOCAL_ALL_ROUTERS.into(),
- &mut packet,
- &ChecksumCapabilities::default(),
- );
- assert_eq!(&*packet.into_inner(), &REPORT_PACKET_BYTES[..]);
- }
- }
|