Blame nameserv.c

Packit 96c956
/*
Packit 96c956
  chronyd/chronyc - Programs for keeping computer clocks accurate.
Packit 96c956
Packit 96c956
 **********************************************************************
Packit 96c956
 * Copyright (C) Richard P. Curnow  1997-2003
Packit 96c956
 * Copyright (C) Miroslav Lichvar  2009-2011
Packit 96c956
 * 
Packit 96c956
 * This program is free software; you can redistribute it and/or modify
Packit 96c956
 * it under the terms of version 2 of the GNU General Public License as
Packit 96c956
 * published by the Free Software Foundation.
Packit 96c956
 * 
Packit 96c956
 * This program is distributed in the hope that it will be useful, but
Packit 96c956
 * WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 96c956
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 96c956
 * General Public License for more details.
Packit 96c956
 * 
Packit 96c956
 * You should have received a copy of the GNU General Public License along
Packit 96c956
 * with this program; if not, write to the Free Software Foundation, Inc.,
Packit 96c956
 * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
Packit 96c956
 * 
Packit 96c956
 **********************************************************************
Packit 96c956
Packit 96c956
  =======================================================================
Packit 96c956
Packit 96c956
  Functions to do name to IP address conversion
Packit 96c956
Packit 96c956
  */
Packit 96c956
Packit 96c956
#include "config.h"
Packit 96c956
Packit 96c956
#include "sysincl.h"
Packit 96c956
Packit 96c956
#include <netdb.h>
Packit 96c956
#include <resolv.h>
Packit 96c956
Packit 96c956
#include "nameserv.h"
Packit 96c956
#include "util.h"
Packit 96c956
Packit 96c956
/* ================================================== */
Packit 96c956
Packit 96c956
static int address_family = IPADDR_UNSPEC;
Packit 96c956
Packit 96c956
void
Packit 96c956
DNS_SetAddressFamily(int family)
Packit 96c956
{
Packit 96c956
  address_family = family;
Packit 96c956
}
Packit 96c956
Packit 96c956
DNS_Status 
Packit 96c956
DNS_Name2IPAddress(const char *name, IPAddr *ip_addrs, int max_addrs)
Packit 96c956
{
Packit 96c956
#ifdef HAVE_GETADDRINFO
Packit 96c956
  struct addrinfo hints, *res, *ai;
Packit 96c956
  int i, result;
Packit 96c956
Packit 96c956
  max_addrs = MIN(max_addrs, DNS_MAX_ADDRESSES);
Packit 96c956
Packit 96c956
  memset(&hints, 0, sizeof (hints));
Packit 96c956
Packit 96c956
  switch (address_family) {
Packit 96c956
    case IPADDR_INET4:
Packit 96c956
      hints.ai_family = AF_INET;
Packit 96c956
      break;
Packit 96c956
#ifdef FEAT_IPV6
Packit 96c956
    case IPADDR_INET6:
Packit 96c956
      hints.ai_family = AF_INET6;
Packit 96c956
      break;
Packit 96c956
#endif
Packit 96c956
    default:
Packit 96c956
      hints.ai_family = AF_UNSPEC;
Packit 96c956
  }
Packit 96c956
  hints.ai_socktype = SOCK_STREAM;
Packit 96c956
Packit 96c956
  result = getaddrinfo(name, NULL, &hints, &res;;
Packit 96c956
Packit 96c956
  if (result) {
Packit 96c956
#ifdef FORCE_DNSRETRY
Packit 96c956
    return DNS_TryAgain;
Packit 96c956
#else
Packit 96c956
    return result == EAI_AGAIN ? DNS_TryAgain : DNS_Failure;
Packit 96c956
#endif
Packit 96c956
  }
Packit 96c956
Packit 96c956
  for (ai = res, i = 0; i < max_addrs && ai != NULL; ai = ai->ai_next) {
Packit 96c956
    switch (ai->ai_family) {
Packit 96c956
      case AF_INET:
Packit 96c956
        if (address_family != IPADDR_UNSPEC && address_family != IPADDR_INET4)
Packit 96c956
          continue;
Packit 96c956
        ip_addrs[i].family = IPADDR_INET4;
Packit 96c956
        ip_addrs[i].addr.in4 = ntohl(((struct sockaddr_in *)ai->ai_addr)->sin_addr.s_addr);
Packit 96c956
        i++;
Packit 96c956
        break;
Packit 96c956
#ifdef FEAT_IPV6
Packit 96c956
      case AF_INET6:
Packit 96c956
        if (address_family != IPADDR_UNSPEC && address_family != IPADDR_INET6)
Packit 96c956
          continue;
Packit 96c956
        ip_addrs[i].family = IPADDR_INET6;
Packit 96c956
        memcpy(&ip_addrs[i].addr.in6, &((struct sockaddr_in6 *)ai->ai_addr)->sin6_addr.s6_addr,
Packit 96c956
               sizeof (ip_addrs->addr.in6));
Packit 96c956
        i++;
Packit 96c956
        break;
Packit 96c956
#endif
Packit 96c956
    }
Packit 96c956
  }
Packit 96c956
Packit 96c956
  for (; i < max_addrs; i++)
Packit 96c956
        ip_addrs[i].family = IPADDR_UNSPEC;
Packit 96c956
Packit 96c956
  freeaddrinfo(res);
Packit 96c956
Packit 96c956
  return !max_addrs || ip_addrs[0].family != IPADDR_UNSPEC ? DNS_Success : DNS_Failure;
Packit 96c956
#else
Packit 96c956
  struct hostent *host;
Packit 96c956
  int i;
Packit 96c956
  
Packit 96c956
  if (address_family != IPADDR_UNSPEC && address_family != IPADDR_INET4)
Packit 96c956
    return DNS_Failure;
Packit 96c956
Packit 96c956
  max_addrs = MIN(max_addrs, DNS_MAX_ADDRESSES);
Packit 96c956
Packit 96c956
  host = gethostbyname(name);
Packit 96c956
Packit 96c956
  if (host == NULL) {
Packit 96c956
    if (h_errno == TRY_AGAIN)
Packit 96c956
      return DNS_TryAgain;
Packit 96c956
  } else {
Packit 96c956
    if (host->h_addrtype != AF_INET || !host->h_addr_list[0])
Packit 96c956
      return DNS_Failure;
Packit 96c956
Packit 96c956
    for (i = 0; host->h_addr_list[i] && i < max_addrs; i++) {
Packit 96c956
      ip_addrs[i].family = IPADDR_INET4;
Packit 96c956
      ip_addrs[i].addr.in4 = ntohl(*(uint32_t *)host->h_addr_list[i]);
Packit 96c956
    }
Packit 96c956
Packit 96c956
    for (; i < max_addrs; i++)
Packit 96c956
      ip_addrs[i].family = IPADDR_UNSPEC;
Packit 96c956
Packit 96c956
    return DNS_Success;
Packit 96c956
  }
Packit 96c956
Packit 96c956
#ifdef FORCE_DNSRETRY
Packit 96c956
  return DNS_TryAgain;
Packit 96c956
#else
Packit 96c956
  return DNS_Failure;
Packit 96c956
#endif
Packit 96c956
Packit 96c956
#endif
Packit 96c956
}
Packit 96c956
Packit 96c956
/* ================================================== */
Packit 96c956
Packit 96c956
int
Packit 96c956
DNS_IPAddress2Name(IPAddr *ip_addr, char *name, int len)
Packit 96c956
{
Packit 96c956
  char *result = NULL;
Packit 96c956
Packit 96c956
#ifdef FEAT_IPV6
Packit 96c956
  struct sockaddr_in6 in6;
Packit 96c956
  socklen_t slen;
Packit 96c956
  char hbuf[NI_MAXHOST];
Packit 96c956
Packit 96c956
  slen = UTI_IPAndPortToSockaddr(ip_addr, 0, (struct sockaddr *)&in6;;
Packit 96c956
  if (!getnameinfo((struct sockaddr *)&in6, slen, hbuf, sizeof (hbuf), NULL, 0, 0))
Packit 96c956
    result = hbuf;
Packit 96c956
#else
Packit 96c956
  struct hostent *host;
Packit 96c956
  uint32_t addr;
Packit 96c956
Packit 96c956
  switch (ip_addr->family) {
Packit 96c956
    case IPADDR_INET4:
Packit 96c956
      addr = htonl(ip_addr->addr.in4);
Packit 96c956
      host = gethostbyaddr((const char *) &addr, sizeof (ip_addr), AF_INET);
Packit 96c956
      break;
Packit 96c956
#ifdef FEAT_IPV6
Packit 96c956
    case IPADDR_INET6:
Packit 96c956
      host = gethostbyaddr((const void *) ip_addr->addr.in6, sizeof (ip_addr->addr.in6), AF_INET6);
Packit 96c956
      break;
Packit 96c956
#endif
Packit 96c956
    default:
Packit 96c956
      host = NULL;
Packit 96c956
  }
Packit 96c956
  if (host)
Packit 96c956
    result = host->h_name;
Packit 96c956
#endif
Packit 96c956
Packit 96c956
  if (result == NULL)
Packit 96c956
    result = UTI_IPToString(ip_addr);
Packit 96c956
  if (snprintf(name, len, "%s", result) >= len)
Packit 96c956
    return 0;
Packit 96c956
Packit 96c956
  return 1;
Packit 96c956
}
Packit 96c956
Packit 96c956
/* ================================================== */
Packit 96c956
Packit 96c956
void
Packit 96c956
DNS_Reload(void)
Packit 96c956
{
Packit 96c956
  res_init();
Packit 96c956
}
Packit 96c956
Packit 96c956
/* ================================================== */
Packit 96c956