Blame ares_options.c

Packit 514978
Packit 514978
/* Copyright 1998 by the Massachusetts Institute of Technology.
Packit 514978
 * Copyright (C) 2008-2013 by Daniel Stenberg
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
Packit 514978
#include "ares_setup.h"
Packit 514978
Packit 514978
#ifdef HAVE_ARPA_INET_H
Packit 514978
#  include <arpa/inet.h>
Packit 514978
#endif
Packit 514978
Packit 514978
#include "ares.h"
Packit 514978
#include "ares_data.h"
Packit 514978
#include "ares_inet_net_pton.h"
Packit 514978
#include "ares_private.h"
Packit 514978
Packit 514978
Packit 514978
int ares_get_servers(ares_channel channel,
Packit 514978
                     struct ares_addr_node **servers)
Packit 514978
{
Packit 514978
  struct ares_addr_node *srvr_head = NULL;
Packit 514978
  struct ares_addr_node *srvr_last = NULL;
Packit 514978
  struct ares_addr_node *srvr_curr;
Packit 514978
  int status = ARES_SUCCESS;
Packit 514978
  int i;
Packit 514978
Packit 514978
  if (!channel)
Packit 514978
    return ARES_ENODATA;
Packit 514978
Packit 514978
  for (i = 0; i < channel->nservers; i++)
Packit 514978
    {
Packit 514978
      /* Allocate storage for this server node appending it to the list */
Packit 514978
      srvr_curr = ares_malloc_data(ARES_DATATYPE_ADDR_NODE);
Packit 514978
      if (!srvr_curr)
Packit 514978
        {
Packit 514978
          status = ARES_ENOMEM;
Packit 514978
          break;
Packit 514978
        }
Packit 514978
      if (srvr_last)
Packit 514978
        {
Packit 514978
          srvr_last->next = srvr_curr;
Packit 514978
        }
Packit 514978
      else
Packit 514978
        {
Packit 514978
          srvr_head = srvr_curr;
Packit 514978
        }
Packit 514978
      srvr_last = srvr_curr;
Packit 514978
Packit 514978
      /* Fill this server node data */
Packit 514978
      srvr_curr->family = channel->servers[i].addr.family;
Packit 514978
      if (srvr_curr->family == AF_INET)
Packit 514978
        memcpy(&srvr_curr->addrV4, &channel->servers[i].addr.addrV4,
Packit 514978
               sizeof(srvr_curr->addrV4));
Packit 514978
      else
Packit 514978
        memcpy(&srvr_curr->addrV6, &channel->servers[i].addr.addrV6,
Packit 514978
               sizeof(srvr_curr->addrV6));
Packit 514978
    }
Packit 514978
Packit 514978
  if (status != ARES_SUCCESS)
Packit 514978
    {
Packit 514978
      if (srvr_head)
Packit 514978
        {
Packit 514978
          ares_free_data(srvr_head);
Packit 514978
          srvr_head = NULL;
Packit 514978
        }
Packit 514978
    }
Packit 514978
Packit 514978
  *servers = srvr_head;
Packit 514978
Packit 514978
  return status;
Packit 514978
}
Packit 514978
Packit 514978
int ares_get_servers_ports(ares_channel channel,
Packit 514978
                           struct ares_addr_port_node **servers)
Packit 514978
{
Packit 514978
  struct ares_addr_port_node *srvr_head = NULL;
Packit 514978
  struct ares_addr_port_node *srvr_last = NULL;
Packit 514978
  struct ares_addr_port_node *srvr_curr;
Packit 514978
  int status = ARES_SUCCESS;
Packit 514978
  int i;
Packit 514978
Packit 514978
  if (!channel)
Packit 514978
    return ARES_ENODATA;
Packit 514978
Packit 514978
  for (i = 0; i < channel->nservers; i++)
Packit 514978
    {
Packit 514978
      /* Allocate storage for this server node appending it to the list */
Packit 514978
      srvr_curr = ares_malloc_data(ARES_DATATYPE_ADDR_PORT_NODE);
Packit 514978
      if (!srvr_curr)
Packit 514978
        {
Packit 514978
          status = ARES_ENOMEM;
Packit 514978
          break;
Packit 514978
        }
Packit 514978
      if (srvr_last)
Packit 514978
        {
Packit 514978
          srvr_last->next = srvr_curr;
Packit 514978
        }
Packit 514978
      else
Packit 514978
        {
Packit 514978
          srvr_head = srvr_curr;
Packit 514978
        }
Packit 514978
      srvr_last = srvr_curr;
Packit 514978
Packit 514978
      /* Fill this server node data */
Packit 514978
      srvr_curr->family = channel->servers[i].addr.family;
Packit 514978
      srvr_curr->udp_port = ntohs((unsigned short)channel->servers[i].addr.udp_port);
Packit 514978
      srvr_curr->tcp_port = ntohs((unsigned short)channel->servers[i].addr.tcp_port);
Packit 514978
      if (srvr_curr->family == AF_INET)
Packit 514978
        memcpy(&srvr_curr->addrV4, &channel->servers[i].addr.addrV4,
Packit 514978
               sizeof(srvr_curr->addrV4));
Packit 514978
      else
Packit 514978
        memcpy(&srvr_curr->addrV6, &channel->servers[i].addr.addrV6,
Packit 514978
               sizeof(srvr_curr->addrV6));
Packit 514978
    }
Packit 514978
Packit 514978
  if (status != ARES_SUCCESS)
Packit 514978
    {
Packit 514978
      if (srvr_head)
Packit 514978
        {
Packit 514978
          ares_free_data(srvr_head);
Packit 514978
          srvr_head = NULL;
Packit 514978
        }
Packit 514978
    }
Packit 514978
Packit 514978
  *servers = srvr_head;
Packit 514978
Packit 514978
  return status;
Packit 514978
}
Packit 514978
Packit 514978
int ares_set_servers(ares_channel channel,
Packit 514978
                     struct ares_addr_node *servers)
Packit 514978
{
Packit 514978
  struct ares_addr_node *srvr;
Packit 514978
  int num_srvrs = 0;
Packit 514978
  int i;
Packit 514978
Packit 514978
  if (ares_library_initialized() != ARES_SUCCESS)
Packit 514978
    return ARES_ENOTINITIALIZED;  /* LCOV_EXCL_LINE: n/a on non-WinSock */
Packit 514978
Packit 514978
  if (!channel)
Packit 514978
    return ARES_ENODATA;
Packit 514978
Packit 514978
  ares__destroy_servers_state(channel);
Packit 514978
Packit 514978
  for (srvr = servers; srvr; srvr = srvr->next)
Packit 514978
    {
Packit 514978
      num_srvrs++;
Packit 514978
    }
Packit 514978
Packit 514978
  if (num_srvrs > 0)
Packit 514978
    {
Packit 514978
      /* Allocate storage for servers state */
Packit 514978
      channel->servers = ares_malloc(num_srvrs * sizeof(struct server_state));
Packit 514978
      if (!channel->servers)
Packit 514978
        {
Packit 514978
          return ARES_ENOMEM;
Packit 514978
        }
Packit 514978
      channel->nservers = num_srvrs;
Packit 514978
      /* Fill servers state address data */
Packit 514978
      for (i = 0, srvr = servers; srvr; i++, srvr = srvr->next)
Packit 514978
        {
Packit 514978
          channel->servers[i].addr.family = srvr->family;
Packit 514978
          channel->servers[i].addr.udp_port = 0;
Packit 514978
          channel->servers[i].addr.tcp_port = 0;
Packit 514978
          if (srvr->family == AF_INET)
Packit 514978
            memcpy(&channel->servers[i].addr.addrV4, &srvr->addrV4,
Packit 514978
                   sizeof(srvr->addrV4));
Packit 514978
          else
Packit 514978
            memcpy(&channel->servers[i].addr.addrV6, &srvr->addrV6,
Packit 514978
                   sizeof(srvr->addrV6));
Packit 514978
        }
Packit 514978
      /* Initialize servers state remaining data */
Packit 514978
      ares__init_servers_state(channel);
Packit 514978
    }
Packit 514978
Packit 514978
  return ARES_SUCCESS;
Packit 514978
}
Packit 514978
Packit 514978
int ares_set_servers_ports(ares_channel channel,
Packit 514978
                           struct ares_addr_port_node *servers)
Packit 514978
{
Packit 514978
  struct ares_addr_port_node *srvr;
Packit 514978
  int num_srvrs = 0;
Packit 514978
  int i;
Packit 514978
Packit 514978
  if (ares_library_initialized() != ARES_SUCCESS)
Packit 514978
    return ARES_ENOTINITIALIZED;  /* LCOV_EXCL_LINE: n/a on non-WinSock */
Packit 514978
Packit 514978
  if (!channel)
Packit 514978
    return ARES_ENODATA;
Packit 514978
Packit 514978
  ares__destroy_servers_state(channel);
Packit 514978
Packit 514978
  for (srvr = servers; srvr; srvr = srvr->next)
Packit 514978
    {
Packit 514978
      num_srvrs++;
Packit 514978
    }
Packit 514978
Packit 514978
  if (num_srvrs > 0)
Packit 514978
    {
Packit 514978
      /* Allocate storage for servers state */
Packit 514978
      channel->servers = ares_malloc(num_srvrs * sizeof(struct server_state));
Packit 514978
      if (!channel->servers)
Packit 514978
        {
Packit 514978
          return ARES_ENOMEM;
Packit 514978
        }
Packit 514978
      channel->nservers = num_srvrs;
Packit 514978
      /* Fill servers state address data */
Packit 514978
      for (i = 0, srvr = servers; srvr; i++, srvr = srvr->next)
Packit 514978
        {
Packit 514978
          channel->servers[i].addr.family = srvr->family;
Packit 514978
          channel->servers[i].addr.udp_port = htons((unsigned short)srvr->udp_port);
Packit 514978
          channel->servers[i].addr.tcp_port = htons((unsigned short)srvr->tcp_port);
Packit 514978
          if (srvr->family == AF_INET)
Packit 514978
            memcpy(&channel->servers[i].addr.addrV4, &srvr->addrV4,
Packit 514978
                   sizeof(srvr->addrV4));
Packit 514978
          else
Packit 514978
            memcpy(&channel->servers[i].addr.addrV6, &srvr->addrV6,
Packit 514978
                   sizeof(srvr->addrV6));
Packit 514978
        }
Packit 514978
      /* Initialize servers state remaining data */
Packit 514978
      ares__init_servers_state(channel);
Packit 514978
    }
Packit 514978
Packit 514978
  return ARES_SUCCESS;
Packit 514978
}
Packit 514978
Packit 514978
/* Incomming string format: host[:port][,host[:port]]... */
Packit 514978
/* IPv6 addresses with ports require square brackets [fe80::1%lo0]:53 */
Packit 514978
static int set_servers_csv(ares_channel channel,
Packit 514978
                           const char* _csv, int use_port)
Packit 514978
{
Packit 514978
  size_t i;
Packit 514978
  char* csv = NULL;
Packit 514978
  char* ptr;
Packit 514978
  char* start_host;
Packit 514978
  int cc = 0;
Packit 514978
  int rv = ARES_SUCCESS;
Packit 514978
  struct ares_addr_port_node *servers = NULL;
Packit 514978
  struct ares_addr_port_node *last = NULL;
Packit 514978
Packit 514978
  if (ares_library_initialized() != ARES_SUCCESS)
Packit 514978
    return ARES_ENOTINITIALIZED;  /* LCOV_EXCL_LINE: n/a on non-WinSock */
Packit 514978
Packit 514978
  if (!channel)
Packit 514978
    return ARES_ENODATA;
Packit 514978
Packit 514978
  ares__destroy_servers_state(channel);
Packit 514978
Packit 514978
  i = strlen(_csv);
Packit 514978
  if (i == 0)
Packit 514978
     return ARES_SUCCESS; /* blank all servers */
Packit 514978
Packit 514978
  csv = ares_malloc(i + 2);
Packit 514978
  if (!csv)
Packit 514978
    return ARES_ENOMEM;
Packit 514978
Packit 514978
  strcpy(csv, _csv);
Packit 514978
  if (csv[i-1] != ',') { /* make parsing easier by ensuring ending ',' */
Packit 514978
    csv[i] = ',';
Packit 514978
    csv[i+1] = 0;
Packit 514978
  }
Packit 514978
Packit 514978
  start_host = csv;
Packit 514978
  for (ptr = csv; *ptr; ptr++) {
Packit 514978
    if (*ptr == ':') {
Packit 514978
      /* count colons to determine if we have an IPv6 number or IPv4 with
Packit 514978
         port */
Packit 514978
      cc++;
Packit 514978
    }
Packit 514978
    else if (*ptr == '[') {
Packit 514978
      /* move start_host if an open square bracket is found wrapping an IPv6
Packit 514978
         address */
Packit 514978
      start_host = ptr + 1;
Packit 514978
    }
Packit 514978
    else if (*ptr == ',') {
Packit 514978
      char* pp = ptr - 1;
Packit 514978
      char* p = ptr;
Packit 514978
      int port = 0;
Packit 514978
      struct in_addr in4;
Packit 514978
      struct ares_in6_addr in6;
Packit 514978
      struct ares_addr_port_node *s = NULL;
Packit 514978
Packit 514978
      *ptr = 0; /* null terminate host:port string */
Packit 514978
      /* Got an entry..see if the port was specified. */
Packit 514978
      if (cc > 0) {
Packit 514978
        while (pp > start_host) {
Packit 514978
          /* a single close square bracket followed by a colon, ']:' indicates
Packit 514978
             an IPv6 address with port */
Packit 514978
          if ((*pp == ']') && (*p == ':'))
Packit 514978
            break; /* found port */
Packit 514978
          /* a single colon, ':' indicates an IPv4 address with port */
Packit 514978
          if ((*pp == ':') && (cc == 1))
Packit 514978
            break; /* found port */
Packit 514978
          if (!(ISDIGIT(*pp) || (*pp == ':'))) {
Packit 514978
            /* Found end of digits before we found :, so wasn't a port */
Packit 514978
            /* must allow ':' for IPv6 case of ']:' indicates we found a port */
Packit 514978
            pp = p = ptr;
Packit 514978
            break;
Packit 514978
          }
Packit 514978
          pp--;
Packit 514978
          p--;
Packit 514978
        }
Packit 514978
        if ((pp != start_host) && ((pp + 1) < ptr)) {
Packit 514978
          /* Found it. Parse over the port number */
Packit 514978
          /* when an IPv6 address is wrapped with square brackets the port
Packit 514978
             starts at pp + 2 */
Packit 514978
          if (*pp == ']')
Packit 514978
            p++; /* move p before ':' */
Packit 514978
          /* p will point to the start of the port */
Packit 514978
          port = (int)strtol(p, NULL, 10);
Packit 514978
          *pp = 0; /* null terminate host */
Packit 514978
        }
Packit 514978
      }
Packit 514978
      /* resolve host, try ipv4 first, rslt is in network byte order */
Packit 514978
      rv = ares_inet_pton(AF_INET, start_host, &in4;;
Packit 514978
      if (!rv) {
Packit 514978
        /* Ok, try IPv6 then */
Packit 514978
        rv = ares_inet_pton(AF_INET6, start_host, &in6;;
Packit 514978
        if (!rv) {
Packit 514978
          rv = ARES_EBADSTR;
Packit 514978
          goto out;
Packit 514978
        }
Packit 514978
        /* was ipv6, add new server */
Packit 514978
        s = ares_malloc(sizeof(*s));
Packit 514978
        if (!s) {
Packit 514978
          rv = ARES_ENOMEM;
Packit 514978
          goto out;
Packit 514978
        }
Packit 514978
        s->family = AF_INET6;
Packit 514978
        memcpy(&s->addr, &in6, sizeof(struct ares_in6_addr));
Packit 514978
      }
Packit 514978
      else {
Packit 514978
        /* was ipv4, add new server */
Packit 514978
        s = ares_malloc(sizeof(*s));
Packit 514978
        if (!s) {
Packit 514978
          rv = ARES_ENOMEM;
Packit 514978
          goto out;
Packit 514978
        }
Packit 514978
        s->family = AF_INET;
Packit 514978
        memcpy(&s->addr, &in4, sizeof(struct in_addr));
Packit 514978
      }
Packit 514978
      if (s) {
Packit 514978
        s->udp_port = use_port ? port: 0;
Packit 514978
        s->tcp_port = s->udp_port;
Packit 514978
        s->next = NULL;
Packit 514978
        if (last) {
Packit 514978
          last->next = s;
Packit 514978
          /* need to move last to maintain the linked list */
Packit 514978
          last = last->next;
Packit 514978
        }
Packit 514978
        else {
Packit 514978
          servers = s;
Packit 514978
          last = s;
Packit 514978
        }
Packit 514978
      }
Packit 514978
Packit 514978
      /* Set up for next one */
Packit 514978
      start_host = ptr + 1;
Packit 514978
      cc = 0;
Packit 514978
    }
Packit 514978
  }
Packit 514978
Packit 514978
  rv = ares_set_servers_ports(channel, servers);
Packit 514978
Packit 514978
  out:
Packit 514978
  if (csv)
Packit 514978
    ares_free(csv);
Packit 514978
  while (servers) {
Packit 514978
    struct ares_addr_port_node *s = servers;
Packit 514978
    servers = servers->next;
Packit 514978
    ares_free(s);
Packit 514978
  }
Packit 514978
Packit 514978
  return rv;
Packit 514978
}
Packit 514978
Packit 514978
int ares_set_servers_csv(ares_channel channel,
Packit 514978
                         const char* _csv)
Packit 514978
{
Packit 514978
  return set_servers_csv(channel, _csv, FALSE);
Packit 514978
}
Packit 514978
Packit 514978
int ares_set_servers_ports_csv(ares_channel channel,
Packit 514978
                               const char* _csv)
Packit 514978
{
Packit 514978
  return set_servers_csv(channel, _csv, TRUE);
Packit 514978
}
Packit 514978