blob: 3d815395cfbc3482c65b7dfa00097870fe53fcd6 [file] [log] [blame]
[email protected]9eb7b11b2012-03-28 20:19:311// Copyright (c) 2012 The Chromium Authors. All rights reserved.
[email protected]76ff86a2011-03-04 03:21:312// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "net/base/ip_endpoint.h"
6
tfarina0dc490e2015-12-22 20:51:467#include "build/build_config.h"
jsbellcea42a52015-11-30 23:50:258
[email protected]76ff86a2011-03-04 03:21:319#if defined(OS_WIN)
10#include <winsock2.h>
tfarinae39d60312015-12-24 14:03:5711#include <ws2bth.h>
[email protected]76ff86a2011-03-04 03:21:3112#elif defined(OS_POSIX)
13#include <netinet/in.h>
14#endif
tfarina0dc490e2015-12-22 20:51:4615
Erik Chenc073e802020-04-22 21:21:5416#include <string.h>
17
tfarina0dc490e2015-12-22 20:51:4618#include <tuple>
19
20#include "base/logging.h"
21#include "base/strings/string_number_conversions.h"
22#include "base/sys_byteorder.h"
23#include "net/base/ip_address.h"
tfarinac5336ca2016-01-11 20:49:3624
25#if defined(OS_WIN)
26#include "net/base/winsock_util.h"
27#endif
[email protected]76ff86a2011-03-04 03:21:3128
29namespace net {
30
[email protected]7054e78f2012-05-07 21:44:5631namespace {
tfarina0dc490e2015-12-22 20:51:4632
[email protected]7054e78f2012-05-07 21:44:5633// By definition, socklen_t is large enough to hold both sizes.
34const socklen_t kSockaddrInSize = sizeof(struct sockaddr_in);
35const socklen_t kSockaddrIn6Size = sizeof(struct sockaddr_in6);
tfarina0dc490e2015-12-22 20:51:4636
tfarinae39d60312015-12-24 14:03:5737// Extracts the address and port portions of a sockaddr.
38bool GetIPAddressFromSockAddr(const struct sockaddr* sock_addr,
39 socklen_t sock_addr_len,
40 const uint8_t** address,
41 size_t* address_len,
42 uint16_t* port) {
43 if (sock_addr->sa_family == AF_INET) {
44 if (sock_addr_len < static_cast<socklen_t>(sizeof(struct sockaddr_in)))
45 return false;
46 const struct sockaddr_in* addr =
47 reinterpret_cast<const struct sockaddr_in*>(sock_addr);
48 *address = reinterpret_cast<const uint8_t*>(&addr->sin_addr);
martijn56506d02016-05-03 18:44:0049 *address_len = IPAddress::kIPv4AddressSize;
tfarinae39d60312015-12-24 14:03:5750 if (port)
51 *port = base::NetToHost16(addr->sin_port);
52 return true;
53 }
54
55 if (sock_addr->sa_family == AF_INET6) {
56 if (sock_addr_len < static_cast<socklen_t>(sizeof(struct sockaddr_in6)))
57 return false;
58 const struct sockaddr_in6* addr =
59 reinterpret_cast<const struct sockaddr_in6*>(sock_addr);
60 *address = reinterpret_cast<const uint8_t*>(&addr->sin6_addr);
martijn56506d02016-05-03 18:44:0061 *address_len = IPAddress::kIPv6AddressSize;
tfarinae39d60312015-12-24 14:03:5762 if (port)
63 *port = base::NetToHost16(addr->sin6_port);
64 return true;
65 }
66
67#if defined(OS_WIN)
68 if (sock_addr->sa_family == AF_BTH) {
69 if (sock_addr_len < static_cast<socklen_t>(sizeof(SOCKADDR_BTH)))
70 return false;
71 const SOCKADDR_BTH* addr = reinterpret_cast<const SOCKADDR_BTH*>(sock_addr);
72 *address = reinterpret_cast<const uint8_t*>(&addr->btAddr);
73 *address_len = kBluetoothAddressSize;
74 if (port)
75 *port = static_cast<uint16_t>(addr->port);
76 return true;
77 }
78#endif
79
80 return false; // Unrecognized |sa_family|.
81}
82
tfarina0dc490e2015-12-22 20:51:4683} // namespace
[email protected]7054e78f2012-05-07 21:44:5684
[email protected]76ff86a2011-03-04 03:21:3185IPEndPoint::IPEndPoint() : port_(0) {}
86
Chris Watkins68b15032017-12-01 03:07:1387IPEndPoint::~IPEndPoint() = default;
[email protected]034714b2011-03-04 04:09:1488
tfarina0dc490e2015-12-22 20:51:4689IPEndPoint::IPEndPoint(const IPAddress& address, uint16_t port)
martijn98acb9b2016-01-27 07:14:0790 : address_(address), port_(port) {}
tfarina0dc490e2015-12-22 20:51:4691
[email protected]76ff86a2011-03-04 03:21:3192IPEndPoint::IPEndPoint(const IPEndPoint& endpoint) {
93 address_ = endpoint.address_;
94 port_ = endpoint.port_;
95}
96
[email protected]e466aaf2012-12-13 01:46:4497AddressFamily IPEndPoint::GetFamily() const {
martijn46f5edc2016-04-12 13:54:2198 return GetAddressFamily(address_);
[email protected]e466aaf2012-12-13 01:46:4499}
100
101int IPEndPoint::GetSockAddrFamily() const {
[email protected]43d4a0262011-03-09 19:26:04102 switch (address_.size()) {
martijn56506d02016-05-03 18:44:00103 case IPAddress::kIPv4AddressSize:
[email protected]43d4a0262011-03-09 19:26:04104 return AF_INET;
martijn56506d02016-05-03 18:44:00105 case IPAddress::kIPv6AddressSize:
[email protected]43d4a0262011-03-09 19:26:04106 return AF_INET6;
[email protected]7054e78f2012-05-07 21:44:56107 default:
[email protected]43d4a0262011-03-09 19:26:04108 NOTREACHED() << "Bad IP address";
[email protected]48e410fb2011-04-07 17:10:46109 return AF_UNSPEC;
[email protected]43d4a0262011-03-09 19:26:04110 }
111}
112
113bool IPEndPoint::ToSockAddr(struct sockaddr* address,
[email protected]7054e78f2012-05-07 21:44:56114 socklen_t* address_length) const {
[email protected]76ff86a2011-03-04 03:21:31115 DCHECK(address);
116 DCHECK(address_length);
117 switch (address_.size()) {
martijn56506d02016-05-03 18:44:00118 case IPAddress::kIPv4AddressSize: {
[email protected]7054e78f2012-05-07 21:44:56119 if (*address_length < kSockaddrInSize)
[email protected]76ff86a2011-03-04 03:21:31120 return false;
[email protected]7054e78f2012-05-07 21:44:56121 *address_length = kSockaddrInSize;
[email protected]76ff86a2011-03-04 03:21:31122 struct sockaddr_in* addr = reinterpret_cast<struct sockaddr_in*>(address);
123 memset(addr, 0, sizeof(struct sockaddr_in));
124 addr->sin_family = AF_INET;
[email protected]9eb7b11b2012-03-28 20:19:31125 addr->sin_port = base::HostToNet16(port_);
martijn56506d02016-05-03 18:44:00126 memcpy(&addr->sin_addr, address_.bytes().data(),
127 IPAddress::kIPv4AddressSize);
[email protected]76ff86a2011-03-04 03:21:31128 break;
129 }
martijn56506d02016-05-03 18:44:00130 case IPAddress::kIPv6AddressSize: {
[email protected]7054e78f2012-05-07 21:44:56131 if (*address_length < kSockaddrIn6Size)
[email protected]76ff86a2011-03-04 03:21:31132 return false;
[email protected]7054e78f2012-05-07 21:44:56133 *address_length = kSockaddrIn6Size;
[email protected]76ff86a2011-03-04 03:21:31134 struct sockaddr_in6* addr6 =
135 reinterpret_cast<struct sockaddr_in6*>(address);
136 memset(addr6, 0, sizeof(struct sockaddr_in6));
137 addr6->sin6_family = AF_INET6;
[email protected]9eb7b11b2012-03-28 20:19:31138 addr6->sin6_port = base::HostToNet16(port_);
martijn56506d02016-05-03 18:44:00139 memcpy(&addr6->sin6_addr, address_.bytes().data(),
140 IPAddress::kIPv6AddressSize);
[email protected]76ff86a2011-03-04 03:21:31141 break;
142 }
[email protected]7054e78f2012-05-07 21:44:56143 default:
144 return false;
[email protected]76ff86a2011-03-04 03:21:31145 }
146 return true;
147}
148
[email protected]0c365402012-09-08 09:44:22149bool IPEndPoint::FromSockAddr(const struct sockaddr* sock_addr,
150 socklen_t sock_addr_len) {
151 DCHECK(sock_addr);
152
wtc69f8ea82015-06-04 00:08:13153 const uint8_t* address;
[email protected]0c365402012-09-08 09:44:22154 size_t address_len;
wtc69f8ea82015-06-04 00:08:13155 uint16_t port;
[email protected]0c365402012-09-08 09:44:22156 if (!GetIPAddressFromSockAddr(sock_addr, sock_addr_len, &address,
157 &address_len, &port)) {
158 return false;
[email protected]76ff86a2011-03-04 03:21:31159 }
[email protected]0c365402012-09-08 09:44:22160
martijn98acb9b2016-01-27 07:14:07161 address_ = net::IPAddress(address, address_len);
[email protected]0c365402012-09-08 09:44:22162 port_ = port;
[email protected]76ff86a2011-03-04 03:21:31163 return true;
164}
165
[email protected]1a2123c62011-03-12 04:42:39166std::string IPEndPoint::ToString() const {
martijn884d81a2016-04-30 13:52:18167 return IPAddressToStringWithPort(address_, port_);
[email protected]7054e78f2012-05-07 21:44:56168}
169
170std::string IPEndPoint::ToStringWithoutPort() const {
martijn98acb9b2016-01-27 07:14:07171 return address_.ToString();
[email protected]1a2123c62011-03-12 04:42:39172}
173
jsbellcea42a52015-11-30 23:50:25174bool IPEndPoint::operator<(const IPEndPoint& other) const {
[email protected]17e06f12011-04-05 23:10:10175 // Sort IPv4 before IPv6.
jsbellcea42a52015-11-30 23:50:25176 if (address_.size() != other.address_.size()) {
177 return address_.size() < other.address_.size();
[email protected]17e06f12011-04-05 23:10:10178 }
jsbellcea42a52015-11-30 23:50:25179 return std::tie(address_, port_) < std::tie(other.address_, other.port_);
[email protected]76ff86a2011-03-04 03:21:31180}
181
jsbellcea42a52015-11-30 23:50:25182bool IPEndPoint::operator==(const IPEndPoint& other) const {
183 return address_ == other.address_ && port_ == other.port_;
[email protected]76ff86a2011-03-04 03:21:31184}
185
Victor Vasilievbee79ea2019-05-15 01:25:48186bool IPEndPoint::operator!=(const IPEndPoint& that) const {
187 return !(*this == that);
188}
189
[email protected]76ff86a2011-03-04 03:21:31190} // namespace net