use byteorder::{ByteOrder, NetworkEndian};
use core::fmt;
use super::{Error, Result};
use crate::phy::ChecksumCapabilities;
use crate::wire::ip::{checksum, pretty_print_ip_payload};
pub use super::IpProtocol as Protocol;
pub const MIN_MTU: usize = 576;
pub const MULTICAST_ALL_SYSTEMS: Address = Address::new(224, 0, 0, 1);
pub const MULTICAST_ALL_ROUTERS: Address = Address::new(224, 0, 0, 2);
const MINIMUM_IHL_BYTES: u8 = 20;
#[derive(Debug, Eq, PartialEq, Ord, PartialOrd, Clone, Copy)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
pub struct Key {
id: u16,
src_addr: Address,
dst_addr: Address,
protocol: Protocol,
}
pub use core::net::Ipv4Addr as Address;
pub(crate) trait AddressExt {
fn x_is_unicast(&self) -> bool;
fn prefix_len(&self) -> Option<u8>;
}
impl AddressExt for Address {
fn x_is_unicast(&self) -> bool {
!(self.is_broadcast() || self.is_multicast() || self.is_unspecified())
}
fn prefix_len(&self) -> Option<u8> {
let mut ones = true;
let mut prefix_len = 0;
for byte in self.octets() {
let mut mask = 0x80;
for _ in 0..8 {
let one = byte & mask != 0;
if ones {
if one {
prefix_len += 1;
} else {
ones = false;
}
} else if one {
return None;
}
mask >>= 1;
}
}
Some(prefix_len)
}
}
#[derive(Debug, Hash, PartialEq, Eq, PartialOrd, Ord, Clone, Copy)]
pub struct Cidr {
address: Address,
prefix_len: u8,
}
impl Cidr {
pub const fn new(address: Address, prefix_len: u8) -> Cidr {
assert!(prefix_len <= 32);
Cidr {
address,
prefix_len,
}
}
pub fn from_netmask(addr: Address, netmask: Address) -> Result<Cidr> {
let netmask = netmask.to_bits();
if netmask.leading_zeros() == 0 && netmask.trailing_zeros() == netmask.count_zeros() {
Ok(Cidr {
address: addr,
prefix_len: netmask.count_ones() as u8,
})
} else {
Err(Error)
}
}
pub const fn address(&self) -> Address {
self.address
}
pub const fn prefix_len(&self) -> u8 {
self.prefix_len
}
pub const fn netmask(&self) -> Address {
if self.prefix_len == 0 {
return Address::new(0, 0, 0, 0);
}
let number = 0xffffffffu32 << (32 - self.prefix_len);
Address::from_bits(number)
}
pub fn broadcast(&self) -> Option<Address> {
let network = self.network();
if network.prefix_len == 31 || network.prefix_len == 32 {
return None;
}
let network_number = network.address.to_bits();
let number = network_number | 0xffffffffu32 >> network.prefix_len;
Some(Address::from_bits(number))
}
pub const fn network(&self) -> Cidr {
Cidr {
address: Address::from_bits(self.address.to_bits() & self.netmask().to_bits()),
prefix_len: self.prefix_len,
}
}
pub fn contains_addr(&self, addr: &Address) -> bool {
self.address.to_bits() & self.netmask().to_bits()
== addr.to_bits() & self.netmask().to_bits()
}
pub fn contains_subnet(&self, subnet: &Cidr) -> bool {
self.prefix_len <= subnet.prefix_len && self.contains_addr(&subnet.address)
}
}
impl fmt::Display for Cidr {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}/{}", self.address, self.prefix_len)
}
}
#[cfg(feature = "defmt")]
impl defmt::Format for Cidr {
fn format(&self, f: defmt::Formatter) {
defmt::write!(f, "{}/{=u8}", self.address, self.prefix_len);
}
}
#[derive(Debug, PartialEq, Eq, Clone)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
pub struct Packet<T: AsRef<[u8]>> {
buffer: T,
}
mod field {
use crate::wire::field::*;
pub const VER_IHL: usize = 0;
pub const DSCP_ECN: usize = 1;
pub const LENGTH: Field = 2..4;
pub const IDENT: Field = 4..6;
pub const FLG_OFF: Field = 6..8;
pub const TTL: usize = 8;
pub const PROTOCOL: usize = 9;
pub const CHECKSUM: Field = 10..12;
pub const SRC_ADDR: Field = 12..16;
pub const DST_ADDR: Field = 16..20;
}
pub const HEADER_LEN: usize = field::DST_ADDR.end;
impl<T: AsRef<[u8]>> Packet<T> {
pub const fn new_unchecked(buffer: T) -> Packet<T> {
Packet { buffer }
}
pub fn new_checked(buffer: T) -> Result<Packet<T>> {
let packet = Self::new_unchecked(buffer);
packet.check_len()?;
Ok(packet)
}
#[allow(clippy::if_same_then_else)]
pub fn check_len(&self) -> Result<()> {
let len = self.buffer.as_ref().len();
if len < field::DST_ADDR.end {
Err(Error)
} else if len < self.header_len() as usize {
Err(Error)
} else if self.header_len() as u16 > self.total_len() {
Err(Error)
} else if len < self.total_len() as usize {
Err(Error)
} else if self.header_len() < MINIMUM_IHL_BYTES {
Err(Error)
} else {
Ok(())
}
}
pub fn into_inner(self) -> T {
self.buffer
}
#[inline]
pub fn version(&self) -> u8 {
let data = self.buffer.as_ref();
data[field::VER_IHL] >> 4
}
#[inline]
pub fn header_len(&self) -> u8 {
let data = self.buffer.as_ref();
(data[field::VER_IHL] & 0x0f) * 4
}
pub fn dscp(&self) -> u8 {
let data = self.buffer.as_ref();
data[field::DSCP_ECN] >> 2
}
pub fn ecn(&self) -> u8 {
let data = self.buffer.as_ref();
data[field::DSCP_ECN] & 0x03
}
#[inline]
pub fn total_len(&self) -> u16 {
let data = self.buffer.as_ref();
NetworkEndian::read_u16(&data[field::LENGTH])
}
#[inline]
pub fn ident(&self) -> u16 {
let data = self.buffer.as_ref();
NetworkEndian::read_u16(&data[field::IDENT])
}
#[inline]
pub fn dont_frag(&self) -> bool {
let data = self.buffer.as_ref();
NetworkEndian::read_u16(&data[field::FLG_OFF]) & 0x4000 != 0
}
#[inline]
pub fn more_frags(&self) -> bool {
let data = self.buffer.as_ref();
NetworkEndian::read_u16(&data[field::FLG_OFF]) & 0x2000 != 0
}
#[inline]
pub fn frag_offset(&self) -> u16 {
let data = self.buffer.as_ref();
NetworkEndian::read_u16(&data[field::FLG_OFF]) << 3
}
#[inline]
pub fn hop_limit(&self) -> u8 {
let data = self.buffer.as_ref();
data[field::TTL]
}
#[inline]
pub fn next_header(&self) -> Protocol {
let data = self.buffer.as_ref();
Protocol::from(data[field::PROTOCOL])
}
#[inline]
pub fn checksum(&self) -> u16 {
let data = self.buffer.as_ref();
NetworkEndian::read_u16(&data[field::CHECKSUM])
}
#[inline]
pub fn src_addr(&self) -> Address {
let data = self.buffer.as_ref();
Address::from_octets(data[field::SRC_ADDR].try_into().unwrap())
}
#[inline]
pub fn dst_addr(&self) -> Address {
let data = self.buffer.as_ref();
Address::from_octets(data[field::DST_ADDR].try_into().unwrap())
}
pub fn verify_checksum(&self) -> bool {
if cfg!(fuzzing) {
return true;
}
let data = self.buffer.as_ref();
checksum::data(&data[..self.header_len() as usize]) == !0
}
pub fn get_key(&self) -> Key {
Key {
id: self.ident(),
src_addr: self.src_addr(),
dst_addr: self.dst_addr(),
protocol: self.next_header(),
}
}
}
impl<'a, T: AsRef<[u8]> + ?Sized> Packet<&'a T> {
#[inline]
pub fn payload(&self) -> &'a [u8] {
let range = self.header_len() as usize..self.total_len() as usize;
let data = self.buffer.as_ref();
&data[range]
}
}
impl<T: AsRef<[u8]> + AsMut<[u8]>> Packet<T> {
#[inline]
pub fn set_version(&mut self, value: u8) {
let data = self.buffer.as_mut();
data[field::VER_IHL] = (data[field::VER_IHL] & !0xf0) | (value << 4);
}
#[inline]
pub fn set_header_len(&mut self, value: u8) {
let data = self.buffer.as_mut();
data[field::VER_IHL] = (data[field::VER_IHL] & !0x0f) | ((value / 4) & 0x0f);
}
pub fn set_dscp(&mut self, value: u8) {
let data = self.buffer.as_mut();
data[field::DSCP_ECN] = (data[field::DSCP_ECN] & !0xfc) | (value << 2)
}
pub fn set_ecn(&mut self, value: u8) {
let data = self.buffer.as_mut();
data[field::DSCP_ECN] = (data[field::DSCP_ECN] & !0x03) | (value & 0x03)
}
#[inline]
pub fn set_total_len(&mut self, value: u16) {
let data = self.buffer.as_mut();
NetworkEndian::write_u16(&mut data[field::LENGTH], value)
}
#[inline]
pub fn set_ident(&mut self, value: u16) {
let data = self.buffer.as_mut();
NetworkEndian::write_u16(&mut data[field::IDENT], value)
}
#[inline]
pub fn clear_flags(&mut self) {
let data = self.buffer.as_mut();
let raw = NetworkEndian::read_u16(&data[field::FLG_OFF]);
let raw = raw & !0xe000;
NetworkEndian::write_u16(&mut data[field::FLG_OFF], raw);
}
#[inline]
pub fn set_dont_frag(&mut self, value: bool) {
let data = self.buffer.as_mut();
let raw = NetworkEndian::read_u16(&data[field::FLG_OFF]);
let raw = if value { raw | 0x4000 } else { raw & !0x4000 };
NetworkEndian::write_u16(&mut data[field::FLG_OFF], raw);
}
#[inline]
pub fn set_more_frags(&mut self, value: bool) {
let data = self.buffer.as_mut();
let raw = NetworkEndian::read_u16(&data[field::FLG_OFF]);
let raw = if value { raw | 0x2000 } else { raw & !0x2000 };
NetworkEndian::write_u16(&mut data[field::FLG_OFF], raw);
}
#[inline]
pub fn set_frag_offset(&mut self, value: u16) {
let data = self.buffer.as_mut();
let raw = NetworkEndian::read_u16(&data[field::FLG_OFF]);
let raw = (raw & 0xe000) | (value >> 3);
NetworkEndian::write_u16(&mut data[field::FLG_OFF], raw);
}
#[inline]
pub fn set_hop_limit(&mut self, value: u8) {
let data = self.buffer.as_mut();
data[field::TTL] = value
}
#[inline]
pub fn set_next_header(&mut self, value: Protocol) {
let data = self.buffer.as_mut();
data[field::PROTOCOL] = value.into()
}
#[inline]
pub fn set_checksum(&mut self, value: u16) {
let data = self.buffer.as_mut();
NetworkEndian::write_u16(&mut data[field::CHECKSUM], value)
}
#[inline]
pub fn set_src_addr(&mut self, value: Address) {
let data = self.buffer.as_mut();
data[field::SRC_ADDR].copy_from_slice(&value.octets())
}
#[inline]
pub fn set_dst_addr(&mut self, value: Address) {
let data = self.buffer.as_mut();
data[field::DST_ADDR].copy_from_slice(&value.octets())
}
pub fn fill_checksum(&mut self) {
self.set_checksum(0);
let checksum = {
let data = self.buffer.as_ref();
!checksum::data(&data[..self.header_len() as usize])
};
self.set_checksum(checksum)
}
#[inline]
pub fn payload_mut(&mut self) -> &mut [u8] {
let range = self.header_len() as usize..self.total_len() as usize;
let data = self.buffer.as_mut();
&mut data[range]
}
}
impl<T: AsRef<[u8]>> AsRef<[u8]> for Packet<T> {
fn as_ref(&self) -> &[u8] {
self.buffer.as_ref()
}
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
pub struct Repr {
pub src_addr: Address,
pub dst_addr: Address,
pub next_header: Protocol,
pub payload_len: usize,
pub hop_limit: u8,
}
impl Repr {
pub fn parse<T: AsRef<[u8]> + ?Sized>(
packet: &Packet<&T>,
checksum_caps: &ChecksumCapabilities,
) -> Result<Repr> {
packet.check_len()?;
if packet.version() != 4 {
return Err(Error);
}
if checksum_caps.ipv4.rx() && !packet.verify_checksum() {
return Err(Error);
}
#[cfg(not(feature = "proto-ipv4-fragmentation"))]
if packet.more_frags() || packet.frag_offset() != 0 {
return Err(Error);
}
let payload_len = packet.total_len() as usize - packet.header_len() as usize;
Ok(Repr {
src_addr: packet.src_addr(),
dst_addr: packet.dst_addr(),
next_header: packet.next_header(),
payload_len,
hop_limit: packet.hop_limit(),
})
}
pub const fn buffer_len(&self) -> usize {
field::DST_ADDR.end
}
pub fn emit<T: AsRef<[u8]> + AsMut<[u8]>>(
&self,
packet: &mut Packet<T>,
checksum_caps: &ChecksumCapabilities,
) {
packet.set_version(4);
packet.set_header_len(field::DST_ADDR.end as u8);
packet.set_dscp(0);
packet.set_ecn(0);
let total_len = packet.header_len() as u16 + self.payload_len as u16;
packet.set_total_len(total_len);
packet.set_ident(0);
packet.clear_flags();
packet.set_more_frags(false);
packet.set_dont_frag(true);
packet.set_frag_offset(0);
packet.set_hop_limit(self.hop_limit);
packet.set_next_header(self.next_header);
packet.set_src_addr(self.src_addr);
packet.set_dst_addr(self.dst_addr);
if checksum_caps.ipv4.tx() {
packet.fill_checksum();
} else {
packet.set_checksum(0);
}
}
}
impl<T: AsRef<[u8]> + ?Sized> fmt::Display for Packet<&T> {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match Repr::parse(self, &ChecksumCapabilities::ignored()) {
Ok(repr) => write!(f, "{repr}"),
Err(err) => {
write!(f, "IPv4 ({err})")?;
write!(
f,
" src={} dst={} proto={} hop_limit={}",
self.src_addr(),
self.dst_addr(),
self.next_header(),
self.hop_limit()
)?;
if self.version() != 4 {
write!(f, " ver={}", self.version())?;
}
if self.header_len() != 20 {
write!(f, " hlen={}", self.header_len())?;
}
if self.dscp() != 0 {
write!(f, " dscp={}", self.dscp())?;
}
if self.ecn() != 0 {
write!(f, " ecn={}", self.ecn())?;
}
write!(f, " tlen={}", self.total_len())?;
if self.dont_frag() {
write!(f, " df")?;
}
if self.more_frags() {
write!(f, " mf")?;
}
if self.frag_offset() != 0 {
write!(f, " off={}", self.frag_offset())?;
}
if self.more_frags() || self.frag_offset() != 0 {
write!(f, " id={}", self.ident())?;
}
Ok(())
}
}
}
}
impl fmt::Display for Repr {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(
f,
"IPv4 src={} dst={} proto={}",
self.src_addr, self.dst_addr, self.next_header
)
}
}
use crate::wire::pretty_print::{PrettyIndent, PrettyPrint};
impl<T: AsRef<[u8]>> PrettyPrint for Packet<T> {
fn pretty_print(
buffer: &dyn AsRef<[u8]>,
f: &mut fmt::Formatter,
indent: &mut PrettyIndent,
) -> fmt::Result {
use crate::wire::ip::checksum::format_checksum;
let checksum_caps = ChecksumCapabilities::ignored();
let (ip_repr, payload) = match Packet::new_checked(buffer) {
Err(err) => return write!(f, "{indent}({err})"),
Ok(ip_packet) => match Repr::parse(&ip_packet, &checksum_caps) {
Err(_) => return Ok(()),
Ok(ip_repr) => {
if ip_packet.more_frags() || ip_packet.frag_offset() != 0 {
write!(
f,
"{}IPv4 Fragment more_frags={} offset={}",
indent,
ip_packet.more_frags(),
ip_packet.frag_offset()
)?;
return Ok(());
} else {
write!(f, "{indent}{ip_repr}")?;
format_checksum(f, ip_packet.verify_checksum(), false)?;
(ip_repr, ip_packet.payload())
}
}
},
};
pretty_print_ip_payload(f, indent, ip_repr, payload)
}
}
#[cfg(test)]
pub(crate) mod test {
use super::*;
#[allow(unused)]
pub(crate) const MOCK_IP_ADDR_1: Address = Address::new(192, 168, 1, 1);
#[allow(unused)]
pub(crate) const MOCK_IP_ADDR_2: Address = Address::new(192, 168, 1, 2);
#[allow(unused)]
pub(crate) const MOCK_IP_ADDR_3: Address = Address::new(192, 168, 1, 3);
#[allow(unused)]
pub(crate) const MOCK_IP_ADDR_4: Address = Address::new(192, 168, 1, 4);
#[allow(unused)]
pub(crate) const MOCK_UNSPECIFIED: Address = Address::UNSPECIFIED;
static PACKET_BYTES: [u8; 30] = [
0x45, 0x00, 0x00, 0x1e, 0x01, 0x02, 0x62, 0x03, 0x1a, 0x01, 0xd5, 0x6e, 0x11, 0x12, 0x13,
0x14, 0x21, 0x22, 0x23, 0x24, 0xaa, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
];
static PAYLOAD_BYTES: [u8; 10] = [0xaa, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff];
#[test]
fn test_deconstruct() {
let packet = Packet::new_unchecked(&PACKET_BYTES[..]);
assert_eq!(packet.version(), 4);
assert_eq!(packet.header_len(), 20);
assert_eq!(packet.dscp(), 0);
assert_eq!(packet.ecn(), 0);
assert_eq!(packet.total_len(), 30);
assert_eq!(packet.ident(), 0x102);
assert!(packet.more_frags());
assert!(packet.dont_frag());
assert_eq!(packet.frag_offset(), 0x203 * 8);
assert_eq!(packet.hop_limit(), 0x1a);
assert_eq!(packet.next_header(), Protocol::Icmp);
assert_eq!(packet.checksum(), 0xd56e);
assert_eq!(packet.src_addr(), Address::new(0x11, 0x12, 0x13, 0x14));
assert_eq!(packet.dst_addr(), Address::new(0x21, 0x22, 0x23, 0x24));
assert!(packet.verify_checksum());
assert_eq!(packet.payload(), &PAYLOAD_BYTES[..]);
}
#[test]
fn test_construct() {
let mut bytes = vec![0xa5; 30];
let mut packet = Packet::new_unchecked(&mut bytes);
packet.set_version(4);
packet.set_header_len(20);
packet.clear_flags();
packet.set_dscp(0);
packet.set_ecn(0);
packet.set_total_len(30);
packet.set_ident(0x102);
packet.set_more_frags(true);
packet.set_dont_frag(true);
packet.set_frag_offset(0x203 * 8);
packet.set_hop_limit(0x1a);
packet.set_next_header(Protocol::Icmp);
packet.set_src_addr(Address::new(0x11, 0x12, 0x13, 0x14));
packet.set_dst_addr(Address::new(0x21, 0x22, 0x23, 0x24));
packet.fill_checksum();
packet.payload_mut().copy_from_slice(&PAYLOAD_BYTES[..]);
assert_eq!(&*packet.into_inner(), &PACKET_BYTES[..]);
}
#[test]
fn test_overlong() {
let mut bytes = vec![];
bytes.extend(&PACKET_BYTES[..]);
bytes.push(0);
assert_eq!(
Packet::new_unchecked(&bytes).payload().len(),
PAYLOAD_BYTES.len()
);
assert_eq!(
Packet::new_unchecked(&mut bytes).payload_mut().len(),
PAYLOAD_BYTES.len()
);
}
#[test]
fn test_total_len_overflow() {
let mut bytes = vec![];
bytes.extend(&PACKET_BYTES[..]);
Packet::new_unchecked(&mut bytes).set_total_len(128);
assert_eq!(Packet::new_checked(&bytes).unwrap_err(), Error);
}
static REPR_PACKET_BYTES: [u8; 24] = [
0x45, 0x00, 0x00, 0x18, 0x00, 0x00, 0x40, 0x00, 0x40, 0x01, 0xd2, 0x79, 0x11, 0x12, 0x13,
0x14, 0x21, 0x22, 0x23, 0x24, 0xaa, 0x00, 0x00, 0xff,
];
static REPR_PAYLOAD_BYTES: [u8; 4] = [0xaa, 0x00, 0x00, 0xff];
const fn packet_repr() -> Repr {
Repr {
src_addr: Address::new(0x11, 0x12, 0x13, 0x14),
dst_addr: Address::new(0x21, 0x22, 0x23, 0x24),
next_header: Protocol::Icmp,
payload_len: 4,
hop_limit: 64,
}
}
#[test]
fn test_parse() {
let packet = Packet::new_unchecked(&REPR_PACKET_BYTES[..]);
let repr = Repr::parse(&packet, &ChecksumCapabilities::default()).unwrap();
assert_eq!(repr, packet_repr());
}
#[test]
fn test_parse_bad_version() {
let mut bytes = vec![0; 24];
bytes.copy_from_slice(&REPR_PACKET_BYTES[..]);
let mut packet = Packet::new_unchecked(&mut bytes);
packet.set_version(6);
packet.fill_checksum();
let packet = Packet::new_unchecked(&*packet.into_inner());
assert_eq!(
Repr::parse(&packet, &ChecksumCapabilities::default()),
Err(Error)
);
}
#[test]
fn test_parse_total_len_less_than_header_len() {
let mut bytes = vec![0; 40];
bytes[0] = 0x09;
assert_eq!(Packet::new_checked(&mut bytes), Err(Error));
}
#[test]
fn test_parse_small_ihl() {
let mut bytes = vec![0; 24];
bytes.copy_from_slice(&REPR_PACKET_BYTES[..]);
let mut packet = Packet::new_unchecked(&mut bytes);
packet.set_header_len(16);
assert_eq!(Packet::new_checked(&mut bytes), Err(Error));
}
#[test]
fn test_emit() {
let repr = packet_repr();
let mut bytes = vec![0xa5; repr.buffer_len() + REPR_PAYLOAD_BYTES.len()];
let mut packet = Packet::new_unchecked(&mut bytes);
repr.emit(&mut packet, &ChecksumCapabilities::default());
packet.payload_mut().copy_from_slice(&REPR_PAYLOAD_BYTES);
assert_eq!(&*packet.into_inner(), &REPR_PACKET_BYTES[..]);
}
#[test]
fn test_unspecified() {
assert!(Address::UNSPECIFIED.is_unspecified());
assert!(!Address::UNSPECIFIED.is_broadcast());
assert!(!Address::UNSPECIFIED.is_multicast());
assert!(!Address::UNSPECIFIED.is_link_local());
assert!(!Address::UNSPECIFIED.is_loopback());
}
#[test]
fn test_broadcast() {
assert!(!Address::BROADCAST.is_unspecified());
assert!(Address::BROADCAST.is_broadcast());
assert!(!Address::BROADCAST.is_multicast());
assert!(!Address::BROADCAST.is_link_local());
assert!(!Address::BROADCAST.is_loopback());
}
#[test]
fn test_cidr() {
let cidr = Cidr::new(Address::new(192, 168, 1, 10), 24);
let inside_subnet = [
[192, 168, 1, 0],
[192, 168, 1, 1],
[192, 168, 1, 2],
[192, 168, 1, 10],
[192, 168, 1, 127],
[192, 168, 1, 255],
];
let outside_subnet = [
[192, 168, 0, 0],
[127, 0, 0, 1],
[192, 168, 2, 0],
[192, 168, 0, 255],
[0, 0, 0, 0],
[255, 255, 255, 255],
];
let subnets = [
([192, 168, 1, 0], 32),
([192, 168, 1, 255], 24),
([192, 168, 1, 10], 30),
];
let not_subnets = [
([192, 168, 1, 10], 23),
([127, 0, 0, 1], 8),
([192, 168, 1, 0], 0),
([192, 168, 0, 255], 32),
];
for addr in inside_subnet.iter().map(|a| Address::from_octets(*a)) {
assert!(cidr.contains_addr(&addr));
}
for addr in outside_subnet.iter().map(|a| Address::from_octets(*a)) {
assert!(!cidr.contains_addr(&addr));
}
for subnet in subnets
.iter()
.map(|&(a, p)| Cidr::new(Address::new(a[0], a[1], a[2], a[3]), p))
{
assert!(cidr.contains_subnet(&subnet));
}
for subnet in not_subnets
.iter()
.map(|&(a, p)| Cidr::new(Address::new(a[0], a[1], a[2], a[3]), p))
{
assert!(!cidr.contains_subnet(&subnet));
}
let cidr_without_prefix = Cidr::new(cidr.address(), 0);
assert!(cidr_without_prefix.contains_addr(&Address::new(127, 0, 0, 1)));
}
#[test]
fn test_cidr_from_netmask() {
assert!(Cidr::from_netmask(Address::new(0, 0, 0, 0), Address::new(1, 0, 2, 0)).is_err());
assert!(Cidr::from_netmask(Address::new(0, 0, 0, 0), Address::new(0, 0, 0, 0)).is_err());
assert_eq!(
Cidr::from_netmask(Address::new(0, 0, 0, 1), Address::new(255, 255, 255, 0)).unwrap(),
Cidr::new(Address::new(0, 0, 0, 1), 24)
);
assert_eq!(
Cidr::from_netmask(Address::new(192, 168, 0, 1), Address::new(255, 255, 0, 0)).unwrap(),
Cidr::new(Address::new(192, 168, 0, 1), 16)
);
assert_eq!(
Cidr::from_netmask(Address::new(172, 16, 0, 1), Address::new(255, 240, 0, 0)).unwrap(),
Cidr::new(Address::new(172, 16, 0, 1), 12)
);
assert_eq!(
Cidr::from_netmask(
Address::new(255, 255, 255, 1),
Address::new(255, 255, 255, 0)
)
.unwrap(),
Cidr::new(Address::new(255, 255, 255, 1), 24)
);
assert_eq!(
Cidr::from_netmask(
Address::new(255, 255, 255, 255),
Address::new(255, 255, 255, 255)
)
.unwrap(),
Cidr::new(Address::new(255, 255, 255, 255), 32)
);
}
#[test]
fn test_cidr_netmask() {
assert_eq!(
Cidr::new(Address::new(0, 0, 0, 0), 0).netmask(),
Address::new(0, 0, 0, 0)
);
assert_eq!(
Cidr::new(Address::new(0, 0, 0, 1), 24).netmask(),
Address::new(255, 255, 255, 0)
);
assert_eq!(
Cidr::new(Address::new(0, 0, 0, 0), 32).netmask(),
Address::new(255, 255, 255, 255)
);
assert_eq!(
Cidr::new(Address::new(127, 0, 0, 0), 8).netmask(),
Address::new(255, 0, 0, 0)
);
assert_eq!(
Cidr::new(Address::new(192, 168, 0, 0), 16).netmask(),
Address::new(255, 255, 0, 0)
);
assert_eq!(
Cidr::new(Address::new(192, 168, 1, 1), 16).netmask(),
Address::new(255, 255, 0, 0)
);
assert_eq!(
Cidr::new(Address::new(192, 168, 1, 1), 17).netmask(),
Address::new(255, 255, 128, 0)
);
assert_eq!(
Cidr::new(Address::new(172, 16, 0, 0), 12).netmask(),
Address::new(255, 240, 0, 0)
);
assert_eq!(
Cidr::new(Address::new(255, 255, 255, 1), 24).netmask(),
Address::new(255, 255, 255, 0)
);
assert_eq!(
Cidr::new(Address::new(255, 255, 255, 255), 32).netmask(),
Address::new(255, 255, 255, 255)
);
}
#[test]
fn test_cidr_broadcast() {
assert_eq!(
Cidr::new(Address::new(0, 0, 0, 0), 0).broadcast().unwrap(),
Address::new(255, 255, 255, 255)
);
assert_eq!(
Cidr::new(Address::new(0, 0, 0, 1), 24).broadcast().unwrap(),
Address::new(0, 0, 0, 255)
);
assert_eq!(Cidr::new(Address::new(0, 0, 0, 0), 32).broadcast(), None);
assert_eq!(
Cidr::new(Address::new(127, 0, 0, 0), 8)
.broadcast()
.unwrap(),
Address::new(127, 255, 255, 255)
);
assert_eq!(
Cidr::new(Address::new(192, 168, 0, 0), 16)
.broadcast()
.unwrap(),
Address::new(192, 168, 255, 255)
);
assert_eq!(
Cidr::new(Address::new(192, 168, 1, 1), 16)
.broadcast()
.unwrap(),
Address::new(192, 168, 255, 255)
);
assert_eq!(
Cidr::new(Address::new(192, 168, 1, 1), 17)
.broadcast()
.unwrap(),
Address::new(192, 168, 127, 255)
);
assert_eq!(
Cidr::new(Address::new(172, 16, 0, 1), 12)
.broadcast()
.unwrap(),
Address::new(172, 31, 255, 255)
);
assert_eq!(
Cidr::new(Address::new(255, 255, 255, 1), 24)
.broadcast()
.unwrap(),
Address::new(255, 255, 255, 255)
);
assert_eq!(
Cidr::new(Address::new(255, 255, 255, 254), 31).broadcast(),
None
);
assert_eq!(
Cidr::new(Address::new(255, 255, 255, 255), 32).broadcast(),
None
);
}
#[test]
fn test_cidr_network() {
assert_eq!(
Cidr::new(Address::new(0, 0, 0, 0), 0).network(),
Cidr::new(Address::new(0, 0, 0, 0), 0)
);
assert_eq!(
Cidr::new(Address::new(0, 0, 0, 1), 24).network(),
Cidr::new(Address::new(0, 0, 0, 0), 24)
);
assert_eq!(
Cidr::new(Address::new(0, 0, 0, 0), 32).network(),
Cidr::new(Address::new(0, 0, 0, 0), 32)
);
assert_eq!(
Cidr::new(Address::new(127, 0, 0, 0), 8).network(),
Cidr::new(Address::new(127, 0, 0, 0), 8)
);
assert_eq!(
Cidr::new(Address::new(192, 168, 0, 0), 16).network(),
Cidr::new(Address::new(192, 168, 0, 0), 16)
);
assert_eq!(
Cidr::new(Address::new(192, 168, 1, 1), 16).network(),
Cidr::new(Address::new(192, 168, 0, 0), 16)
);
assert_eq!(
Cidr::new(Address::new(192, 168, 1, 1), 17).network(),
Cidr::new(Address::new(192, 168, 0, 0), 17)
);
assert_eq!(
Cidr::new(Address::new(172, 16, 0, 1), 12).network(),
Cidr::new(Address::new(172, 16, 0, 0), 12)
);
assert_eq!(
Cidr::new(Address::new(255, 255, 255, 1), 24).network(),
Cidr::new(Address::new(255, 255, 255, 0), 24)
);
assert_eq!(
Cidr::new(Address::new(255, 255, 255, 255), 32).network(),
Cidr::new(Address::new(255, 255, 255, 255), 32)
);
}
}