Merge pull request #3939 from PastaPastaPasta/backport-addrv2

Backport partial addrv2 support
This commit is contained in:
UdjinM6 2021-01-29 11:39:00 +03:00 committed by GitHub
commit 7099dea2e4
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 146 additions and 53 deletions

View File

@ -24,19 +24,35 @@ CNetAddr::CNetAddr()
void CNetAddr::SetIP(const CNetAddr& ipIn) void CNetAddr::SetIP(const CNetAddr& ipIn)
{ {
m_net = ipIn.m_net;
memcpy(ip, ipIn.ip, sizeof(ip)); memcpy(ip, ipIn.ip, sizeof(ip));
} }
void CNetAddr::SetLegacyIPv6(const uint8_t ipv6[16])
{
if (memcmp(ipv6, pchIPv4, sizeof(pchIPv4)) == 0) {
m_net = NET_IPV4;
} else if (memcmp(ipv6, pchOnionCat, sizeof(pchOnionCat)) == 0) {
m_net = NET_ONION;
} else if (memcmp(ipv6, g_internal_prefix, sizeof(g_internal_prefix)) == 0) {
m_net = NET_INTERNAL;
} else {
m_net = NET_IPV6;
}
memcpy(ip, ipv6, 16);
}
void CNetAddr::SetRaw(Network network, const uint8_t *ip_in) void CNetAddr::SetRaw(Network network, const uint8_t *ip_in)
{ {
switch(network) switch(network)
{ {
case NET_IPV4: case NET_IPV4:
m_net = NET_IPV4;
memcpy(ip, pchIPv4, 12); memcpy(ip, pchIPv4, 12);
memcpy(ip+12, ip_in, 4); memcpy(ip+12, ip_in, 4);
break; break;
case NET_IPV6: case NET_IPV6:
memcpy(ip, ip_in, 16); SetLegacyIPv6(ip_in);
break; break;
default: default:
assert(!"invalid network"); assert(!"invalid network");
@ -48,6 +64,7 @@ bool CNetAddr::SetInternal(const std::string &name)
if (name.empty()) { if (name.empty()) {
return false; return false;
} }
m_net = NET_INTERNAL;
unsigned char hash[32] = {}; unsigned char hash[32] = {};
CSHA256().Write((const unsigned char*)name.data(), name.size()).Finalize(hash); CSHA256().Write((const unsigned char*)name.data(), name.size()).Finalize(hash);
memcpy(ip, g_internal_prefix, sizeof(g_internal_prefix)); memcpy(ip, g_internal_prefix, sizeof(g_internal_prefix));
@ -61,6 +78,7 @@ bool CNetAddr::SetSpecial(const std::string &strName)
std::vector<unsigned char> vchAddr = DecodeBase32(strName.substr(0, strName.size() - 6).c_str()); std::vector<unsigned char> vchAddr = DecodeBase32(strName.substr(0, strName.size() - 6).c_str());
if (vchAddr.size() != 16-sizeof(pchOnionCat)) if (vchAddr.size() != 16-sizeof(pchOnionCat))
return false; return false;
m_net = NET_ONION;
memcpy(ip, pchOnionCat, sizeof(pchOnionCat)); memcpy(ip, pchOnionCat, sizeof(pchOnionCat));
for (unsigned int i=0; i<16-sizeof(pchOnionCat); i++) for (unsigned int i=0; i<16-sizeof(pchOnionCat); i++)
ip[i + sizeof(pchOnionCat)] = vchAddr[i]; ip[i + sizeof(pchOnionCat)] = vchAddr[i];
@ -95,15 +113,9 @@ bool CNetAddr::IsBindAny() const
return true; return true;
} }
bool CNetAddr::IsIPv4() const bool CNetAddr::IsIPv4() const { return m_net == NET_IPV4; }
{
return (memcmp(ip, pchIPv4, sizeof(pchIPv4)) == 0);
}
bool CNetAddr::IsIPv6() const bool CNetAddr::IsIPv6() const { return m_net == NET_IPV6; }
{
return (!IsIPv4() && !IsTor() && !IsInternal());
}
bool CNetAddr::IsRFC1918() const bool CNetAddr::IsRFC1918() const
{ {
@ -137,52 +149,58 @@ bool CNetAddr::IsRFC5737() const
bool CNetAddr::IsRFC3849() const bool CNetAddr::IsRFC3849() const
{ {
return GetByte(15) == 0x20 && GetByte(14) == 0x01 && GetByte(13) == 0x0D && GetByte(12) == 0xB8; return IsIPv6() && GetByte(15) == 0x20 && GetByte(14) == 0x01 &&
GetByte(13) == 0x0D && GetByte(12) == 0xB8;
} }
bool CNetAddr::IsRFC3964() const bool CNetAddr::IsRFC3964() const
{ {
return (GetByte(15) == 0x20 && GetByte(14) == 0x02); return IsIPv6() && GetByte(15) == 0x20 && GetByte(14) == 0x02;
} }
bool CNetAddr::IsRFC6052() const bool CNetAddr::IsRFC6052() const
{ {
static const unsigned char pchRFC6052[] = {0,0x64,0xFF,0x9B,0,0,0,0,0,0,0,0}; static const unsigned char pchRFC6052[] = {0,0x64,0xFF,0x9B,0,0,0,0,0,0,0,0};
return (memcmp(ip, pchRFC6052, sizeof(pchRFC6052)) == 0); return IsIPv6() && memcmp(ip, pchRFC6052, sizeof(pchRFC6052)) == 0;
} }
bool CNetAddr::IsRFC4380() const bool CNetAddr::IsRFC4380() const
{ {
return (GetByte(15) == 0x20 && GetByte(14) == 0x01 && GetByte(13) == 0 && GetByte(12) == 0); return IsIPv6() && GetByte(15) == 0x20 && GetByte(14) == 0x01 && GetByte(13) == 0 &&
GetByte(12) == 0;
} }
bool CNetAddr::IsRFC4862() const bool CNetAddr::IsRFC4862() const
{ {
static const unsigned char pchRFC4862[] = {0xFE,0x80,0,0,0,0,0,0}; static const unsigned char pchRFC4862[] = {0xFE,0x80,0,0,0,0,0,0};
return (memcmp(ip, pchRFC4862, sizeof(pchRFC4862)) == 0); return IsIPv6() && memcmp(ip, pchRFC4862, sizeof(pchRFC4862)) == 0;
} }
bool CNetAddr::IsRFC4193() const bool CNetAddr::IsRFC4193() const
{ {
return ((GetByte(15) & 0xFE) == 0xFC); return IsIPv6() && (GetByte(15) & 0xFE) == 0xFC;
} }
bool CNetAddr::IsRFC6145() const bool CNetAddr::IsRFC6145() const
{ {
static const unsigned char pchRFC6145[] = {0,0,0,0,0,0,0,0,0xFF,0xFF,0,0}; static const unsigned char pchRFC6145[] = {0,0,0,0,0,0,0,0,0xFF,0xFF,0,0};
return (memcmp(ip, pchRFC6145, sizeof(pchRFC6145)) == 0); return IsIPv6() && memcmp(ip, pchRFC6145, sizeof(pchRFC6145)) == 0;
} }
bool CNetAddr::IsRFC4843() const bool CNetAddr::IsRFC4843() const
{ {
return (GetByte(15) == 0x20 && GetByte(14) == 0x01 && GetByte(13) == 0x00 && (GetByte(12) & 0xF0) == 0x10); return IsIPv6() && GetByte(15) == 0x20 && GetByte(14) == 0x01 &&
GetByte(13) == 0x00 && (GetByte(12) & 0xF0) == 0x10;
} }
bool CNetAddr::IsTor() const bool CNetAddr::IsRFC7343() const
{ {
return (memcmp(ip, pchOnionCat, sizeof(pchOnionCat)) == 0); return IsIPv6() && GetByte(15) == 0x20 && GetByte(14) == 0x01 &&
GetByte(13) == 0x00 && (GetByte(12) & 0xF0) == 0x20;
} }
bool CNetAddr::IsTor() const { return m_net == NET_ONION; }
bool CNetAddr::IsLocal() const bool CNetAddr::IsLocal() const
{ {
// IPv4 loopback // IPv4 loopback
@ -191,7 +209,7 @@ bool CNetAddr::IsLocal() const
// IPv6 loopback (::1/128) // IPv6 loopback (::1/128)
static const unsigned char pchLocal[16] = {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1}; static const unsigned char pchLocal[16] = {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1};
if (memcmp(ip, pchLocal, 16) == 0) if (IsIPv6() && memcmp(ip, pchLocal, 16) == 0)
return true; return true;
return false; return false;
@ -205,12 +223,12 @@ bool CNetAddr::IsValid() const
// header20 vectorlen3 addr26 addr26 addr26 header20 vectorlen3 addr26 addr26 addr26... // header20 vectorlen3 addr26 addr26 addr26 header20 vectorlen3 addr26 addr26 addr26...
// so if the first length field is garbled, it reads the second batch // so if the first length field is garbled, it reads the second batch
// of addr misaligned by 3 bytes. // of addr misaligned by 3 bytes.
if (memcmp(ip, pchIPv4+3, sizeof(pchIPv4)-3) == 0) if (IsIPv6() && memcmp(ip, pchIPv4+3, sizeof(pchIPv4)-3) == 0)
return false; return false;
// unspecified IPv6 address (::/128) // unspecified IPv6 address (::/128)
unsigned char ipNone6[16] = {}; unsigned char ipNone6[16] = {};
if (memcmp(ip, ipNone6, 16) == 0) if (IsIPv6() && memcmp(ip, ipNone6, 16) == 0)
return false; return false;
// documentation IPv6 address // documentation IPv6 address
@ -242,12 +260,12 @@ bool CNetAddr::IsRoutable() const
return false; return false;
if (!fAllowPrivateNet && IsRFC1918()) if (!fAllowPrivateNet && IsRFC1918())
return false; return false;
return !(IsRFC2544() || IsRFC3927() || IsRFC4862() || IsRFC6598() || IsRFC5737() || (IsRFC4193() && !IsTor()) || IsRFC4843() || IsLocal() || IsInternal()); return !(IsRFC2544() || IsRFC3927() || IsRFC4862() || IsRFC6598() || IsRFC5737() || (IsRFC4193() && !IsTor()) || IsRFC4843() || IsRFC7343() || IsLocal() || IsInternal());
} }
bool CNetAddr::IsInternal() const bool CNetAddr::IsInternal() const
{ {
return memcmp(ip, g_internal_prefix, sizeof(g_internal_prefix)) == 0; return m_net == NET_INTERNAL;
} }
enum Network CNetAddr::GetNetwork() const enum Network CNetAddr::GetNetwork() const
@ -258,13 +276,7 @@ enum Network CNetAddr::GetNetwork() const
if (!IsRoutable()) if (!IsRoutable())
return NET_UNROUTABLE; return NET_UNROUTABLE;
if (IsIPv4()) return m_net;
return NET_IPV4;
if (IsTor())
return NET_ONION;
return NET_IPV6;
} }
std::string CNetAddr::ToStringIP(bool fUseGetnameinfo) const std::string CNetAddr::ToStringIP(bool fUseGetnameinfo) const
@ -301,12 +313,12 @@ std::string CNetAddr::ToString() const
bool operator==(const CNetAddr& a, const CNetAddr& b) bool operator==(const CNetAddr& a, const CNetAddr& b)
{ {
return (memcmp(a.ip, b.ip, 16) == 0); return a.m_net == b.m_net && memcmp(a.ip, b.ip, 16) == 0;
} }
bool operator<(const CNetAddr& a, const CNetAddr& b) bool operator<(const CNetAddr& a, const CNetAddr& b)
{ {
return (memcmp(a.ip, b.ip, 16) < 0); return a.m_net < b.m_net || (a.m_net == b.m_net && memcmp(a.ip, b.ip, 16) < 0);
} }
bool CNetAddr::GetInAddr(struct in_addr* pipv4Addr) const bool CNetAddr::GetInAddr(struct in_addr* pipv4Addr) const
@ -568,12 +580,10 @@ bool CService::GetSockAddr(struct sockaddr* paddr, socklen_t *addrlen) const
std::vector<unsigned char> CService::GetKey() const std::vector<unsigned char> CService::GetKey() const
{ {
std::vector<unsigned char> vKey; auto key = GetAddrBytes();
vKey.resize(18); key.push_back(port / 0x100);
memcpy(vKey.data(), ip, 16); key.push_back(port & 0x0FF);
vKey[16] = port / 0x100; return key;
vKey[17] = port & 0x0FF;
return vKey;
} }
std::string CService::ToStringPort() const std::string CService::ToStringPort() const
@ -658,7 +668,7 @@ CSubNet::CSubNet(const CNetAddr &addr):
bool CSubNet::Match(const CNetAddr &addr) const bool CSubNet::Match(const CNetAddr &addr) const
{ {
if (!valid || !addr.IsValid()) if (!valid || !addr.IsValid() || network.m_net != addr.m_net)
return false; return false;
for(int x=0; x<16; ++x) for(int x=0; x<16; ++x)
if ((addr.ip[x] & netmask[x]) != network.ip[x]) if ((addr.ip[x] & netmask[x]) != network.ip[x])

View File

@ -19,21 +19,50 @@
extern bool fAllowPrivateNet; extern bool fAllowPrivateNet;
/**
* A network type.
* @note An address may belong to more than one network, for example `10.0.0.1`
* belongs to both `NET_UNROUTABLE` and `NET_IPV4`.
* Keep these sequential starting from 0 and `NET_MAX` as the last entry.
* We have loops like `for (int i = 0; i < NET_MAX; i++)` that expect to iterate
* over all enum values and also `GetExtNetwork()` "extends" this enum by
* introducing standalone constants starting from `NET_MAX`.
*/
enum Network enum Network
{ {
/// Addresses from these networks are not publicly routable on the global Internet.
NET_UNROUTABLE = 0, NET_UNROUTABLE = 0,
/// IPv4
NET_IPV4, NET_IPV4,
/// IPv6
NET_IPV6, NET_IPV6,
/// TORv2
NET_ONION, NET_ONION,
/// A set of dummy addresses that map a name to an IPv6 address. These
/// addresses belong to RFC4193's fc00::/7 subnet (unique-local addresses).
/// We use them to map a string or FQDN to an IPv6 address in CAddrMan to
/// keep track of which DNS seeds were used.
NET_INTERNAL, NET_INTERNAL,
/// Dummy value to indicate the number of NET_* constants.
NET_MAX, NET_MAX,
}; };
/** IP address (IPv6, or IPv4 using mapped IPv6 range (::FFFF:0:0/96)) */ /**
* Network address.
*/
class CNetAddr class CNetAddr
{ {
protected: protected:
/**
* Network to which this address belongs.
*/
Network m_net{NET_IPV6};
unsigned char ip[16]; // in network byte order unsigned char ip[16]; // in network byte order
uint32_t scopeId{0}; // for scoped/link-local ipv6 addresses uint32_t scopeId{0}; // for scoped/link-local ipv6 addresses
@ -42,6 +71,14 @@ class CNetAddr
explicit CNetAddr(const struct in_addr& ipv4Addr); explicit CNetAddr(const struct in_addr& ipv4Addr);
void SetIP(const CNetAddr& ip); void SetIP(const CNetAddr& ip);
/**
* Set from a legacy IPv6 address.
* Legacy IPv6 address may be a normal IPv6 address, or another address
* (e.g. IPv4) disguised as IPv6. This encoding is used in the legacy
* `addr` encoding.
*/
void SetLegacyIPv6(const uint8_t ipv6[16]);
private: private:
/** /**
* Set raw IPv4 or IPv6 address (in network byte order) * Set raw IPv4 or IPv6 address (in network byte order)
@ -69,7 +106,8 @@ class CNetAddr
bool IsRFC3964() const; // IPv6 6to4 tunnelling (2002::/16) bool IsRFC3964() const; // IPv6 6to4 tunnelling (2002::/16)
bool IsRFC4193() const; // IPv6 unique local (FC00::/7) bool IsRFC4193() const; // IPv6 unique local (FC00::/7)
bool IsRFC4380() const; // IPv6 Teredo tunnelling (2001::/32) bool IsRFC4380() const; // IPv6 Teredo tunnelling (2001::/32)
bool IsRFC4843() const; // IPv6 ORCHID (2001:10::/28) bool IsRFC4843() const; // IPv6 ORCHID (deprecated) (2001:10::/28)
bool IsRFC7343() const; // IPv6 ORCHIDv2 (2001:20::/28)
bool IsRFC4862() const; // IPv6 autoconfig (FE80::/64) bool IsRFC4862() const; // IPv6 autoconfig (FE80::/64)
bool IsRFC6052() const; // IPv6 well-known prefix (64:FF9B::/96) bool IsRFC6052() const; // IPv6 well-known prefix (64:FF9B::/96)
bool IsRFC6145() const; // IPv6 IPv4-translated address (::FFFF:0:0:0/96) bool IsRFC6145() const; // IPv6 IPv4-translated address (::FFFF:0:0:0/96)
@ -85,6 +123,7 @@ class CNetAddr
uint64_t GetHash() const; uint64_t GetHash() const;
bool GetInAddr(struct in_addr* pipv4Addr) const; bool GetInAddr(struct in_addr* pipv4Addr) const;
std::vector<unsigned char> GetGroup() const; std::vector<unsigned char> GetGroup() const;
std::vector<unsigned char> GetAddrBytes() const { return {std::begin(ip), std::end(ip)}; }
int GetReachabilityFrom(const CNetAddr *paddrPartner = nullptr) const; int GetReachabilityFrom(const CNetAddr *paddrPartner = nullptr) const;
explicit CNetAddr(const struct in6_addr& pipv6Addr, const uint32_t scope = 0); explicit CNetAddr(const struct in6_addr& pipv6Addr, const uint32_t scope = 0);
@ -94,11 +133,26 @@ class CNetAddr
friend bool operator!=(const CNetAddr& a, const CNetAddr& b) { return !(a == b); } friend bool operator!=(const CNetAddr& a, const CNetAddr& b) { return !(a == b); }
friend bool operator<(const CNetAddr& a, const CNetAddr& b); friend bool operator<(const CNetAddr& a, const CNetAddr& b);
ADD_SERIALIZE_METHODS; /**
* Serialize to a stream.
*/
template <typename Stream>
void Serialize(Stream& s) const
{
s << ip;
}
template <typename Stream, typename Operation> /**
inline void SerializationOp(Stream& s, Operation ser_action) { * Unserialize from a stream.
READWRITE(ip); */
template <typename Stream>
void Unserialize(Stream& s)
{
unsigned char ip_temp[sizeof(ip)];
s >> ip_temp;
// Use SetLegacyIPv6() so that m_net is set correctly. For example
// ::FFFF:0102:0304 should be set as m_net=NET_IPV4 (1.2.3.4).
SetLegacyIPv6(ip_temp);
} }
friend class CSubNet; friend class CSubNet;
@ -167,12 +221,28 @@ class CService : public CNetAddr
CService(const struct in6_addr& ipv6Addr, unsigned short port); CService(const struct in6_addr& ipv6Addr, unsigned short port);
explicit CService(const struct sockaddr_in6& addr); explicit CService(const struct sockaddr_in6& addr);
ADD_SERIALIZE_METHODS; /**
* Serialize to a stream.
*/
template <typename Stream>
void Serialize(Stream& s) const
{
s << ip;
s << WrapBigEndian(port);
}
template <typename Stream, typename Operation> /**
inline void SerializationOp(Stream& s, Operation ser_action) { * Unserialize from a stream.
READWRITE(ip); */
READWRITE(WrapBigEndian(port)); template <typename Stream>
void Unserialize(Stream& s)
{
unsigned char ip_temp[sizeof(ip)];
s >> ip_temp;
// Use SetLegacyIPv6() so that m_net is set correctly. For example
// ::FFFF:0102:0304 should be set as m_net=NET_IPV4 (1.2.3.4).
SetLegacyIPv6(ip_temp);
s >> WrapBigEndian(port);
} }
}; };

View File

@ -60,6 +60,7 @@ BOOST_AUTO_TEST_CASE(netbase_properties)
BOOST_CHECK(ResolveIP("FC00::").IsRFC4193()); BOOST_CHECK(ResolveIP("FC00::").IsRFC4193());
BOOST_CHECK(ResolveIP("2001::2").IsRFC4380()); BOOST_CHECK(ResolveIP("2001::2").IsRFC4380());
BOOST_CHECK(ResolveIP("2001:10::").IsRFC4843()); BOOST_CHECK(ResolveIP("2001:10::").IsRFC4843());
BOOST_CHECK(ResolveIP("2001:20::").IsRFC7343());
BOOST_CHECK(ResolveIP("FE80::").IsRFC4862()); BOOST_CHECK(ResolveIP("FE80::").IsRFC4862());
BOOST_CHECK(ResolveIP("64:FF9B::").IsRFC6052()); BOOST_CHECK(ResolveIP("64:FF9B::").IsRFC6052());
BOOST_CHECK(ResolveIP("FD87:D87E:EB43:edb1:8e4:3588:e546:35ca").IsTor()); BOOST_CHECK(ResolveIP("FD87:D87E:EB43:edb1:8e4:3588:e546:35ca").IsTor());
@ -135,6 +136,14 @@ BOOST_AUTO_TEST_CASE(onioncat_test)
} }
BOOST_AUTO_TEST_CASE(embedded_test)
{
CNetAddr addr1(ResolveIP("1.2.3.4"));
CNetAddr addr2(ResolveIP("::FFFF:0102:0304"));
BOOST_CHECK(addr2.IsIPv4());
BOOST_CHECK_EQUAL(addr1.ToString(), addr2.ToString());
}
BOOST_AUTO_TEST_CASE(subnet_test) BOOST_AUTO_TEST_CASE(subnet_test)
{ {
@ -155,9 +164,13 @@ BOOST_AUTO_TEST_CASE(subnet_test)
BOOST_CHECK(ResolveSubNet("1.2.2.1/24").Match(ResolveIP("1.2.2.4"))); BOOST_CHECK(ResolveSubNet("1.2.2.1/24").Match(ResolveIP("1.2.2.4")));
BOOST_CHECK(ResolveSubNet("1.2.2.110/31").Match(ResolveIP("1.2.2.111"))); BOOST_CHECK(ResolveSubNet("1.2.2.110/31").Match(ResolveIP("1.2.2.111")));
BOOST_CHECK(ResolveSubNet("1.2.2.20/26").Match(ResolveIP("1.2.2.63"))); BOOST_CHECK(ResolveSubNet("1.2.2.20/26").Match(ResolveIP("1.2.2.63")));
// All-Matching IPv6 Matches arbitrary IPv4 and IPv6 // All-Matching IPv6 Matches arbitrary IPv6
BOOST_CHECK(ResolveSubNet("::/0").Match(ResolveIP("1:2:3:4:5:6:7:1234"))); BOOST_CHECK(ResolveSubNet("::/0").Match(ResolveIP("1:2:3:4:5:6:7:1234")));
BOOST_CHECK(ResolveSubNet("::/0").Match(ResolveIP("1.2.3.4"))); // But not `::` or `0.0.0.0` because they are considered invalid addresses
BOOST_CHECK(!ResolveSubNet("::/0").Match(ResolveIP("::")));
BOOST_CHECK(!ResolveSubNet("::/0").Match(ResolveIP("0.0.0.0")));
// Addresses from one network (IPv4) don't belong to subnets of another network (IPv6)
BOOST_CHECK(!ResolveSubNet("::/0").Match(ResolveIP("1.2.3.4")));
// All-Matching IPv4 does not Match IPv6 // All-Matching IPv4 does not Match IPv6
BOOST_CHECK(!ResolveSubNet("0.0.0.0/0").Match(ResolveIP("1:2:3:4:5:6:7:1234"))); BOOST_CHECK(!ResolveSubNet("0.0.0.0/0").Match(ResolveIP("1:2:3:4:5:6:7:1234")));
// Invalid subnets Match nothing (not even invalid addresses) // Invalid subnets Match nothing (not even invalid addresses)