Skip to content

Commit 3cd378c

Browse files
authored
Rollup merge of #119081 - jstasiak:is-ipv4-mapped, r=dtolnay
Add Ipv6Addr::is_ipv4_mapped This change consists of cherry-picking the content from the original PR[1], which got closed due to inactivity, and applying the following changes: * Resolving merge conflicts (obviously) * Linked to to_ipv4_mapped instead of to_ipv4 in the documentation (seems more appropriate) * Added the must_use and rustc_const_unstable attributes the original didn't have I think it's a reasonably useful method to have. [1] #86490
2 parents e8d1c2e + 605504c commit 3cd378c

File tree

2 files changed

+36
-1
lines changed

2 files changed

+36
-1
lines changed

library/core/src/net/ip_addr.rs

+25
Original file line numberDiff line numberDiff line change
@@ -1785,6 +1785,31 @@ impl Ipv6Addr {
17851785
(self.segments()[0] & 0xff00) == 0xff00
17861786
}
17871787

1788+
/// Returns [`true`] if the address is an IPv4-mapped address (`::ffff:0:0/96`).
1789+
///
1790+
/// IPv4-mapped addresses can be converted to their canonical IPv4 address with
1791+
/// [`to_ipv4_mapped`](Ipv6Addr::to_ipv4_mapped).
1792+
///
1793+
/// # Examples
1794+
/// ```
1795+
/// #![feature(ip)]
1796+
///
1797+
/// use std::net::{Ipv4Addr, Ipv6Addr};
1798+
///
1799+
/// let ipv4_mapped = Ipv4Addr::new(192, 0, 2, 255).to_ipv6_mapped();
1800+
/// assert_eq!(ipv4_mapped.is_ipv4_mapped(), true);
1801+
/// assert_eq!(Ipv6Addr::new(0, 0, 0, 0, 0, 0xffff, 0xc000, 0x2ff).is_ipv4_mapped(), true);
1802+
///
1803+
/// assert_eq!(Ipv6Addr::new(0x2001, 0xdb8, 0, 0, 0, 0, 0, 0).is_ipv4_mapped(), false);
1804+
/// ```
1805+
#[rustc_const_unstable(feature = "const_ipv6", issue = "76205")]
1806+
#[unstable(feature = "ip", issue = "27709")]
1807+
#[must_use]
1808+
#[inline]
1809+
pub const fn is_ipv4_mapped(&self) -> bool {
1810+
matches!(self.segments(), [0, 0, 0, 0, 0, 0xffff, _, _])
1811+
}
1812+
17881813
/// Converts this address to an [`IPv4` address] if it's an [IPv4-mapped] address,
17891814
/// as defined in [IETF RFC 4291 section 2.5.5.2], otherwise returns [`None`].
17901815
///

library/core/tests/net/ip_addr.rs

+11-1
Original file line numberDiff line numberDiff line change
@@ -516,6 +516,7 @@ fn ipv6_properties() {
516516
| multicast_realm_local
517517
| multicast_site_local
518518
| multicast_organization_local;
519+
let ipv4_mapped: u32 = 1 << 17;
519520

520521
if ($mask & unspecified) == unspecified {
521522
assert!(ip!($s).is_unspecified());
@@ -592,6 +593,11 @@ fn ipv6_properties() {
592593
assert_eq!(ip!($s).multicast_scope().unwrap(),
593594
Ipv6MulticastScope::Global);
594595
}
596+
if ($mask & ipv4_mapped) == ipv4_mapped {
597+
assert!(ip!($s).is_ipv4_mapped());
598+
} else {
599+
assert!(!ip!($s).is_ipv4_mapped());
600+
}
595601
}
596602
}
597603

@@ -610,6 +616,7 @@ fn ipv6_properties() {
610616
let multicast_site_local: u32 = 1 << 13;
611617
let multicast_organization_local: u32 = 1 << 14;
612618
let multicast_global: u32 = 1 << 15;
619+
let ipv4_mapped: u32 = 1 << 17;
613620

614621
check!("::", &[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0], unspecified);
615622

@@ -622,7 +629,7 @@ fn ipv6_properties() {
622629
check!(
623630
"::ffff:127.0.0.1",
624631
&[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff, 0x7f, 0, 0, 1],
625-
unicast_global
632+
unicast_global | ipv4_mapped
626633
);
627634

628635
check!(
@@ -996,6 +1003,9 @@ fn ipv6_const() {
9961003
const IS_MULTICAST: bool = IP_ADDRESS.is_multicast();
9971004
assert!(!IS_MULTICAST);
9981005

1006+
const IS_IPV4_MAPPED: bool = IP_ADDRESS.is_ipv4_mapped();
1007+
assert!(!IS_IPV4_MAPPED);
1008+
9991009
const IP_V4: Option<Ipv4Addr> = IP_ADDRESS.to_ipv4();
10001010
assert_eq!(IP_V4.unwrap(), Ipv4Addr::new(0, 0, 0, 1));
10011011
}

0 commit comments

Comments
 (0)