|
@@ -135,6 +135,23 @@ impl Address {
|
|
|
*self == Self::LOOPBACK
|
|
|
}
|
|
|
|
|
|
+ /// Query whether the IPv6 address is an [IPv4 mapped IPv6 address].
|
|
|
+ ///
|
|
|
+ /// [IPv4 mapped IPv6 address]: https://tools.ietf.org/html/rfc4291#section-2.5.5.2
|
|
|
+ pub fn is_ipv4_mapped(&self) -> bool {
|
|
|
+ self.0[0..12] == [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff]
|
|
|
+ }
|
|
|
+
|
|
|
+ #[cfg(feature = "proto-ipv4")]
|
|
|
+ /// Convert an IPv4 mapped IPv6 address to an IPv4 address.
|
|
|
+ pub fn as_ipv4(&self) -> Option<::wire::ipv4::Address> {
|
|
|
+ if self.is_ipv4_mapped() {
|
|
|
+ Some(::wire::ipv4::Address::new(self.0[12], self.0[13], self.0[14], self.0[15]))
|
|
|
+ } else {
|
|
|
+ None
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
/// Helper function used to mask an addres given a prefix.
|
|
|
///
|
|
|
/// # Panics
|
|
@@ -156,6 +173,10 @@ impl Address {
|
|
|
|
|
|
impl fmt::Display for Address {
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
+ if self.is_ipv4_mapped() {
|
|
|
+ return write!(f, "::ffff:{}.{}.{}.{}", self.0[12], self.0[13], self.0[14], self.0[15])
|
|
|
+ }
|
|
|
+
|
|
|
// The string representation of an IPv6 address should
|
|
|
// collapse a series of 16 bit sections that evaluate
|
|
|
// to 0 to "::"
|
|
@@ -204,6 +225,16 @@ impl fmt::Display for Address {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+#[cfg(feature = "proto-ipv4")]
|
|
|
+/// Convert the given IPv4 address into a IPv4-mapped IPv6 address
|
|
|
+impl From<::wire::ipv4::Address> for Address {
|
|
|
+ fn from(address: ::wire::ipv4::Address) -> Self {
|
|
|
+ let octets = address.0;
|
|
|
+ Address([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff,
|
|
|
+ octets[0], octets[1], octets[2], octets[3]])
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
/// A specification of an IPv6 CIDR block, containing an address and a variable-length
|
|
|
/// subnet masking prefix length.
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
|
|
@@ -606,6 +637,9 @@ mod test {
|
|
|
use super::{Packet, Protocol, Repr};
|
|
|
use wire::pretty_print::{PrettyPrinter};
|
|
|
|
|
|
+ #[cfg(feature = "proto-ipv4")]
|
|
|
+ use wire::ipv4::Address as Ipv4Address;
|
|
|
+
|
|
|
static LINK_LOCAL_ADDR: Address = Address([0xfe, 0x80, 0x00, 0x00,
|
|
|
0x00, 0x00, 0x00, 0x00,
|
|
|
0x00, 0x00, 0x00, 0x00,
|
|
@@ -646,6 +680,14 @@ mod test {
|
|
|
format!("{}", LINK_LOCAL_ADDR));
|
|
|
assert_eq!("fe80::7f00:0:1",
|
|
|
format!("{}", Address::new(0xfe80, 0, 0, 0, 0, 0x7f00, 0x0000, 0x0001)));
|
|
|
+ assert_eq!("::",
|
|
|
+ format!("{}", Address::UNSPECIFIED));
|
|
|
+ assert_eq!("::1",
|
|
|
+ format!("{}", Address::LOOPBACK));
|
|
|
+
|
|
|
+ #[cfg(feature = "proto-ipv4")]
|
|
|
+ assert_eq!("::ffff:192.168.1.1",
|
|
|
+ format!("{}", Address::from(Ipv4Address::new(192, 168, 1, 1))));
|
|
|
}
|
|
|
|
|
|
#[test]
|
|
@@ -703,6 +745,31 @@ mod test {
|
|
|
assert_eq!(addr.mask(127), [0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]);
|
|
|
}
|
|
|
|
|
|
+ #[cfg(feature = "proto-ipv4")]
|
|
|
+ #[test]
|
|
|
+ fn test_is_ipv4_mapped() {
|
|
|
+ assert_eq!(false, Address::UNSPECIFIED.is_ipv4_mapped());
|
|
|
+ assert_eq!(true, Address::from(Ipv4Address::new(192, 168, 1, 1)).is_ipv4_mapped());
|
|
|
+ }
|
|
|
+
|
|
|
+ #[cfg(feature = "proto-ipv4")]
|
|
|
+ #[test]
|
|
|
+ fn test_as_ipv4() {
|
|
|
+ assert_eq!(None, Address::UNSPECIFIED.as_ipv4());
|
|
|
+
|
|
|
+ let ipv4 = Ipv4Address::new(192, 168, 1, 1);
|
|
|
+ assert_eq!(Some(ipv4), Address::from(ipv4).as_ipv4());
|
|
|
+ }
|
|
|
+
|
|
|
+ #[cfg(feature = "proto-ipv4")]
|
|
|
+ #[test]
|
|
|
+ fn test_from_ipv4_address() {
|
|
|
+ assert_eq!(Address([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff, 192, 168, 1, 1]),
|
|
|
+ Address::from(Ipv4Address::new(192, 168, 1, 1)));
|
|
|
+ assert_eq!(Address([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff, 222, 1, 41, 90]),
|
|
|
+ Address::from(Ipv4Address::new(222, 1, 41, 90)));
|
|
|
+ }
|
|
|
+
|
|
|
#[test]
|
|
|
fn test_cidr() {
|
|
|
let cidr = Cidr::new(LINK_LOCAL_ADDR, 64);
|