mirror of
https://github.com/oxen-io/lokinet.git
synced 2024-10-31 09:20:21 +00:00
49b9ad7197
* partial tun code refactor * take out the trash * move vpn platform code into llarp/vpn/platform.cpp * fix hive build * fix win32 * fix memory leak on win32 * reduce cpu use * make macos compile * win32 patches: * use wepoll for zmq * use all cores on windows iocp read loop * fix zmq patch for windows * clean up cmake for win32 * add uninstall before reinstall option to win32 installer * more ipv6 stuff * make it compile * fix up route poker * remove an unneeded code block in macos wtf * always use call to system * fix route poker behavior on macos * disable ipv6 on windows for now * cpu perf improvement: * colease calls to Router::PumpLL to 1 per event loop wakeup * set up THEN add addresses * emulate proactor event loop on win32 * remove excessively verbose error message * fix issue #1499 * exclude uv_poll from win32 so that it can start up * update logtag to include directory * create minidump on windows if there was a crash * make windows happy * use dmp suffix on minidump files * typo fix * address feedback from jason * use PROJECT_SOURCE_DIR instead of CMAKE_SOURCE_DIR * quote $@ in apply-patches in case path has spaces in it * address feedback from tom * remove llarp/ev/pipe * add comments for clairification * make event loop queue size constant named
130 lines
3.1 KiB
C++
130 lines
3.1 KiB
C++
#pragma once
|
|
|
|
#include <ev/vpn.hpp>
|
|
#include <unistd.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/types.h>
|
|
#include <fcntl.h>
|
|
#include <vpn/common.hpp>
|
|
#include <linux/if_tun.h>
|
|
|
|
namespace llarp::vpn
|
|
{
|
|
struct in6_ifreq
|
|
{
|
|
in6_addr addr;
|
|
uint32_t prefixlen;
|
|
unsigned int ifindex;
|
|
};
|
|
|
|
class LinuxInterface : public NetworkInterface
|
|
{
|
|
const int m_fd;
|
|
const InterfaceInfo m_Info;
|
|
|
|
public:
|
|
LinuxInterface(InterfaceInfo info)
|
|
: NetworkInterface{}, m_fd{::open("/dev/net/tun", O_RDWR)}, m_Info{std::move(info)}
|
|
|
|
{
|
|
if (m_fd == -1)
|
|
throw std::runtime_error("cannot open /dev/net/tun " + std::string{strerror(errno)});
|
|
|
|
ifreq ifr{};
|
|
in6_ifreq ifr6{};
|
|
ifr.ifr_flags = IFF_TUN | IFF_NO_PI;
|
|
std::copy_n(
|
|
m_Info.ifname.c_str(),
|
|
std::min(m_Info.ifname.size(), sizeof(ifr.ifr_name)),
|
|
ifr.ifr_name);
|
|
if (::ioctl(m_fd, TUNSETIFF, &ifr) == -1)
|
|
throw std::runtime_error("cannot set interface name: " + std::string{strerror(errno)});
|
|
IOCTL control{AF_INET};
|
|
|
|
control.ioctl(SIOCGIFFLAGS, &ifr);
|
|
const int flags = ifr.ifr_flags;
|
|
control.ioctl(SIOCGIFINDEX, &ifr);
|
|
const int ifindex = ifr.ifr_ifindex;
|
|
|
|
IOCTL control6{AF_INET6};
|
|
for (const auto& ifaddr : m_Info.addrs)
|
|
{
|
|
if (ifaddr.fam == AF_INET)
|
|
{
|
|
ifr.ifr_addr.sa_family = AF_INET;
|
|
const nuint32_t addr = ToNet(net::TruncateV6(ifaddr.range.addr));
|
|
((sockaddr_in*)&ifr.ifr_addr)->sin_addr.s_addr = addr.n;
|
|
control.ioctl(SIOCSIFADDR, &ifr);
|
|
|
|
const nuint32_t mask = ToNet(net::TruncateV6(ifaddr.range.netmask_bits));
|
|
((sockaddr_in*)&ifr.ifr_netmask)->sin_addr.s_addr = mask.n;
|
|
control.ioctl(SIOCSIFNETMASK, &ifr);
|
|
}
|
|
if (ifaddr.fam == AF_INET6)
|
|
{
|
|
ifr6.addr = net::HUIntToIn6(ifaddr.range.addr);
|
|
ifr6.prefixlen = llarp::bits::count_bits(ifaddr.range.netmask_bits);
|
|
ifr6.ifindex = ifindex;
|
|
control6.ioctl(SIOCSIFADDR, &ifr6);
|
|
}
|
|
}
|
|
ifr.ifr_flags = flags | IFF_UP | IFF_NO_PI;
|
|
control.ioctl(SIOCSIFFLAGS, &ifr);
|
|
}
|
|
|
|
virtual ~LinuxInterface()
|
|
{
|
|
if (m_fd != -1)
|
|
::close(m_fd);
|
|
}
|
|
|
|
int
|
|
PollFD() const override
|
|
{
|
|
return m_fd;
|
|
}
|
|
|
|
net::IPPacket
|
|
ReadNextPacket() override
|
|
{
|
|
net::IPPacket pkt;
|
|
const auto sz = read(m_fd, pkt.buf, sizeof(pkt.buf));
|
|
if (sz >= 0)
|
|
pkt.sz = std::min(sz, ssize_t{sizeof(pkt.buf)});
|
|
return pkt;
|
|
}
|
|
|
|
bool
|
|
WritePacket(net::IPPacket pkt) override
|
|
{
|
|
const auto sz = write(m_fd, pkt.buf, pkt.sz);
|
|
if (sz <= 0)
|
|
return false;
|
|
return sz == static_cast<ssize_t>(pkt.sz);
|
|
}
|
|
|
|
bool
|
|
HasNextPacket() override
|
|
{
|
|
return false;
|
|
}
|
|
|
|
std::string
|
|
IfName() const override
|
|
{
|
|
return m_Info.ifname;
|
|
}
|
|
};
|
|
|
|
class LinuxPlatform : public Platform
|
|
{
|
|
public:
|
|
std::shared_ptr<NetworkInterface>
|
|
ObtainInterface(InterfaceInfo info) override
|
|
{
|
|
return std::make_shared<LinuxInterface>(std::move(info));
|
|
};
|
|
};
|
|
|
|
} // namespace llarp::vpn
|