async_resolv.c revision a0c764a36f2f432e6063de84be6f6af7e96ec159
/*
SSSD
Async resolver
Authors:
Martin Nagy <mnagy@redhat.com>
Jakub Hrozek <jhrozek@redhat.com>
Copyright (C) Red Hat, Inc 2009
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <ares.h>
#include <talloc.h>
#include <tevent.h>
#include <errno.h>
#include <netdb.h>
#include <stddef.h>
#include <string.h>
#include <unistd.h>
#include "config.h"
#include "resolv/async_resolv.h"
#include "util/dlinklist.h"
/*
* Macro DNS__32BIT reads a network long (32 bit) given in network
* byte order, and returns its value as an unsigned int. Copied
* from c-ares source code.
*/
#define DNS__32BIT(p) ((unsigned int) \
(((unsigned int)((unsigned char)(p)[0]) << 24U) | \
((unsigned int)((unsigned char)(p)[1]) << 16U) | \
((unsigned int)((unsigned char)(p)[2]) << 8U) | \
((unsigned int)((unsigned char)(p)[3]))))
#define RESOLV_TIMEOUTMS 2000
struct fd_watch {
int fd;
struct resolv_ctx *ctx;
};
struct resolv_ctx {
struct tevent_context *ev_ctx;
/* List of file descriptors that are watched by tevent. */
/* Time in milliseconds before canceling a DNS request */
int timeout;
/* The timeout watcher periodically calls ares_process_fd() to check
* if our pending requests didn't timeout. */
int pending_requests;
struct tevent_timer *timeout_watcher;
};
struct request_watch {
struct tevent_req *req;
struct resolv_request *rr;
};
struct resolv_request {
struct resolv_ctx *ctx;
struct request_watch *rwatch;
struct tevent_timer *request_timeout;
};
static int
return_code(int ares_code)
{
switch (ares_code) {
case ARES_SUCCESS:
return EOK;
case ARES_ENOMEM:
return ENOMEM;
case ARES_EFILE:
default:
return EIO;
}
}
const char *
resolv_strerror(int ares_code)
{
return ares_strerror(ares_code);
}
static int
fd_watch_destructor(struct fd_watch *f)
{
f->fd = -1;
return 0;
}
static void
{
"Invalid ares channel - this is likely a bug\n");
return;
}
if (flags & TEVENT_FD_READ) {
}
if (flags & TEVENT_FD_WRITE) {
}
}
static void
static void
{
if (ctx->timeout_watcher) {
return;
}
}
/* Enforce a minimum of 1 second. */
} else {
}
ctx);
}
}
static void
{
/* NULLify the timeout_watcher so we don't
* free it in the _done() function if it
* gets called. Now that we're already in
* the handler, tevent will take care of
* freeing it when it returns.
*/
if (ctx->pending_requests > 0) {
}
}
static void
struct tevent_timer *te,
{
struct resolv_request *rreq;
return;
}
}
static int
{
return 0;
}
static struct resolv_request *
struct tevent_req *req)
{
struct resolv_request *rreq;
/* Intentionally allocating on ctx, because the request might go away
* before c-ares returns */
if (!rreq) {
return NULL;
}
rreq);
return NULL;
}
/* The watch will go away when the request finishes */
return NULL;
}
return rreq;
}
static struct resolv_request *
struct tevent_req *req)
{
struct resolv_request *rreq;
ctx->pending_requests++;
return rreq;
}
static void
{
/* Unlink the watch if the request is still active */
}
if (ctx->pending_requests <= 0) {
return;
}
ctx->pending_requests--;
if (ctx->pending_requests == 0) {
}
}
/*
* When ares is ready to read or write to a file descriptor, it will
* call this callback. If both read and write are 0, it means that ares
* will soon close the socket. We are mainly using this function to register
* new file descriptors with tevent.
*/
static void
{
int flags;
/* The socket is about to get closed. */
fd_event_close(ctx, s);
return;
}
/* Are we already watching this file descriptor? */
while (watch) {
return;
}
}
}
static void
{
/* The file descriptor is new, register it with tevent. */
"Out of memory allocating fd_watch structure\n");
return;
}
return;
}
}
static void
{
/* Remove the socket from list */
while (watch) {
return;
}
}
}
static int
{
return -1;
}
/* Set ctx->channel to NULL first, so that callbacks that get
* ARES_EDESTRUCTION won't retry. */
return 0;
}
static int
{
int ret;
struct ares_options options;
/* FIXME: the options would contain
* the nameservers to contact, the domains
* to search... => get from confdb
*/
/* Only affects ares_gethostbyname */
if (ret != ARES_SUCCESS) {
return return_code(ret);
}
if (old_channel != NULL) {
}
return EOK;
}
int
{
int ret;
struct resolv_ctx *ctx;
if (timeout < 1) {
"The timeout is too short, DNS operations are going to fail. "
"This is a bug outside unit tests\n");
}
return ENOMEM;
goto done;
}
return EOK;
done:
return ret;
}
void
{
}
static errno_t
struct ares_addrttl *attl)
{
return EOK;
}
static errno_t
struct ares_addr6ttl *a6ttl)
{
return EOK;
}
static struct resolv_hostent *
{
struct resolv_hostent *ret;
int len;
int i;
return NULL;
}
goto fail;
}
}
goto fail;
}
for (i = 0; i < len; i++) {
goto fail;
}
}
}
return ret;
fail:
return NULL;
}
struct resolv_hostent *
{
struct resolv_hostent *ret;
int len;
int i;
return NULL;
}
goto fail;
}
for (i = 0; i < len; i++) {
struct resolv_addr);
goto fail;
}
src->h_addr_list[i],
goto fail;
}
}
}
return ret;
fail:
return NULL;
}
struct resolv_hostent *
int family, void *ares_ttl_data,
int num_ares_ttl_data)
{
struct resolv_hostent *ret;
int i;
return NULL;
}
if (num_ares_ttl_data > 0) {
num_ares_ttl_data + 1);
goto fail;
}
for (i = 0; i < num_ares_ttl_data; i++) {
struct resolv_addr);
goto fail;
}
switch (family) {
case AF_INET:
&((struct ares_addrttl *) ares_ttl_data)[i]);
break;
case AF_INET6:
&((struct ares_addr6ttl *) ares_ttl_data)[i]);
break;
default:
"Unknown address family %d\n", family);
goto fail;
}
goto fail;
}
}
}
return ret;
fail:
return NULL;
}
/* =================== Resolve host name in files =========================*/
struct gethostbyname_files_state {
struct resolv_ctx *resolv_ctx;
/* Part of the query. */
const char *name;
int family;
/* query result */
struct resolv_hostent *rhostent;
/* returned by ares. */
int status;
};
/* Fake up an async interface even though files would
* always be blocking */
static struct tevent_req *
struct tevent_context *ev,
struct resolv_ctx *ctx,
const char *name,
int family)
{
struct tevent_req *req;
struct gethostbyname_files_state *state;
struct gethostbyname_files_state);
return NULL;
}
"Trying to resolve %s record of '%s' in files\n",
&hostent);
goto done;
}
/* Just say we didn't find anything and let the caller decide
* about retrying */
goto done;
} else {
goto done;
}
done:
return req;
}
static errno_t
{
struct gethostbyname_files_state);
/* Fill in even in case of error as status contains the
* c-ares return code */
if (status) {
}
if (rhostent) {
}
return EOK;
}
/* ==================== Resolve host name in DNS =========================*/
struct gethostbyname_dns_state {
struct resolv_ctx *resolv_ctx;
struct tevent_context *ev;
/* Part of the query. */
const char *name;
int family;
/* query result */
struct resolv_hostent *rhostent;
/* These are returned by ares. */
int status;
int timeouts;
int retrying;
};
static void
static void
struct gethostbyname_dns_state *state);
static void
static int
static struct tevent_req *
int family)
{
struct gethostbyname_dns_state *state;
"Invalid ares channel - this is likely a bug\n");
return NULL;
}
return NULL;
}
/* We need to have a wrapper around ares async calls, because
* they can in some cases call it's callback immediately.
* This would not let our caller to set a callback for req. */
"Failed to add critical timer to run next operation!\n");
return NULL;
}
return req;
}
static void
{
struct tevent_req);
struct gethostbyname_dns_state);
if (!tevent_wakeup_recv(subreq)) {
return;
}
"Invalid ares channel - this is likely a bug\n");
return;
}
}
static void
struct gethostbyname_dns_state *state)
{
struct resolv_request *rreq;
if (!rreq) {
return;
}
}
static void
{
struct gethostbyname_dns_state *state;
struct tevent_req *req;
/* The tevent request was cancelled while the ares call was still in
* progress so nobody cares about the result now. Quit. */
return;
}
/* If resolv.conf changed during processing of a request we might
* destroy the old channel before the request has a chance to finish.
* We must resend the request in this case */
return;
}
/* Just say we didn't find anything and let the caller decide
* about retrying */
return;
}
if (status != ARES_SUCCESS) {
/* Any other error indicates a server error,
* so don't bother trying again
*/
return;
}
return;
}
}
static int
{
int naddrttls;
case AF_INET:
if (!addr) {
goto fail;
}
(struct ares_addrttl *) addr,
&naddrttls);
break;
case AF_INET6:
if (!addr) {
goto fail;
}
(struct ares_addr6ttl *) addr,
&naddrttls);
break;
default:
ret = EAFNOSUPPORT;
goto fail;
}
goto fail;
}
/* The address list is NULL. This is probably a bug in
* c-ares, but we need to handle it gracefully.
*/
return ENOENT;
}
}
return return_code(status);
fail:
return ret;
}
static int
struct resolv_hostent **rhostent)
{
struct gethostbyname_dns_state);
/* Fill in even in case of error as status contains the
* c-ares return code */
if (status) {
}
if (timeouts) {
}
if (rhostent) {
}
return EOK;
}
/*******************************************************************
* Get host by name. *
*******************************************************************/
struct gethostbyname_state {
struct resolv_ctx *resolv_ctx;
struct tevent_context *ev;
/* Part of the query. */
const char *name;
int family;
/* In which order to use IPv4, or v6 */
enum restrict_family family_order;
/* Known hosts databases and index to the current one */
enum host_database *db;
int dbi;
/* These are returned by ares. The hostent struct will be freed
* when the user callback returns. */
struct resolv_hostent *rhostent;
int status;
int timeouts;
int retrying;
};
static errno_t
struct resolv_hostent **_rhostent);
static inline int
static bool
resolv_is_address(const char *name);
static errno_t
struct tevent_req *
enum restrict_family family_order,
enum host_database *db)
{
struct tevent_req *req;
struct gethostbyname_state *state;
"Invalid ares channel - this is likely a bug\n");
return NULL;
}
return NULL;
}
goto fail;
}
/* Do not attempt to resolve IP addresses */
"Canot create a fake hostent structure\n");
goto fail;
}
return req;
}
goto fail;
}
return req;
fail:
return NULL;
}
static bool
resolv_is_address(const char *name)
{
int ret;
if (ret != 0) {
if (ret == -2) {
"[%s] does not look like an IP address\n", name);
} else {
}
} else { /* ret == 0 */
}
return ret == 0;
}
static errno_t
struct resolv_hostent **_rhostent)
{
struct resolv_hostent *rhostent;
int family;
if (!rhostent) {
goto done;
}
goto done;
}
struct resolv_addr);
goto done;
}
sizeof(struct in6_addr));
goto done;
}
if (ret != 1) {
if (ret != 1) {
"Could not parse address as neither v4 nor v6\n");
goto done;
}
}
done:
return ret;
}
static inline int
{
switch(family_order) {
case IPV4_ONLY:
case IPV4_FIRST:
return AF_INET;
case IPV6_ONLY:
case IPV6_FIRST:
return AF_INET6;
}
"Unknown address family order %d\n", family_order);
return -1;
}
static int
{
return EOK;
return EOK;
} else {
/* No more address families for this DB, check if
* there is another DB to try */
return EOK;
}
}
return ENOENT;
}
static void
static errno_t
{
struct gethostbyname_state);
struct tevent_req *subreq;
case DB_FILES:
break;
case DB_DNS:
break;
default:
return EINVAL;
}
return ENOMEM;
}
return EOK;
}
static void
{
struct tevent_req);
struct gethostbyname_state);
case DB_FILES:
/* files is synchronous, there can be no timeouts */
break;
case DB_DNS:
break;
default:
return;
}
}
return;
}
return;
/* In case we killed the request before c-ares answered */
}
return;
}
}
int
struct resolv_hostent **rhostent)
{
/* Fill in even in case of error as status contains the
* c-ares return code */
if (status) {
}
if (timeouts) {
}
if (rhostent) {
}
return EOK;
}
char *
struct resolv_hostent *hostent,
unsigned int addrindex)
{
char *address;
return NULL;
}
errno = 0;
return NULL;
}
return address;
}
char *
{
char *straddr;
int i;
char hexbyte[3];
if (!straddr) {
return NULL;
}
for (i = 15; i >= 0; i--) {
}
"%u.%u.%u.%u.in-addr.arpa.",
(address[3]),
(address[2]),
(address[1]),
(address[0]));
} else {
return NULL;
}
return straddr;
}
struct sockaddr_storage *
struct resolv_hostent *hostent,
{
struct sockaddr_storage *sockaddr;
return NULL;
}
case AF_INET:
sizeof(struct in_addr));
break;
case AF_INET6:
sizeof(struct in6_addr));
break;
default:
return NULL;
}
return sockaddr;
}
/*
* A simple helper function that will take an array of struct ares_srv_reply that
* was allocated by malloc() in c-ares and copies it using talloc. The old one
* is freed and the talloc one is put into 'reply_list' instead.
*/
static int
{
/* Nothing to do, but not an error */
if (!old_list) {
return EOK;
}
/* Copy the linked list */
while (old_list) {
/* Special case for the first node */
if (!new_list) {
return ENOMEM;
}
} else {
return ENOMEM;
}
}
return ENOMEM;
}
}
/* Free the old one (uses malloc). */
/* And now put our own new_list in place. */
*reply_list = new_list;
return EOK;
}
/*******************************************************************
* Get SRV record *
*******************************************************************/
struct getsrv_state {
struct tevent_context *ev;
struct resolv_ctx *resolv_ctx;
/* the SRV query - for example _ldap._tcp.example.com */
const char *query;
/* parsed data returned by ares */
struct ares_srv_reply *reply_list;
int status;
int timeouts;
int retrying;
};
static void
static void
struct getsrv_state *state);
struct tevent_req *
{
struct getsrv_state *state;
"Trying to resolve SRV record of '%s'\n", query);
"Invalid ares channel - this is likely a bug\n");
return NULL;
}
return NULL;
"Failed to add critical timer to run next operation!\n");
return NULL;
}
return req;
}
/*
* Implemented based on http://tools.ietf.org/html/rfc2181#section-5
*
* Especially:
* 5.2. TTLs of RRs in an RRSet
* Consequently the use of differing TTLs in an RRSet is hereby
* deprecated, the TTLs of all RRs in an RRSet must be the same.
* ...
* Should an authoritative source send such a malformed RRSet, the
* client should treat the RRs for all purposes as if all TTLs in the
* RRSet had been set to the value of the lowest TTL in the RRSet.
*
* On success, returns true and sets the TTL in the _ttl parameter. On
* failure, returns false and _ttl is undefined.
*/
static bool
{
const unsigned char *aptr;
int ret;
long len;
unsigned int rr_len;
unsigned int ancount;
unsigned int i;
/* Read the number of RRs and then skip past the header */
if (alen < NS_HFIXEDSZ) {
return false;
}
if (ancount == 0) {
return false;
}
/* We only care about len from the question data,
* so that we can move past hostname */
if (ret != ARES_SUCCESS) {
return false;
}
/* Skip past the question */
return false;
}
/* Examine each RR in turn and read the lowest TTL */
for (i = 0; i < ancount; i++) {
/* Decode the RR up to the data field. */
if (ret != ARES_SUCCESS) {
return false;
}
return false;
}
return false;
}
if (ttl > 0) {
} else {
}
}
return true;
}
static void
{
struct tevent_req *req;
struct getsrv_state *state;
int ret;
bool ok;
struct ares_srv_reply *reply_list;
/* The tevent request was cancelled while the ares call was still in
* progress so nobody cares about the result now. Quit. */
return;
}
return;
}
if (status != ARES_SUCCESS) {
goto fail;
}
if (ret != ARES_SUCCESS) {
goto fail;
}
goto fail;
}
if (ok == false) {
}
return;
fail:
}
int
{
if (status)
if (timeouts)
if (reply_list)
if (ttl) {
}
return EOK;
}
static void
{
struct tevent_req);
struct getsrv_state);
if (!tevent_wakeup_recv(subreq)) {
return;
}
"Invalid ares channel - this is likely a bug\n");
return;
}
}
static void
struct getsrv_state *state)
{
struct resolv_request *rreq;
if (!rreq) {
return;
}
}
/* TXT parsing is not used anywhere in the code yet, so we disable it
* for now
*/
#ifdef BUILD_TXT
/*
* A simple helper function that will take an array of struct txt_reply that
* was allocated by malloc() in c-ares and copies it using talloc. The old one
* is freed and the talloc one is put into 'reply_list' instead.
*/
static int
{
/* Nothing to do, but not an error */
if (!old_list) {
return EOK;
}
/* Copy the linked list */
while (old_list) {
/* Special case for the first node */
if (!new_list) {
return ENOMEM;
}
} else {
return ENOMEM;
}
}
return ENOMEM;
}
}
/* And now put our own new_list in place. */
*reply_list = new_list;
return EOK;
}
/*******************************************************************
* Get TXT record *
*******************************************************************/
struct gettxt_state {
struct tevent_context *ev;
struct resolv_ctx *resolv_ctx;
/* the TXT query */
const char *query;
/* parsed data returned by ares */
struct ares_txt_reply *reply_list;
int status;
int timeouts;
int retrying;
};
static void
static void
struct gettxt_state *state);
struct tevent_req *
{
struct gettxt_state *state;
"Trying to resolve TXT record of '%s'\n", query);
"Invalid ares channel - this is likely a bug\n");
return NULL;
}
return NULL;
"Failed to add critical timer to run next operation!\n");
return NULL;
}
return req;
}
static void
{
struct tevent_req *req;
struct gettxt_state *state;
int ret;
struct ares_txt_reply *reply_list;
/* The tevent request was cancelled while the ares call was still in
* progress so nobody cares about the result now. Quit. */
return;
}
return;
}
if (status != ARES_SUCCESS) {
goto fail;
}
if (status != ARES_SUCCESS) {
goto fail;
}
goto fail;
}
return;
fail:
}
int
{
if (status)
if (timeouts)
if (reply_list)
return EOK;
}
static void
{
struct tevent_req);
struct gettxt_state);
if (!tevent_wakeup_recv(subreq)) {
return;
}
"Invalid ares channel - this is likely a bug\n");
return;
}
}
static void
struct gettxt_state *state)
{
struct resolv_request *rreq;
if (!rreq) {
return;
}
}
#endif
{
if (!list) {
return NULL;
}
prev = single_step;
}
return single_step;
}
struct ares_srv_reply *right)
{
struct ares_srv_reply *l, *r;
if (!left)
return right;
if (!right)
return left;
r = right;
} else {
l = left;
}
while(l && r) {
res = l;
l = l->next;
} else {
res = r;
r = r->next;
}
}
return res_start;
}
/**
* sort linked list of struct ares_srv_reply by priority using merge sort.
*
* Merge sort is ideal for sorting linked lists as there is no problem
* with absence of random access into the list. The complexity is O(n log n)
*
* For reference, see Robert Sedgewick's "Algorithms in C", Addison-Wesley,
* ISBN 0-201-51425
*/
{
struct ares_srv_reply *half;
return list;
return list;
}
static int reply_weight_rearrange(int len,
struct ares_srv_reply **start,
struct ares_srv_reply **end)
{
int i;
int *totals;
int ret;
if (len <= 1) {
return EOK;
}
if (!totals) {
return ENOMEM;
}
/* promote all servers with weight==0 to the top */
r = *(start);
while (r != NULL) {
/* remove from the old list */
/* add to the head of the new list */
tmp = r;
r = r->next;
} else {
prev = r;
r = r->next;
}
}
/* Commpute the sum of the weights of those RRs, and with each RR
* associate the running sum in the selected order.
*/
total = 0;
}
/* choose a uniform random number between 0 and the sum computed
* (inclusive), and select the RR whose running sum value is the
* first in the selected order which is greater than or equal to
* the random number selected.
*/
break;
prev = r;
}
goto done;
}
/* remove r from the old list */
if (prev) {
} else {
}
/* add r to the end of the new list */
if (!new_start) {
new_start = r;
new_end = r;
} else {
new_end = r;
}
}
/* return the rearranged list */
done:
return ret;
}
int
{
int ret;
int len;
/* RFC 2782 says: If there is precisely one SRV RR, and its Target is "."
* (the root domain), abort.
*/
"DNS returned only the root domain, aborting\n");
return EIO;
}
/* sort the list by priority */
while (pri_start) {
/* Find nodes with the same priority */
len = 1;
len++;
}
/* rearrange each priority level according to the weight field */
if (ret) {
"Error rearranging priority level [%d]: %s\n",
return ret;
}
/* Hook the level back into the list */
if (prev_end) {
} else {
}
/* Move on to the next level */
}
return EOK;
}