mirror of
https://github.com/zerotier/ZeroTierOne.git
synced 2025-01-02 19:26:41 +00:00
168 lines
4.1 KiB
C++
168 lines
4.1 KiB
C++
/*
|
|
* Copyright (c)2019 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: 2025-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.
|
|
*/
|
|
/****/
|
|
|
|
#include "EthernetTap.hpp"
|
|
#include "OSUtils.hpp"
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#ifdef ZT_SDK
|
|
|
|
#include "../controller/EmbeddedNetworkController.hpp"
|
|
#include "../node/Node.hpp"
|
|
#include "../include/VirtualTap.hpp"
|
|
|
|
#else
|
|
|
|
#ifdef __APPLE__
|
|
#include <sys/sysctl.h>
|
|
#include "MacEthernetTap.hpp"
|
|
#include "MacKextEthernetTap.hpp"
|
|
#endif // __APPLE__
|
|
|
|
#ifdef __LINUX__
|
|
#include "LinuxEthernetTap.hpp"
|
|
#endif // __LINUX__
|
|
|
|
#ifdef __WINDOWS__
|
|
#include "WindowsEthernetTap.hpp"
|
|
#endif // __WINDOWS__
|
|
|
|
#ifdef __FreeBSD__
|
|
#include "BSDEthernetTap.hpp"
|
|
#endif // __FreeBSD__
|
|
|
|
#ifdef __NetBSD__
|
|
#include "NetBSDEthernetTap.hpp"
|
|
#endif // __NetBSD__
|
|
|
|
#ifdef __OpenBSD__
|
|
#include "BSDEthernetTap.hpp"
|
|
#endif // __OpenBSD__
|
|
|
|
#endif
|
|
|
|
namespace ZeroTier {
|
|
|
|
std::shared_ptr<EthernetTap> EthernetTap::newInstance(
|
|
const char *tapDeviceType, // OS-specific, NULL for default
|
|
const char *homePath,
|
|
const MAC &mac,
|
|
unsigned int mtu,
|
|
unsigned int metric,
|
|
uint64_t nwid,
|
|
const char *friendlyName,
|
|
void (*handler)(void *,void *,uint64_t,const MAC &,const MAC &,unsigned int,unsigned int,const void *,unsigned int),
|
|
void *arg)
|
|
{
|
|
|
|
#ifdef ZT_SDK
|
|
|
|
return std::shared_ptr<EthernetTap>(new VirtualTap(homePath,mac,mtu,metric,nwid,friendlyName,handler,arg));
|
|
|
|
#else // not ZT_SDK
|
|
|
|
#ifdef __APPLE__
|
|
char osrelease[256];
|
|
size_t size = sizeof(osrelease);
|
|
if (sysctlbyname("kern.osrelease",osrelease,&size,nullptr,0) == 0) {
|
|
char *dotAt = strchr(osrelease,'.');
|
|
if (dotAt) {
|
|
*dotAt = (char)0;
|
|
// The "feth" virtual Ethernet device type appeared in Darwin 17.x.x. Older versions
|
|
// (Sierra and earlier) must use the a kernel extension.
|
|
if (strtol(osrelease,(char **)0,10) < 17) {
|
|
return std::shared_ptr<EthernetTap>(new MacKextEthernetTap(homePath,mac,mtu,metric,nwid,friendlyName,handler,arg));
|
|
} else {
|
|
return std::shared_ptr<EthernetTap>(new MacEthernetTap(homePath,mac,mtu,metric,nwid,friendlyName,handler,arg));
|
|
}
|
|
}
|
|
}
|
|
#endif // __APPLE__
|
|
|
|
#ifdef __LINUX__
|
|
return std::shared_ptr<EthernetTap>(new LinuxEthernetTap(homePath,mac,mtu,metric,nwid,friendlyName,handler,arg));
|
|
#endif // __LINUX__
|
|
|
|
#ifdef __WINDOWS__
|
|
HRESULT hres = CoInitializeEx(0, COINIT_MULTITHREADED);
|
|
if (FAILED(hres)) {
|
|
throw std::runtime_error("WinEthernetTap: COM initialization failed");
|
|
}
|
|
|
|
static bool _comInit = false;
|
|
static Mutex _comInit_m;
|
|
|
|
{
|
|
Mutex::Lock l(_comInit_m);
|
|
if (!_comInit) {
|
|
hres = CoInitializeSecurity(
|
|
NULL,
|
|
-1,
|
|
NULL,
|
|
NULL,
|
|
RPC_C_AUTHN_LEVEL_PKT,
|
|
RPC_C_IMP_LEVEL_IMPERSONATE,
|
|
NULL,
|
|
EOAC_NONE,
|
|
NULL
|
|
);
|
|
if (FAILED(hres)) {
|
|
CoUninitialize();
|
|
fprintf(stderr, "WinEthernetTap: Failed to initialize security");
|
|
throw std::runtime_error("WinEthernetTap: Failed to initialize security");
|
|
}
|
|
_comInit = true;
|
|
}
|
|
}
|
|
return std::shared_ptr<EthernetTap>(new WindowsEthernetTap(homePath,mac,mtu,metric,nwid,friendlyName,handler,arg));
|
|
#endif // __WINDOWS__
|
|
|
|
#ifdef __FreeBSD__
|
|
return std::shared_ptr<EthernetTap>(new BSDEthernetTap(homePath,mac,mtu,metric,nwid,friendlyName,handler,arg));
|
|
#endif // __FreeBSD__
|
|
|
|
#ifdef __NetBSD__
|
|
return std::shared_ptr<EthernetTap>(new NetBSDEthernetTap(homePath,mac,mtu,metric,nwid,friendlyName,handler,arg));
|
|
#endif // __NetBSD__
|
|
|
|
#ifdef __OpenBSD__
|
|
return std::shared_ptr<EthernetTap>(new BSDEthernetTap(homePath,mac,mtu,metric,nwid,friendlyName,handler,arg));
|
|
#endif // __OpenBSD__
|
|
|
|
#endif // ZT_SDK?
|
|
|
|
return std::shared_ptr<EthernetTap>();
|
|
}
|
|
|
|
EthernetTap::EthernetTap() {}
|
|
EthernetTap::~EthernetTap() {}
|
|
|
|
bool EthernetTap::addIps(std::vector<InetAddress> ips)
|
|
{
|
|
for(std::vector<InetAddress>::const_iterator i(ips.begin());i!=ips.end();++i) {
|
|
if (!addIp(*i))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
std::string EthernetTap::friendlyName() const
|
|
{
|
|
// Most platforms do not have this.
|
|
return std::string();
|
|
}
|
|
|
|
} // namespace ZeroTier
|