1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324
|
// Copyright 2015 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifdef UNSAFE_BUFFERS_BUILD
// TODO(crbug.com/40285824): Remove this and convert code to safer constructs.
#pragma allow_unsafe_buffers
#endif
// This is a copy of net/base/ip_address.h circa 2023. It should be used only by
// components/feedback/redaction_tool/.
// We need a copy because the components/feedback/redaction_tool source code is
// shared into ChromeOS and needs to have no dependencies outside of base/.
#ifndef COMPONENTS_FEEDBACK_REDACTION_TOOL_IP_ADDRESS_H_
#define COMPONENTS_FEEDBACK_REDACTION_TOOL_IP_ADDRESS_H_
#include <stddef.h>
#include <stdint.h>
#include <algorithm>
#include <array>
#include <string>
#include <string_view>
#include <vector>
#include "base/check_op.h"
#include "base/values.h"
namespace redaction_internal {
// Helper class to represent the sequence of bytes in an IP address.
// A vector<uint8_t> would be simpler but incurs heap allocation, so
// IPAddressBytes uses a fixed size array.
class IPAddressBytes {
public:
IPAddressBytes();
IPAddressBytes(const uint8_t* data, size_t data_len);
IPAddressBytes(const IPAddressBytes& other);
~IPAddressBytes();
// Copies |data_len| elements from |data| into this object.
void Assign(const uint8_t* data, size_t data_len);
// Returns the number of elements in the underlying array.
size_t size() const { return size_; }
// Sets the size to be |size|. Does not actually change the size
// of the underlying array or zero-initialize the bytes.
void Resize(size_t size) {
DCHECK_LE(size, 16u);
size_ = static_cast<uint8_t>(size);
}
// Returns true if the underlying array is empty.
bool empty() const { return size_ == 0; }
// Returns a pointer to the underlying array of bytes.
const uint8_t* data() const { return bytes_.data(); }
uint8_t* data() { return bytes_.data(); }
// Returns a pointer to the first element.
const uint8_t* begin() const { return data(); }
uint8_t* begin() { return data(); }
// Returns a pointer past the last element.
const uint8_t* end() const { return data() + size_; }
uint8_t* end() { return data() + size_; }
// Returns a reference to the last element.
uint8_t& back() {
DCHECK(!empty());
return bytes_[size_ - 1];
}
const uint8_t& back() const {
DCHECK(!empty());
return bytes_[size_ - 1];
}
// Appends |val| to the end and increments the size.
void push_back(uint8_t val) {
DCHECK_GT(16, size_);
bytes_[size_++] = val;
}
// Returns a reference to the byte at index |pos|.
uint8_t& operator[](size_t pos) {
DCHECK_LT(pos, size_);
return bytes_[pos];
}
const uint8_t& operator[](size_t pos) const {
DCHECK_LT(pos, size_);
return bytes_[pos];
}
bool operator<(const IPAddressBytes& other) const;
bool operator==(const IPAddressBytes& other) const;
size_t EstimateMemoryUsage() const;
private:
// Underlying sequence of bytes
std::array<uint8_t, 16> bytes_;
// Number of elements in |bytes_|. Should be either kIPv4AddressSize
// or kIPv6AddressSize or 0.
uint8_t size_;
};
class IPAddress {
public:
enum : size_t { kIPv4AddressSize = 4, kIPv6AddressSize = 16 };
// Creates a zero-sized, invalid address.
IPAddress();
IPAddress(const IPAddress& other);
// Copies the input address to |ip_address_|.
explicit IPAddress(const IPAddressBytes& address);
// Copies the input address to |ip_address_| taking an additional length
// parameter. The input is expected to be in network byte order.
IPAddress(const uint8_t* address, size_t address_len);
// Initializes |ip_address_| from the 4 bX bytes to form an IPv4 address.
// The bytes are expected to be in network byte order.
IPAddress(uint8_t b0, uint8_t b1, uint8_t b2, uint8_t b3);
// Initializes |ip_address_| from the 16 bX bytes to form an IPv6 address.
// The bytes are expected to be in network byte order.
IPAddress(uint8_t b0,
uint8_t b1,
uint8_t b2,
uint8_t b3,
uint8_t b4,
uint8_t b5,
uint8_t b6,
uint8_t b7,
uint8_t b8,
uint8_t b9,
uint8_t b10,
uint8_t b11,
uint8_t b12,
uint8_t b13,
uint8_t b14,
uint8_t b15);
~IPAddress();
// Returns true if the IP has |kIPv4AddressSize| elements.
bool IsIPv4() const;
// Returns true if the IP has |kIPv6AddressSize| elements.
bool IsIPv6() const;
// Returns true if the IP is either an IPv4 or IPv6 address. This function
// only checks the address length.
bool IsValid() const;
// Returns true if the IP is not in a range reserved by the IANA for
// local networks. Works with both IPv4 and IPv6 addresses.
// IPv4-mapped-to-IPv6 addresses are considered publicly routable.
bool IsPubliclyRoutable() const;
// Returns true if the IP is "zero" (e.g. the 0.0.0.0 IPv4 address).
bool IsZero() const;
// Returns true if |ip_address_| is an IPv4-mapped IPv6 address.
bool IsIPv4MappedIPv6() const;
// Returns true if |ip_address_| is 127.0.0.1/8 or ::1/128
bool IsLoopback() const;
// Returns true if |ip_address_| is 169.254.0.0/16 or fe80::/10, or
// ::ffff:169.254.0.0/112 (IPv4 mapped IPv6 link-local).
bool IsLinkLocal() const;
// The size in bytes of |ip_address_|.
size_t size() const { return ip_address_.size(); }
// Returns true if the IP is an empty, zero-sized (invalid) address.
bool empty() const { return ip_address_.empty(); }
// Returns the canonical string representation of an IP address.
// For example: "192.168.0.1" or "::1". Returns the empty string when
// |ip_address_| is invalid.
std::string ToString() const;
// Parses an IP address literal (either IPv4 or IPv6) to its numeric value.
// Returns true on success and fills |ip_address_| with the numeric value.
//
// When parsing fails, the original value of |this| will be overwritten such
// that |this->empty()| and |!this->IsValid()|.
[[nodiscard]] bool AssignFromIPLiteral(std::string_view ip_literal);
// Returns the underlying bytes.
const IPAddressBytes& bytes() const { return ip_address_; }
// Copies the bytes to a new vector. Generally callers should be using
// |bytes()| and the IPAddressBytes abstraction. This method is provided as a
// convenience for call sites that existed prior to the introduction of
// IPAddressBytes.
std::vector<uint8_t> CopyBytesToVector() const;
// Returns an IPAddress instance representing the 127.0.0.1 address.
static IPAddress IPv4Localhost();
// Returns an IPAddress instance representing the ::1 address.
static IPAddress IPv6Localhost();
// Returns an IPAddress made up of |num_zero_bytes| zeros.
static IPAddress AllZeros(size_t num_zero_bytes);
// Returns an IPAddress instance representing the 0.0.0.0 address.
static IPAddress IPv4AllZeros();
// Returns an IPAddress instance representing the :: address.
static IPAddress IPv6AllZeros();
friend bool operator==(const IPAddress&, const IPAddress&) = default;
bool operator<(const IPAddress& that) const;
// Must be a valid address (per IsValid()).
base::Value ToValue() const;
size_t EstimateMemoryUsage() const;
private:
IPAddressBytes ip_address_;
// This class is copyable and assignable.
};
using IPAddressList = std::vector<IPAddress>;
// Returns the canonical string representation of an IP address along with its
// port. For example: "192.168.0.1:99" or "[::1]:80".
std::string IPAddressToStringWithPort(const IPAddress& address, uint16_t port);
// Returns the address as a sequence of bytes in network-byte-order.
std::string IPAddressToPackedString(const IPAddress& address);
// Converts an IPv4 address to an IPv4-mapped IPv6 address.
// For example 192.168.0.1 would be converted to ::ffff:192.168.0.1.
IPAddress ConvertIPv4ToIPv4MappedIPv6(const IPAddress& address);
// Converts an IPv4-mapped IPv6 address to IPv4 address. Should only be called
// on IPv4-mapped IPv6 addresses.
IPAddress ConvertIPv4MappedIPv6ToIPv4(const IPAddress& address);
// Compares an IP address to see if it falls within the specified IP block.
// Returns true if it does, false otherwise.
//
// The IP block is given by (|ip_prefix|, |prefix_length_in_bits|) -- any
// IP address whose |prefix_length_in_bits| most significant bits match
// |ip_prefix| will be matched.
//
// In cases when an IPv4 address is being compared to an IPv6 address prefix
// and vice versa, the IPv4 addresses will be converted to IPv4-mapped
// (IPv6) addresses.
bool IPAddressMatchesPrefix(const IPAddress& ip_address,
const IPAddress& ip_prefix,
size_t prefix_length_in_bits);
// Checks whether |address| starts with |prefix|. This provides similar
// functionality as IPAddressMatchesPrefix() but doesn't perform automatic IPv4
// to IPv4MappedIPv6 conversions and only checks against full bytes.
template <size_t N>
bool IPAddressStartsWith(const IPAddress& address, const uint8_t (&prefix)[N]) {
if (address.size() < N) {
return false;
}
return std::equal(prefix, prefix + N, address.bytes().begin());
}
// According to RFC6052 Section 2.2 IPv4-Embedded IPv6 Address Format.
// https://www.rfc-editor.org/rfc/rfc6052#section-2.2
// +--+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
// |PL| 0-------------32--40--48--56--64--72--80--88--96--104---------|
// +--+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
// |32| prefix |v4(32) | u | suffix |
// +--+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
// |40| prefix |v4(24) | u |(8)| suffix |
// +--+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
// |48| prefix |v4(16) | u | (16) | suffix |
// +--+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
// |56| prefix |(8)| u | v4(24) | suffix |
// +--+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
// |64| prefix | u | v4(32) | suffix |
// +--+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
// |96| prefix | v4(32) |
// +--+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
//
// The NAT64/DNS64 translation prefixes has one of the following lengths.
enum class Dns64PrefixLength {
k32bit,
k40bit,
k48bit,
k56bit,
k64bit,
k96bit,
kInvalid
};
// Extracts the NAT64 translation prefix from the IPv6 address using the well
// known address ipv4only.arpa 192.0.0.170 and 192.0.0.171.
// Returns prefix length on success, or Dns64PrefixLength::kInvalid on failure
// (when the ipv4only.arpa IPv4 address is not found)
Dns64PrefixLength ExtractPref64FromIpv4onlyArpaAAAA(const IPAddress& address);
// Converts an IPv4 address to an IPv4-embedded IPv6 address using the given
// prefix. For example 192.168.0.1 and 64:ff9b::/96 would be converted to
// 64:ff9b::192.168.0.1
// Returns converted IPv6 address when prefix_length is not
// Dns64PrefixLength::kInvalid, and returns the original IPv4 address when
// prefix_length is Dns64PrefixLength::kInvalid.
IPAddress ConvertIPv4ToIPv4EmbeddedIPv6(const IPAddress& ipv4_address,
const IPAddress& ipv6_address,
Dns64PrefixLength prefix_length);
} // namespace redaction_internal
#endif // COMPONENTS_FEEDBACK_REDACTION_TOOL_IP_ADDRESS_H_
|