xref: /aosp_15_r20/external/cronet/net/base/ip_endpoint.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2012 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #include "net/base/ip_endpoint.h"
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #include <string.h>
8*6777b538SAndroid Build Coastguard Worker 
9*6777b538SAndroid Build Coastguard Worker #include <optional>
10*6777b538SAndroid Build Coastguard Worker #include <ostream>
11*6777b538SAndroid Build Coastguard Worker #include <tuple>
12*6777b538SAndroid Build Coastguard Worker #include <utility>
13*6777b538SAndroid Build Coastguard Worker 
14*6777b538SAndroid Build Coastguard Worker #include "base/check.h"
15*6777b538SAndroid Build Coastguard Worker #include "base/check_op.h"
16*6777b538SAndroid Build Coastguard Worker #include "base/containers/span.h"
17*6777b538SAndroid Build Coastguard Worker #include "base/notreached.h"
18*6777b538SAndroid Build Coastguard Worker #include "base/numerics/safe_conversions.h"
19*6777b538SAndroid Build Coastguard Worker #include "base/strings/string_number_conversions.h"
20*6777b538SAndroid Build Coastguard Worker #include "base/sys_byteorder.h"
21*6777b538SAndroid Build Coastguard Worker #include "base/values.h"
22*6777b538SAndroid Build Coastguard Worker #include "build/build_config.h"
23*6777b538SAndroid Build Coastguard Worker #include "net/base/ip_address.h"
24*6777b538SAndroid Build Coastguard Worker #include "net/base/sys_addrinfo.h"
25*6777b538SAndroid Build Coastguard Worker 
26*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN)
27*6777b538SAndroid Build Coastguard Worker #include <winsock2.h>
28*6777b538SAndroid Build Coastguard Worker 
29*6777b538SAndroid Build Coastguard Worker #include <ws2bth.h>
30*6777b538SAndroid Build Coastguard Worker 
31*6777b538SAndroid Build Coastguard Worker #include "net/base/winsock_util.h"  // For kBluetoothAddressSize
32*6777b538SAndroid Build Coastguard Worker #endif
33*6777b538SAndroid Build Coastguard Worker 
34*6777b538SAndroid Build Coastguard Worker namespace net {
35*6777b538SAndroid Build Coastguard Worker 
36*6777b538SAndroid Build Coastguard Worker namespace {
37*6777b538SAndroid Build Coastguard Worker 
38*6777b538SAndroid Build Coastguard Worker // Value dictionary keys
39*6777b538SAndroid Build Coastguard Worker constexpr std::string_view kValueAddressKey = "address";
40*6777b538SAndroid Build Coastguard Worker constexpr std::string_view kValuePortKey = "port";
41*6777b538SAndroid Build Coastguard Worker 
42*6777b538SAndroid Build Coastguard Worker }  // namespace
43*6777b538SAndroid Build Coastguard Worker 
44*6777b538SAndroid Build Coastguard Worker // static
FromValue(const base::Value & value)45*6777b538SAndroid Build Coastguard Worker std::optional<IPEndPoint> IPEndPoint::FromValue(const base::Value& value) {
46*6777b538SAndroid Build Coastguard Worker   const base::Value::Dict* dict = value.GetIfDict();
47*6777b538SAndroid Build Coastguard Worker   if (!dict)
48*6777b538SAndroid Build Coastguard Worker     return std::nullopt;
49*6777b538SAndroid Build Coastguard Worker 
50*6777b538SAndroid Build Coastguard Worker   const base::Value* address_value = dict->Find(kValueAddressKey);
51*6777b538SAndroid Build Coastguard Worker   if (!address_value)
52*6777b538SAndroid Build Coastguard Worker     return std::nullopt;
53*6777b538SAndroid Build Coastguard Worker   std::optional<IPAddress> address = IPAddress::FromValue(*address_value);
54*6777b538SAndroid Build Coastguard Worker   if (!address.has_value())
55*6777b538SAndroid Build Coastguard Worker     return std::nullopt;
56*6777b538SAndroid Build Coastguard Worker   // Expect IPAddress to only allow deserializing valid addresses.
57*6777b538SAndroid Build Coastguard Worker   DCHECK(address.value().IsValid());
58*6777b538SAndroid Build Coastguard Worker 
59*6777b538SAndroid Build Coastguard Worker   std::optional<int> port = dict->FindInt(kValuePortKey);
60*6777b538SAndroid Build Coastguard Worker   if (!port.has_value() ||
61*6777b538SAndroid Build Coastguard Worker       !base::IsValueInRangeForNumericType<uint16_t>(port.value())) {
62*6777b538SAndroid Build Coastguard Worker     return std::nullopt;
63*6777b538SAndroid Build Coastguard Worker   }
64*6777b538SAndroid Build Coastguard Worker 
65*6777b538SAndroid Build Coastguard Worker   return IPEndPoint(address.value(),
66*6777b538SAndroid Build Coastguard Worker                     base::checked_cast<uint16_t>(port.value()));
67*6777b538SAndroid Build Coastguard Worker }
68*6777b538SAndroid Build Coastguard Worker 
69*6777b538SAndroid Build Coastguard Worker IPEndPoint::IPEndPoint() = default;
70*6777b538SAndroid Build Coastguard Worker 
71*6777b538SAndroid Build Coastguard Worker IPEndPoint::~IPEndPoint() = default;
72*6777b538SAndroid Build Coastguard Worker 
IPEndPoint(const IPAddress & address,uint16_t port)73*6777b538SAndroid Build Coastguard Worker IPEndPoint::IPEndPoint(const IPAddress& address, uint16_t port)
74*6777b538SAndroid Build Coastguard Worker     : address_(address), port_(port) {}
75*6777b538SAndroid Build Coastguard Worker 
76*6777b538SAndroid Build Coastguard Worker IPEndPoint::IPEndPoint(const IPEndPoint& endpoint) = default;
77*6777b538SAndroid Build Coastguard Worker 
port() const78*6777b538SAndroid Build Coastguard Worker uint16_t IPEndPoint::port() const {
79*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN)
80*6777b538SAndroid Build Coastguard Worker   DCHECK_NE(address_.size(), kBluetoothAddressSize);
81*6777b538SAndroid Build Coastguard Worker #endif
82*6777b538SAndroid Build Coastguard Worker   return port_;
83*6777b538SAndroid Build Coastguard Worker }
84*6777b538SAndroid Build Coastguard Worker 
GetFamily() const85*6777b538SAndroid Build Coastguard Worker AddressFamily IPEndPoint::GetFamily() const {
86*6777b538SAndroid Build Coastguard Worker   return GetAddressFamily(address_);
87*6777b538SAndroid Build Coastguard Worker }
88*6777b538SAndroid Build Coastguard Worker 
GetSockAddrFamily() const89*6777b538SAndroid Build Coastguard Worker int IPEndPoint::GetSockAddrFamily() const {
90*6777b538SAndroid Build Coastguard Worker   switch (address_.size()) {
91*6777b538SAndroid Build Coastguard Worker     case IPAddress::kIPv4AddressSize:
92*6777b538SAndroid Build Coastguard Worker       return AF_INET;
93*6777b538SAndroid Build Coastguard Worker     case IPAddress::kIPv6AddressSize:
94*6777b538SAndroid Build Coastguard Worker       return AF_INET6;
95*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN)
96*6777b538SAndroid Build Coastguard Worker     case kBluetoothAddressSize:
97*6777b538SAndroid Build Coastguard Worker       return AF_BTH;
98*6777b538SAndroid Build Coastguard Worker #endif
99*6777b538SAndroid Build Coastguard Worker     default:
100*6777b538SAndroid Build Coastguard Worker       NOTREACHED() << "Bad IP address";
101*6777b538SAndroid Build Coastguard Worker       return AF_UNSPEC;
102*6777b538SAndroid Build Coastguard Worker   }
103*6777b538SAndroid Build Coastguard Worker }
104*6777b538SAndroid Build Coastguard Worker 
ToSockAddr(struct sockaddr * address,socklen_t * address_length) const105*6777b538SAndroid Build Coastguard Worker bool IPEndPoint::ToSockAddr(struct sockaddr* address,
106*6777b538SAndroid Build Coastguard Worker                             socklen_t* address_length) const {
107*6777b538SAndroid Build Coastguard Worker   // By definition, socklen_t is large enough to hold both sizes.
108*6777b538SAndroid Build Coastguard Worker   constexpr socklen_t kSockaddrInSize =
109*6777b538SAndroid Build Coastguard Worker       static_cast<socklen_t>(sizeof(struct sockaddr_in));
110*6777b538SAndroid Build Coastguard Worker   constexpr socklen_t kSockaddrIn6Size =
111*6777b538SAndroid Build Coastguard Worker       static_cast<socklen_t>(sizeof(struct sockaddr_in6));
112*6777b538SAndroid Build Coastguard Worker 
113*6777b538SAndroid Build Coastguard Worker   DCHECK(address);
114*6777b538SAndroid Build Coastguard Worker   DCHECK(address_length);
115*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN)
116*6777b538SAndroid Build Coastguard Worker   DCHECK_NE(address_.size(), kBluetoothAddressSize);
117*6777b538SAndroid Build Coastguard Worker #endif
118*6777b538SAndroid Build Coastguard Worker   switch (address_.size()) {
119*6777b538SAndroid Build Coastguard Worker     case IPAddress::kIPv4AddressSize: {
120*6777b538SAndroid Build Coastguard Worker       if (*address_length < kSockaddrInSize)
121*6777b538SAndroid Build Coastguard Worker         return false;
122*6777b538SAndroid Build Coastguard Worker       *address_length = kSockaddrInSize;
123*6777b538SAndroid Build Coastguard Worker       struct sockaddr_in* addr = reinterpret_cast<struct sockaddr_in*>(address);
124*6777b538SAndroid Build Coastguard Worker       memset(addr, 0, sizeof(struct sockaddr_in));
125*6777b538SAndroid Build Coastguard Worker       addr->sin_family = AF_INET;
126*6777b538SAndroid Build Coastguard Worker       addr->sin_port = base::HostToNet16(port_);
127*6777b538SAndroid Build Coastguard Worker       memcpy(&addr->sin_addr, address_.bytes().data(),
128*6777b538SAndroid Build Coastguard Worker              IPAddress::kIPv4AddressSize);
129*6777b538SAndroid Build Coastguard Worker       break;
130*6777b538SAndroid Build Coastguard Worker     }
131*6777b538SAndroid Build Coastguard Worker     case IPAddress::kIPv6AddressSize: {
132*6777b538SAndroid Build Coastguard Worker       if (*address_length < kSockaddrIn6Size)
133*6777b538SAndroid Build Coastguard Worker         return false;
134*6777b538SAndroid Build Coastguard Worker       *address_length = kSockaddrIn6Size;
135*6777b538SAndroid Build Coastguard Worker       struct sockaddr_in6* addr6 =
136*6777b538SAndroid Build Coastguard Worker           reinterpret_cast<struct sockaddr_in6*>(address);
137*6777b538SAndroid Build Coastguard Worker       memset(addr6, 0, sizeof(struct sockaddr_in6));
138*6777b538SAndroid Build Coastguard Worker       addr6->sin6_family = AF_INET6;
139*6777b538SAndroid Build Coastguard Worker       addr6->sin6_port = base::HostToNet16(port_);
140*6777b538SAndroid Build Coastguard Worker       memcpy(&addr6->sin6_addr, address_.bytes().data(),
141*6777b538SAndroid Build Coastguard Worker              IPAddress::kIPv6AddressSize);
142*6777b538SAndroid Build Coastguard Worker       break;
143*6777b538SAndroid Build Coastguard Worker     }
144*6777b538SAndroid Build Coastguard Worker     default:
145*6777b538SAndroid Build Coastguard Worker       return false;
146*6777b538SAndroid Build Coastguard Worker   }
147*6777b538SAndroid Build Coastguard Worker   return true;
148*6777b538SAndroid Build Coastguard Worker }
149*6777b538SAndroid Build Coastguard Worker 
FromSockAddr(const struct sockaddr * sock_addr,socklen_t sock_addr_len)150*6777b538SAndroid Build Coastguard Worker bool IPEndPoint::FromSockAddr(const struct sockaddr* sock_addr,
151*6777b538SAndroid Build Coastguard Worker                               socklen_t sock_addr_len) {
152*6777b538SAndroid Build Coastguard Worker   DCHECK(sock_addr);
153*6777b538SAndroid Build Coastguard Worker   switch (sock_addr->sa_family) {
154*6777b538SAndroid Build Coastguard Worker     case AF_INET: {
155*6777b538SAndroid Build Coastguard Worker       if (sock_addr_len < static_cast<socklen_t>(sizeof(struct sockaddr_in)))
156*6777b538SAndroid Build Coastguard Worker         return false;
157*6777b538SAndroid Build Coastguard Worker       const struct sockaddr_in* addr =
158*6777b538SAndroid Build Coastguard Worker           reinterpret_cast<const struct sockaddr_in*>(sock_addr);
159*6777b538SAndroid Build Coastguard Worker       *this = IPEndPoint(
160*6777b538SAndroid Build Coastguard Worker           // `s_addr` is a `uint32_t`, but it is already in network byte order.
161*6777b538SAndroid Build Coastguard Worker           IPAddress(base::as_bytes(base::span_from_ref(addr->sin_addr.s_addr))),
162*6777b538SAndroid Build Coastguard Worker           base::NetToHost16(addr->sin_port));
163*6777b538SAndroid Build Coastguard Worker       return true;
164*6777b538SAndroid Build Coastguard Worker     }
165*6777b538SAndroid Build Coastguard Worker     case AF_INET6: {
166*6777b538SAndroid Build Coastguard Worker       if (sock_addr_len < static_cast<socklen_t>(sizeof(struct sockaddr_in6)))
167*6777b538SAndroid Build Coastguard Worker         return false;
168*6777b538SAndroid Build Coastguard Worker       const struct sockaddr_in6* addr =
169*6777b538SAndroid Build Coastguard Worker           reinterpret_cast<const struct sockaddr_in6*>(sock_addr);
170*6777b538SAndroid Build Coastguard Worker       *this = IPEndPoint(IPAddress(addr->sin6_addr.s6_addr),
171*6777b538SAndroid Build Coastguard Worker                          base::NetToHost16(addr->sin6_port));
172*6777b538SAndroid Build Coastguard Worker       return true;
173*6777b538SAndroid Build Coastguard Worker     }
174*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN)
175*6777b538SAndroid Build Coastguard Worker     case AF_BTH: {
176*6777b538SAndroid Build Coastguard Worker       if (sock_addr_len < static_cast<socklen_t>(sizeof(SOCKADDR_BTH)))
177*6777b538SAndroid Build Coastguard Worker         return false;
178*6777b538SAndroid Build Coastguard Worker       const SOCKADDR_BTH* addr =
179*6777b538SAndroid Build Coastguard Worker           reinterpret_cast<const SOCKADDR_BTH*>(sock_addr);
180*6777b538SAndroid Build Coastguard Worker       *this = IPEndPoint();
181*6777b538SAndroid Build Coastguard Worker       // A bluetooth address is 6 bytes, but btAddr is a ULONGLONG, so we take a
182*6777b538SAndroid Build Coastguard Worker       // prefix of it.
183*6777b538SAndroid Build Coastguard Worker       address_ = IPAddress(base::as_bytes(base::span_from_ref(addr->btAddr))
184*6777b538SAndroid Build Coastguard Worker                                .first(kBluetoothAddressSize));
185*6777b538SAndroid Build Coastguard Worker       // Intentionally ignoring Bluetooth port. It is a ULONG, but
186*6777b538SAndroid Build Coastguard Worker       // `IPEndPoint::port_` is a uint16_t. See https://crbug.com/1231273.
187*6777b538SAndroid Build Coastguard Worker       return true;
188*6777b538SAndroid Build Coastguard Worker     }
189*6777b538SAndroid Build Coastguard Worker #endif
190*6777b538SAndroid Build Coastguard Worker   }
191*6777b538SAndroid Build Coastguard Worker   return false;  // Unrecognized |sa_family|.
192*6777b538SAndroid Build Coastguard Worker }
193*6777b538SAndroid Build Coastguard Worker 
ToString() const194*6777b538SAndroid Build Coastguard Worker std::string IPEndPoint::ToString() const {
195*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN)
196*6777b538SAndroid Build Coastguard Worker   DCHECK_NE(address_.size(), kBluetoothAddressSize);
197*6777b538SAndroid Build Coastguard Worker #endif
198*6777b538SAndroid Build Coastguard Worker   return IPAddressToStringWithPort(address_, port_);
199*6777b538SAndroid Build Coastguard Worker }
200*6777b538SAndroid Build Coastguard Worker 
ToStringWithoutPort() const201*6777b538SAndroid Build Coastguard Worker std::string IPEndPoint::ToStringWithoutPort() const {
202*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(IS_WIN)
203*6777b538SAndroid Build Coastguard Worker   DCHECK_NE(address_.size(), kBluetoothAddressSize);
204*6777b538SAndroid Build Coastguard Worker #endif
205*6777b538SAndroid Build Coastguard Worker   return address_.ToString();
206*6777b538SAndroid Build Coastguard Worker }
207*6777b538SAndroid Build Coastguard Worker 
operator <(const IPEndPoint & other) const208*6777b538SAndroid Build Coastguard Worker bool IPEndPoint::operator<(const IPEndPoint& other) const {
209*6777b538SAndroid Build Coastguard Worker   // Sort IPv4 before IPv6.
210*6777b538SAndroid Build Coastguard Worker   if (address_.size() != other.address_.size()) {
211*6777b538SAndroid Build Coastguard Worker     return address_.size() < other.address_.size();
212*6777b538SAndroid Build Coastguard Worker   }
213*6777b538SAndroid Build Coastguard Worker   return std::tie(address_, port_) < std::tie(other.address_, other.port_);
214*6777b538SAndroid Build Coastguard Worker }
215*6777b538SAndroid Build Coastguard Worker 
operator ==(const IPEndPoint & other) const216*6777b538SAndroid Build Coastguard Worker bool IPEndPoint::operator==(const IPEndPoint& other) const {
217*6777b538SAndroid Build Coastguard Worker   return address_ == other.address_ && port_ == other.port_;
218*6777b538SAndroid Build Coastguard Worker }
219*6777b538SAndroid Build Coastguard Worker 
operator !=(const IPEndPoint & that) const220*6777b538SAndroid Build Coastguard Worker bool IPEndPoint::operator!=(const IPEndPoint& that) const {
221*6777b538SAndroid Build Coastguard Worker   return !(*this == that);
222*6777b538SAndroid Build Coastguard Worker }
223*6777b538SAndroid Build Coastguard Worker 
ToValue() const224*6777b538SAndroid Build Coastguard Worker base::Value IPEndPoint::ToValue() const {
225*6777b538SAndroid Build Coastguard Worker   base::Value::Dict dict;
226*6777b538SAndroid Build Coastguard Worker 
227*6777b538SAndroid Build Coastguard Worker   DCHECK(address_.IsValid());
228*6777b538SAndroid Build Coastguard Worker   dict.Set(kValueAddressKey, address_.ToValue());
229*6777b538SAndroid Build Coastguard Worker   dict.Set(kValuePortKey, port_);
230*6777b538SAndroid Build Coastguard Worker 
231*6777b538SAndroid Build Coastguard Worker   return base::Value(std::move(dict));
232*6777b538SAndroid Build Coastguard Worker }
233*6777b538SAndroid Build Coastguard Worker 
operator <<(std::ostream & os,const IPEndPoint & ip_endpoint)234*6777b538SAndroid Build Coastguard Worker std::ostream& operator<<(std::ostream& os, const IPEndPoint& ip_endpoint) {
235*6777b538SAndroid Build Coastguard Worker   return os << ip_endpoint.ToString();
236*6777b538SAndroid Build Coastguard Worker }
237*6777b538SAndroid Build Coastguard Worker 
238*6777b538SAndroid Build Coastguard Worker }  // namespace net
239