pipewire/src/modules/network-utils.h

130 lines
3 KiB
C
Raw Normal View History

#ifndef NETWORK_UTILS_H
#define NETWORK_UTILS_H
#include <stdio.h>
#include <netdb.h>
#include <string.h>
#include <arpa/inet.h>
#include <net/if.h>
#include <errno.h>
static inline int pw_net_parse_address(const char *address, uint16_t port,
struct sockaddr_storage *addr, socklen_t *len)
{
struct addrinfo hints;
struct addrinfo *result, *rp;
int res;
char port_str[6];
snprintf(port_str, sizeof(port_str), "%u", port);
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_DGRAM;
hints.ai_flags = AI_NUMERICHOST | AI_NUMERICSERV;
res = getaddrinfo(address, port_str, &hints, &result);
if (res != 0)
return -EINVAL;
for (rp = result; rp != NULL; rp = rp->ai_next) {
memcpy(addr, rp->ai_addr, rp->ai_addrlen);
*len = rp->ai_addrlen;
break;
}
freeaddrinfo(result);
return 0;
}
static inline uint16_t pw_net_parse_port(const char *str, uint16_t def)
{
uint32_t val;
if (spa_atou32(str, &val, 0) && val <= 65535u)
return val;
return def;
}
static inline int pw_net_parse_address_port(const char *address,
const char *default_address, uint16_t default_port,
struct sockaddr_storage *addr, socklen_t *len)
{
uint16_t port;
char *br = NULL, *col, *n;
n = strdupa(address);
col = strrchr(n, ':');
if (n[0] == '[') {
br = strchr(n, ']');
if (br == NULL)
return -EINVAL;
n++;
*br = 0;
}
if (br && col && col < br)
col = NULL;
if (col) {
*col = '\0';
port = pw_net_parse_port(col+1, default_port);
} else {
port = pw_net_parse_port(n, default_port);
n = strdupa(default_address ? default_address : "0.0.0.0");
}
return pw_net_parse_address(n, port, addr, len);
}
static inline int pw_net_get_ip(const struct sockaddr_storage *sa, char *ip, size_t len, bool *ip4, uint16_t *port)
{
if (sa->ss_family == AF_INET) {
struct sockaddr_in *in = (struct sockaddr_in*)sa;
inet_ntop(sa->ss_family, &in->sin_addr, ip, len);
if (port)
*port = ntohs(in->sin_port);
} else if (sa->ss_family == AF_INET6) {
struct sockaddr_in6 *in = (struct sockaddr_in6*)sa;
inet_ntop(sa->ss_family, &in->sin6_addr, ip, len);
if (port)
*port = ntohs(in->sin6_port);
if (in->sin6_scope_id == 0 || len <= 1)
goto finish;
size_t curlen = strlen(ip);
if (len-(curlen+1) >= IFNAMSIZ) {
ip += curlen+1;
ip[-1] = '%';
if (if_indextoname(in->sin6_scope_id, ip) == NULL)
ip[-1] = 0;
}
} else
return -EINVAL;
finish:
if (ip4)
*ip4 = sa->ss_family == AF_INET;
return 0;
}
static inline char *pw_net_get_ip_fmt(const struct sockaddr_storage *sa, char *ip, size_t len)
{
if (pw_net_get_ip(sa, ip, len, NULL, NULL) != 0)
snprintf(ip, len, "invalid ip");
return ip;
}
static inline bool pw_net_addr_is_any(struct sockaddr_storage *addr)
{
if (addr->ss_family == AF_INET) {
struct sockaddr_in *sa = (struct sockaddr_in*)addr;
return sa->sin_addr.s_addr == INADDR_ANY;
} else if (addr->ss_family == AF_INET6) {
struct sockaddr_in6 *sa = (struct sockaddr_in6*)addr;
return memcmp(&sa->sin6_addr, &in6addr_any, sizeof(sa->sin6_addr));
}
return false;
}
#endif /* NETWORK_UTILS_H */