blob: e6f265d0e2654373deb66fb5cd4779bd7733d1b5 [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// 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 <string.h>
#include <optional>
#include <ostream>
#include <tuple>
#include <utility>
#include "base/check.h"
#include "base/check_op.h"
#include "base/containers/span.h"
#include "base/notreached.h"
#include "base/numerics/safe_conversions.h"
#include "base/strings/string_number_conversions.h"
#include "base/sys_byteorder.h"
#include "base/values.h"
#include "build/build_config.h"
#include "net/base/ip_address.h"
#include "net/base/ip_address_util.h"
#include "net/base/sys_addrinfo.h"
#if BUILDFLAG(IS_WIN)
#include <winsock2.h>
#include <winternl.h>
#include <netioapi.h>
#include <ntstatus.h>
#include <ws2bth.h>
#include "net/base/winsock_util.h" // For kBluetoothAddressSize
#elif BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA)
#include <net/if.h>
#endif
namespace net {
namespace {
// Value dictionary keys
constexpr std::string_view kValueAddressKey = "address";
constexpr std::string_view kValuePortKey = "port";
constexpr std::string_view kInterfaceName = "interface_name";
} // namespace
IPEndPoint::IndexToNameFunc IPEndPoint::index_to_name_func_for_testing_ =
nullptr;
IPEndPoint::NameToIndexFunc IPEndPoint::name_to_index_func_for_testing_ =
nullptr;
// static
void IPEndPoint::SetNameToIndexFuncForTesting(NameToIndexFunc func) {
name_to_index_func_for_testing_ = func;
}
void IPEndPoint::SetIndexToNameFuncForTesting(IndexToNameFunc func) {
index_to_name_func_for_testing_ = func;
}
// static
std::optional<uint32_t> IPEndPoint::ScopeIdFromDict(
const base::Value::Dict& dict) {
const std::string* name = dict.FindString(kInterfaceName);
if (!name) {
return std::nullopt;
}
unsigned int index = 0;
if (name_to_index_func_for_testing_) {
index = name_to_index_func_for_testing_(name->c_str());
} else {
index = if_nametoindex(name->c_str());
}
return index;
}
// static
base::Value IPEndPoint::ScopeIdToValue(std::optional<uint32_t> scope_id) {
if (!scope_id.has_value()) {
return base::Value();
}
char* name = nullptr;
char buf[IF_NAMESIZE + 1] = {0};
if (index_to_name_func_for_testing_) {
name = index_to_name_func_for_testing_(scope_id.value(),
base::span<char>(buf));
} else {
name = if_indextoname(scope_id.value(), buf);
}
if (!name) {
return base::Value();
}
return base::Value(name);
}
// static
std::optional<IPEndPoint> IPEndPoint::FromValue(const base::Value& value) {
const base::Value::Dict* dict = value.GetIfDict();
if (!dict)
return std::nullopt;
const base::Value* address_value = dict->Find(kValueAddressKey);
if (!address_value)
return std::nullopt;
std::optional<IPAddress> address = IPAddress::FromValue(*address_value);
if (!address.has_value())
return std::nullopt;
// Expect IPAddress to only allow deserializing valid addresses.
DCHECK(address.value().IsValid());
std::optional<int> port = dict->FindInt(kValuePortKey);
if (!port.has_value() ||
!base::IsValueInRangeForNumericType<uint16_t>(port.value())) {
return std::nullopt;
}
IPEndPoint endpoint(address.value(),
base::checked_cast<uint16_t>(port.value()));
std::optional<uint32_t> scope_id = ScopeIdFromDict(*dict);
if (scope_id.has_value()) {
if (scope_id.value() == 0 || !endpoint.IsIPv6LinkLocal() ||
!base::IsValueInRangeForNumericType<uint32_t>(scope_id.value())) {
return std::nullopt;
}
endpoint.scope_id_ = scope_id.value();
}
return endpoint;
}
IPEndPoint::IPEndPoint() = default;
IPEndPoint::~IPEndPoint() = default;
IPEndPoint::IPEndPoint(const IPAddress& address,
uint16_t port,
std::optional<uint32_t> scope_id)
: address_(address), port_(port), scope_id_(scope_id) {}
IPEndPoint::IPEndPoint(const IPEndPoint& endpoint) = default;
uint16_t IPEndPoint::port() const {
#if BUILDFLAG(IS_WIN)
DCHECK_NE(address_.size(), kBluetoothAddressSize);
#endif
return port_;
}
AddressFamily IPEndPoint::GetFamily() const {
return GetAddressFamily(address_);
}
int IPEndPoint::GetSockAddrFamily() const {
switch (address_.size()) {
case IPAddress::kIPv4AddressSize:
return AF_INET;
case IPAddress::kIPv6AddressSize:
return AF_INET6;
#if BUILDFLAG(IS_WIN)
case kBluetoothAddressSize:
return AF_BTH;
#endif
default:
NOTREACHED() << "Bad IP address";
}
}
bool IPEndPoint::ToSockAddr(struct sockaddr* address,
socklen_t* address_length) const {
// By definition, socklen_t is large enough to hold both sizes.
constexpr socklen_t kSockaddrInSize =
static_cast<socklen_t>(sizeof(struct sockaddr_in));
constexpr socklen_t kSockaddrIn6Size =
static_cast<socklen_t>(sizeof(struct sockaddr_in6));
DCHECK(address);
DCHECK(address_length);
#if BUILDFLAG(IS_WIN)
DCHECK_NE(address_.size(), kBluetoothAddressSize);
#endif
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);
// Zero out address struct.
*addr = {};
addr->sin_family = AF_INET;
addr->sin_port = base::HostToNet16(port_);
addr->sin_addr = ToInAddr(address_);
break;
}
case IPAddress::kIPv6AddressSize: {
if (*address_length < kSockaddrIn6Size)
return false;
*address_length = kSockaddrIn6Size;
struct sockaddr_in6* addr6 =
reinterpret_cast<struct sockaddr_in6*>(address);
// Zero out address struct.
*addr6 = {};
addr6->sin6_family = AF_INET6;
addr6->sin6_port = base::HostToNet16(port_);
addr6->sin6_addr = ToIn6Addr(address_);
if (IsIPv6LinkLocal() && scope_id_) {
addr6->sin6_scope_id = *scope_id_;
}
break;
}
default:
return false;
}
return true;
}
bool IPEndPoint::FromSockAddr(const struct sockaddr* sock_addr,
socklen_t sock_addr_len) {
DCHECK(sock_addr);
switch (sock_addr->sa_family) {
case 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);
*this = IPEndPoint(
// `s_addr` is a `uint32_t`, but it is already in network byte order.
IPAddress(base::as_bytes(base::span_from_ref(addr->sin_addr.s_addr))),
base::NetToHost16(addr->sin_port));
return true;
}
case 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);
*this = IPEndPoint(IPAddress(addr->sin6_addr.s6_addr),
base::NetToHost16(addr->sin6_port));
if (IsIPv6LinkLocal() && addr->sin6_scope_id != 0) {
scope_id_ = addr->sin6_scope_id;
}
return true;
}
#if BUILDFLAG(IS_WIN)
case 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);
*this = IPEndPoint();
// A bluetooth address is 6 bytes, but btAddr is a ULONGLONG, so we take a
// prefix of it.
address_ = IPAddress(base::as_bytes(base::span_from_ref(addr->btAddr))
.first(kBluetoothAddressSize));
// Intentionally ignoring Bluetooth port. It is a ULONG, but
// `IPEndPoint::port_` is a uint16_t. See https://crbug.com/1231273.
return true;
}
#endif
}
return false; // Unrecognized |sa_family|.
}
std::string IPEndPoint::ToString() const {
#if BUILDFLAG(IS_WIN)
DCHECK_NE(address_.size(), kBluetoothAddressSize);
#endif
return IPAddressToStringWithPort(address_, port_);
}
std::string IPEndPoint::ToStringWithoutPort() const {
#if BUILDFLAG(IS_WIN)
DCHECK_NE(address_.size(), kBluetoothAddressSize);
#endif
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_, scope_id_) <
std::tie(other.address_, other.port_, other.scope_id_);
}
base::Value IPEndPoint::ToValue() const {
base::Value::Dict dict;
DCHECK(address_.IsValid());
dict.Set(kValueAddressKey, address_.ToValue());
dict.Set(kValuePortKey, port_);
base::Value interface_name = ScopeIdToValue(scope_id_);
if (!interface_name.is_none()) {
DCHECK(IsIPv6LinkLocal());
dict.Set(kInterfaceName, std::move(interface_name));
}
return base::Value(std::move(dict));
}
bool IPEndPoint::IsIPv6LinkLocal() const {
return address_.IsValid() && address_.IsIPv6() &&
!address_.IsIPv4MappedIPv6() && address_.IsLinkLocal();
}
std::ostream& operator<<(std::ostream& os, const IPEndPoint& ip_endpoint) {
return os << ip_endpoint.ToString();
}
} // namespace net