123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207 |
- /*
- * Copyright (c)2013-2020 ZeroTier, Inc.
- *
- * Use of this software is governed by the Business Source License included
- * in the LICENSE.TXT file in the project's root directory.
- *
- * Change Date: 2024-01-01
- *
- * On the date above, in accordance with the Business Source License, use
- * of this software will be governed by version 2.0 of the Apache License.
- */
- /****/
- #ifndef ZT_ADDRESS_HPP
- #define ZT_ADDRESS_HPP
- #include "Constants.hpp"
- #include "Utils.hpp"
- #include "TriviallyCopyable.hpp"
- #define ZT_ADDRESS_STRING_SIZE_MAX 11
- namespace ZeroTier {
- /**
- * A ZeroTier address
- */
- class Address : public TriviallyCopyable
- {
- public:
- ZT_INLINE Address() noexcept : _a(0) {}
- explicit ZT_INLINE Address(const uint64_t a) noexcept : _a(a) {}
- explicit ZT_INLINE Address(const uint8_t b[5]) noexcept : _a(((uint64_t)b[0] << 32U) | ((uint64_t)b[1] << 24U) | ((uint64_t)b[2] << 16U) | ((uint64_t)b[3] << 8U) | (uint64_t)b[4]) {}
- ZT_INLINE Address &operator=(const uint64_t a) noexcept { _a = a; return *this; }
- /**
- * @param bits Raw address -- 5 bytes, big-endian byte order
- * @param len Length of array
- */
- ZT_INLINE void setTo(const uint8_t b[5]) noexcept
- {
- _a = ((uint64_t)b[0] << 32U) | ((uint64_t)b[1] << 24U) | ((uint64_t)b[2] << 16U) | ((uint64_t)b[3] << 8U) | (uint64_t)b[4];
- }
- /**
- * @param bits Buffer to hold 5-byte address in big-endian byte order
- * @param len Length of array
- */
- ZT_INLINE void copyTo(uint8_t b[5]) const noexcept
- {
- const uint64_t a = _a;
- b[0] = (uint8_t)(a >> 32U);
- b[1] = (uint8_t)(a >> 24U);
- b[2] = (uint8_t)(a >> 16U);
- b[3] = (uint8_t)(a >> 8U);
- b[4] = (uint8_t)a;
- }
- /**
- * @return Integer containing address (0 to 2^40)
- */
- ZT_INLINE uint64_t toInt() const noexcept { return _a; }
- /**
- * Set address to zero/NIL
- */
- ZT_INLINE void zero() noexcept { _a = 0; }
- /**
- * @return Hash code for use with Hashtable
- */
- ZT_INLINE unsigned long hashCode() const noexcept { return (unsigned long)_a; }
- /**
- * @param s String with at least 11 characters of space available (10 + terminating NULL)
- * @return Hexadecimal string
- */
- ZT_INLINE char *toString(char s[ZT_ADDRESS_STRING_SIZE_MAX]) const noexcept
- {
- const uint64_t a = _a;
- const unsigned int m = 0xf;
- s[0] = Utils::HEXCHARS[(unsigned int)(a >> 36U) & m];
- s[1] = Utils::HEXCHARS[(unsigned int)(a >> 32U) & m];
- s[2] = Utils::HEXCHARS[(unsigned int)(a >> 28U) & m];
- s[3] = Utils::HEXCHARS[(unsigned int)(a >> 24U) & m];
- s[4] = Utils::HEXCHARS[(unsigned int)(a >> 20U) & m];
- s[5] = Utils::HEXCHARS[(unsigned int)(a >> 16U) & m];
- s[6] = Utils::HEXCHARS[(unsigned int)(a >> 12U) & m];
- s[7] = Utils::HEXCHARS[(unsigned int)(a >> 8U) & m];
- s[8] = Utils::HEXCHARS[(unsigned int)(a >> 4U) & m];
- s[9] = Utils::HEXCHARS[(unsigned int)a & m];
- s[10] = 0;
- return s;
- }
- /**
- * Check if this address is reserved
- *
- * The all-zero null address and any address beginning with 0xff are
- * reserved. (0xff is reserved for future use to designate possibly
- * longer addresses, addresses based on IPv6 innards, etc.)
- *
- * @return True if address is reserved and may not be used
- */
- ZT_INLINE bool isReserved() const noexcept { return ((!_a) || ((_a >> 32U) == ZT_ADDRESS_RESERVED_PREFIX)); }
- ZT_INLINE operator bool() const noexcept { return (_a != 0); }
- ZT_INLINE bool operator==(const Address &a) const noexcept { return _a == a._a; }
- ZT_INLINE bool operator!=(const Address &a) const noexcept { return _a != a._a; }
- ZT_INLINE bool operator>(const Address &a) const noexcept { return _a > a._a; }
- ZT_INLINE bool operator<(const Address &a) const noexcept { return _a < a._a; }
- ZT_INLINE bool operator>=(const Address &a) const noexcept { return _a >= a._a; }
- ZT_INLINE bool operator<=(const Address &a) const noexcept { return _a <= a._a; }
- #if 0
- /**
- * Create a list of the first N bits of a list of unique addresses with N as the minimum unique size
- *
- * The list is stored in a space-efficient packed bit format.
- *
- * @param start Starting Address iterator/pointer
- * @param end Ending Address iterator/pointer
- * @param list Pointer to location to write list
- * @param listCapacityBytes Number of bytes available for list
- * @return Number of bytes written or -1 on overflow or other error
- * @tparam I Input iterator type
- */
- template<typename I>
- static inline int createMinPrefixList(I start,I end,uint8_t *list,const int listCapacityBytes)
- {
- std::vector<Address> sortedAddrs(start,end);
- if (sortedAddrs.empty())
- return 0;
- if (listCapacityBytes == 0)
- return -1;
- std::sort(sortedAddrs.begin(),sortedAddrs.end());
- unsigned int bits = (unsigned int)fmaxf(log2f((float)(sortedAddrs.size() * 2)),3.0F);
- uint64_t mask;
- try_additional_bits: {
- mask = 0xffffffffffffffffULL >> (64 - bits);
- std::vector<Address>::iterator a(sortedAddrs.begin());
- uint64_t aa = *(a++) & mask;
- aa |= (uint64_t)(aa == 0);
- uint64_t lastMaskedAddress = aa;
- while (a != sortedAddrs.end()) {
- aa = *(a++) & mask;
- aa |= (uint64_t)(aa == 0);
- if (aa == lastMaskedAddress) {
- ++bits;
- goto try_additional_bits;
- }
- lastMaskedAddress = aa;
- }
- }
- int l = 0;
- unsigned int bitPtr = 0;
- for(I a(start);a!=end;) {
- uint64_t aa = *(a++) & mask;
- aa |= (uint64_t)(aa == 0);
- unsigned int br = bits;
- if (bitPtr > 0) {
- unsigned int w = 8 - bitPtr;
- if (w > br) w = br;
- list[l] = (list[l] << w) | (((uint8_t)aa) & (0xff >> (8 - w)));
- bitPtr += w;
- if (bitPtr == 8) {
- bitPtr = 0;
- if (l >= listCapacityBytes)
- return -1;
- ++l;
- }
- aa >>= w;
- br -= w;
- }
- while (br >= 8) {
- if (l >= listCapacityBytes)
- return -1;
- list[l++] = (uint8_t)aa;
- br -= 8;
- aa >>= 8;
- }
- if (br > 0) {
- list[l] = (uint8_t)aa;
- bitPtr = br;
- }
- }
- if (bitPtr > 0) {
- if (l >= listCapacityBytes)
- return -1;
- ++l;
- }
- return l;
- }
- #endif
- private:
- uint64_t _a;
- };
- } // namespace ZeroTier
- #endif
|