Initial import from FreeBSD RELENG_4:
[dragonfly.git] / lib / libc / net / res_query.c
1 /*
2  * Copyright (c) 1988, 1993
3  *    The Regents of the University of California.  All rights reserved.
4  * 
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  * 3. All advertising materials mentioning features or use of this software
14  *    must display the following acknowledgement:
15  *      This product includes software developed by the University of
16  *      California, Berkeley and its contributors.
17  * 4. Neither the name of the University nor the names of its contributors
18  *    may be used to endorse or promote products derived from this software
19  *    without specific prior written permission.
20  * 
21  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
22  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
25  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31  * SUCH DAMAGE.
32  */
33
34 /*
35  * Portions Copyright (c) 1993 by Digital Equipment Corporation.
36  * 
37  * Permission to use, copy, modify, and distribute this software for any
38  * purpose with or without fee is hereby granted, provided that the above
39  * copyright notice and this permission notice appear in all copies, and that
40  * the name of Digital Equipment Corporation not be used in advertising or
41  * publicity pertaining to distribution of the document or software without
42  * specific, written prior permission.
43  * 
44  * THE SOFTWARE IS PROVIDED "AS IS" AND DIGITAL EQUIPMENT CORP. DISCLAIMS ALL
45  * WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES
46  * OF MERCHANTABILITY AND FITNESS.   IN NO EVENT SHALL DIGITAL EQUIPMENT
47  * CORPORATION BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
48  * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
49  * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
50  * ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
51  * SOFTWARE.
52  */
53
54 /*
55  * Portions Copyright (c) 1996 by Internet Software Consortium.
56  *
57  * Permission to use, copy, modify, and distribute this software for any
58  * purpose with or without fee is hereby granted, provided that the above
59  * copyright notice and this permission notice appear in all copies.
60  *
61  * THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS
62  * ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
63  * OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE
64  * CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
65  * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
66  * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
67  * ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
68  * SOFTWARE.
69  */
70
71 #if defined(LIBC_SCCS) && !defined(lint)
72 static char sccsid[] = "@(#)res_query.c 8.1 (Berkeley) 6/4/93";
73 static char orig_rcsid[] = "From: Id: res_query.c,v 8.14 1997/06/09 17:47:05 halley Exp $";
74 static char rcsid[] = "$FreeBSD: src/lib/libc/net/res_query.c,v 1.19.2.2 2002/07/07 11:34:42 robert Exp $";
75 #endif /* LIBC_SCCS and not lint */
76
77 #include <sys/types.h>
78 #include <sys/param.h>
79 #include <netinet/in.h>
80 #include <arpa/inet.h>
81 #include <arpa/nameser.h>
82 #include <ctype.h>
83 #include <errno.h>
84 #include <netdb.h>
85 #include <resolv.h>
86 #include <stdio.h>
87 #include <stdlib.h>
88 #include <string.h>
89
90 #include "res_config.h"
91
92 #if PACKETSZ > 1024
93 #define MAXPACKET       PACKETSZ
94 #else
95 #define MAXPACKET       1024
96 #endif
97
98 /*
99  * Formulate a normal query, send, and await answer.
100  * Returned answer is placed in supplied buffer "answer".
101  * Perform preliminary check of answer, returning success only
102  * if no error is indicated and the answer count is nonzero.
103  * Return the size of the response on success, -1 on error.
104  * Error number is left in h_errno.
105  *
106  * Caller must parse answer and determine whether it answers the question.
107  */
108 int
109 res_query(name, class, type, answer, anslen)
110         const char *name;       /* domain name */
111         int class, type;        /* class and type of query */
112         u_char *answer;         /* buffer to put answer */
113         int anslen;             /* size of answer buffer */
114 {
115         u_char buf[MAXPACKET];
116         HEADER *hp = (HEADER *) answer;
117         int n;
118
119         hp->rcode = NOERROR;    /* default */
120
121         if ((_res.options & RES_INIT) == 0 && res_init() == -1) {
122                 h_errno = NETDB_INTERNAL;
123                 return (-1);
124         }
125 #ifdef DEBUG
126         if (_res.options & RES_DEBUG)
127                 printf(";; res_query(%s, %d, %d)\n", name, class, type);
128 #endif
129
130         n = res_mkquery(QUERY, name, class, type, NULL, 0, NULL,
131                         buf, sizeof(buf));
132         if (n > 0 && (_res.options & RES_USE_EDNS0) != 0)
133                 n = res_opt(n, buf, sizeof(buf), anslen);
134         if (n <= 0) {
135 #ifdef DEBUG
136                 if (_res.options & RES_DEBUG)
137                         printf(";; res_query: mkquery failed\n");
138 #endif
139                 h_errno = NO_RECOVERY;
140                 return (n);
141         }
142         n = res_send(buf, n, answer, anslen);
143         if (n < 0) {
144 #ifdef DEBUG
145                 if (_res.options & RES_DEBUG)
146                         printf(";; res_query: send error\n");
147 #endif
148                 h_errno = TRY_AGAIN;
149                 return (n);
150         }
151
152         if (hp->rcode != NOERROR || ntohs(hp->ancount) == 0) {
153 #ifdef DEBUG
154                 if (_res.options & RES_DEBUG)
155                         printf(";; rcode = %d, ancount=%d\n", hp->rcode,
156                             ntohs(hp->ancount));
157 #endif
158                 switch (hp->rcode) {
159                 case NXDOMAIN:
160                         h_errno = HOST_NOT_FOUND;
161                         break;
162                 case SERVFAIL:
163                         h_errno = TRY_AGAIN;
164                         break;
165                 case NOERROR:
166                         h_errno = NO_DATA;
167                         break;
168                 case FORMERR:
169                 case NOTIMP:
170                 case REFUSED:
171                 default:
172                         h_errno = NO_RECOVERY;
173                         break;
174                 }
175                 return (-1);
176         }
177         return (n);
178 }
179
180 /*
181  * Formulate a normal query, send, and retrieve answer in supplied buffer.
182  * Return the size of the response on success, -1 on error.
183  * If enabled, implement search rules until answer or unrecoverable failure
184  * is detected.  Error code, if any, is left in h_errno.
185  */
186 int
187 res_search(name, class, type, answer, anslen)
188         const char *name;       /* domain name */
189         int class, type;        /* class and type of query */
190         u_char *answer;         /* buffer to put answer */
191         int anslen;             /* size of answer */
192 {
193         const char *cp, * const *domain;
194         HEADER *hp = (HEADER *) answer;
195         u_int dots;
196         int trailing_dot, ret, saved_herrno;
197         int got_nodata = 0, got_servfail = 0, tried_as_is = 0;
198
199         if ((_res.options & RES_INIT) == 0 && res_init() == -1) {
200                 h_errno = NETDB_INTERNAL;
201                 return (-1);
202         }
203         errno = 0;
204         h_errno = HOST_NOT_FOUND;       /* default, if we never query */
205         dots = 0;
206         for (cp = name; *cp; cp++)
207                 dots += (*cp == '.');
208         trailing_dot = 0;
209         if (cp > name && *--cp == '.')
210                 trailing_dot++;
211
212         /* If there aren't any dots, it could be a user-level alias */
213         if (!dots && (cp = hostalias(name)) != NULL)
214                 return (res_query(cp, class, type, answer, anslen));
215
216         /*
217          * If there are dots in the name already, let's just give it a try
218          * 'as is'.  The threshold can be set with the "ndots" option.
219          */
220         saved_herrno = -1;
221         if (dots >= _res.ndots) {
222                 ret = res_querydomain(name, NULL, class, type, answer, anslen);
223                 if (ret > 0)
224                         return (ret);
225                 saved_herrno = h_errno;
226                 tried_as_is++;
227         }
228
229         /*
230          * We do at least one level of search if
231          *      - there is no dot and RES_DEFNAME is set, or
232          *      - there is at least one dot, there is no trailing dot,
233          *        and RES_DNSRCH is set.
234          */
235         if ((!dots && (_res.options & RES_DEFNAMES)) ||
236             (dots && !trailing_dot && (_res.options & RES_DNSRCH))) {
237                 int done = 0;
238
239                 for (domain = (const char * const *)_res.dnsrch;
240                      *domain && !done;
241                      domain++) {
242
243                         ret = res_querydomain(name, *domain, class, type,
244                                               answer, anslen);
245                         if (ret > 0)
246                                 return (ret);
247
248                         /*
249                          * If no server present, give up.
250                          * If name isn't found in this domain,
251                          * keep trying higher domains in the search list
252                          * (if that's enabled).
253                          * On a NO_DATA error, keep trying, otherwise
254                          * a wildcard entry of another type could keep us
255                          * from finding this entry higher in the domain.
256                          * If we get some other error (negative answer or
257                          * server failure), then stop searching up,
258                          * but try the input name below in case it's
259                          * fully-qualified.
260                          */
261                         if (errno == ECONNREFUSED) {
262                                 h_errno = TRY_AGAIN;
263                                 return (-1);
264                         }
265
266                         switch (h_errno) {
267                         case NO_DATA:
268                                 got_nodata++;
269                                 /* FALLTHROUGH */
270                         case HOST_NOT_FOUND:
271                                 /* keep trying */
272                                 break;
273                         case TRY_AGAIN:
274                                 if (hp->rcode == SERVFAIL) {
275                                         /* try next search element, if any */
276                                         got_servfail++;
277                                         break;
278                                 }
279                                 /* FALLTHROUGH */
280                         default:
281                                 /* anything else implies that we're done */
282                                 done++;
283                         }
284
285                         /* if we got here for some reason other than DNSRCH,
286                          * we only wanted one iteration of the loop, so stop.
287                          */
288                         if (!(_res.options & RES_DNSRCH))
289                                 done++;
290                 }
291         }
292
293         /*
294          * If we have not already tried the name "as is", do that now.
295          * note that we do this regardless of how many dots were in the
296          * name or whether it ends with a dot unless NOTLDQUERY is set.
297          */
298         if (!tried_as_is && (dots || !(_res.options & RES_NOTLDQUERY))) {
299                 ret = res_querydomain(name, NULL, class, type, answer, anslen);
300                 if (ret > 0)
301                         return (ret);
302         }
303
304         /* if we got here, we didn't satisfy the search.
305          * if we did an initial full query, return that query's h_errno
306          * (note that we wouldn't be here if that query had succeeded).
307          * else if we ever got a nodata, send that back as the reason.
308          * else send back meaningless h_errno, that being the one from
309          * the last DNSRCH we did.
310          */
311         if (saved_herrno != -1)
312                 h_errno = saved_herrno;
313         else if (got_nodata)
314                 h_errno = NO_DATA;
315         else if (got_servfail)
316                 h_errno = TRY_AGAIN;
317         return (-1);
318 }
319
320 /*
321  * Perform a call on res_query on the concatenation of name and domain,
322  * removing a trailing dot from name if domain is NULL.
323  */
324 int
325 res_querydomain(name, domain, class, type, answer, anslen)
326         const char *name, *domain;
327         int class, type;        /* class and type of query */
328         u_char *answer;         /* buffer to put answer */
329         int anslen;             /* size of answer */
330 {
331         char nbuf[MAXDNAME];
332         const char *longname = nbuf;
333         int n, d;
334
335         if ((_res.options & RES_INIT) == 0 && res_init() == -1) {
336                 h_errno = NETDB_INTERNAL;
337                 return (-1);
338         }
339 #ifdef DEBUG
340         if (_res.options & RES_DEBUG)
341                 printf(";; res_querydomain(%s, %s, %d, %d)\n",
342                        name, domain?domain:"<Nil>", class, type);
343 #endif
344         if (domain == NULL) {
345                 /*
346                  * Check for trailing '.';
347                  * copy without '.' if present.
348                  */
349                 n = strlen(name);
350                 if (n >= MAXDNAME) {
351                         h_errno = NO_RECOVERY;
352                         return (-1);
353                 }
354                 n--;
355                 if (n >= 0 && name[n] == '.') {
356                         strncpy(nbuf, name, n);
357                         nbuf[n] = '\0';
358                 } else
359                         longname = name;
360         } else {
361                 n = strlen(name);
362                 d = strlen(domain);
363                 if (n + d + 1 >= MAXDNAME) {
364                         h_errno = NO_RECOVERY;
365                         return (-1);
366                 }
367                 sprintf(nbuf, "%s.%s", name, domain);
368         }
369         return (res_query(longname, class, type, answer, anslen));
370 }
371
372 const char *
373 hostalias(name)
374         const char *name;
375 {
376         register char *cp1, *cp2;
377         FILE *fp;
378         char *file;
379         char buf[BUFSIZ];
380         static char abuf[MAXDNAME];
381
382         if (_res.options & RES_NOALIASES)
383                 return (NULL);
384         if (issetugid())
385                 return (NULL);
386         file = getenv("HOSTALIASES");
387         if (file == NULL || (fp = fopen(file, "r")) == NULL)
388                 return (NULL);
389         setbuf(fp, NULL);
390         buf[sizeof(buf) - 1] = '\0';
391         while (fgets(buf, sizeof(buf), fp)) {
392                 for (cp1 = buf; *cp1 && !isspace((unsigned char)*cp1); ++cp1)
393                         ;
394                 if (!*cp1)
395                         break;
396                 *cp1 = '\0';
397                 if (!strcasecmp(buf, name)) {
398                         while (isspace((unsigned char)*++cp1))
399                                 ;
400                         if (!*cp1)
401                                 break;
402                         for (cp2 = cp1 + 1; *cp2 && !isspace((unsigned char)*cp2); ++cp2)
403                                 ;
404                         abuf[sizeof(abuf) - 1] = *cp2 = '\0';
405                         strncpy(abuf, cp1, sizeof(abuf) - 1);
406                         fclose(fp);
407                         return (abuf);
408                 }
409         }
410         fclose(fp);
411         return (NULL);
412 }
413
414 /*
415  * Weak aliases for applications that use certain private entry points,
416  * and fail to include <resolv.h>.
417  */
418 #undef res_query
419 __weak_reference(__res_query, res_query);
420 #undef res_search
421 __weak_reference(__res_search, res_search);
422 #undef res_querydomain
423 __weak_reference(__res_querydomain, res_querydomain);