/* * ZeroTier One - Global Peer to Peer Ethernet * Copyright (C) 2012-2013 ZeroTier Networks LLC * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * * -- * * ZeroTier may be used and distributed under the terms of the GPLv3, which * are available at: http://www.gnu.org/licenses/gpl-3.0.html * * If you would like to embed ZeroTier into a commercial application or * redistribute it in a modified binary form, please contact ZeroTier Networks * LLC. Start here: http://www.zerotier.com/ */ #ifndef _ZT_ADDRESS_HPP #define _ZT_ADDRESS_HPP #include #include #include #include #include #include "Constants.hpp" #include "Utils.hpp" #include "MAC.hpp" #include "Buffer.hpp" namespace ZeroTier { /** * A ZeroTier address */ class Address { public: Address() throw() : _a(0) { } Address(const Address &a) throw() : _a(a._a) { } Address(uint64_t a) throw() : _a(a & 0xffffffffffULL) { } Address(const char *s) throw() { unsigned char foo[ZT_ADDRESS_LENGTH]; setTo(foo,Utils::unhex(s,foo,ZT_ADDRESS_LENGTH)); } Address(const std::string &s) throw() { unsigned char foo[ZT_ADDRESS_LENGTH]; setTo(foo,Utils::unhex(s.c_str(),foo,ZT_ADDRESS_LENGTH)); } /** * @param bits Raw address -- 5 bytes, big-endian byte order * @param len Length of array */ Address(const void *bits,unsigned int len) throw() { setTo(bits,len); } inline Address &operator=(const Address &a) throw() { _a = a._a; return *this; } inline Address &operator=(const uint64_t a) throw() { _a = (a & 0xffffffffffULL); return *this; } /** * @param bits Raw address -- 5 bytes, big-endian byte order * @param len Length of array */ inline void setTo(const void *bits,unsigned int len) throw() { if (len < ZT_ADDRESS_LENGTH) { _a = 0; return; } const unsigned char *b = (const unsigned char *)bits; uint64_t a = ((uint64_t)*b++) << 32; a |= ((uint64_t)*b++) << 24; a |= ((uint64_t)*b++) << 16; a |= ((uint64_t)*b++) << 8; a |= ((uint64_t)*b); _a = a; } /** * @param bits Buffer to hold 5-byte address in big-endian byte order * @param len Length of array */ inline void copyTo(void *bits,unsigned int len) const throw() { if (len < ZT_ADDRESS_LENGTH) return; unsigned char *b = (unsigned char *)bits; *(b++) = (unsigned char)((_a >> 32) & 0xff); *(b++) = (unsigned char)((_a >> 24) & 0xff); *(b++) = (unsigned char)((_a >> 16) & 0xff); *(b++) = (unsigned char)((_a >> 8) & 0xff); *b = (unsigned char)(_a & 0xff); } /** * Append to a buffer in big-endian byte order * * @param b Buffer to append to */ template inline void appendTo(Buffer &b) const throw(std::out_of_range) { b.append((unsigned char)((_a >> 32) & 0xff)); b.append((unsigned char)((_a >> 24) & 0xff)); b.append((unsigned char)((_a >> 16) & 0xff)); b.append((unsigned char)((_a >> 8) & 0xff)); b.append((unsigned char)(_a & 0xff)); } /** * @return String containing address as 5 binary bytes */ inline std::string toBinaryString() const { std::string b; b.push_back((char)((_a >> 32) & 0xff)); b.push_back((char)((_a >> 24) & 0xff)); b.push_back((char)((_a >> 16) & 0xff)); b.push_back((char)((_a >> 8) & 0xff)); b.push_back((char)(_a & 0xff)); return b; } /** * @return Integer containing address (0 to 2^40) */ inline uint64_t toInt() const throw() { return _a; } /** * Derive a MAC whose first octet is the ZeroTier LAN standard * * @return Ethernet MAC derived from address */ inline MAC toMAC() const throw() { MAC m; m.data[0] = ZT_MAC_FIRST_OCTET; copyTo(m.data + 1,ZT_ADDRESS_LENGTH); return m; } /** * @param mac MAC address to check * @return True if this address would have this MAC */ inline bool wouldHaveMac(const MAC &mac) const throw() { if (mac.data[0] != ZT_MAC_FIRST_OCTET) return false; if (mac.data[1] != (unsigned char)((_a >> 32) & 0xff)) return false; if (mac.data[2] != (unsigned char)((_a >> 24) & 0xff)) return false; if (mac.data[3] != (unsigned char)((_a >> 16) & 0xff)) return false; if (mac.data[4] != (unsigned char)((_a >> 8) & 0xff)) return false; if (mac.data[5] != (unsigned char)(_a & 0xff)) return false; return true; } /** * @return Hexadecimal string */ inline std::string toString() const { char buf[16]; Utils::snprintf(buf,sizeof(buf),"%.10llx",(unsigned long long)_a); return std::string(buf); }; /** * @return True if this address is not zero */ inline operator bool() const throw() { return (_a != 0); } /** * Set to null/zero */ inline void zero() throw() { _a = 0; } /** * 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 */ inline bool isReserved() const throw() { return ((!_a)||((_a >> 32) == ZT_ADDRESS_RESERVED_PREFIX)); } /** * @param i Value from 0 to 4 (inclusive) * @return Byte at said position (address interpreted in big-endian order) */ inline unsigned char operator[](unsigned int i) const throw() { return (unsigned char)((_a >> (32 - (i * 8))) & 0xff); } inline bool operator==(const Address &a) const throw() { return (_a == a._a); } inline bool operator!=(const Address &a) const throw() { return (_a != a._a); } inline bool operator>(const Address &a) const throw() { return (_a > a._a); } inline bool operator<(const Address &a) const throw() { return (_a < a._a); } inline bool operator>=(const Address &a) const throw() { return (_a >= a._a); } inline bool operator<=(const Address &a) const throw() { return (_a <= a._a); } private: uint64_t _a; }; } // namespace ZeroTier #endif