X-Git-Url: http://www.chiark.greenend.org.uk/ucgi/~mdw/git/disorder/blobdiff_plain/9e42afcd09973d589b58be8f503f05b9f292c3b4..6735af5fd1a4ae3978c0c31ac6026f15742c0a90:/lib/addr.c diff --git a/lib/addr.c b/lib/addr.c index 1b62251..802204a 100644 --- a/lib/addr.c +++ b/lib/addr.c @@ -20,6 +20,8 @@ #include "common.h" +#include + #include #if HAVE_SYS_SOCKET_H # include @@ -339,31 +341,70 @@ void netaddress_format(const struct netaddress *na, * @param passive True if passive (bindable) address is desired * @param protocol Protocol number desired (e.g. @c IPPROTO_TCP) * @return List of suitable addresses or NULL + * + * Free the address using netaddress_freeaddrinfo() because it might not + * have come from getaddrinfo() directly. */ struct addrinfo *netaddress_resolve(const struct netaddress *na, int passive, int protocol) { struct addrinfo *res, hints[1]; + struct sockaddr_un *sun; char service[64]; int rc; char errbuf[1024]; - memset(hints, 0, sizeof hints); - hints->ai_family = na->af; - hints->ai_protocol = protocol; - hints->ai_flags = passive ? AI_PASSIVE : 0; - byte_snprintf(service, sizeof service, "%d", na->port); - rc = getaddrinfo(na->address, service, hints, &res); - if(rc) { - disorder_error(0, "getaddrinfo %s %d: %s", - na->address ? na->address : "*", - na->port, - format_error(ec_getaddrinfo, rc, errbuf, sizeof errbuf)); - return NULL; +#if HAVE_SYS_UN_H + if (na->af == AF_UNIX) { + /* `getaddrinfo' won't work, so we make our own one */ + res = xmalloc(sizeof(*res)); + res->ai_flags = 0; + res->ai_family = AF_UNIX; + res->ai_socktype = (protocol == IPPROTO_UDP ? SOCK_DGRAM : SOCK_STREAM); + res->ai_protocol = 0; + res->ai_addrlen = offsetof(struct sockaddr_un, sun_path) + + strlen(na->address) + 1; + sun = xmalloc(res->ai_addrlen); + sun->sun_family = AF_UNIX; + strcpy(sun->sun_path, na->address); + res->ai_addr = (struct sockaddr *)sun; + res->ai_canonname = sun->sun_path; + res->ai_next = 0; + } else +#endif + { + /* get the system to do the heavy lifting */ + memset(hints, 0, sizeof hints); + hints->ai_family = na->af; + hints->ai_protocol = protocol; + hints->ai_flags = passive ? AI_PASSIVE : 0; + byte_snprintf(service, sizeof service, "%d", na->port); + rc = getaddrinfo(na->address, service, hints, &res); + if(rc) { + disorder_error(0, "getaddrinfo %s %d: %s", + na->address ? na->address : "*", + na->port, + format_error(ec_getaddrinfo, rc, errbuf, sizeof errbuf)); + return NULL; + } + assert(res->ai_family != AF_UNIX); } return res; } +/** @brief Free an address-info list from netaddress_resovle() + * @param res Address-info list + */ +void netaddress_freeaddrinfo(struct addrinfo *res) { +#if HAVE_SYS_UN_H + if(res->ai_family == AF_UNIX) { + xfree(res->ai_addr); + xfree(res); + } else +#endif + freeaddrinfo(res); +} + /* Local Variables: c-basic-offset:2