blob: 0ac4bc8b0e65edb25d3d9bd7ce62143df5bd03b0 [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "net/base/ip_endpoint.h"
#include "build/build_config.h"
#if defined(OS_WIN)
#include <winsock2.h>
#include <ws2bth.h>
#elif defined(OS_POSIX)
#include <netinet/in.h>
#endif
#include <tuple>
#include "base/logging.h"
#include "base/strings/string_number_conversions.h"
#include "base/sys_byteorder.h"
#include "net/base/ip_address.h"
#if defined(OS_WIN)
#include "net/base/winsock_util.h"
#endif
#if defined(STARBOARD)
#include "starboard/memory.h"
#include "starboard/types.h"
#endif
namespace net {
#if defined(STARBOARD)
bool GetIPAddressFromSbSocketAddress(const SbSocketAddress* address,
const unsigned char** out_address_data,
size_t* out_address_len,
uint16_t* out_port) {
DCHECK(address);
DCHECK(out_address_data);
DCHECK(out_address_len);
if (out_port) {
*out_port = address->port;
}
*out_address_data = address->address;
switch (address->type) {
case kSbSocketAddressTypeIpv4:
*out_address_len = IPAddress::kIPv4AddressSize;
break;
case kSbSocketAddressTypeIpv6:
*out_address_len = IPAddress::kIPv6AddressSize;
break;
default:
NOTREACHED();
return false;
}
return true;
}
#else // defined(STARBOARD)
namespace {
// By definition, socklen_t is large enough to hold both sizes.
const socklen_t kSockaddrInSize = sizeof(struct sockaddr_in);
const socklen_t kSockaddrIn6Size = sizeof(struct sockaddr_in6);
// Extracts the address and port portions of a sockaddr.
bool GetIPAddressFromSockAddr(const struct sockaddr* sock_addr,
socklen_t sock_addr_len,
const uint8_t** address,
size_t* address_len,
uint16_t* port) {
if (sock_addr->sa_family == AF_INET) {
if (sock_addr_len < static_cast<socklen_t>(sizeof(struct sockaddr_in)))
return false;
const struct sockaddr_in* addr =
reinterpret_cast<const struct sockaddr_in*>(sock_addr);
*address = reinterpret_cast<const uint8_t*>(&addr->sin_addr);
*address_len = IPAddress::kIPv4AddressSize;
if (port)
*port = base::NetToHost16(addr->sin_port);
return true;
}
if (sock_addr->sa_family == AF_INET6) {
if (sock_addr_len < static_cast<socklen_t>(sizeof(struct sockaddr_in6)))
return false;
const struct sockaddr_in6* addr =
reinterpret_cast<const struct sockaddr_in6*>(sock_addr);
*address = reinterpret_cast<const uint8_t*>(&addr->sin6_addr);
*address_len = IPAddress::kIPv6AddressSize;
if (port)
*port = base::NetToHost16(addr->sin6_port);
return true;
}
#if defined(OS_WIN)
if (sock_addr->sa_family == AF_BTH) {
if (sock_addr_len < static_cast<socklen_t>(sizeof(SOCKADDR_BTH)))
return false;
const SOCKADDR_BTH* addr = reinterpret_cast<const SOCKADDR_BTH*>(sock_addr);
*address = reinterpret_cast<const uint8_t*>(&addr->btAddr);
*address_len = kBluetoothAddressSize;
if (port)
*port = static_cast<uint16_t>(addr->port);
return true;
}
#endif
return false; // Unrecognized |sa_family|.
}
} // namespace
#endif // defined(STARBOARD)
IPEndPoint::IPEndPoint() : port_(0) {}
IPEndPoint::~IPEndPoint() = default;
IPEndPoint::IPEndPoint(const IPAddress& address, uint16_t port)
: address_(address), port_(port) {}
IPEndPoint::IPEndPoint(const IPEndPoint& endpoint) {
address_ = endpoint.address_;
port_ = endpoint.port_;
}
AddressFamily IPEndPoint::GetFamily() const {
return GetAddressFamily(address_);
}
#if !defined(STARBOARD)
int IPEndPoint::GetSockAddrFamily() const {
switch (address_.size()) {
case IPAddress::kIPv4AddressSize:
return AF_INET;
case IPAddress::kIPv6AddressSize:
return AF_INET6;
default:
NOTREACHED() << "Bad IP address";
return AF_UNSPEC;
}
}
#endif // !defined(STARBOARD)
#if defined(STARBOARD)
// static
IPEndPoint IPEndPoint::GetForAllInterfaces(int port) {
// Directly construct the 0.0.0.0 address with the given port.
IPAddress address(0, 0, 0, 0);
return IPEndPoint(address, port);
}
bool IPEndPoint::ToSbSocketAddress(SbSocketAddress* out_address) const {
DCHECK(out_address);
out_address->port = port_;
memset(out_address->address, 0, sizeof(out_address->address));
switch (GetFamily()) {
case ADDRESS_FAMILY_IPV4:
out_address->type = kSbSocketAddressTypeIpv4;
memcpy(&out_address->address, address_.bytes().data(),
IPAddress::kIPv4AddressSize);
break;
case ADDRESS_FAMILY_IPV6:
out_address->type = kSbSocketAddressTypeIpv6;
memcpy(&out_address->address, address_.bytes().data(),
IPAddress::kIPv6AddressSize);
break;
default:
NOTREACHED();
return false;
}
return true;
}
bool IPEndPoint::FromSbSocketAddress(const SbSocketAddress* address) {
DCHECK(address);
const uint8_t* address_data;
size_t address_len;
uint16_t port;
if (!GetIPAddressFromSbSocketAddress(address, &address_data, &address_len,
&port)) {
return false;
}
address_ = net::IPAddress(address_data, address_len);
port_ = port;
return true;
}
#else // defined(STARBOARD)
bool IPEndPoint::ToSockAddr(struct sockaddr* address,
socklen_t* address_length) const {
DCHECK(address);
DCHECK(address_length);
switch (address_.size()) {
case IPAddress::kIPv4AddressSize: {
if (*address_length < kSockaddrInSize)
return false;
*address_length = kSockaddrInSize;
struct sockaddr_in* addr = reinterpret_cast<struct sockaddr_in*>(address);
memset(addr, 0, sizeof(struct sockaddr_in));
addr->sin_family = AF_INET;
addr->sin_port = base::HostToNet16(port_);
memcpy(&addr->sin_addr, address_.bytes().data(),
IPAddress::kIPv4AddressSize);
break;
}
case IPAddress::kIPv6AddressSize: {
if (*address_length < kSockaddrIn6Size)
return false;
*address_length = kSockaddrIn6Size;
struct sockaddr_in6* addr6 =
reinterpret_cast<struct sockaddr_in6*>(address);
memset(addr6, 0, sizeof(struct sockaddr_in6));
addr6->sin6_family = AF_INET6;
addr6->sin6_port = base::HostToNet16(port_);
memcpy(&addr6->sin6_addr, address_.bytes().data(),
IPAddress::kIPv6AddressSize);
break;
}
default:
return false;
}
return true;
}
bool IPEndPoint::FromSockAddr(const struct sockaddr* sock_addr,
socklen_t sock_addr_len) {
DCHECK(sock_addr);
const uint8_t* address;
size_t address_len;
uint16_t port;
if (!GetIPAddressFromSockAddr(sock_addr, sock_addr_len, &address,
&address_len, &port)) {
return false;
}
address_ = net::IPAddress(address, address_len);
port_ = port;
return true;
}
#endif // defined(STARBOARD)
std::string IPEndPoint::ToString() const {
return IPAddressToStringWithPort(address_, port_);
}
std::string IPEndPoint::ToStringWithoutPort() const {
return address_.ToString();
}
bool IPEndPoint::operator<(const IPEndPoint& other) const {
// Sort IPv4 before IPv6.
if (address_.size() != other.address_.size()) {
return address_.size() < other.address_.size();
}
return std::tie(address_, port_) < std::tie(other.address_, other.port_);
}
bool IPEndPoint::operator==(const IPEndPoint& other) const {
return address_ == other.address_ && port_ == other.port_;
}
} // namespace net