You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
392 lines
12 KiB
392 lines
12 KiB
/* |
|
* ZeroTier One - Network Virtualization Everywhere |
|
* Copyright (C) 2011-2016 ZeroTier, Inc. https://www.zerotier.com/ |
|
* |
|
* 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 <http://www.gnu.org/licenses/>. |
|
*/ |
|
|
|
#ifndef ZT_NETWORK_HPP |
|
#define ZT_NETWORK_HPP |
|
|
|
#include <stdint.h> |
|
|
|
#include "../include/ZeroTierOne.h" |
|
|
|
#include <string> |
|
#include <map> |
|
#include <vector> |
|
#include <algorithm> |
|
#include <stdexcept> |
|
|
|
#include "Constants.hpp" |
|
#include "NonCopyable.hpp" |
|
#include "Hashtable.hpp" |
|
#include "Address.hpp" |
|
#include "Mutex.hpp" |
|
#include "SharedPtr.hpp" |
|
#include "AtomicCounter.hpp" |
|
#include "MulticastGroup.hpp" |
|
#include "MAC.hpp" |
|
#include "Dictionary.hpp" |
|
#include "Multicaster.hpp" |
|
#include "Membership.hpp" |
|
#include "NetworkConfig.hpp" |
|
#include "CertificateOfMembership.hpp" |
|
|
|
#define ZT_NETWORK_MAX_INCOMING_UPDATES 3 |
|
#define ZT_NETWORK_MAX_UPDATE_CHUNKS ((ZT_NETWORKCONFIG_DICT_CAPACITY / 1024) + 1) |
|
|
|
namespace ZeroTier { |
|
|
|
class RuntimeEnvironment; |
|
class Peer; |
|
|
|
/** |
|
* A virtual LAN |
|
*/ |
|
class Network : NonCopyable |
|
{ |
|
friend class SharedPtr<Network>; |
|
|
|
public: |
|
/** |
|
* Broadcast multicast group: ff:ff:ff:ff:ff:ff / 0 |
|
*/ |
|
static const MulticastGroup BROADCAST; |
|
|
|
/** |
|
* Compute primary controller device ID from network ID |
|
*/ |
|
static inline Address controllerFor(uint64_t nwid) throw() { return Address(nwid >> 24); } |
|
|
|
/** |
|
* Construct a new network |
|
* |
|
* Note that init() should be called immediately after the network is |
|
* constructed to actually configure the port. |
|
* |
|
* @param renv Runtime environment |
|
* @param nwid Network ID |
|
* @param uptr Arbitrary pointer used by externally-facing API (for user use) |
|
*/ |
|
Network(const RuntimeEnvironment *renv,uint64_t nwid,void *uptr); |
|
|
|
~Network(); |
|
|
|
inline uint64_t id() const { return _id; } |
|
inline Address controller() const { return Address(_id >> 24); } |
|
inline bool multicastEnabled() const { return (_config.multicastLimit > 0); } |
|
inline bool hasConfig() const { return (_config); } |
|
inline uint64_t lastConfigUpdate() const throw() { return _lastConfigUpdate; } |
|
inline ZT_VirtualNetworkStatus status() const { Mutex::Lock _l(_lock); return _status(); } |
|
inline const NetworkConfig &config() const { return _config; } |
|
inline const MAC &mac() const { return _mac; } |
|
|
|
/** |
|
* Apply filters to an outgoing packet |
|
* |
|
* This applies filters from our network config and, if that doesn't match, |
|
* our capabilities in ascending order of capability ID. Additional actions |
|
* such as TEE may be taken, and credentials may be pushed, so this is not |
|
* side-effect-free. It's basically step one in sending something over VL2. |
|
* |
|
* @param noTee If true, do not TEE anything anywhere (for two-pass filtering as done with multicast and bridging) |
|
* @param ztSource Source ZeroTier address |
|
* @param ztDest Destination ZeroTier address |
|
* @param macSource Ethernet layer source address |
|
* @param macDest Ethernet layer destination address |
|
* @param frameData Ethernet frame data |
|
* @param frameLen Ethernet frame payload length |
|
* @param etherType 16-bit ethernet type ID |
|
* @param vlanId 16-bit VLAN ID |
|
* @return True if packet should be sent, false if dropped or redirected |
|
*/ |
|
bool filterOutgoingPacket( |
|
const bool noTee, |
|
const Address &ztSource, |
|
const Address &ztDest, |
|
const MAC &macSource, |
|
const MAC &macDest, |
|
const uint8_t *frameData, |
|
const unsigned int frameLen, |
|
const unsigned int etherType, |
|
const unsigned int vlanId); |
|
|
|
/** |
|
* Apply filters to an incoming packet |
|
* |
|
* This applies filters from our network config and, if that doesn't match, |
|
* the peer's capabilities in ascending order of capability ID. If there is |
|
* a match certain actions may be taken such as sending a copy of the packet |
|
* to a TEE or REDIRECT target. |
|
* |
|
* @param sourcePeer Source Peer |
|
* @param ztDest Destination ZeroTier address |
|
* @param macSource Ethernet layer source address |
|
* @param macDest Ethernet layer destination address |
|
* @param frameData Ethernet frame data |
|
* @param frameLen Ethernet frame payload length |
|
* @param etherType 16-bit ethernet type ID |
|
* @param vlanId 16-bit VLAN ID |
|
* @return 0 == drop, 1 == accept, 2 == accept even if bridged |
|
*/ |
|
int filterIncomingPacket( |
|
const SharedPtr<Peer> &sourcePeer, |
|
const Address &ztDest, |
|
const MAC &macSource, |
|
const MAC &macDest, |
|
const uint8_t *frameData, |
|
const unsigned int frameLen, |
|
const unsigned int etherType, |
|
const unsigned int vlanId); |
|
|
|
/** |
|
* Check whether we are subscribed to a multicast group |
|
* |
|
* @param mg Multicast group |
|
* @param includeBridgedGroups If true, also check groups we've learned via bridging |
|
* @return True if this network endpoint / peer is a member |
|
*/ |
|
bool subscribedToMulticastGroup(const MulticastGroup &mg,bool includeBridgedGroups) const; |
|
|
|
/** |
|
* Subscribe to a multicast group |
|
* |
|
* @param mg New multicast group |
|
*/ |
|
void multicastSubscribe(const MulticastGroup &mg); |
|
|
|
/** |
|
* Unsubscribe from a multicast group |
|
* |
|
* @param mg Multicast group |
|
*/ |
|
void multicastUnsubscribe(const MulticastGroup &mg); |
|
|
|
/** |
|
* Handle an inbound network config chunk |
|
* |
|
* This is called from IncomingPacket to handle incoming network config |
|
* chunks via OK(NETWORK_CONFIG_REQUEST) or NETWORK_CONFIG. It verifies |
|
* each chunk and once assembled applies the configuration. |
|
* |
|
* @param chunk Packet containing chunk |
|
* @param ptr Index of chunk and related fields in packet |
|
* @return Update ID if update was fully assembled and accepted or 0 otherwise |
|
*/ |
|
uint64_t handleConfigChunk(const Packet &chunk,unsigned int ptr); |
|
|
|
/** |
|
* Set network configuration |
|
* |
|
* @param nconf Network configuration |
|
* @param saveToDisk Save to disk? Used during loading, should usually be true otherwise. |
|
* @return 0 == bad, 1 == accepted but duplicate/unchanged, 2 == accepted and new |
|
*/ |
|
int setConfiguration(const NetworkConfig &nconf,bool saveToDisk); |
|
|
|
/** |
|
* Set netconf failure to 'access denied' -- called in IncomingPacket when controller reports this |
|
*/ |
|
inline void setAccessDenied() |
|
{ |
|
Mutex::Lock _l(_lock); |
|
_netconfFailure = NETCONF_FAILURE_ACCESS_DENIED; |
|
} |
|
|
|
/** |
|
* Set netconf failure to 'not found' -- called by IncomingPacket when controller reports this |
|
*/ |
|
inline void setNotFound() |
|
{ |
|
Mutex::Lock _l(_lock); |
|
_netconfFailure = NETCONF_FAILURE_NOT_FOUND; |
|
} |
|
|
|
/** |
|
* Causes this network to request an updated configuration from its master node now |
|
*/ |
|
void requestConfiguration(); |
|
|
|
/** |
|
* Determine whether this peer is permitted to communicate on this network |
|
*/ |
|
bool gate(const SharedPtr<Peer> &peer); |
|
|
|
/** |
|
* Do periodic cleanup and housekeeping tasks |
|
*/ |
|
void clean(); |
|
|
|
/** |
|
* Push state to members such as multicast group memberships and latest COM (if needed) |
|
*/ |
|
inline void sendUpdatesToMembers() |
|
{ |
|
Mutex::Lock _l(_lock); |
|
_sendUpdatesToMembers((const MulticastGroup *)0); |
|
} |
|
|
|
/** |
|
* Find the node on this network that has this MAC behind it (if any) |
|
* |
|
* @param mac MAC address |
|
* @return ZeroTier address of bridge to this MAC |
|
*/ |
|
inline Address findBridgeTo(const MAC &mac) const |
|
{ |
|
Mutex::Lock _l(_lock); |
|
const Address *const br = _remoteBridgeRoutes.get(mac); |
|
return ((br) ? *br : Address()); |
|
} |
|
|
|
/** |
|
* Set a bridge route |
|
* |
|
* @param mac MAC address of destination |
|
* @param addr Bridge this MAC is reachable behind |
|
*/ |
|
void learnBridgeRoute(const MAC &mac,const Address &addr); |
|
|
|
/** |
|
* Learn a multicast group that is bridged to our tap device |
|
* |
|
* @param mg Multicast group |
|
* @param now Current time |
|
*/ |
|
void learnBridgedMulticastGroup(const MulticastGroup &mg,uint64_t now); |
|
|
|
/** |
|
* Validate a credential and learn it if it passes certificate and other checks |
|
*/ |
|
Membership::AddCredentialResult addCredential(const CertificateOfMembership &com); |
|
|
|
/** |
|
* Validate a credential and learn it if it passes certificate and other checks |
|
*/ |
|
inline Membership::AddCredentialResult addCredential(const Capability &cap) |
|
{ |
|
if (cap.networkId() != _id) |
|
return Membership::ADD_REJECTED; |
|
Mutex::Lock _l(_lock); |
|
return _membership(cap.issuedTo()).addCredential(RR,_config,cap); |
|
} |
|
|
|
/** |
|
* Validate a credential and learn it if it passes certificate and other checks |
|
*/ |
|
inline Membership::AddCredentialResult addCredential(const Tag &tag) |
|
{ |
|
if (tag.networkId() != _id) |
|
return Membership::ADD_REJECTED; |
|
Mutex::Lock _l(_lock); |
|
return _membership(tag.issuedTo()).addCredential(RR,_config,tag); |
|
} |
|
|
|
/** |
|
* Validate a credential and learn it if it passes certificate and other checks |
|
*/ |
|
Membership::AddCredentialResult addCredential(const Address &sentFrom,const Revocation &rev); |
|
|
|
/** |
|
* Force push credentials (COM, etc.) to a peer now |
|
* |
|
* @param to Destination peer address |
|
* @param now Current time |
|
*/ |
|
inline void pushCredentialsNow(const Address &to,const uint64_t now) |
|
{ |
|
Mutex::Lock _l(_lock); |
|
_membership(to).pushCredentials(RR,now,to,_config,-1,true); |
|
} |
|
|
|
/** |
|
* Destroy this network |
|
* |
|
* This causes the network to disable itself, destroy its tap device, and on |
|
* delete to delete all trace of itself on disk and remove any persistent tap |
|
* device instances. Call this when a network is being removed from the system. |
|
*/ |
|
void destroy(); |
|
|
|
/** |
|
* Get this network's config for export via the ZT core API |
|
* |
|
* @param ec Buffer to fill with externally-visible network configuration |
|
*/ |
|
inline void externalConfig(ZT_VirtualNetworkConfig *ec) const |
|
{ |
|
Mutex::Lock _l(_lock); |
|
_externalConfig(ec); |
|
} |
|
|
|
/** |
|
* @return Externally usable pointer-to-pointer exported via the core API |
|
*/ |
|
inline void **userPtr() throw() { return &_uPtr; } |
|
|
|
private: |
|
ZT_VirtualNetworkStatus _status() const; |
|
void _externalConfig(ZT_VirtualNetworkConfig *ec) const; // assumes _lock is locked |
|
bool _gate(const SharedPtr<Peer> &peer); |
|
void _sendUpdatesToMembers(const MulticastGroup *const newMulticastGroup); |
|
void _announceMulticastGroupsTo(const Address &peer,const std::vector<MulticastGroup> &allMulticastGroups); |
|
std::vector<MulticastGroup> _allMulticastGroups() const; |
|
Membership &_membership(const Address &a); |
|
|
|
const RuntimeEnvironment *const RR; |
|
void *_uPtr; |
|
const uint64_t _id; |
|
uint64_t _lastAnnouncedMulticastGroupsUpstream; |
|
MAC _mac; // local MAC address |
|
bool _portInitialized; |
|
|
|
std::vector< MulticastGroup > _myMulticastGroups; // multicast groups that we belong to (according to tap) |
|
Hashtable< MulticastGroup,uint64_t > _multicastGroupsBehindMe; // multicast groups that seem to be behind us and when we last saw them (if we are a bridge) |
|
Hashtable< MAC,Address > _remoteBridgeRoutes; // remote addresses where given MACs are reachable (for tracking devices behind remote bridges) |
|
|
|
NetworkConfig _config; |
|
uint64_t _lastConfigUpdate; |
|
|
|
struct _IncomingConfigChunk |
|
{ |
|
uint64_t ts; |
|
uint64_t updateId; |
|
uint64_t haveChunkIds[ZT_NETWORK_MAX_UPDATE_CHUNKS]; |
|
unsigned long haveChunks; |
|
unsigned long haveBytes; |
|
Dictionary<ZT_NETWORKCONFIG_DICT_CAPACITY> data; |
|
}; |
|
_IncomingConfigChunk _incomingConfigChunks[ZT_NETWORK_MAX_INCOMING_UPDATES]; |
|
|
|
bool _destroyed; |
|
|
|
enum { |
|
NETCONF_FAILURE_NONE, |
|
NETCONF_FAILURE_ACCESS_DENIED, |
|
NETCONF_FAILURE_NOT_FOUND, |
|
NETCONF_FAILURE_INIT_FAILED |
|
} _netconfFailure; |
|
int _portError; // return value from port config callback |
|
|
|
Hashtable<Address,Membership> _memberships; |
|
|
|
Mutex _lock; |
|
|
|
AtomicCounter __refCount; |
|
}; |
|
|
|
} // naemspace ZeroTier |
|
|
|
#endif
|
|
|