فهرست منبع

Tests: Add basic interface tests

 - Add tests for the following
   - ICMP error responses are not sent in response to broadcast requests
   - ARP requests are responded to and inserted into the cache
   - ARP requests for someone else are not responded to, but the sender
     is still inserted in the cache
Dan Robertson 7 سال پیش
والد
کامیت
7a2271dfd6
6فایلهای تغییر یافته به همراه149 افزوده شده و 5 حذف شده
  1. 144 0
      src/iface/ethernet.rs
  2. 1 1
      src/wire/arp.rs
  3. 1 1
      src/wire/icmpv4.rs
  4. 1 1
      src/wire/ipv4.rs
  5. 1 1
      src/wire/tcp.rs
  6. 1 1
      src/wire/udp.rs

+ 144 - 0
src/iface/ethernet.rs

@@ -32,6 +32,7 @@ pub struct Interface<'a, 'b, 'c, DeviceT: Device + 'a> {
     ipv4_gateway:   Option<Ipv4Address>,
 }
 
+#[derive(Debug, PartialEq)]
 enum Packet<'a> {
     None,
     Arp(ArpRepr),
@@ -644,3 +645,146 @@ impl<'a, 'b, 'c, DeviceT: Device + 'a> Interface<'a, 'b, 'c, DeviceT> {
         })
     }
 }
+
+#[cfg(test)]
+mod tests {
+    use std::boxed::Box;
+    use super::Packet;
+    use phy::{Loopback, ChecksumCapabilities};
+    use wire::{ArpOperation, ArpPacket, ArpRepr};
+    use wire::{EthernetAddress, EthernetFrame, EthernetProtocol};
+    use wire::{IpAddress, IpCidr, IpProtocol, IpRepr};
+    use wire::{Ipv4Address, Ipv4Repr};
+    use iface::{ArpCache, SliceArpCache, EthernetInterface};
+    use socket::SocketSet;
+
+    fn create_loopback<'a, 'b>() ->
+            (EthernetInterface<'a, 'static, 'b, Loopback>, SocketSet<'static, 'a, 'b>) {
+        // Create a basic device
+        let device = Loopback::new();
+
+        let arp_cache = SliceArpCache::new(vec![Default::default(); 8]);
+
+        let ip_addr = IpCidr::new(IpAddress::v4(127, 0, 0, 1), 8);
+        (EthernetInterface::new(
+            Box::new(device), Box::new(arp_cache) as Box<ArpCache>,
+            EthernetAddress::default(), [ip_addr], None), SocketSet::new(vec![]))
+    }
+
+    #[test]
+    fn no_icmp_to_broadcast() {
+        let (mut iface, mut socket_set) = create_loopback();
+
+        let mut eth_bytes = vec![0u8; 34];
+
+        // Unknown Ipv4 Protocol
+        //
+        // Because the destination is the broadcast address
+        // this should not trigger and Destination Unreachable
+        // response. See RFC 1122 § 3.2.2.
+        let repr = IpRepr::Ipv4(Ipv4Repr {
+            src_addr:    Ipv4Address([0x7f, 0x00, 0x00, 0x01]),
+            dst_addr:    Ipv4Address::BROADCAST,
+            protocol:    IpProtocol::Unknown(0x0c),
+            payload_len: 0,
+            ttl:         0x40
+        });
+
+        let frame = {
+            let mut frame = EthernetFrame::new(&mut eth_bytes);
+            frame.set_dst_addr(EthernetAddress::BROADCAST);
+            frame.set_src_addr(EthernetAddress([0x52, 0x54, 0x00, 0x00, 0x00, 0x00]));
+            frame.set_ethertype(EthernetProtocol::Ipv4);
+            repr.emit(frame.payload_mut(), &ChecksumCapabilities::default());
+            EthernetFrame::new(&*frame.into_inner())
+        };
+
+        // Ensure that the unknown protocol frame does not trigger an
+        // ICMP error response when the destination address is a
+        // broadcast address
+        assert_eq!(iface.process_ipv4(&mut socket_set, 0, &frame),
+                   Ok(Packet::None));
+    }
+
+    #[test]
+    fn handle_valid_arp_request() {
+        let (mut iface, mut socket_set) = create_loopback();
+
+        let mut eth_bytes = vec![0u8; 42];
+
+        let local_ip_addr = Ipv4Address([0x7f, 0x00, 0x00, 0x01]);
+        let remote_ip_addr = Ipv4Address([0x7f, 0x00, 0x00, 0x02]);
+        let local_hw_addr = EthernetAddress([0x00, 0x00, 0x00, 0x00, 0x00, 0x00]);
+        let remote_hw_addr = EthernetAddress([0x52, 0x54, 0x00, 0x00, 0x00, 0x00]);
+
+        let repr = ArpRepr::EthernetIpv4 {
+            operation: ArpOperation::Request,
+            source_hardware_addr: remote_hw_addr,
+            source_protocol_addr: remote_ip_addr,
+            target_hardware_addr: EthernetAddress::default(),
+            target_protocol_addr: local_ip_addr,
+        };
+
+        let mut frame = EthernetFrame::new(&mut eth_bytes);
+        frame.set_dst_addr(EthernetAddress::BROADCAST);
+        frame.set_src_addr(remote_hw_addr);
+        frame.set_ethertype(EthernetProtocol::Arp);
+        {
+            let mut packet = ArpPacket::new(frame.payload_mut());
+            repr.emit(&mut packet);
+        }
+
+        // Ensure an ARP Request for us triggers an ARP Reply
+        assert_eq!(iface.process_ethernet(&mut socket_set, 0, frame.into_inner()),
+                   Ok(Packet::Arp(ArpRepr::EthernetIpv4 {
+                       operation: ArpOperation::Reply,
+                       source_hardware_addr: local_hw_addr,
+                       source_protocol_addr: local_ip_addr,
+                       target_hardware_addr: remote_hw_addr,
+                       target_protocol_addr: remote_ip_addr
+                   })));
+
+        // Ensure the address of the requestor was entered in the cache
+        assert_eq!(iface.lookup_hardware_addr(0,
+                                              &IpAddress::Ipv4(local_ip_addr),
+                                              &IpAddress::Ipv4(remote_ip_addr)),
+                   Ok(remote_hw_addr));
+    }
+
+    #[test]
+    fn handle_other_arp_request() {
+        let (mut iface, mut socket_set) = create_loopback();
+
+        let mut eth_bytes = vec![0u8; 42];
+
+        let remote_ip_addr = Ipv4Address([0x7f, 0x00, 0x00, 0x02]);
+        let remote_hw_addr = EthernetAddress([0x52, 0x54, 0x00, 0x00, 0x00, 0x00]);
+
+        let repr = ArpRepr::EthernetIpv4 {
+            operation: ArpOperation::Request,
+            source_hardware_addr: remote_hw_addr,
+            source_protocol_addr: remote_ip_addr,
+            target_hardware_addr: EthernetAddress::default(),
+            target_protocol_addr: Ipv4Address([0x7f, 0x00, 0x00, 0x03]),
+        };
+
+        let mut frame = EthernetFrame::new(&mut eth_bytes);
+        frame.set_dst_addr(EthernetAddress::BROADCAST);
+        frame.set_src_addr(remote_hw_addr);
+        frame.set_ethertype(EthernetProtocol::Arp);
+        {
+            let mut packet = ArpPacket::new(frame.payload_mut());
+            repr.emit(&mut packet);
+        }
+
+        // Ensure an ARP Request for someone else does not trigger an ARP Reply
+        assert_eq!(iface.process_ethernet(&mut socket_set, 0, frame.into_inner()),
+                   Ok(Packet::None));
+
+        // Ensure the address of the requestor was entered in the cache
+        assert_eq!(iface.lookup_hardware_addr(0,
+                                              &IpAddress::Ipv4(Ipv4Address([0x7f, 0x00, 0x00, 0x01])),
+                                              &IpAddress::Ipv4(remote_ip_addr)),
+                   Ok(remote_hw_addr));
+    }
+}

+ 1 - 1
src/wire/arp.rs

@@ -21,7 +21,7 @@ enum_with_unknown! {
 }
 
 /// A read/write wrapper around an Address Resolution Protocol packet buffer.
-#[derive(Debug)]
+#[derive(Debug, PartialEq)]
 pub struct Packet<T: AsRef<[u8]>> {
     buffer: T
 }

+ 1 - 1
src/wire/icmpv4.rs

@@ -166,7 +166,7 @@ enum_with_unknown! {
 }
 
 /// A read/write wrapper around an Internet Control Message Protocol version 4 packet buffer.
-#[derive(Debug)]
+#[derive(Debug, PartialEq)]
 pub struct Packet<T: AsRef<[u8]>> {
     buffer: T
 }

+ 1 - 1
src/wire/ipv4.rs

@@ -129,7 +129,7 @@ impl fmt::Display for Cidr {
 }
 
 /// A read/write wrapper around an Internet Protocol version 4 packet buffer.
-#[derive(Debug)]
+#[derive(Debug, PartialEq)]
 pub struct Packet<T: AsRef<[u8]>> {
     buffer: T
 }

+ 1 - 1
src/wire/tcp.rs

@@ -62,7 +62,7 @@ impl cmp::PartialOrd for SeqNumber {
 }
 
 /// A read/write wrapper around a Transmission Control Protocol packet buffer.
-#[derive(Debug)]
+#[derive(Debug, PartialEq)]
 pub struct Packet<T: AsRef<[u8]>> {
     buffer: T
 }

+ 1 - 1
src/wire/udp.rs

@@ -7,7 +7,7 @@ use super::{IpProtocol, IpAddress};
 use super::ip::checksum;
 
 /// A read/write wrapper around an User Datagram Protocol packet buffer.
-#[derive(Debug)]
+#[derive(Debug, PartialEq)]
 pub struct Packet<T: AsRef<[u8]>> {
     buffer: T
 }