mirror of
https://github.com/oxen-io/lokinet.git
synced 2024-11-15 12:13:24 +00:00
6f31d5108b
If we get back an IPv6 address as the first gateway then we won't have the expected IPv4 gateway that the route poker needs to operate. This iterates through them separately so that we treat the IPv4 and IPv6 sides of an address as separate interfaces which should allow the route poker to find the one it wants (and just skip the IPv6 one).
207 lines
5.5 KiB
C++
207 lines
5.5 KiB
C++
#include "net.hpp"
|
|
|
|
#include "net_if.hpp"
|
|
#include <stdexcept>
|
|
#include <llarp/constants/platform.hpp>
|
|
|
|
#include "ip.hpp"
|
|
#include "ip_range.hpp"
|
|
#include <llarp/util/logging.hpp>
|
|
#include <llarp/util/str.hpp>
|
|
|
|
#include <iphlpapi.h>
|
|
#include <llarp/win32/exception.hpp>
|
|
|
|
#include <cstdio>
|
|
#include <list>
|
|
#include <type_traits>
|
|
|
|
namespace llarp::net
|
|
{
|
|
class Platform_Impl : public Platform
|
|
{
|
|
/// visit all adapters (not addresses). windows serves net info per adapter unlink posix which
|
|
/// gives a list of all distinct addresses.
|
|
template <typename Visit_t>
|
|
void
|
|
iter_adapters(Visit_t&& visit, int af = AF_UNSPEC) const
|
|
{
|
|
ULONG err;
|
|
ULONG sz = 15000; // MS-recommended so that it "never fails", but often fails with a too
|
|
// large error.
|
|
std::unique_ptr<byte_t[]> ptr;
|
|
PIP_ADAPTER_ADDRESSES addr;
|
|
int tries = 0;
|
|
do
|
|
{
|
|
ptr = std::make_unique<byte_t[]>(sz);
|
|
addr = reinterpret_cast<PIP_ADAPTER_ADDRESSES>(ptr.get());
|
|
err = GetAdaptersAddresses(
|
|
af, GAA_FLAG_INCLUDE_GATEWAYS | GAA_FLAG_INCLUDE_PREFIX, nullptr, addr, &sz);
|
|
} while (err == ERROR_BUFFER_OVERFLOW and ++tries < 4);
|
|
|
|
if (err != ERROR_SUCCESS)
|
|
throw llarp::win32::error{err, "GetAdaptersAddresses()"};
|
|
|
|
for (; addr; addr = addr->Next)
|
|
visit(addr);
|
|
}
|
|
|
|
template <typename adapter_t>
|
|
bool
|
|
adapter_has_ip(adapter_t* a, ipaddr_t ip) const
|
|
{
|
|
for (auto* addr = a->FirstUnicastAddress; addr; addr = addr->Next)
|
|
{
|
|
SockAddr saddr{*addr->Address.lpSockaddr};
|
|
LogDebug(fmt::format("'{}' has address '{}'", a->AdapterName, saddr));
|
|
if (saddr.getIP() == ip)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
template <typename adapter_t>
|
|
bool
|
|
adapter_has_fam(adapter_t* a, int af) const
|
|
{
|
|
for (auto* addr = a->FirstUnicastAddress; addr; addr = addr->Next)
|
|
{
|
|
SockAddr saddr{*addr->Address.lpSockaddr};
|
|
if (saddr.Family() == af)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public:
|
|
std::optional<int>
|
|
GetInterfaceIndex(ipaddr_t ip) const override
|
|
{
|
|
std::optional<int> found;
|
|
int af{AF_INET};
|
|
if (std::holds_alternative<ipv6addr_t>(ip))
|
|
af = AF_INET6;
|
|
iter_adapters(
|
|
[&found, ip, this](auto* adapter) {
|
|
if (found)
|
|
return;
|
|
|
|
LogDebug(fmt::format(
|
|
"visit adapter looking for '{}': '{}' idx={}",
|
|
ip,
|
|
adapter->AdapterName,
|
|
adapter->IfIndex));
|
|
if (adapter_has_ip(adapter, ip))
|
|
{
|
|
found = adapter->IfIndex;
|
|
}
|
|
},
|
|
af);
|
|
return found;
|
|
}
|
|
|
|
std::optional<llarp::SockAddr>
|
|
GetInterfaceAddr(std::string_view name, int af) const override
|
|
{
|
|
std::optional<SockAddr> found;
|
|
iter_adapters([name = std::string{name}, af, &found, this](auto* a) {
|
|
if (found)
|
|
return;
|
|
if (std::string{a->AdapterName} != name)
|
|
return;
|
|
|
|
if (adapter_has_fam(a, af))
|
|
found = SockAddr{*a->FirstUnicastAddress->Address.lpSockaddr};
|
|
});
|
|
return found;
|
|
}
|
|
|
|
std::optional<SockAddr>
|
|
AllInterfaces(SockAddr fallback) const override
|
|
{
|
|
// windows seems to not give a shit about source address
|
|
return fallback.isIPv6() ? SockAddr{"[::]"} : SockAddr{"0.0.0.0"};
|
|
}
|
|
|
|
std::optional<std::string>
|
|
FindFreeTun() const override
|
|
{
|
|
return "lokitun0";
|
|
}
|
|
|
|
std::optional<std::string>
|
|
GetBestNetIF(int) const override
|
|
{
|
|
// TODO: implement me ?
|
|
return std::nullopt;
|
|
}
|
|
|
|
std::optional<IPRange>
|
|
FindFreeRange() const override
|
|
{
|
|
std::list<IPRange> currentRanges;
|
|
iter_adapters([¤tRanges](auto* i) {
|
|
for (auto* addr = i->FirstUnicastAddress; addr; addr = addr->Next)
|
|
{
|
|
SockAddr saddr{*addr->Address.lpSockaddr};
|
|
currentRanges.emplace_back(
|
|
saddr.asIPv6(),
|
|
ipaddr_netmask_bits(addr->OnLinkPrefixLength, addr->Address.lpSockaddr->sa_family));
|
|
}
|
|
});
|
|
|
|
return IPRange::FindPrivateRange(currentRanges);
|
|
}
|
|
|
|
std::string
|
|
LoopbackInterfaceName() const override
|
|
{
|
|
// todo: implement me? does windows even have a loopback?
|
|
return "";
|
|
}
|
|
|
|
bool
|
|
HasInterfaceAddress(ipaddr_t ip) const override
|
|
{
|
|
return GetInterfaceIndex(ip) != std::nullopt;
|
|
}
|
|
|
|
std::vector<InterfaceInfo>
|
|
AllNetworkInterfaces() const override
|
|
{
|
|
std::vector<InterfaceInfo> all;
|
|
for (int af : {AF_INET, AF_INET6})
|
|
iter_adapters(
|
|
[&all](auto* a) {
|
|
auto& cur = all.emplace_back();
|
|
cur.index = a->IfIndex;
|
|
cur.name = a->AdapterName;
|
|
for (auto* addr = a->FirstUnicastAddress; addr; addr = addr->Next)
|
|
{
|
|
SockAddr saddr{*addr->Address.lpSockaddr};
|
|
cur.addrs.emplace_back(
|
|
saddr.asIPv6(),
|
|
ipaddr_netmask_bits(
|
|
addr->OnLinkPrefixLength, addr->Address.lpSockaddr->sa_family));
|
|
}
|
|
if (auto* addr = a->FirstGatewayAddress)
|
|
{
|
|
SockAddr gw{*addr->Address.lpSockaddr};
|
|
cur.gateway = gw.getIP();
|
|
}
|
|
},
|
|
af);
|
|
return all;
|
|
}
|
|
};
|
|
|
|
const Platform_Impl g_plat{};
|
|
|
|
const Platform*
|
|
Platform::Default_ptr()
|
|
{
|
|
return &g_plat;
|
|
}
|
|
} // namespace llarp::net
|