mirror of
https://github.com/kunkundi/crossdesk.git
synced 2025-10-27 04:35:34 +08:00
311 lines
8.1 KiB
C
311 lines
8.1 KiB
C
/**
|
|
* Copyright (c) 2020 Paul-Louis Ageneau
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
*/
|
|
|
|
#include "addr.h"
|
|
#include "log.h"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
socklen_t addr_get_len(const struct sockaddr *sa) {
|
|
switch (sa->sa_family) {
|
|
case AF_INET:
|
|
return sizeof(struct sockaddr_in);
|
|
case AF_INET6:
|
|
return sizeof(struct sockaddr_in6);
|
|
default:
|
|
JLOG_WARN("Unknown address family %hu", sa->sa_family);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
uint16_t addr_get_port(const struct sockaddr *sa) {
|
|
switch (sa->sa_family) {
|
|
case AF_INET:
|
|
return ntohs(((struct sockaddr_in *)sa)->sin_port);
|
|
case AF_INET6:
|
|
return ntohs(((struct sockaddr_in6 *)sa)->sin6_port);
|
|
default:
|
|
JLOG_WARN("Unknown address family %hu", sa->sa_family);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
int addr_set_port(struct sockaddr *sa, uint16_t port) {
|
|
switch (sa->sa_family) {
|
|
case AF_INET:
|
|
((struct sockaddr_in *)sa)->sin_port = htons(port);
|
|
return 0;
|
|
case AF_INET6:
|
|
((struct sockaddr_in6 *)sa)->sin6_port = htons(port);
|
|
return 0;
|
|
default:
|
|
JLOG_WARN("Unknown address family %hu", sa->sa_family);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
bool addr_is_any(const struct sockaddr *sa) {
|
|
switch (sa->sa_family) {
|
|
case AF_INET: {
|
|
const struct sockaddr_in *sin = (const struct sockaddr_in *)sa;
|
|
const uint8_t *b = (const uint8_t *)&sin->sin_addr;
|
|
for (int i = 0; i < 4; ++i)
|
|
if (b[i] != 0)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
case AF_INET6: {
|
|
const struct sockaddr_in6 *sin6 = (const struct sockaddr_in6 *)sa;
|
|
if (IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr)) {
|
|
const uint8_t *b = (const uint8_t *)&sin6->sin6_addr + 12;
|
|
for (int i = 0; i < 4; ++i)
|
|
if (b[i] != 0)
|
|
return false;
|
|
} else {
|
|
const uint8_t *b = (const uint8_t *)&sin6->sin6_addr;
|
|
for (int i = 0; i < 16; ++i)
|
|
if (b[i] != 0)
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool addr_is_local(const struct sockaddr *sa) {
|
|
switch (sa->sa_family) {
|
|
case AF_INET: {
|
|
const struct sockaddr_in *sin = (const struct sockaddr_in *)sa;
|
|
const uint8_t *b = (const uint8_t *)&sin->sin_addr;
|
|
if (b[0] == 127) // loopback
|
|
return true;
|
|
if (b[0] == 169 && b[1] == 254) // link-local
|
|
return true;
|
|
return false;
|
|
}
|
|
case AF_INET6: {
|
|
const struct sockaddr_in6 *sin6 = (const struct sockaddr_in6 *)sa;
|
|
if (IN6_IS_ADDR_LOOPBACK(&sin6->sin6_addr)) {
|
|
return true;
|
|
}
|
|
if (IN6_IS_ADDR_LINKLOCAL(&sin6->sin6_addr)) {
|
|
return true;
|
|
}
|
|
if (IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr)) {
|
|
const uint8_t *b = (const uint8_t *)&sin6->sin6_addr + 12;
|
|
if (b[0] == 127) // loopback
|
|
return true;
|
|
if (b[0] == 169 && b[1] == 254) // link-local
|
|
return true;
|
|
return false;
|
|
}
|
|
return false;
|
|
}
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool addr_unmap_inet6_v4mapped(struct sockaddr *sa, socklen_t *len) {
|
|
if (sa->sa_family != AF_INET6)
|
|
return false;
|
|
|
|
const struct sockaddr_in6 *sin6 = (const struct sockaddr_in6 *)sa;
|
|
if (!IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr))
|
|
return false;
|
|
|
|
struct sockaddr_in6 copy = *sin6;
|
|
sin6 = ©
|
|
|
|
struct sockaddr_in *sin = (struct sockaddr_in *)sa;
|
|
memset(sin, 0, sizeof(*sin));
|
|
sin->sin_family = AF_INET;
|
|
sin->sin_port = sin6->sin6_port;
|
|
memcpy(&sin->sin_addr, ((const uint8_t *)&sin6->sin6_addr) + 12, 4);
|
|
*len = sizeof(*sin);
|
|
return true;
|
|
}
|
|
|
|
bool addr_map_inet6_v4mapped(struct sockaddr_storage *ss, socklen_t *len) {
|
|
if (ss->ss_family != AF_INET)
|
|
return false;
|
|
|
|
const struct sockaddr_in *sin = (const struct sockaddr_in *)ss;
|
|
struct sockaddr_in copy = *sin;
|
|
sin = ©
|
|
|
|
struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)ss;
|
|
memset(sin6, 0, sizeof(*sin6));
|
|
sin6->sin6_family = AF_INET6;
|
|
sin6->sin6_port = sin->sin_port;
|
|
uint8_t *b = (uint8_t *)&sin6->sin6_addr;
|
|
memset(b, 0, 10);
|
|
memset(b + 10, 0xFF, 2);
|
|
memcpy(b + 12, (const uint8_t *)&sin->sin_addr, 4);
|
|
*len = sizeof(*sin6);
|
|
return true;
|
|
}
|
|
|
|
bool addr_is_equal(const struct sockaddr *a, const struct sockaddr *b, bool compare_ports) {
|
|
if (a->sa_family != b->sa_family)
|
|
return false;
|
|
|
|
switch (a->sa_family) {
|
|
case AF_INET: {
|
|
const struct sockaddr_in *ain = (const struct sockaddr_in *)a;
|
|
const struct sockaddr_in *bin = (const struct sockaddr_in *)b;
|
|
if (memcmp(&ain->sin_addr, &bin->sin_addr, 4) != 0)
|
|
return false;
|
|
if (compare_ports && ain->sin_port != bin->sin_port)
|
|
return false;
|
|
break;
|
|
}
|
|
case AF_INET6: {
|
|
const struct sockaddr_in6 *ain6 = (const struct sockaddr_in6 *)a;
|
|
const struct sockaddr_in6 *bin6 = (const struct sockaddr_in6 *)b;
|
|
if (memcmp(&ain6->sin6_addr, &bin6->sin6_addr, 16) != 0)
|
|
return false;
|
|
if (compare_ports && ain6->sin6_port != bin6->sin6_port)
|
|
return false;
|
|
break;
|
|
}
|
|
default:
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
int addr_to_string(const struct sockaddr *sa, char *buffer, size_t size) {
|
|
socklen_t salen = addr_get_len(sa);
|
|
if (salen == 0)
|
|
goto error;
|
|
|
|
char host[ADDR_MAX_NUMERICHOST_LEN];
|
|
char service[ADDR_MAX_NUMERICSERV_LEN];
|
|
if (getnameinfo(sa, salen, host, ADDR_MAX_NUMERICHOST_LEN, service, ADDR_MAX_NUMERICSERV_LEN,
|
|
NI_NUMERICHOST | NI_NUMERICSERV | NI_DGRAM)) {
|
|
JLOG_ERROR("getnameinfo failed, errno=%d", sockerrno);
|
|
goto error;
|
|
}
|
|
|
|
int len = snprintf(buffer, size, "%s:%s", host, service);
|
|
if (len < 0 || (size_t)len >= size)
|
|
goto error;
|
|
|
|
return len;
|
|
|
|
error:
|
|
// Make sure we still write a valid null-terminated string
|
|
snprintf(buffer, size, "?");
|
|
return -1;
|
|
}
|
|
|
|
// djb2 hash function
|
|
#define DJB2_INIT 5381
|
|
static void djb2(unsigned long *hash, int i) {
|
|
*hash = ((*hash << 5) + *hash) + i; // hash * 33 + i
|
|
}
|
|
|
|
unsigned long addr_hash(const struct sockaddr *sa, bool with_port) {
|
|
unsigned long hash = DJB2_INIT;
|
|
|
|
djb2(&hash, sa->sa_family);
|
|
switch (sa->sa_family) {
|
|
case AF_INET: {
|
|
const struct sockaddr_in *sin = (const struct sockaddr_in *)sa;
|
|
const uint8_t *b = (const uint8_t *)&sin->sin_addr;
|
|
for (int i = 0; i < 4; ++i)
|
|
djb2(&hash, b[i]);
|
|
if (with_port) {
|
|
djb2(&hash, sin->sin_port >> 8);
|
|
djb2(&hash, sin->sin_port & 0xFF);
|
|
}
|
|
break;
|
|
}
|
|
case AF_INET6: {
|
|
const struct sockaddr_in6 *sin6 = (const struct sockaddr_in6 *)sa;
|
|
const uint8_t *b = (const uint8_t *)&sin6->sin6_addr;
|
|
for (int i = 0; i < 16; ++i)
|
|
djb2(&hash, b[i]);
|
|
if (with_port) {
|
|
djb2(&hash, sin6->sin6_port >> 8);
|
|
djb2(&hash, sin6->sin6_port & 0xFF);
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return hash;
|
|
}
|
|
|
|
int addr_resolve(const char *hostname, const char *service, addr_record_t *records, size_t count) {
|
|
addr_record_t *end = records + count;
|
|
|
|
struct addrinfo hints;
|
|
memset(&hints, 0, sizeof(hints));
|
|
hints.ai_family = AF_UNSPEC;
|
|
hints.ai_socktype = SOCK_DGRAM;
|
|
hints.ai_protocol = IPPROTO_UDP;
|
|
hints.ai_flags = AI_ADDRCONFIG;
|
|
struct addrinfo *ai_list = NULL;
|
|
if (getaddrinfo(hostname, service, &hints, &ai_list)) {
|
|
JLOG_WARN("Address resolution failed for %s:%s", hostname, service);
|
|
return -1;
|
|
}
|
|
|
|
int ret = 0;
|
|
for (struct addrinfo *ai = ai_list; ai; ai = ai->ai_next) {
|
|
if (ai->ai_family == AF_INET || ai->ai_family == AF_INET6) {
|
|
++ret;
|
|
if (records != end) {
|
|
memcpy(&records->addr, ai->ai_addr, ai->ai_addrlen);
|
|
records->len = (socklen_t)ai->ai_addrlen;
|
|
++records;
|
|
}
|
|
}
|
|
}
|
|
|
|
freeaddrinfo(ai_list);
|
|
return ret;
|
|
}
|
|
|
|
bool addr_is_numeric_hostname(const char *hostname) {
|
|
struct addrinfo hints;
|
|
memset(&hints, 0, sizeof(hints));
|
|
hints.ai_family = AF_UNSPEC;
|
|
hints.ai_socktype = SOCK_DGRAM;
|
|
hints.ai_protocol = IPPROTO_UDP;
|
|
hints.ai_flags = AI_NUMERICHOST|AI_NUMERICSERV;
|
|
struct addrinfo *ai_list = NULL;
|
|
if (getaddrinfo(hostname, "9", &hints, &ai_list))
|
|
return false;
|
|
|
|
freeaddrinfo(ai_list);
|
|
return true;
|
|
}
|
|
|
|
bool addr_record_is_equal(const addr_record_t *a, const addr_record_t *b, bool compare_ports) {
|
|
return addr_is_equal((const struct sockaddr *)&a->addr, (const struct sockaddr *)&b->addr,
|
|
compare_ports);
|
|
}
|
|
|
|
int addr_record_to_string(const addr_record_t *record, char *buffer, size_t size) {
|
|
return addr_to_string((const struct sockaddr *)&record->addr, buffer, size);
|
|
}
|
|
|
|
unsigned long addr_record_hash(const addr_record_t *record, bool with_port) {
|
|
return addr_hash((const struct sockaddr *)&record->addr, with_port);
|
|
}
|