Skip to content

Commit 423a930

Browse files
authored
Rollup merge of #87708 - the8472:canonical_v6, r=dtolnay
Add convenience method for handling ipv4-mapped addresses by canonicalizing them This simplifies checking common properties in an address-family-agnostic way since #86335 commits to not checking IPv4 semantics of IPv4-mapped addresses in the `Ipv6Addr` property methods.
2 parents ebebc7a + a5cdff3 commit 423a930

File tree

1 file changed

+45
-0
lines changed

1 file changed

+45
-0
lines changed

library/std/src/net/ip.rs

+45
Original file line numberDiff line numberDiff line change
@@ -379,6 +379,29 @@ impl IpAddr {
379379
pub const fn is_ipv6(&self) -> bool {
380380
matches!(self, IpAddr::V6(_))
381381
}
382+
383+
/// Converts this address to an `IpAddr::V4` if it is a IPv4-mapped IPv6 addresses, otherwise it
384+
/// return `self` as-is.
385+
///
386+
/// # Examples
387+
///
388+
/// ```
389+
/// #![feature(ip)]
390+
/// use std::net::{IpAddr, Ipv4Addr, Ipv6Addr};
391+
///
392+
/// assert_eq!(IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)).to_canonical().is_loopback(), true);
393+
/// assert_eq!(IpAddr::V6(Ipv6Addr::new(0, 0, 0, 0, 0, 0xffff, 0x7f00, 0x1)).is_loopback(), false);
394+
/// assert_eq!(IpAddr::V6(Ipv6Addr::new(0, 0, 0, 0, 0, 0xffff, 0x7f00, 0x1)).to_canonical().is_loopback(), true);
395+
/// ```
396+
#[inline]
397+
#[rustc_const_unstable(feature = "const_ip", issue = "76205")]
398+
#[unstable(feature = "ip", issue = "27709")]
399+
pub const fn to_canonical(&self) -> IpAddr {
400+
match self {
401+
&v4 @ IpAddr::V4(_) => v4,
402+
IpAddr::V6(v6) => v6.to_canonical(),
403+
}
404+
}
382405
}
383406

384407
impl Ipv4Addr {
@@ -1598,6 +1621,28 @@ impl Ipv6Addr {
15981621
}
15991622
}
16001623

1624+
/// Converts this address to an `IpAddr::V4` if it is a IPv4-mapped addresses, otherwise it
1625+
/// returns self wrapped in a `IpAddr::V6`.
1626+
///
1627+
/// # Examples
1628+
///
1629+
/// ```
1630+
/// #![feature(ip)]
1631+
/// use std::net::Ipv6Addr;
1632+
///
1633+
/// assert_eq!(Ipv6Addr::new(0, 0, 0, 0, 0, 0xffff, 0x7f00, 0x1).is_loopback(), false);
1634+
/// assert_eq!(Ipv6Addr::new(0, 0, 0, 0, 0, 0xffff, 0x7f00, 0x1).to_canonical().is_loopback(), true);
1635+
/// ```
1636+
#[inline]
1637+
#[rustc_const_unstable(feature = "const_ipv6", issue = "76205")]
1638+
#[unstable(feature = "ip", issue = "27709")]
1639+
pub const fn to_canonical(&self) -> IpAddr {
1640+
if let Some(mapped) = self.to_ipv4_mapped() {
1641+
return IpAddr::V4(mapped);
1642+
}
1643+
IpAddr::V6(*self)
1644+
}
1645+
16011646
/// Returns the sixteen eight-bit integers the IPv6 address consists of.
16021647
///
16031648
/// ```

0 commit comments

Comments
 (0)