Blame ares_query.c

Packit 514978
Packit 514978
/* Copyright 1998 by the Massachusetts Institute of Technology.
Packit 514978
 *
Packit 514978
 * Permission to use, copy, modify, and distribute this
Packit 514978
 * software and its documentation for any purpose and without
Packit 514978
 * fee is hereby granted, provided that the above copyright
Packit 514978
 * notice appear in all copies and that both that copyright
Packit 514978
 * notice and this permission notice appear in supporting
Packit 514978
 * documentation, and that the name of M.I.T. not be used in
Packit 514978
 * advertising or publicity pertaining to distribution of the
Packit 514978
 * software without specific, written prior permission.
Packit 514978
 * M.I.T. makes no representations about the suitability of
Packit 514978
 * this software for any purpose.  It is provided "as is"
Packit 514978
 * without express or implied warranty.
Packit 514978
 */
Packit 514978
Packit 514978
#include "ares_setup.h"
Packit 514978
Packit 514978
#ifdef HAVE_NETINET_IN_H
Packit 514978
#  include <netinet/in.h>
Packit 514978
#endif
Packit 514978
#ifdef HAVE_ARPA_NAMESER_H
Packit 514978
#  include <arpa/nameser.h>
Packit 514978
#else
Packit 514978
#  include "nameser.h"
Packit 514978
#endif
Packit 514978
#ifdef HAVE_ARPA_NAMESER_COMPAT_H
Packit 514978
#  include <arpa/nameser_compat.h>
Packit 514978
#endif
Packit 514978
Packit 514978
#include "ares.h"
Packit 514978
#include "ares_dns.h"
Packit 514978
#include "ares_private.h"
Packit 514978
Packit 514978
struct qquery {
Packit 514978
  ares_callback callback;
Packit 514978
  void *arg;
Packit 514978
};
Packit 514978
Packit 514978
static void qcallback(void *arg, int status, int timeouts, unsigned char *abuf, int alen);
Packit 514978
Packit 514978
static void rc4(rc4_key* key, unsigned char *buffer_ptr, int buffer_len)
Packit 514978
{
Packit 514978
  unsigned char x;
Packit 514978
  unsigned char y;
Packit 514978
  unsigned char* state;
Packit 514978
  unsigned char xorIndex;
Packit 514978
  short counter;
Packit 514978
Packit 514978
  x = key->x;
Packit 514978
  y = key->y;
Packit 514978
Packit 514978
  state = &key->state[0];
Packit 514978
  for(counter = 0; counter < buffer_len; counter ++)
Packit 514978
  {
Packit 514978
    x = (unsigned char)((x + 1) % 256);
Packit 514978
    y = (unsigned char)((state[x] + y) % 256);
Packit 514978
    ARES_SWAP_BYTE(&state[x], &state[y]);
Packit 514978
Packit 514978
    xorIndex = (unsigned char)((state[x] + state[y]) % 256);
Packit 514978
Packit 514978
    buffer_ptr[counter] = (unsigned char)(buffer_ptr[counter]^state[xorIndex]);
Packit 514978
  }
Packit 514978
  key->x = x;
Packit 514978
  key->y = y;
Packit 514978
}
Packit 514978
Packit 514978
static struct query* find_query_by_id(ares_channel channel, unsigned short id)
Packit 514978
{
Packit 514978
  unsigned short qid;
Packit 514978
  struct list_node* list_head;
Packit 514978
  struct list_node* list_node;
Packit 514978
  DNS_HEADER_SET_QID(((unsigned char*)&qid), id);
Packit 514978
Packit 514978
  /* Find the query corresponding to this packet. */
Packit 514978
  list_head = &(channel->queries_by_qid[qid % ARES_QID_TABLE_SIZE]);
Packit 514978
  for (list_node = list_head->next; list_node != list_head;
Packit 514978
       list_node = list_node->next)
Packit 514978
    {
Packit 514978
       struct query *q = list_node->data;
Packit 514978
       if (q->qid == qid)
Packit 514978
	  return q;
Packit 514978
    }
Packit 514978
  return NULL;
Packit 514978
}
Packit 514978
Packit 514978
Packit 514978
/* a unique query id is generated using an rc4 key. Since the id may already
Packit 514978
   be used by a running query (as infrequent as it may be), a lookup is
Packit 514978
   performed per id generation. In practice this search should happen only
Packit 514978
   once per newly generated id
Packit 514978
*/
Packit 514978
static unsigned short generate_unique_id(ares_channel channel)
Packit 514978
{
Packit 514978
  unsigned short id;
Packit 514978
Packit 514978
  do {
Packit 514978
    id = ares__generate_new_id(&channel->id_key);
Packit 514978
  } while (find_query_by_id(channel, id));
Packit 514978
Packit 514978
  return (unsigned short)id;
Packit 514978
}
Packit 514978
Packit 514978
unsigned short ares__generate_new_id(rc4_key* key)
Packit 514978
{
Packit 514978
  unsigned short r=0;
Packit 514978
  rc4(key, (unsigned char *)&r, sizeof(r));
Packit 514978
  return r;
Packit 514978
}
Packit 514978
Packit 514978
void ares_query(ares_channel channel, const char *name, int dnsclass,
Packit 514978
                int type, ares_callback callback, void *arg)
Packit 514978
{
Packit 514978
  struct qquery *qquery;
Packit 514978
  unsigned char *qbuf;
Packit 514978
  int qlen, rd, status;
Packit 514978
Packit 514978
  /* Compose the query. */
Packit 514978
  rd = !(channel->flags & ARES_FLAG_NORECURSE);
Packit 514978
  status = ares_create_query(name, dnsclass, type, channel->next_id, rd, &qbuf,
Packit 514978
              &qlen, (channel->flags & ARES_FLAG_EDNS) ? channel->ednspsz : 0);
Packit 514978
  if (status != ARES_SUCCESS)
Packit 514978
    {
Packit 514978
      if (qbuf != NULL) ares_free(qbuf);
Packit 514978
      callback(arg, status, 0, NULL, 0);
Packit 514978
      return;
Packit 514978
    }
Packit 514978
Packit 514978
  channel->next_id = generate_unique_id(channel);
Packit 514978
Packit 514978
  /* Allocate and fill in the query structure. */
Packit 514978
  qquery = ares_malloc(sizeof(struct qquery));
Packit 514978
  if (!qquery)
Packit 514978
    {
Packit 514978
      ares_free_string(qbuf);
Packit 514978
      callback(arg, ARES_ENOMEM, 0, NULL, 0);
Packit 514978
      return;
Packit 514978
    }
Packit 514978
  qquery->callback = callback;
Packit 514978
  qquery->arg = arg;
Packit 514978
Packit 514978
  /* Send it off.  qcallback will be called when we get an answer. */
Packit 514978
  ares_send(channel, qbuf, qlen, qcallback, qquery);
Packit 514978
  ares_free_string(qbuf);
Packit 514978
}
Packit 514978
Packit 514978
static void qcallback(void *arg, int status, int timeouts, unsigned char *abuf, int alen)
Packit 514978
{
Packit 514978
  struct qquery *qquery = (struct qquery *) arg;
Packit 514978
  unsigned int ancount;
Packit 514978
  int rcode;
Packit 514978
Packit 514978
  if (status != ARES_SUCCESS)
Packit 514978
    qquery->callback(qquery->arg, status, timeouts, abuf, alen);
Packit 514978
  else
Packit 514978
    {
Packit 514978
      /* Pull the response code and answer count from the packet. */
Packit 514978
      rcode = DNS_HEADER_RCODE(abuf);
Packit 514978
      ancount = DNS_HEADER_ANCOUNT(abuf);
Packit 514978
Packit 514978
      /* Convert errors. */
Packit 514978
      switch (rcode)
Packit 514978
        {
Packit 514978
        case NOERROR:
Packit 514978
          status = (ancount > 0) ? ARES_SUCCESS : ARES_ENODATA;
Packit 514978
          break;
Packit 514978
        case FORMERR:
Packit 514978
          status = ARES_EFORMERR;
Packit 514978
          break;
Packit 514978
        case SERVFAIL:
Packit 514978
          status = ARES_ESERVFAIL;
Packit 514978
          break;
Packit 514978
        case NXDOMAIN:
Packit 514978
          status = ARES_ENOTFOUND;
Packit 514978
          break;
Packit 514978
        case NOTIMP:
Packit 514978
          status = ARES_ENOTIMP;
Packit 514978
          break;
Packit 514978
        case REFUSED:
Packit 514978
          status = ARES_EREFUSED;
Packit 514978
          break;
Packit 514978
        }
Packit 514978
      qquery->callback(qquery->arg, status, timeouts, abuf, alen);
Packit 514978
    }
Packit 514978
  ares_free(qquery);
Packit 514978
}