mirror of https://github.com/c-ares/c-ares.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
264 lines
6.7 KiB
264 lines
6.7 KiB
/* $Id$ */ |
|
|
|
/* Copyright 1998, 2009 by the Massachusetts Institute of Technology. |
|
* |
|
* Permission to use, copy, modify, and distribute this |
|
* software and its documentation for any purpose and without |
|
* fee is hereby granted, provided that the above copyright |
|
* notice appear in all copies and that both that copyright |
|
* notice and this permission notice appear in supporting |
|
* documentation, and that the name of M.I.T. not be used in |
|
* advertising or publicity pertaining to distribution of the |
|
* software without specific, written prior permission. |
|
* M.I.T. makes no representations about the suitability of |
|
* this software for any purpose. It is provided "as is" |
|
* without express or implied warranty. |
|
*/ |
|
|
|
#include "setup.h" |
|
|
|
#ifdef HAVE_SYS_SOCKET_H |
|
# include <sys/socket.h> |
|
#endif |
|
#ifdef HAVE_NETINET_IN_H |
|
# include <netinet/in.h> |
|
#endif |
|
#ifdef HAVE_NETDB_H |
|
# include <netdb.h> |
|
#endif |
|
#ifdef HAVE_ARPA_INET_H |
|
# include <arpa/inet.h> |
|
#endif |
|
|
|
#include "ares.h" |
|
#include "inet_net_pton.h" |
|
#include "ares_private.h" |
|
|
|
int ares__get_hostent(FILE *fp, int family, struct hostent **host) |
|
{ |
|
char *line = NULL, *p, *q, **alias; |
|
char *txtaddr, *txthost, *txtalias; |
|
int status; |
|
size_t addrlen, linesize, naliases; |
|
struct ares_addr addr; |
|
struct hostent *hostent = NULL; |
|
|
|
*host = NULL; /* Assume failure */ |
|
|
|
/* Validate family */ |
|
switch (family) { |
|
case AF_INET: |
|
case AF_INET6: |
|
case AF_UNSPEC: |
|
break; |
|
default: |
|
return ARES_EBADFAMILY; |
|
} |
|
|
|
while ((status = ares__read_line(fp, &line, &linesize)) == ARES_SUCCESS) |
|
{ |
|
|
|
/* Trim line comment. */ |
|
p = line; |
|
while (*p && (*p != '#')) |
|
p++; |
|
*p = '\0'; |
|
|
|
/* Trim trailing whitespace. */ |
|
q = p - 1; |
|
while ((q >= line) && ISSPACE(*q)) |
|
q--; |
|
*++q = '\0'; |
|
|
|
/* Skip leading whitespace. */ |
|
p = line; |
|
while (*p && ISSPACE(*p)) |
|
p++; |
|
if (!*p) |
|
/* Ignore line if empty. */ |
|
continue; |
|
|
|
/* Pointer to start of IPv4 or IPv6 address part. */ |
|
txtaddr = p; |
|
|
|
/* Advance past address part. */ |
|
while (*p && !ISSPACE(*p)) |
|
p++; |
|
if (!*p) |
|
/* Ignore line if reached end of line. */ |
|
continue; |
|
|
|
/* Null terminate address part. */ |
|
*p = '\0'; |
|
|
|
/* Advance to host name */ |
|
p++; |
|
while (*p && ISSPACE(*p)) |
|
p++; |
|
if (!*p) |
|
/* Ignore line if reached end of line. */ |
|
continue; |
|
|
|
/* Pointer to start of host name. */ |
|
txthost = p; |
|
|
|
/* Advance past host name. */ |
|
while (*p && !ISSPACE(*p)) |
|
p++; |
|
|
|
/* Pointer to start of first alias. */ |
|
txtalias = NULL; |
|
if (*p) |
|
{ |
|
q = p + 1; |
|
while (*q && ISSPACE(*q)) |
|
q++; |
|
if (*q) |
|
txtalias = q; |
|
} |
|
|
|
/* Null terminate host name. */ |
|
*p = '\0'; |
|
|
|
/* find out number of aliases. */ |
|
naliases = 0; |
|
if (txtalias) |
|
{ |
|
p = txtalias; |
|
while (*p) |
|
{ |
|
while (*p && !ISSPACE(*p)) |
|
p++; |
|
while (*p && ISSPACE(*p)) |
|
p++; |
|
naliases++; |
|
} |
|
} |
|
|
|
/* Convert address string to network address for the requested family. */ |
|
addrlen = 0; |
|
addr.family = AF_UNSPEC; |
|
addr.addrV4.s_addr = INADDR_NONE; |
|
if ((family == AF_INET) || (family == AF_UNSPEC)) |
|
{ |
|
addr.addrV4.s_addr = inet_addr(txtaddr); |
|
if (addr.addrV4.s_addr != INADDR_NONE) |
|
{ |
|
/* Actual network address family and length. */ |
|
addr.family = AF_INET; |
|
addrlen = sizeof(struct in_addr); |
|
} |
|
} |
|
if ((family == AF_INET6) || ((family == AF_UNSPEC) && (!addrlen))) |
|
{ |
|
if (ares_inet_pton(AF_INET6, txtaddr, &addr.addrV6) > 0) |
|
{ |
|
/* Actual network address family and length. */ |
|
addr.family = AF_INET6; |
|
addrlen = sizeof(struct in6_addr); |
|
} |
|
} |
|
if (!addrlen) |
|
/* Ignore line if invalid address string for the requested family. */ |
|
continue; |
|
|
|
/* |
|
** Actual address family possible values are AF_INET and AF_INET6 only. |
|
*/ |
|
|
|
/* Allocate memory for the hostent structure. */ |
|
hostent = malloc(sizeof(struct hostent)); |
|
if (!hostent) |
|
break; |
|
|
|
/* Initialize fields for out of memory condition. */ |
|
hostent->h_aliases = NULL; |
|
hostent->h_addr_list = NULL; |
|
|
|
/* Copy official host name. */ |
|
hostent->h_name = strdup(txthost); |
|
if (!hostent->h_name) |
|
break; |
|
|
|
/* Copy network address. */ |
|
hostent->h_addr_list = malloc(2 * sizeof(char *)); |
|
if (!hostent->h_addr_list) |
|
break; |
|
hostent->h_addr_list[1] = NULL; |
|
hostent->h_addr_list[0] = malloc(addrlen); |
|
if (!hostent->h_addr_list[0]) |
|
break; |
|
if (addr.family == AF_INET) |
|
memcpy(hostent->h_addr_list[0], &addr.addrV4, sizeof(struct in_addr)); |
|
else |
|
memcpy(hostent->h_addr_list[0], &addr.addrV6, sizeof(struct in6_addr)); |
|
|
|
/* Copy aliases. */ |
|
hostent->h_aliases = malloc((naliases + 1) * sizeof(char *)); |
|
if (!hostent->h_aliases) |
|
break; |
|
alias = hostent->h_aliases; |
|
while (naliases) |
|
*(alias + naliases--) = NULL; |
|
*alias = NULL; |
|
while (txtalias) |
|
{ |
|
p = txtalias; |
|
while (*p && !ISSPACE(*p)) |
|
p++; |
|
q = p; |
|
while (*q && ISSPACE(*q)) |
|
q++; |
|
*p = '\0'; |
|
if ((*alias = strdup(txtalias)) == NULL) |
|
break; |
|
alias++; |
|
txtalias = *q ? q : NULL; |
|
} |
|
if (txtalias) |
|
/* Alias memory allocation failure. */ |
|
break; |
|
|
|
/* Copy actual network address family and length. */ |
|
hostent->h_addrtype = addr.family; |
|
hostent->h_length = (int)addrlen; |
|
|
|
/* Free line buffer. */ |
|
free(line); |
|
|
|
/* Return hostent successfully */ |
|
*host = hostent; |
|
return ARES_SUCCESS; |
|
|
|
} |
|
|
|
/* If allocated, free line buffer. */ |
|
if (line) |
|
free(line); |
|
|
|
if (status == ARES_SUCCESS) |
|
{ |
|
/* Memory allocation failure; clean up. */ |
|
if (hostent) |
|
{ |
|
if (hostent->h_name) |
|
free((char *) hostent->h_name); |
|
if (hostent->h_aliases) |
|
{ |
|
for (alias = hostent->h_aliases; *alias; alias++) |
|
free(*alias); |
|
free(hostent->h_aliases); |
|
} |
|
if (hostent->h_addr_list) |
|
{ |
|
if (hostent->h_addr_list[0]) |
|
free(hostent->h_addr_list[0]); |
|
free(hostent->h_addr_list); |
|
} |
|
free(hostent); |
|
} |
|
return ARES_ENOMEM; |
|
} |
|
|
|
return status; |
|
}
|
|
|