2013-07-04 16:56:19 -04:00
/*
2015-02-17 13:11:34 -08:00
* ZeroTier One - Network Virtualization Everywhere
* Copyright ( C ) 2011 - 2015 ZeroTier , Inc .
2013-07-04 16:56:19 -04:00
*
* 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/>.
*
* - -
*
* 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/
*/
2013-08-06 00:05:39 -04:00
# include <stdio.h>
# include <string.h>
2013-07-29 13:56:20 -04:00
# include <stdlib.h>
# include <math.h>
2014-04-07 15:39:33 -07:00
# include "Constants.hpp"
2013-10-18 12:01:48 -04:00
# include "Network.hpp"
2013-07-29 13:56:20 -04:00
# include "RuntimeEnvironment.hpp"
2013-07-04 16:56:19 -04:00
# include "Switch.hpp"
2013-07-30 11:14:53 -04:00
# include "Packet.hpp"
2013-10-16 17:47:26 -04:00
# include "Buffer.hpp"
2015-04-15 15:12:09 -07:00
# include "NetworkController.hpp"
2013-07-04 16:56:19 -04:00
2015-07-23 09:50:10 -07:00
# include "../version.h"
2013-07-04 16:56:19 -04:00
namespace ZeroTier {
2015-04-06 15:47:57 -07:00
const ZeroTier : : MulticastGroup Network : : BROADCAST ( ZeroTier : : MAC ( 0xffffffffffffULL ) , 0 ) ;
2014-05-23 15:13:34 -07:00
2015-04-01 19:09:18 -07:00
Network : : Network ( const RuntimeEnvironment * renv , uint64_t nwid ) :
RR ( renv ) ,
_id ( nwid ) ,
_mac ( renv - > identity . address ( ) , nwid ) ,
_enabled ( true ) ,
2015-04-15 13:09:20 -07:00
_portInitialized ( false ) ,
2015-04-01 19:09:18 -07:00
_lastConfigUpdate ( 0 ) ,
_destroyed ( false ) ,
2015-04-06 16:52:52 -07:00
_netconfFailure ( NETCONF_FAILURE_NONE ) ,
_portError ( 0 )
2013-08-08 10:41:17 -04:00
{
2015-04-01 19:09:18 -07:00
char confn [ 128 ] , mcdbn [ 128 ] ;
Utils : : snprintf ( confn , sizeof ( confn ) , " networks.d/%.16llx.conf " , _id ) ;
Utils : : snprintf ( mcdbn , sizeof ( mcdbn ) , " networks.d/%.16llx.mcerts " , _id ) ;
if ( _id = = ZT_TEST_NETWORK_ID ) {
applyConfiguration ( NetworkConfig : : createTestNetworkConfig ( RR - > identity . address ( ) ) ) ;
// Save a one-byte CR to persist membership in the test network
RR - > node - > dataStorePut ( confn , " \n " , 1 , false ) ;
} else {
bool gotConf = false ;
try {
std : : string conf ( RR - > node - > dataStoreGet ( confn ) ) ;
if ( conf . length ( ) ) {
setConfiguration ( Dictionary ( conf ) , false ) ;
2015-04-14 15:32:05 -07:00
_lastConfigUpdate = 0 ; // we still want to re-request a new config from the network
2015-04-01 19:09:18 -07:00
gotConf = true ;
}
} catch ( . . . ) { } // ignore invalids, we'll re-request
if ( ! gotConf ) {
// Save a one-byte CR to persist membership while we request a real netconf
RR - > node - > dataStorePut ( confn , " \n " , 1 , false ) ;
}
try {
std : : string mcdb ( RR - > node - > dataStoreGet ( mcdbn ) ) ;
if ( mcdb . length ( ) > 6 ) {
const char * p = mcdb . data ( ) ;
const char * e = p + mcdb . length ( ) ;
if ( ! memcmp ( " ZTMCD0 " , p , 6 ) ) {
p + = 6 ;
while ( p ! = e ) {
CertificateOfMembership com ;
com . deserialize2 ( p , e ) ;
if ( ! com )
break ;
2015-09-04 14:14:32 -07:00
_certInfo [ com . issuedTo ( ) ] . com = com ;
2015-04-01 19:09:18 -07:00
}
}
}
} catch ( . . . ) { } // ignore invalid MCDB, we'll re-learn from peers
2013-08-08 10:41:17 -04:00
}
2015-04-01 19:09:18 -07:00
2015-04-15 13:09:20 -07:00
if ( ! _portInitialized ) {
ZT1_VirtualNetworkConfig ctmp ;
_externalConfig ( & ctmp ) ;
_portError = RR - > node - > configureVirtualNetworkPort ( _id , ZT1_VIRTUAL_NETWORK_CONFIG_OPERATION_UP , & ctmp ) ;
_portInitialized = true ;
}
2013-08-08 10:41:17 -04:00
}
2013-08-06 01:28:56 -04:00
Network : : ~ Network ( )
{
2015-04-06 18:27:24 -07:00
ZT1_VirtualNetworkConfig ctmp ;
_externalConfig ( & ctmp ) ;
2015-04-06 16:52:52 -07:00
2015-04-01 19:15:21 -07:00
char n [ 128 ] ;
2014-08-21 17:49:05 -07:00
if ( _destroyed ) {
2015-04-06 18:27:24 -07:00
RR - > node - > configureVirtualNetworkPort ( _id , ZT1_VIRTUAL_NETWORK_CONFIG_OPERATION_DESTROY , & ctmp ) ;
2015-04-01 19:09:18 -07:00
Utils : : snprintf ( n , sizeof ( n ) , " networks.d/%.16llx.conf " , _id ) ;
RR - > node - > dataStoreDelete ( n ) ;
Utils : : snprintf ( n , sizeof ( n ) , " networks.d/%.16llx.mcerts " , _id ) ;
RR - > node - > dataStoreDelete ( n ) ;
2013-08-06 01:28:56 -04:00
} else {
2015-04-06 18:27:24 -07:00
RR - > node - > configureVirtualNetworkPort ( _id , ZT1_VIRTUAL_NETWORK_CONFIG_OPERATION_DOWN , & ctmp ) ;
2013-08-06 01:28:56 -04:00
clean ( ) ;
2015-04-01 19:15:21 -07:00
Utils : : snprintf ( n , sizeof ( n ) , " networks.d/%.16llx.mcerts " , _id ) ;
2015-09-04 14:14:32 -07:00
Mutex : : Lock _l ( _lock ) ;
if ( ( ! _config ) | | ( _config - > isPublic ( ) ) | | ( _certInfo . empty ( ) ) ) {
2015-04-01 19:15:21 -07:00
RR - > node - > dataStoreDelete ( n ) ;
2015-09-04 14:14:32 -07:00
} else {
std : : string buf ( " ZTMCD0 " ) ;
Hashtable < Address , _RemoteMemberCertificateInfo > : : Iterator i ( _certInfo ) ;
Address * a = ( Address * ) 0 ;
_RemoteMemberCertificateInfo * ci = ( _RemoteMemberCertificateInfo * ) 0 ;
while ( i . next ( a , ci ) ) {
if ( ci - > com )
ci - > com . serialize2 ( buf ) ;
}
RR - > node - > dataStorePut ( n , buf , true ) ;
2015-04-01 19:15:21 -07:00
}
2013-08-06 01:28:56 -04:00
}
}
2015-04-06 19:41:55 -07:00
bool Network : : subscribedToMulticastGroup ( const MulticastGroup & mg , bool includeBridgedGroups ) const
{
Mutex : : Lock _l ( _lock ) ;
if ( std : : binary_search ( _myMulticastGroups . begin ( ) , _myMulticastGroups . end ( ) , mg ) )
return true ;
else if ( includeBridgedGroups )
2015-09-04 13:42:19 -07:00
return _multicastGroupsBehindMe . contains ( mg ) ;
2015-04-06 19:41:55 -07:00
else return false ;
}
2015-04-06 18:27:24 -07:00
void Network : : multicastSubscribe ( const MulticastGroup & mg )
2014-06-13 14:06:34 -07:00
{
2015-04-15 13:09:20 -07:00
{
Mutex : : Lock _l ( _lock ) ;
if ( std : : binary_search ( _myMulticastGroups . begin ( ) , _myMulticastGroups . end ( ) , mg ) )
return ;
_myMulticastGroups . push_back ( mg ) ;
std : : sort ( _myMulticastGroups . begin ( ) , _myMulticastGroups . end ( ) ) ;
}
2015-04-07 19:35:16 -07:00
_announceMulticastGroups ( ) ;
2015-04-06 18:27:24 -07:00
}
2014-10-03 18:27:42 -07:00
2015-04-06 18:27:24 -07:00
void Network : : multicastUnsubscribe ( const MulticastGroup & mg )
{
2015-04-06 18:28:18 -07:00
Mutex : : Lock _l ( _lock ) ;
std : : vector < MulticastGroup > nmg ;
for ( std : : vector < MulticastGroup > : : const_iterator i ( _myMulticastGroups . begin ( ) ) ; i ! = _myMulticastGroups . end ( ) ; + + i ) {
if ( * i ! = mg )
nmg . push_back ( * i ) ;
2014-10-03 18:27:42 -07:00
}
2015-04-06 18:28:18 -07:00
if ( nmg . size ( ) ! = _myMulticastGroups . size ( ) )
_myMulticastGroups . swap ( nmg ) ;
2015-04-06 18:27:24 -07:00
}
2014-10-03 18:27:42 -07:00
2014-10-03 16:14:34 -07:00
bool Network : : applyConfiguration ( const SharedPtr < NetworkConfig > & conf )
2013-07-29 13:56:20 -04:00
{
2015-04-15 13:09:20 -07:00
if ( _destroyed ) // sanity check
2014-08-21 17:49:05 -07:00
return false ;
2013-09-04 09:27:56 -04:00
try {
2014-10-03 16:14:34 -07:00
if ( ( conf - > networkId ( ) = = _id ) & & ( conf - > issuedTo ( ) = = RR - > identity . address ( ) ) ) {
2015-04-06 16:52:52 -07:00
ZT1_VirtualNetworkConfig ctmp ;
2015-04-15 13:09:20 -07:00
bool portInitialized ;
{
Mutex : : Lock _l ( _lock ) ;
_config = conf ;
_lastConfigUpdate = RR - > node - > now ( ) ;
_netconfFailure = NETCONF_FAILURE_NONE ;
_externalConfig ( & ctmp ) ;
portInitialized = _portInitialized ;
_portInitialized = true ;
}
_portError = RR - > node - > configureVirtualNetworkPort ( _id , ( portInitialized ) ? ZT1_VIRTUAL_NETWORK_CONFIG_OPERATION_CONFIG_UPDATE : ZT1_VIRTUAL_NETWORK_CONFIG_OPERATION_UP , & ctmp ) ;
2014-01-27 23:13:36 -08:00
return true ;
2013-10-18 12:01:48 -04:00
} else {
2015-04-08 16:49:21 -07:00
TRACE ( " ignored invalid configuration for network %.16llx (configuration contains mismatched network ID or issued-to address) " , ( unsigned long long ) _id ) ;
2013-08-06 00:05:39 -04:00
}
2013-10-18 12:01:48 -04:00
} catch ( std : : exception & exc ) {
2015-04-08 16:49:21 -07:00
TRACE ( " ignored invalid configuration for network %.16llx (%s) " , ( unsigned long long ) _id , exc . what ( ) ) ;
2013-09-04 09:27:56 -04:00
} catch ( . . . ) {
2015-04-08 16:49:21 -07:00
TRACE ( " ignored invalid configuration for network %.16llx (unknown exception) " , ( unsigned long long ) _id ) ;
2013-07-30 11:14:53 -04:00
}
2014-01-27 23:13:36 -08:00
return false ;
2013-07-29 13:56:20 -04:00
}
2015-01-05 17:47:59 -08:00
int Network : : setConfiguration ( const Dictionary & conf , bool saveToDisk )
2014-10-03 16:14:34 -07:00
{
try {
2015-04-06 15:47:57 -07:00
const SharedPtr < NetworkConfig > newConfig ( new NetworkConfig ( conf ) ) ; // throws if invalid
2015-01-05 17:47:59 -08:00
{
Mutex : : Lock _l ( _lock ) ;
if ( ( _config ) & & ( * _config = = * newConfig ) )
2015-01-09 16:35:20 -05:00
return 1 ; // OK config, but duplicate of what we already have
2015-01-05 17:47:59 -08:00
}
2014-10-03 16:14:34 -07:00
if ( applyConfiguration ( newConfig ) ) {
if ( saveToDisk ) {
2015-04-01 19:09:18 -07:00
char n [ 128 ] ;
Utils : : snprintf ( n , sizeof ( n ) , " networks.d/%.16llx.conf " , _id ) ;
RR - > node - > dataStorePut ( n , conf . toString ( ) , true ) ;
2014-10-03 16:14:34 -07:00
}
2015-01-05 17:47:59 -08:00
return 2 ; // OK and configuration has changed
2014-10-03 16:14:34 -07:00
}
} catch ( . . . ) {
2015-04-08 16:49:21 -07:00
TRACE ( " ignored invalid configuration for network %.16llx (dictionary decode failed) " , ( unsigned long long ) _id ) ;
2014-10-03 16:14:34 -07:00
}
2015-01-05 17:47:59 -08:00
return 0 ;
2014-10-03 16:14:34 -07:00
}
2013-07-29 13:56:20 -04:00
void Network : : requestConfiguration ( )
{
2015-04-15 15:12:09 -07:00
if ( _id = = ZT_TEST_NETWORK_ID ) // pseudo-network-ID, uses locally generated static config
2014-10-03 16:14:34 -07:00
return ;
2014-09-24 13:53:03 -07:00
if ( controller ( ) = = RR - > identity . address ( ) ) {
2015-04-15 15:12:09 -07:00
if ( RR - > localNetworkController ) {
2015-04-01 19:09:18 -07:00
SharedPtr < NetworkConfig > nconf ( config2 ( ) ) ;
Dictionary newconf ;
2015-09-08 11:35:55 -07:00
switch ( RR - > localNetworkController - > doNetworkConfigRequest ( InetAddress ( ) , RR - > identity , RR - > identity , _id , Dictionary ( ) , newconf ) ) {
2015-04-15 15:12:09 -07:00
case NetworkController : : NETCONF_QUERY_OK :
2015-04-01 19:09:18 -07:00
this - > setConfiguration ( newconf , true ) ;
return ;
2015-04-15 15:12:09 -07:00
case NetworkController : : NETCONF_QUERY_OBJECT_NOT_FOUND :
2015-04-01 19:09:18 -07:00
this - > setNotFound ( ) ;
return ;
2015-04-15 15:12:09 -07:00
case NetworkController : : NETCONF_QUERY_ACCESS_DENIED :
2015-04-01 19:09:18 -07:00
this - > setAccessDenied ( ) ;
return ;
default :
return ;
}
} else {
this - > setNotFound ( ) ;
return ;
}
2013-08-05 16:06:16 -04:00
}
2013-09-24 17:35:05 -04:00
2015-04-15 15:12:09 -07:00
TRACE ( " requesting netconf for network %.16llx from controller %s " , ( unsigned long long ) _id , controller ( ) . toString ( ) . c_str ( ) ) ;
2015-07-23 09:50:10 -07:00
// TODO: in the future we will include things like join tokens here, etc.
Dictionary metaData ;
metaData . setHex ( ZT_NETWORKCONFIG_REQUEST_METADATA_KEY_NODE_MAJOR_VERSION , ZEROTIER_ONE_VERSION_MAJOR ) ;
metaData . setHex ( ZT_NETWORKCONFIG_REQUEST_METADATA_KEY_NODE_MINOR_VERSION , ZEROTIER_ONE_VERSION_MINOR ) ;
metaData . setHex ( ZT_NETWORKCONFIG_REQUEST_METADATA_KEY_NODE_REVISION , ZEROTIER_ONE_VERSION_REVISION ) ;
std : : string mds ( metaData . toString ( ) ) ;
2014-09-24 13:53:03 -07:00
Packet outp ( controller ( ) , RR - > identity . address ( ) , Packet : : VERB_NETWORK_CONFIG_REQUEST ) ;
2013-07-30 11:14:53 -04:00
outp . append ( ( uint64_t ) _id ) ;
2015-07-23 09:50:10 -07:00
outp . append ( ( uint16_t ) mds . length ( ) ) ;
outp . append ( ( const void * ) mds . data ( ) , ( unsigned int ) mds . length ( ) ) ;
2015-01-05 17:51:50 -08:00
{
Mutex : : Lock _l ( _lock ) ;
if ( _config )
2015-04-01 19:09:18 -07:00
outp . append ( ( uint64_t ) _config - > revision ( ) ) ;
2015-01-05 17:51:50 -08:00
else outp . append ( ( uint64_t ) 0 ) ;
}
2015-07-07 10:02:48 -07:00
RR - > sw - > send ( outp , true , 0 ) ;
2013-07-29 13:56:20 -04:00
}
2015-07-07 08:14:41 -07:00
bool Network : : validateAndAddMembershipCertificate ( const CertificateOfMembership & cert )
2013-08-06 00:05:39 -04:00
{
2013-10-25 13:43:04 -04:00
if ( ! cert ) // sanity check
2015-07-07 08:14:41 -07:00
return false ;
2013-10-25 13:43:04 -04:00
2014-11-13 12:40:51 -08:00
Mutex : : Lock _l ( _lock ) ;
2015-09-04 14:14:32 -07:00
{
const _RemoteMemberCertificateInfo * ci = _certInfo . get ( cert . issuedTo ( ) ) ;
if ( ( ci ) & & ( ci - > com = = cert ) )
return true ; // we already have it
}
2014-11-13 12:40:51 -08:00
// Check signature, log and return if cert is invalid
2015-07-07 08:14:41 -07:00
if ( cert . signedBy ( ) ! = controller ( ) ) {
TRACE ( " rejected network membership certificate for %.16llx signed by %s: signer not a controller of this network " , ( unsigned long long ) _id , cert . signedBy ( ) . toString ( ) . c_str ( ) ) ;
return false ; // invalid signer
}
if ( cert . signedBy ( ) = = RR - > identity . address ( ) ) {
// We are the controller: RR->identity.address() == controller() == cert.signedBy()
// So, verify that we signed th cert ourself
if ( ! cert . verify ( RR - > identity ) ) {
TRACE ( " rejected network membership certificate for %.16llx self signed by %s: signature check failed " , ( unsigned long long ) _id , cert . signedBy ( ) . toString ( ) . c_str ( ) ) ;
return false ; // invalid signature
2014-10-14 12:37:35 -07:00
}
2015-07-07 08:14:41 -07:00
} else {
2014-10-14 12:37:35 -07:00
2015-07-07 08:14:41 -07:00
SharedPtr < Peer > signer ( RR - > topology - > getPeer ( cert . signedBy ( ) ) ) ;
2014-10-14 12:37:35 -07:00
2015-07-07 08:14:41 -07:00
if ( ! signer ) {
// This would be rather odd, since this is our controller... could happen
// if we get packets before we've gotten config.
RR - > sw - > requestWhois ( cert . signedBy ( ) ) ;
return false ; // signer unknown
}
2015-06-26 07:22:13 +02:00
2015-07-07 08:14:41 -07:00
if ( ! cert . verify ( signer - > identity ( ) ) ) {
TRACE ( " rejected network membership certificate for %.16llx signed by %s: signature check failed " , ( unsigned long long ) _id , cert . signedBy ( ) . toString ( ) . c_str ( ) ) ;
return false ; // invalid signature
2014-10-14 12:37:35 -07:00
}
2014-09-30 17:26:34 -07:00
}
2015-09-04 14:14:32 -07:00
// If we made it past authentication, add or update cert in our cert info store
_certInfo [ cert . issuedTo ( ) ] . com = cert ;
2015-07-07 08:14:41 -07:00
return true ;
2013-08-06 00:05:39 -04:00
}
2014-10-09 12:42:25 -07:00
bool Network : : peerNeedsOurMembershipCertificate ( const Address & to , uint64_t now )
{
Mutex : : Lock _l ( _lock ) ;
if ( ( _config ) & & ( ! _config - > isPublic ( ) ) & & ( _config - > com ( ) ) ) {
2015-09-04 14:14:32 -07:00
_RemoteMemberCertificateInfo & ci = _certInfo [ to ] ;
if ( ( now - ci . lastPushed ) > ( ZT_NETWORK_AUTOCONF_DELAY / 2 ) ) {
ci . lastPushed = now ;
2015-01-06 17:16:54 -08:00
return true ;
2014-10-09 12:42:25 -07:00
}
}
return false ;
}
2013-07-29 17:11:00 -04:00
void Network : : clean ( )
{
2015-04-08 15:26:45 -07:00
const uint64_t now = RR - > node - > now ( ) ;
2014-09-30 16:28:25 -07:00
Mutex : : Lock _l ( _lock ) ;
2014-08-21 17:49:05 -07:00
2014-09-30 16:28:25 -07:00
if ( _destroyed )
return ;
2014-08-21 17:49:05 -07:00
2014-09-30 16:28:25 -07:00
if ( ( _config ) & & ( _config - > isPublic ( ) ) ) {
// Open (public) networks do not track certs or cert pushes at all.
2015-09-04 14:14:32 -07:00
_certInfo . clear ( ) ;
2014-09-30 16:28:25 -07:00
} else if ( _config ) {
2015-09-04 14:14:32 -07:00
// Clean obsolete entries from private network cert info table
Hashtable < Address , _RemoteMemberCertificateInfo > : : Iterator i ( _certInfo ) ;
Address * a = ( Address * ) 0 ;
_RemoteMemberCertificateInfo * ci = ( _RemoteMemberCertificateInfo * ) 0 ;
const uint64_t forgetIfBefore = now - ( ZT_PEER_ACTIVITY_TIMEOUT * 16 ) ; // arbitrary reasonable cutoff
while ( i . next ( a , ci ) ) {
if ( ( ci - > lastPushed < forgetIfBefore ) & & ( ! ci - > com . agreesWith ( _config - > com ( ) ) ) )
_certInfo . erase ( * a ) ;
2013-08-03 12:53:46 -04:00
}
2013-07-29 17:11:00 -04:00
}
2014-09-30 16:28:25 -07:00
// Clean learned multicast groups if we haven't heard from them in a while
2015-09-04 13:42:19 -07:00
{
Hashtable < MulticastGroup , uint64_t > : : Iterator i ( _multicastGroupsBehindMe ) ;
MulticastGroup * mg = ( MulticastGroup * ) 0 ;
uint64_t * ts = ( uint64_t * ) 0 ;
while ( i . next ( mg , ts ) ) {
if ( ( now - * ts ) > ( ZT_MULTICAST_LIKE_EXPIRE * 2 ) )
_multicastGroupsBehindMe . erase ( * mg ) ;
}
2014-06-13 14:06:34 -07:00
}
}
2014-09-26 12:23:43 -07:00
void Network : : learnBridgeRoute ( const MAC & mac , const Address & addr )
2013-07-04 16:56:19 -04:00
{
2014-09-26 12:23:43 -07:00
Mutex : : Lock _l ( _lock ) ;
_remoteBridgeRoutes [ mac ] = addr ;
2013-10-16 17:47:26 -04:00
2015-09-04 13:53:48 -07:00
// Anti-DOS circuit breaker to prevent nodes from spamming us with absurd numbers of bridge routes
2014-09-26 12:23:43 -07:00
while ( _remoteBridgeRoutes . size ( ) > ZT_MAX_BRIDGE_ROUTES ) {
2015-09-04 13:53:48 -07:00
Hashtable < Address , unsigned long > counts ;
2014-09-26 12:23:43 -07:00
Address maxAddr ;
unsigned long maxCount = 0 ;
2015-09-04 13:53:48 -07:00
MAC * m = ( MAC * ) 0 ;
Address * a = ( Address * ) 0 ;
// Find the address responsible for the most entries
{
Hashtable < MAC , Address > : : Iterator i ( _remoteBridgeRoutes ) ;
while ( i . next ( m , a ) ) {
const unsigned long c = + + counts [ * a ] ;
if ( c > maxCount ) {
maxCount = c ;
maxAddr = * a ;
}
2014-09-26 12:23:43 -07:00
}
}
2015-09-04 13:53:48 -07:00
// Kill this address from our table, since it's most likely spamming us
{
Hashtable < MAC , Address > : : Iterator i ( _remoteBridgeRoutes ) ;
while ( i . next ( m , a ) ) {
if ( * a = = maxAddr )
_remoteBridgeRoutes . erase ( * m ) ;
}
2014-09-26 12:23:43 -07:00
}
2013-07-04 16:56:19 -04:00
}
}
2015-04-06 18:27:24 -07:00
void Network : : learnBridgedMulticastGroup ( const MulticastGroup & mg , uint64_t now )
{
Mutex : : Lock _l ( _lock ) ;
2015-09-04 13:42:19 -07:00
const unsigned long tmp = ( unsigned long ) _multicastGroupsBehindMe . size ( ) ;
_multicastGroupsBehindMe . set ( mg , now ) ;
2015-04-06 18:27:24 -07:00
if ( tmp ! = _multicastGroupsBehindMe . size ( ) )
_announceMulticastGroups ( ) ;
}
2014-09-26 12:23:43 -07:00
void Network : : setEnabled ( bool enabled )
2013-10-07 17:00:53 -04:00
{
2014-09-26 12:23:43 -07:00
Mutex : : Lock _l ( _lock ) ;
2015-04-14 17:57:51 -07:00
if ( _enabled ! = enabled ) {
_enabled = enabled ;
ZT1_VirtualNetworkConfig ctmp ;
_externalConfig ( & ctmp ) ;
_portError = RR - > node - > configureVirtualNetworkPort ( _id , ZT1_VIRTUAL_NETWORK_CONFIG_OPERATION_CONFIG_UPDATE , & ctmp ) ;
}
2014-09-26 12:23:43 -07:00
}
2013-10-07 17:00:53 -04:00
2014-09-26 12:23:43 -07:00
void Network : : destroy ( )
{
Mutex : : Lock _l ( _lock ) ;
_enabled = false ;
_destroyed = true ;
2013-10-07 17:00:53 -04:00
}
2015-04-06 16:52:52 -07:00
ZT1_VirtualNetworkStatus Network : : _status ( ) const
{
// assumes _lock is locked
if ( _portError )
return ZT1_NETWORK_STATUS_PORT_ERROR ;
switch ( _netconfFailure ) {
case NETCONF_FAILURE_ACCESS_DENIED :
return ZT1_NETWORK_STATUS_ACCESS_DENIED ;
case NETCONF_FAILURE_NOT_FOUND :
return ZT1_NETWORK_STATUS_NOT_FOUND ;
case NETCONF_FAILURE_NONE :
2015-04-15 13:09:20 -07:00
return ( ( _config ) ? ZT1_NETWORK_STATUS_OK : ZT1_NETWORK_STATUS_REQUESTING_CONFIGURATION ) ;
2015-04-06 16:52:52 -07:00
default :
return ZT1_NETWORK_STATUS_PORT_ERROR ;
}
}
void Network : : _externalConfig ( ZT1_VirtualNetworkConfig * ec ) const
{
// assumes _lock is locked
ec - > nwid = _id ;
2015-04-14 15:16:04 -07:00
ec - > mac = _mac . toInt ( ) ;
2015-04-06 16:52:52 -07:00
if ( _config )
Utils : : scopy ( ec - > name , sizeof ( ec - > name ) , _config - > name ( ) . c_str ( ) ) ;
else ec - > name [ 0 ] = ( char ) 0 ;
ec - > status = _status ( ) ;
ec - > type = ( _config ) ? ( _config - > isPrivate ( ) ? ZT1_NETWORK_TYPE_PRIVATE : ZT1_NETWORK_TYPE_PUBLIC ) : ZT1_NETWORK_TYPE_PRIVATE ;
ec - > mtu = ZT_IF_MTU ;
ec - > dhcp = 0 ;
ec - > bridge = ( _config ) ? ( ( _config - > allowPassiveBridging ( ) | | ( std : : find ( _config - > activeBridges ( ) . begin ( ) , _config - > activeBridges ( ) . end ( ) , RR - > identity . address ( ) ) ! = _config - > activeBridges ( ) . end ( ) ) ) ? 1 : 0 ) : 0 ;
ec - > broadcastEnabled = ( _config ) ? ( _config - > enableBroadcast ( ) ? 1 : 0 ) : 0 ;
ec - > portError = _portError ;
2015-04-14 17:57:51 -07:00
ec - > enabled = ( _enabled ) ? 1 : 0 ;
2015-04-06 16:52:52 -07:00
ec - > netconfRevision = ( _config ) ? ( unsigned long ) _config - > revision ( ) : 0 ;
2015-04-14 15:16:04 -07:00
ec - > multicastSubscriptionCount = std : : min ( ( unsigned int ) _myMulticastGroups . size ( ) , ( unsigned int ) ZT1_MAX_NETWORK_MULTICAST_SUBSCRIPTIONS ) ;
2015-04-06 16:52:52 -07:00
for ( unsigned int i = 0 ; i < ec - > multicastSubscriptionCount ; + + i ) {
ec - > multicastSubscriptions [ i ] . mac = _myMulticastGroups [ i ] . mac ( ) . toInt ( ) ;
ec - > multicastSubscriptions [ i ] . adi = _myMulticastGroups [ i ] . adi ( ) ;
}
if ( _config ) {
ec - > assignedAddressCount = ( unsigned int ) _config - > staticIps ( ) . size ( ) ;
for ( unsigned long i = 0 ; i < ZT1_MAX_ZT_ASSIGNED_ADDRESSES ; + + i ) {
if ( i < _config - > staticIps ( ) . size ( ) )
memcpy ( & ( ec - > assignedAddresses [ i ] ) , & ( _config - > staticIps ( ) [ i ] ) , sizeof ( struct sockaddr_storage ) ) ;
}
} else ec - > assignedAddressCount = 0 ;
}
2015-05-13 16:55:18 -07:00
bool Network : : _isAllowed ( const Address & peer ) const
{
// Assumes _lock is locked
try {
if ( ! _config )
return false ;
if ( _config - > isPublic ( ) )
return true ;
2015-09-04 14:14:32 -07:00
const _RemoteMemberCertificateInfo * ci = _certInfo . get ( peer ) ;
if ( ! ci )
return false ;
return _config - > com ( ) . agreesWith ( ci - > com ) ;
2015-05-13 16:55:18 -07:00
} catch ( std : : exception & exc ) {
TRACE ( " isAllowed() check failed for peer %s: unexpected exception: %s " , peer . toString ( ) . c_str ( ) , exc . what ( ) ) ;
} catch ( . . . ) {
TRACE ( " isAllowed() check failed for peer %s: unexpected exception: unknown exception " , peer . toString ( ) . c_str ( ) ) ;
}
return false ; // default position on any failure
}
2015-05-25 14:21:05 -07:00
std : : vector < MulticastGroup > Network : : _allMulticastGroups ( ) const
{
// Assumes _lock is locked
2015-06-01 11:56:15 -07:00
std : : vector < MulticastGroup > mgs ;
mgs . reserve ( _myMulticastGroups . size ( ) + _multicastGroupsBehindMe . size ( ) + 1 ) ;
mgs . insert ( mgs . end ( ) , _myMulticastGroups . begin ( ) , _myMulticastGroups . end ( ) ) ;
2015-09-04 13:42:19 -07:00
_multicastGroupsBehindMe . appendKeys ( mgs ) ;
2015-05-25 14:21:05 -07:00
if ( ( _config ) & & ( _config - > enableBroadcast ( ) ) )
mgs . push_back ( Network : : BROADCAST ) ;
std : : sort ( mgs . begin ( ) , mgs . end ( ) ) ;
2015-06-29 10:21:28 -07:00
mgs . erase ( std : : unique ( mgs . begin ( ) , mgs . end ( ) ) , mgs . end ( ) ) ;
2015-05-25 14:21:05 -07:00
return mgs ;
}
2015-04-06 18:27:24 -07:00
// Used in Network::_announceMulticastGroups()
class _AnnounceMulticastGroupsToPeersWithActiveDirectPaths
{
public :
_AnnounceMulticastGroupsToPeersWithActiveDirectPaths ( const RuntimeEnvironment * renv , Network * nw ) :
RR ( renv ) ,
2015-04-08 15:26:45 -07:00
_now ( renv - > node - > now ( ) ) ,
2015-04-06 18:27:24 -07:00
_network ( nw ) ,
2015-06-19 10:23:25 -07:00
_rootAddresses ( renv - > topology - > rootAddresses ( ) ) ,
2015-05-25 14:21:05 -07:00
_allMulticastGroups ( nw - > _allMulticastGroups ( ) )
2015-04-06 18:27:24 -07:00
{ }
inline void operator ( ) ( Topology & t , const SharedPtr < Peer > & p )
{
2015-07-31 09:50:55 -07:00
if ( ( ( p - > hasActiveDirectPath ( _now ) ) & & ( ( _network - > _isAllowed ( p - > address ( ) ) ) | | ( p - > address ( ) = = _network - > controller ( ) ) ) ) | | ( std : : find ( _rootAddresses . begin ( ) , _rootAddresses . end ( ) , p - > address ( ) ) ! = _rootAddresses . end ( ) ) ) {
2015-04-06 18:27:24 -07:00
Packet outp ( p - > address ( ) , RR - > identity . address ( ) , Packet : : VERB_MULTICAST_LIKE ) ;
2015-05-25 14:21:05 -07:00
for ( std : : vector < MulticastGroup > : : iterator mg ( _allMulticastGroups . begin ( ) ) ; mg ! = _allMulticastGroups . end ( ) ; + + mg ) {
if ( ( outp . size ( ) + 18 ) > = ZT_UDP_DEFAULT_PAYLOAD_MTU ) {
2015-04-06 18:27:24 -07:00
outp . armor ( p - > key ( ) , true ) ;
p - > send ( RR , outp . data ( ) , outp . size ( ) , _now ) ;
outp . reset ( p - > address ( ) , RR - > identity . address ( ) , Packet : : VERB_MULTICAST_LIKE ) ;
}
// network ID, MAC, ADI
outp . append ( ( uint64_t ) _network - > id ( ) ) ;
mg - > mac ( ) . appendTo ( outp ) ;
outp . append ( ( uint32_t ) mg - > adi ( ) ) ;
}
if ( outp . size ( ) > ZT_PROTO_MIN_PACKET_LENGTH ) {
outp . armor ( p - > key ( ) , true ) ;
p - > send ( RR , outp . data ( ) , outp . size ( ) , _now ) ;
}
}
}
private :
const RuntimeEnvironment * RR ;
uint64_t _now ;
Network * _network ;
2015-06-19 10:23:25 -07:00
std : : vector < Address > _rootAddresses ;
2015-05-25 14:21:05 -07:00
std : : vector < MulticastGroup > _allMulticastGroups ;
2015-04-06 18:27:24 -07:00
} ;
void Network : : _announceMulticastGroups ( )
{
2015-05-13 16:55:18 -07:00
// Assumes _lock is locked
2015-04-06 18:27:24 -07:00
_AnnounceMulticastGroupsToPeersWithActiveDirectPaths afunc ( RR , this ) ;
RR - > topology - > eachPeer < _AnnounceMulticastGroupsToPeersWithActiveDirectPaths & > ( afunc ) ;
}
2013-07-04 16:56:19 -04:00
} // namespace ZeroTier