112 lines
3.3 KiB
C++
112 lines
3.3 KiB
C++
#include "NetworkAddresses.hh"
|
|
|
|
#include <arpa/inet.h>
|
|
#include <errno.h>
|
|
#include <ifaddrs.h>
|
|
#include <netdb.h>
|
|
#include <netinet/in.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/types.h>
|
|
|
|
#include <memory>
|
|
#include <phosg/Encoding.hh>
|
|
#include <phosg/Network.hh>
|
|
#include <phosg/Strings.hh>
|
|
#include <stdexcept>
|
|
|
|
using namespace std;
|
|
|
|
uint32_t resolve_address(const char* address) {
|
|
struct addrinfo* res0;
|
|
if (getaddrinfo(address, nullptr, nullptr, &res0)) {
|
|
auto e = phosg::string_for_error(errno);
|
|
throw runtime_error(phosg::string_printf(
|
|
"can\'t resolve hostname %s: %s", address, e.c_str()));
|
|
}
|
|
|
|
std::unique_ptr<struct addrinfo, void (*)(struct addrinfo*)> res0_unique(res0, freeaddrinfo);
|
|
struct addrinfo* res4 = nullptr;
|
|
for (struct addrinfo* res = res0; res; res = res->ai_next) {
|
|
if (res->ai_family == AF_INET) {
|
|
res4 = res;
|
|
}
|
|
}
|
|
if (!res4) {
|
|
throw runtime_error(phosg::string_printf(
|
|
"can\'t resolve hostname %s: no usable data", address));
|
|
}
|
|
|
|
struct sockaddr_in* res_sin = (struct sockaddr_in*)res4->ai_addr;
|
|
return ntohl(res_sin->sin_addr.s_addr);
|
|
}
|
|
|
|
map<string, uint32_t> get_local_addresses() {
|
|
struct ifaddrs* ifa_raw;
|
|
if (getifaddrs(&ifa_raw)) {
|
|
auto s = phosg::string_for_error(errno);
|
|
throw runtime_error(phosg::string_printf("failed to get interface addresses: %s", s.c_str()));
|
|
}
|
|
|
|
unique_ptr<struct ifaddrs, void (*)(struct ifaddrs*)> ifa(ifa_raw, freeifaddrs);
|
|
|
|
map<string, uint32_t> ret;
|
|
for (struct ifaddrs* i = ifa.get(); i; i = i->ifa_next) {
|
|
if (!i->ifa_addr) {
|
|
continue;
|
|
}
|
|
|
|
auto* sin = reinterpret_cast<sockaddr_in*>(i->ifa_addr);
|
|
if (sin->sin_family != AF_INET) {
|
|
continue;
|
|
}
|
|
|
|
ret.emplace(i->ifa_name, ntohl(sin->sin_addr.s_addr));
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
bool is_loopback_address(uint32_t addr) {
|
|
return ((addr & 0xFF000000) == 0x7F000000); // 127.0.0.0/8
|
|
}
|
|
|
|
bool is_local_address(uint32_t addr) {
|
|
return is_loopback_address(addr) || // 127.0.0.0/8
|
|
((addr & 0xFF000000) == 0x0A000000) || // 10.0.0.0/8
|
|
((addr & 0xFFF00000) == 0xAC100000) || // 172.16.0.0/12
|
|
((addr & 0xFFFF0000) == 0xC0A80000) || // 192.168.0.0/16
|
|
((addr & 0xFFFF0000) == 0xA9FE0000); // 169.254.0.0/16
|
|
}
|
|
|
|
bool is_local_address(const sockaddr_storage& daddr) {
|
|
if (daddr.ss_family != AF_INET) {
|
|
return false;
|
|
}
|
|
const sockaddr_in* sin = reinterpret_cast<const sockaddr_in*>(&daddr);
|
|
return is_local_address(ntohl(sin->sin_addr.s_addr));
|
|
}
|
|
|
|
string string_for_address(uint32_t address) {
|
|
return phosg::string_printf("%hhu.%hhu.%hhu.%hhu",
|
|
static_cast<uint8_t>(address >> 24), static_cast<uint8_t>(address >> 16),
|
|
static_cast<uint8_t>(address >> 8), static_cast<uint8_t>(address));
|
|
}
|
|
|
|
uint32_t address_for_string(const char* address) {
|
|
return ntohl(inet_addr(address));
|
|
}
|
|
|
|
uint64_t devolution_phone_number_for_netloc(uint32_t addr, uint16_t port) {
|
|
// It seems the address part of the number is fixed-width, but the port is
|
|
// not. Why did they do it this way?
|
|
if (port & 0xF000) {
|
|
return (static_cast<uint64_t>(addr) << 16) | port;
|
|
} else if (port & 0x0F00) {
|
|
return (static_cast<uint64_t>(addr) << 12) | port;
|
|
} else if (port & 0x00F0) {
|
|
return (static_cast<uint64_t>(addr) << 8) | port;
|
|
} else {
|
|
return (static_cast<uint64_t>(addr) << 4) | port;
|
|
}
|
|
}
|