2014-09-26 05:08:52 +00:00
|
|
|
/*
|
2015-02-17 21:11:34 +00:00
|
|
|
* ZeroTier One - Network Virtualization Everywhere
|
2016-01-12 22:04:55 +00:00
|
|
|
* Copyright (C) 2011-2016 ZeroTier, Inc. https://www.zerotier.com/
|
2014-09-26 05:08:52 +00: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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Constants.hpp"
|
2014-10-09 19:42:25 +00:00
|
|
|
#include "RuntimeEnvironment.hpp"
|
2014-09-26 05:08:52 +00:00
|
|
|
#include "OutboundMulticast.hpp"
|
|
|
|
#include "Switch.hpp"
|
2014-10-09 19:42:25 +00:00
|
|
|
#include "Network.hpp"
|
2014-09-26 21:18:25 +00:00
|
|
|
#include "CertificateOfMembership.hpp"
|
2015-04-08 22:42:23 +00:00
|
|
|
#include "Node.hpp"
|
2014-09-26 05:08:52 +00:00
|
|
|
|
|
|
|
namespace ZeroTier {
|
|
|
|
|
2014-10-01 21:05:25 +00:00
|
|
|
void OutboundMulticast::init(
|
2014-10-10 00:58:31 +00:00
|
|
|
const RuntimeEnvironment *RR,
|
2014-10-01 21:05:25 +00:00
|
|
|
uint64_t timestamp,
|
|
|
|
uint64_t nwid,
|
|
|
|
const CertificateOfMembership *com,
|
|
|
|
unsigned int limit,
|
|
|
|
unsigned int gatherLimit,
|
|
|
|
const MAC &src,
|
|
|
|
const MulticastGroup &dest,
|
|
|
|
unsigned int etherType,
|
|
|
|
const void *payload,
|
|
|
|
unsigned int len)
|
2014-09-26 05:08:52 +00:00
|
|
|
{
|
|
|
|
_timestamp = timestamp;
|
|
|
|
_nwid = nwid;
|
2014-10-01 21:05:25 +00:00
|
|
|
_limit = limit;
|
2014-09-30 15:38:03 +00:00
|
|
|
|
2014-10-09 19:42:25 +00:00
|
|
|
uint8_t flags = 0;
|
|
|
|
if (gatherLimit) flags |= 0x02;
|
|
|
|
if (src) flags |= 0x04;
|
2014-09-30 15:38:03 +00:00
|
|
|
|
2014-10-10 01:32:05 +00:00
|
|
|
/*
|
2014-10-10 00:58:31 +00:00
|
|
|
TRACE(">>MC %.16llx INIT %.16llx/%s limit %u gatherLimit %u from %s to %s length %u com==%d",
|
|
|
|
(unsigned long long)this,
|
|
|
|
nwid,
|
|
|
|
dest.toString().c_str(),
|
|
|
|
limit,
|
|
|
|
gatherLimit,
|
|
|
|
(src) ? src.toString().c_str() : MAC(RR->identity.address(),nwid).toString().c_str(),
|
|
|
|
dest.toString().c_str(),
|
|
|
|
len,
|
|
|
|
(com) ? 1 : 0);
|
2014-10-10 01:32:05 +00:00
|
|
|
*/
|
2014-10-10 00:58:31 +00:00
|
|
|
|
|
|
|
_packetNoCom.setSource(RR->identity.address());
|
2014-10-09 19:42:25 +00:00
|
|
|
_packetNoCom.setVerb(Packet::VERB_MULTICAST_FRAME);
|
|
|
|
_packetNoCom.append((uint64_t)nwid);
|
|
|
|
_packetNoCom.append(flags);
|
|
|
|
if (gatherLimit) _packetNoCom.append((uint32_t)gatherLimit);
|
|
|
|
if (src) src.appendTo(_packetNoCom);
|
|
|
|
dest.mac().appendTo(_packetNoCom);
|
|
|
|
_packetNoCom.append((uint32_t)dest.adi());
|
|
|
|
_packetNoCom.append((uint16_t)etherType);
|
|
|
|
_packetNoCom.append(payload,len);
|
|
|
|
_packetNoCom.compress();
|
2014-09-30 15:38:03 +00:00
|
|
|
|
2014-10-09 19:42:25 +00:00
|
|
|
if (com) {
|
|
|
|
_haveCom = true;
|
|
|
|
flags |= 0x01;
|
|
|
|
|
2014-10-10 00:58:31 +00:00
|
|
|
_packetWithCom.setSource(RR->identity.address());
|
2014-10-09 19:42:25 +00:00
|
|
|
_packetWithCom.setVerb(Packet::VERB_MULTICAST_FRAME);
|
|
|
|
_packetWithCom.append((uint64_t)nwid);
|
|
|
|
_packetWithCom.append(flags);
|
|
|
|
com->serialize(_packetWithCom);
|
|
|
|
if (gatherLimit) _packetWithCom.append((uint32_t)gatherLimit);
|
|
|
|
if (src) src.appendTo(_packetWithCom);
|
|
|
|
dest.mac().appendTo(_packetWithCom);
|
|
|
|
_packetWithCom.append((uint32_t)dest.adi());
|
|
|
|
_packetWithCom.append((uint16_t)etherType);
|
|
|
|
_packetWithCom.append(payload,len);
|
|
|
|
_packetWithCom.compress();
|
|
|
|
} else _haveCom = false;
|
2014-09-26 05:08:52 +00:00
|
|
|
}
|
|
|
|
|
2014-10-09 19:42:25 +00:00
|
|
|
void OutboundMulticast::sendOnly(const RuntimeEnvironment *RR,const Address &toAddr)
|
2014-09-26 05:08:52 +00:00
|
|
|
{
|
2014-10-09 19:42:25 +00:00
|
|
|
if (_haveCom) {
|
2015-10-01 18:11:52 +00:00
|
|
|
SharedPtr<Peer> peer(RR->topology->getPeer(toAddr));
|
|
|
|
if ( (!peer) || (peer->needsOurNetworkMembershipCertificate(_nwid,RR->node->now(),true)) ) {
|
|
|
|
//TRACE(">>MC %.16llx -> %s (with COM)",(unsigned long long)this,toAddr.toString().c_str());
|
2014-10-09 19:42:25 +00:00
|
|
|
_packetWithCom.newInitializationVector();
|
|
|
|
_packetWithCom.setDestination(toAddr);
|
2015-06-02 00:50:44 +00:00
|
|
|
RR->sw->send(_packetWithCom,true,_nwid);
|
2014-10-09 19:42:25 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2015-07-06 21:53:27 +00:00
|
|
|
|
2014-10-10 01:32:05 +00:00
|
|
|
//TRACE(">>MC %.16llx -> %s (without COM)",(unsigned long long)this,toAddr.toString().c_str());
|
2014-10-09 19:42:25 +00:00
|
|
|
_packetNoCom.newInitializationVector();
|
|
|
|
_packetNoCom.setDestination(toAddr);
|
2015-06-02 00:50:44 +00:00
|
|
|
RR->sw->send(_packetNoCom,true,_nwid);
|
2014-09-26 05:08:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace ZeroTier
|