|  | // Copyright (c) 2016 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. | 
|  |  | 
|  | #ifndef QUICHE_COMMON_QUICHE_IP_ADDRESS_H_ | 
|  | #define QUICHE_COMMON_QUICHE_IP_ADDRESS_H_ | 
|  |  | 
|  | #include <cstdint> | 
|  | #if defined(_WIN32) | 
|  | #include <winsock2.h> | 
|  | #include <ws2tcpip.h> | 
|  | #else | 
|  | #include <arpa/inet.h> | 
|  | #include <netinet/in.h> | 
|  | #include <sys/socket.h> | 
|  | #include <sys/types.h> | 
|  | #endif | 
|  |  | 
|  | #include <ostream> | 
|  | #include <string> | 
|  |  | 
|  | #include "quiche/common/platform/api/quiche_export.h" | 
|  | #include "quiche/common/quiche_ip_address_family.h" | 
|  |  | 
|  | namespace quiche { | 
|  |  | 
|  | // Represents an IP address. | 
|  | class QUICHE_EXPORT QuicheIpAddress { | 
|  | public: | 
|  | // Sizes of IP addresses of different types, in bytes. | 
|  | enum : size_t { | 
|  | kIPv4AddressSize = 32 / 8, | 
|  | kIPv6AddressSize = 128 / 8, | 
|  | kMaxAddressSize = kIPv6AddressSize, | 
|  | }; | 
|  |  | 
|  | // TODO(fayang): Remove Loopback*() and use TestLoopback*() in tests. | 
|  | static QuicheIpAddress Loopback4(); | 
|  | static QuicheIpAddress Loopback6(); | 
|  | static QuicheIpAddress Any4(); | 
|  | static QuicheIpAddress Any6(); | 
|  |  | 
|  | QuicheIpAddress(); | 
|  | QuicheIpAddress(const QuicheIpAddress& other) = default; | 
|  | explicit QuicheIpAddress(const in_addr& ipv4_address); | 
|  | explicit QuicheIpAddress(const in6_addr& ipv6_address); | 
|  | QuicheIpAddress& operator=(const QuicheIpAddress& other) = default; | 
|  | QuicheIpAddress& operator=(QuicheIpAddress&& other) = default; | 
|  | QUICHE_EXPORT friend bool operator==(QuicheIpAddress lhs, | 
|  | QuicheIpAddress rhs); | 
|  | QUICHE_EXPORT friend bool operator!=(QuicheIpAddress lhs, | 
|  | QuicheIpAddress rhs); | 
|  |  | 
|  | bool IsInitialized() const; | 
|  | IpAddressFamily address_family() const; | 
|  | int AddressFamilyToInt() const; | 
|  | // Returns the address as a sequence of bytes in network-byte-order. IPv4 will | 
|  | // be 4 bytes. IPv6 will be 16 bytes. | 
|  | std::string ToPackedString() const; | 
|  | // Returns string representation of the address. | 
|  | std::string ToString() const; | 
|  | // Normalizes the address representation with respect to IPv4 addresses, i.e, | 
|  | // mapped IPv4 addresses ("::ffff:X.Y.Z.Q") are converted to pure IPv4 | 
|  | // addresses.  All other IPv4, IPv6, and empty values are left unchanged. | 
|  | QuicheIpAddress Normalized() const; | 
|  | // Returns an address suitable for use in IPv6-aware contexts.  This is the | 
|  | // opposite of NormalizeIPAddress() above.  IPv4 addresses are converted into | 
|  | // their IPv4-mapped address equivalents (e.g. 192.0.2.1 becomes | 
|  | // ::ffff:192.0.2.1).  IPv6 addresses are a noop (they are returned | 
|  | // unchanged). | 
|  | QuicheIpAddress DualStacked() const; | 
|  | bool FromPackedString(const char* data, size_t length); | 
|  | bool FromString(std::string str); | 
|  | bool IsIPv4() const; | 
|  | bool IsIPv6() const; | 
|  | bool InSameSubnet(const QuicheIpAddress& other, int subnet_length); | 
|  |  | 
|  | in_addr GetIPv4() const; | 
|  | in6_addr GetIPv6() const; | 
|  |  | 
|  | private: | 
|  | union { | 
|  | in_addr v4; | 
|  | in6_addr v6; | 
|  | uint8_t bytes[kMaxAddressSize]; | 
|  | char chars[kMaxAddressSize]; | 
|  | } address_; | 
|  | IpAddressFamily family_; | 
|  | }; | 
|  |  | 
|  | inline std::ostream& operator<<(std::ostream& os, | 
|  | const QuicheIpAddress address) { | 
|  | os << address.ToString(); | 
|  | return os; | 
|  | } | 
|  |  | 
|  | // Represents an IP prefix, which is an IP address and a prefix length in bits. | 
|  | class QUICHE_EXPORT QuicheIpPrefix { | 
|  | public: | 
|  | QuicheIpPrefix(); | 
|  | explicit QuicheIpPrefix(const QuicheIpAddress& address); | 
|  | explicit QuicheIpPrefix(const QuicheIpAddress& address, | 
|  | uint8_t prefix_length); | 
|  |  | 
|  | QuicheIpAddress address() const { return address_; } | 
|  | uint8_t prefix_length() const { return prefix_length_; } | 
|  | // Human-readable string representation of the prefix suitable for logging. | 
|  | std::string ToString() const; | 
|  |  | 
|  | QuicheIpPrefix(const QuicheIpPrefix& other) = default; | 
|  | QuicheIpPrefix& operator=(const QuicheIpPrefix& other) = default; | 
|  | QuicheIpPrefix& operator=(QuicheIpPrefix&& other) = default; | 
|  | QUICHE_EXPORT friend bool operator==(const QuicheIpPrefix& lhs, | 
|  | const QuicheIpPrefix& rhs); | 
|  | QUICHE_EXPORT friend bool operator!=(const QuicheIpPrefix& lhs, | 
|  | const QuicheIpPrefix& rhs); | 
|  |  | 
|  | private: | 
|  | QuicheIpAddress address_; | 
|  | uint8_t prefix_length_; | 
|  | }; | 
|  |  | 
|  | inline std::ostream& operator<<(std::ostream& os, const QuicheIpPrefix prefix) { | 
|  | os << prefix.ToString(); | 
|  | return os; | 
|  | } | 
|  |  | 
|  | }  // namespace quiche | 
|  |  | 
|  | #endif  // QUICHE_COMMON_QUICHE_IP_ADDRESS_H_ |