Преглед на файлове

Add has_solicited_node to EthernetInterface

 - Add a function to EthernetInterface useful for determining if a
   received packet with the source address being a solicited node
   address is the solicited node address for an IPv6 address assigned
   to the interface.
 - Add SOLICITED_NODES_PREFIX to Ipv6Cidr
 - Fix some nits

Closes: #175
Approved by: whitequark
Dan Robertson преди 7 години
родител
ревизия
89999a6981
променени са 2 файла, в които са добавени 53 реда и са изтрити 9 реда
  1. 37 3
      src/iface/ethernet.rs
  2. 16 6
      src/wire/ipv6.rs

+ 37 - 3
src/iface/ethernet.rs

@@ -11,7 +11,7 @@ use wire::pretty_print::PrettyPrinter;
 use wire::{EthernetAddress, EthernetProtocol, EthernetFrame};
 use wire::{IpAddress, IpProtocol, IpRepr, IpCidr};
 #[cfg(feature = "proto-ipv6")]
-use wire::{Ipv6Packet, Ipv6Repr, IPV6_MIN_MTU};
+use wire::{Ipv6Address, Ipv6Packet, Ipv6Repr, IPV6_MIN_MTU};
 #[cfg(feature = "proto-ipv4")]
 use wire::{Ipv4Address, Ipv4Packet, Ipv4Repr, IPV4_MIN_MTU};
 #[cfg(feature = "proto-ipv4")]
@@ -130,7 +130,7 @@ impl<'b, 'c, DeviceT> InterfaceBuilder<'b, 'c, DeviceT>
     /// [ip_addrs].
     ///
     /// # Panics
-    /// This function panics if any of the addresses is not unicast.
+    /// This function panics if any of the addresses are not unicast.
     ///
     /// [ip_addrs]: struct.EthernetInterface.html#method.ip_addrs
     pub fn ip_addrs<T>(mut self, ip_addrs: T) -> InterfaceBuilder<'b, 'c, DeviceT>
@@ -267,10 +267,29 @@ impl<'b, 'c, DeviceT> Interface<'b, 'c, DeviceT>
         self.inner.ip_addrs.as_ref()
     }
 
+    /// Determine if the given `Ipv6Address` is the solicited node
+    /// multicast address for a IPv6 addresses assigned to the interface.
+    /// See [RFC 4291 § 2.7.1] for more details.
+    ///
+    /// [RFC 4291 § 2.7.1]: https://tools.ietf.org/html/rfc4291#section-2.7.1
+    #[cfg(feature = "proto-ipv6")]
+    pub fn has_solicited_node(&self, addr: Ipv6Address) -> bool {
+        self.inner.ip_addrs.iter().find(|cidr| {
+            match *cidr {
+                &IpCidr::Ipv6(cidr) if cidr.address() != Ipv6Address::LOOPBACK=> {
+                    // Take the lower order 24 bits of the IPv6 address and
+                    // append those bits to FF02:0:0:0:0:1:FF00::/104.
+                    addr.as_bytes()[14..] == cidr.address().as_bytes()[14..]
+                }
+                _ => false,
+            }
+        }).is_some()
+    }
+
     /// Update the IP addresses of the interface.
     ///
     /// # Panics
-    /// This function panics if any of the addresses is not unicast.
+    /// This function panics if any of the addresses are not unicast.
     pub fn update_ip_addrs<F: FnOnce(&mut ManagedSlice<'c, IpCidr>)>(&mut self, f: F) {
         f(&mut self.inner.ip_addrs);
         InterfaceInner::check_ip_addrs(&self.inner.ip_addrs)
@@ -1663,6 +1682,21 @@ mod test {
         }
     }
 
+    #[test]
+    #[cfg(feature = "proto-ipv6")]
+    fn test_solicited_node_addrs() {
+        let (mut iface, _) = create_loopback();
+        let mut new_addrs = vec![IpCidr::new(IpAddress::v6(0xfe80, 0, 0, 0, 1, 2, 0, 2), 64),
+                                 IpCidr::new(IpAddress::v6(0xfe80, 0, 0, 0, 3, 4, 0, 0xffff), 64)];
+        iface.update_ip_addrs(|addrs| {
+            new_addrs.extend(addrs.to_vec());
+            *addrs = From::from(new_addrs);
+        });
+        assert!(iface.has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0x0002)));
+        assert!(iface.has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0xffff)));
+        assert!(!iface.has_solicited_node(Ipv6Address::new(0xff02, 0, 0, 0, 0, 1, 0xff00, 0x0001)));
+    }
+
     #[test]
     #[cfg(feature = "proto-ipv6")]
     fn test_icmpv6_nxthdr_unknown() {

+ 16 - 6
src/wire/ipv6.rs

@@ -26,22 +26,22 @@ impl Address {
     ///
     /// [all routers multicast address]: https://tools.ietf.org/html/rfc4291#section-2.7.1
     pub const LINK_LOCAL_ALL_NODES: Address =
-        Address([0xff, 0x02, 0x00, 0x0, 0x00, 0x00, 0x00, 0x0,
-                 0x00, 0x00, 0x00, 0x0, 0x00, 0x00, 0x00, 0x1]);
+        Address([0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+                 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01]);
 
     /// The link-local [all nodes multicast address].
     ///
     /// [all nodes multicast address]: https://tools.ietf.org/html/rfc4291#section-2.7.1
     pub const LINK_LOCAL_ALL_ROUTERS: Address =
-        Address([0xff, 0x02, 0x00, 0x0, 0x00, 0x00, 0x00, 0x0,
-                 0x00, 0x00, 0x00, 0x0, 0x00, 0x00, 0x00, 0x2]);
+        Address([0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+                 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02]);
 
     /// The [loopback address].
     ///
     /// [loopback address]: https://tools.ietf.org/html/rfc4291#section-2.5.3
     pub const LOOPBACK: Address =
-        Address([0x00, 0x00, 0x00, 0x0, 0x00, 0x00, 0x00, 0x0,
-                 0x00, 0x00, 0x00, 0x0, 0x00, 0x00, 0x00, 0x1]);
+        Address([0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+                 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01]);
 
     /// Construct an IPv6 address from parts.
     pub fn new(a0: u16, a1: u16, a2: u16, a3: u16,
@@ -258,6 +258,16 @@ pub struct Cidr {
 }
 
 impl Cidr {
+    /// The [solicited node prefix].
+    ///
+    /// [solicited node prefix]: https://tools.ietf.org/html/rfc4291#section-2.7.1
+    pub const SOLICITED_NODE_PREFIX: Cidr =
+        Cidr {
+            address: Address([0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+                              0x00, 0x00, 0x00, 0x01, 0xff, 0x00, 0x00, 0x00]),
+            prefix_len: 104
+        };
+
     /// Create an IPv6 CIDR block from the given address and prefix length.
     ///
     /// # Panics