• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* Copyright (C) 2019 by Andrew Selivanov
2  *
3  * Permission to use, copy, modify, and distribute this
4  * software and its documentation for any purpose and without
5  * fee is hereby granted, provided that the above copyright
6  * notice appear in all copies and that both that copyright
7  * notice and this permission notice appear in supporting
8  * documentation, and that the name of M.I.T. not be used in
9  * advertising or publicity pertaining to distribution of the
10  * software without specific, written prior permission.
11  * M.I.T. makes no representations about the suitability of
12  * this software for any purpose.  It is provided "as is"
13  * without express or implied warranty.
14  */
15 
16 #include "ares_setup.h"
17 
18 #ifdef HAVE_NETINET_IN_H
19 #  include <netinet/in.h>
20 #endif
21 #ifdef HAVE_NETDB_H
22 #  include <netdb.h>
23 #endif
24 #ifdef HAVE_ARPA_INET_H
25 #  include <arpa/inet.h>
26 #endif
27 
28 #include "ares.h"
29 #include "ares_inet_net_pton.h"
30 #include "ares_nowarn.h"
31 #include "ares_private.h"
32 
33 #define MAX_ALIASES 40
34 
ares__readaddrinfo(FILE * fp,const char * name,unsigned short port,const struct ares_addrinfo_hints * hints,struct ares_addrinfo * ai)35 int ares__readaddrinfo(FILE *fp,
36                        const char *name,
37                        unsigned short port,
38                        const struct ares_addrinfo_hints *hints,
39                        struct ares_addrinfo *ai)
40 {
41   char *line = NULL, *p, *q;
42   char *txtaddr, *txthost, *txtalias;
43   char *aliases[MAX_ALIASES];
44   unsigned int i, alias_count;
45   int status = ARES_SUCCESS;
46   size_t linesize;
47   struct ares_addrinfo_cname *cname = NULL, *cnames = NULL;
48   struct ares_addrinfo_node *nodes = NULL;
49   int match_with_alias, match_with_canonical;
50   int want_cname = hints->ai_flags & ARES_AI_CANONNAME;
51 
52   /* Validate family */
53   switch (hints->ai_family) {
54     case AF_INET:
55     case AF_INET6:
56     case AF_UNSPEC:
57       break;
58     default:
59       return ARES_EBADFAMILY;
60   }
61 
62   ai->name = ares_strdup(name);
63   if(!ai->name)
64     {
65       status = ARES_ENOMEM;
66       goto fail;
67     }
68 
69   while ((status = ares__read_line(fp, &line, &linesize)) == ARES_SUCCESS)
70     {
71       match_with_alias = 0;
72       match_with_canonical = 0;
73       alias_count = 0;
74       /* Trim line comment. */
75       p = line;
76       while (*p && (*p != '#'))
77         p++;
78       *p = '\0';
79 
80       /* Trim trailing whitespace. */
81       q = p - 1;
82       while ((q >= line) && ISSPACE(*q))
83         q--;
84       *++q = '\0';
85 
86       /* Skip leading whitespace. */
87       p = line;
88       while (*p && ISSPACE(*p))
89         p++;
90       if (!*p)
91         /* Ignore line if empty. */
92         continue;
93 
94       /* Pointer to start of IPv4 or IPv6 address part. */
95       txtaddr = p;
96 
97       /* Advance past address part. */
98       while (*p && !ISSPACE(*p))
99         p++;
100       if (!*p)
101         /* Ignore line if reached end of line. */
102         continue;
103 
104       /* Null terminate address part. */
105       *p = '\0';
106 
107       /* Advance to host name */
108       p++;
109       while (*p && ISSPACE(*p))
110         p++;
111       if (!*p)
112         /* Ignore line if reached end of line. */
113         continue;  /* LCOV_EXCL_LINE: trailing whitespace already stripped */
114 
115       /* Pointer to start of host name. */
116       txthost = p;
117 
118       /* Advance past host name. */
119       while (*p && !ISSPACE(*p))
120         p++;
121 
122       /* Pointer to start of first alias. */
123       txtalias = NULL;
124       if (*p)
125         {
126           q = p + 1;
127           while (*q && ISSPACE(*q))
128             q++;
129           if (*q)
130             txtalias = q;
131         }
132 
133       /* Null terminate host name. */
134       *p = '\0';
135 
136       /* Find out if host name matches with canonical host name. */
137       if (strcasecmp(txthost, name) == 0)
138         {
139           match_with_canonical = 1;
140         }
141 
142       /* Find out if host name matches with one of the aliases. */
143       while (txtalias)
144         {
145           p = txtalias;
146           while (*p && !ISSPACE(*p))
147             p++;
148           q = p;
149           while (*q && ISSPACE(*q))
150             q++;
151           *p = '\0';
152           if (strcasecmp(txtalias, name) == 0)
153             {
154               match_with_alias = 1;
155               if (!want_cname)
156                 break;
157             }
158           if (alias_count < MAX_ALIASES)
159             {
160               aliases[alias_count++] = txtalias;
161             }
162           txtalias = *q ? q : NULL;
163         }
164 
165       /* Try next line if host does not match. */
166       if (!match_with_alias && !match_with_canonical)
167         {
168           continue;
169         }
170 
171       /*
172        * Convert address string to network address for the requested families.
173        * Actual address family possible values are AF_INET and AF_INET6 only.
174        */
175       if ((hints->ai_family == AF_INET) || (hints->ai_family == AF_UNSPEC))
176         {
177           struct in_addr addr4;
178           if (ares_inet_pton(AF_INET, txtaddr, &addr4) == 1)
179             {
180               status = ares_append_ai_node(AF_INET, port, 0, &addr4, &nodes);
181               if (status != ARES_SUCCESS)
182                 {
183                   goto fail;
184                 }
185             }
186         }
187       if ((hints->ai_family == AF_INET6) || (hints->ai_family == AF_UNSPEC))
188         {
189           struct ares_in6_addr addr6;
190           if (ares_inet_pton(AF_INET6, txtaddr, &addr6) == 1)
191             {
192               status = ares_append_ai_node(AF_INET6, port, 0, &addr6, &nodes);
193               if (status != ARES_SUCCESS)
194                 {
195                   goto fail;
196                 }
197             }
198         }
199 
200       if (status != ARES_SUCCESS)
201         /* Ignore line if invalid address string for the requested family. */
202         continue;
203 
204       if (want_cname)
205         {
206           for (i = 0; i < alias_count; ++i)
207             {
208               cname = ares__append_addrinfo_cname(&cnames);
209               if (!cname)
210                 {
211                   status = ARES_ENOMEM;
212                   goto fail;
213                 }
214               cname->alias = ares_strdup(aliases[i]);
215               cname->name = ares_strdup(txthost);
216             }
217           /* No aliases, cname only. */
218           if(!alias_count)
219             {
220               cname = ares__append_addrinfo_cname(&cnames);
221               if (!cname)
222                 {
223                   status = ARES_ENOMEM;
224                   goto fail;
225                 }
226               cname->name = ares_strdup(txthost);
227             }
228         }
229     }
230 
231   /* Last read failed. */
232   if (status == ARES_ENOMEM)
233     {
234       goto fail;
235     }
236 
237   /* Free line buffer. */
238   ares_free(line);
239   ares__addrinfo_cat_cnames(&ai->cnames, cnames);
240   ares__addrinfo_cat_nodes(&ai->nodes, nodes);
241 
242   return nodes ? ARES_SUCCESS : ARES_ENOTFOUND;
243 
244 fail:
245   ares_free(line);
246   ares__freeaddrinfo_cnames(cnames);
247   ares__freeaddrinfo_nodes(nodes);
248   ares_free(ai->name);
249   ai->name = NULL;
250   return status;
251 }
252