1
1
Fork 0
mirror of https://github.com/oxen-io/lokinet synced 2023-12-14 06:53:00 +01:00
lokinet/llarp/net.hpp

192 lines
4.4 KiB
C++
Raw Normal View History

2018-05-18 15:17:58 +02:00
#ifndef LLARP_NET_HPP
#define LLARP_NET_HPP
#include <llarp/address_info.h>
2018-05-18 15:17:58 +02:00
#include <llarp/net.h>
#include <string>
2018-05-23 15:49:00 +02:00
#include "mem.hpp"
2018-05-18 15:17:58 +02:00
bool
operator==(const sockaddr& a, const sockaddr& b);
2018-05-18 15:17:58 +02:00
bool
operator<(const sockaddr_in6& a, const sockaddr_in6& b);
2018-05-18 15:54:15 +02:00
bool
operator<(const in6_addr& a, const in6_addr& b);
2018-05-18 15:54:15 +02:00
namespace llarp
{
struct Addr
{
2018-05-23 15:49:00 +02:00
sockaddr_in6 _addr;
2018-05-29 14:15:48 +02:00
sockaddr_in _addr4;
~Addr(){};
Addr(){};
Addr(const Addr& other)
{
2018-05-23 15:49:00 +02:00
memcpy(&_addr, &other._addr, sizeof(sockaddr_in6));
2018-05-29 15:40:26 +02:00
memcpy(&_addr4, &other._addr4, sizeof(sockaddr_in));
2018-05-23 15:49:00 +02:00
}
in6_addr*
addr6()
{
2018-05-23 22:37:43 +02:00
return (in6_addr*)&_addr.sin6_addr.s6_addr[0];
2018-05-23 15:49:00 +02:00
}
in_addr*
addr4()
{
return (in_addr*)&_addr.sin6_addr.s6_addr[12];
}
const in6_addr*
addr6() const
{
2018-05-23 22:37:43 +02:00
return (const in6_addr*)&_addr.sin6_addr.s6_addr[0];
2018-05-23 15:49:00 +02:00
}
const in_addr*
addr4() const
{
return (const in_addr*)&_addr.sin6_addr.s6_addr[12];
2018-05-18 19:10:48 +02:00
}
Addr(const llarp_ai& other)
{
2018-05-23 15:49:00 +02:00
memcpy(addr6(), other.ip.s6_addr, 16);
_addr.sin6_port = htons(other.port);
2018-05-29 14:15:48 +02:00
auto ptr = &_addr.sin6_addr.s6_addr[0];
// TODO: detect SIIT better
2018-05-29 15:40:26 +02:00
if(ptr[11] == 0xff && ptr[10] == 0xff && ptr[9] == 0 && ptr[8] == 0
&& ptr[7] == 0 && ptr[6] == 0 && ptr[5] == 0 && ptr[4] == 0
&& ptr[3] == 0 && ptr[2] == 0 && ptr[1] == 0 && ptr[0] == 0)
2018-05-29 14:15:48 +02:00
{
_addr4.sin_family = AF_INET;
_addr4.sin_port = htons(other.port);
_addr.sin6_family = AF_INET;
memcpy(&_addr4.sin_addr.s_addr, addr4(), sizeof(in_addr));
}
else
_addr.sin6_family = AF_INET6;
2018-05-18 19:10:48 +02:00
}
Addr(const sockaddr& other)
{
2018-05-23 15:49:00 +02:00
llarp::Zero(&_addr, sizeof(sockaddr_in6));
_addr.sin6_family = other.sa_family;
uint8_t* addrptr = _addr.sin6_addr.s6_addr;
uint16_t* port = &_addr.sin6_port;
2018-05-18 15:54:15 +02:00
switch(other.sa_family)
{
case AF_INET:
// SIIT
memcpy(12 + addrptr, &((const sockaddr_in*)(&other))->sin_addr,
sizeof(in_addr));
2018-05-29 14:15:48 +02:00
addrptr[11] = 0xff;
addrptr[10] = 0xff;
*port = ((sockaddr_in*)(&other))->sin_port;
_addr4.sin_family = AF_INET;
_addr4.sin_port = *port;
memcpy(&_addr4.sin_addr.s_addr, addr4(), sizeof(in_addr));
break;
case AF_INET6:
2018-05-23 15:49:00 +02:00
memcpy(addrptr, &((const sockaddr_in6*)(&other))->sin6_addr.s6_addr,
16);
*port = ((sockaddr_in6*)(&other))->sin6_port;
break;
// TODO : sockaddr_ll
default:
break;
2018-05-18 15:54:15 +02:00
}
2018-05-18 19:10:48 +02:00
}
std::string
to_string() const
2018-05-21 16:28:15 +02:00
{
std::string str;
char tmp[128];
socklen_t sz;
const void* ptr = nullptr;
2018-05-23 15:49:00 +02:00
if(af() == AF_INET6)
{
str += "[";
sz = sizeof(sockaddr_in6);
2018-05-23 15:49:00 +02:00
ptr = addr6();
}
else
{
sz = sizeof(sockaddr_in);
ptr = addr4();
}
2018-05-23 15:49:00 +02:00
if(inet_ntop(af(), ptr, tmp, sz))
{
str += tmp;
2018-05-23 15:49:00 +02:00
if(af() == AF_INET6)
str += "]";
}
2018-05-23 15:49:00 +02:00
return str + ":" + std::to_string(port());
2018-05-21 16:28:15 +02:00
}
operator const sockaddr*() const
2018-05-18 19:10:48 +02:00
{
2018-05-29 14:15:48 +02:00
if(af() == AF_INET)
return (const sockaddr*)&_addr4;
else
return (const sockaddr*)&_addr;
2018-05-18 15:54:15 +02:00
}
void
2018-05-23 15:49:00 +02:00
CopyInto(sockaddr* other) const
{
void *dst, *src;
in_port_t* ptr;
size_t slen;
2018-05-23 15:49:00 +02:00
switch(af())
{
case AF_INET:
2018-05-23 15:49:00 +02:00
dst = (void*)&((sockaddr_in*)other)->sin_addr.s_addr;
src = (void*)&_addr.sin6_addr.s6_addr[12];
ptr = &((sockaddr_in*)other)->sin_port;
slen = sizeof(in_addr);
break;
case AF_INET6:
2018-05-23 15:49:00 +02:00
dst = (void*)((sockaddr_in6*)other)->sin6_addr.s6_addr;
src = (void*)_addr.sin6_addr.s6_addr;
ptr = &((sockaddr_in6*)other)->sin6_port;
slen = sizeof(in6_addr);
break;
default:
return;
}
memcpy(dst, src, slen);
2018-05-23 15:49:00 +02:00
*ptr = htons(port());
other->sa_family = af();
}
int
af() const
{
return _addr.sin6_family;
}
uint16_t
port() const
{
return ntohs(_addr.sin6_port);
}
bool
operator<(const Addr& other) const
2018-05-18 15:54:15 +02:00
{
2018-05-29 18:01:28 +02:00
return af() < other.af() && memcmp(addr6(), other.addr6(), 16) < 0
2018-05-23 15:49:00 +02:00
&& port() < other.port();
2018-05-18 15:54:15 +02:00
}
};
}
2018-05-18 15:17:58 +02:00
#endif