OSDN Git Service

2003-05-28 Jeff Johnston <jjohnstn@redhat.com>
[pf3gnuchains/pf3gnuchains4x.git] / newlib / libc / sys / linux / 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 #endif /* LIBC_SCCS and not lint */
75 #include <sys/cdefs.h>
76 #include <sys/types.h>
77
78 #include <sys/types.h>
79 #include <sys/param.h>
80 #include <netinet/in.h>
81 #include <arpa/inet.h>
82 #include <arpa/nameser.h>
83 #include <ctype.h>
84 #include <errno.h>
85 #include <netdb.h>
86 #include <resolv.h>
87 #include <stdio.h>
88 #include <stdlib.h>
89 #include <string.h>
90
91 #include "res_config.h"
92
93 #if PACKETSZ > 1024
94 #define MAXPACKET       PACKETSZ
95 #else
96 #define MAXPACKET       1024
97 #endif
98
99 /*
100  * Formulate a normal query, send, and await answer.
101  * Returned answer is placed in supplied buffer "answer".
102  * Perform preliminary check of answer, returning success only
103  * if no error is indicated and the answer count is nonzero.
104  * Return the size of the response on success, -1 on error.
105  * Error number is left in h_errno.
106  *
107  * Caller must parse answer and determine whether it answers the question.
108  */
109 int
110 res_query(name, class, type, answer, anslen)
111         const char *name;       /* domain name */
112         int class, type;        /* class and type of query */
113         u_char *answer;         /* buffer to put answer */
114         int anslen;             /* size of answer buffer */
115 {
116         u_char buf[MAXPACKET];
117         HEADER *hp = (HEADER *) answer;
118         int n;
119
120         hp->rcode = NOERROR;    /* default */
121
122         if ((_res.options & RES_INIT) == 0 && res_init() == -1) {
123                 h_errno = NETDB_INTERNAL;
124                 return (-1);
125         }
126 #ifdef DEBUG
127         if (_res.options & RES_DEBUG)
128                 printf(";; res_query(%s, %d, %d)\n", name, class, type);
129 #endif
130
131         n = res_mkquery(QUERY, name, class, type, NULL, 0, NULL,
132                         buf, sizeof(buf));
133         if (n > 0 && (_res.options & RES_USE_EDNS0) != 0)
134                 n = res_opt(n, buf, sizeof(buf), anslen);
135         if (n <= 0) {
136 #ifdef DEBUG
137                 if (_res.options & RES_DEBUG)
138                         printf(";; res_query: mkquery failed\n");
139 #endif
140                 h_errno = NO_RECOVERY;
141                 return (n);
142         }
143         n = res_send(buf, n, answer, anslen);
144         if (n < 0) {
145 #ifdef DEBUG
146                 if (_res.options & RES_DEBUG)
147                         printf(";; res_query: send error\n");
148 #endif
149                 h_errno = TRY_AGAIN;
150                 return (n);
151         }
152
153         if (hp->rcode != NOERROR || ntohs(hp->ancount) == 0) {
154 #ifdef DEBUG
155                 if (_res.options & RES_DEBUG)
156                         printf(";; rcode = %d, ancount=%d\n", hp->rcode,
157                             ntohs(hp->ancount));
158 #endif
159                 switch (hp->rcode) {
160                 case NXDOMAIN:
161                         h_errno = HOST_NOT_FOUND;
162                         break;
163                 case SERVFAIL:
164                         h_errno = TRY_AGAIN;
165                         break;
166                 case NOERROR:
167                         h_errno = NO_DATA;
168                         break;
169                 case FORMERR:
170                 case NOTIMP:
171                 case REFUSED:
172                 default:
173                         h_errno = NO_RECOVERY;
174                         break;
175                 }
176                 return (-1);
177         }
178         return (n);
179 }
180
181 /*
182  * Formulate a normal query, send, and retrieve answer in supplied buffer.
183  * Return the size of the response on success, -1 on error.
184  * If enabled, implement search rules until answer or unrecoverable failure
185  * is detected.  Error code, if any, is left in h_errno.
186  */
187 int
188 res_search(name, class, type, answer, anslen)
189         const char *name;       /* domain name */
190         int class, type;        /* class and type of query */
191         u_char *answer;         /* buffer to put answer */
192         int anslen;             /* size of answer */
193 {
194         const char *cp, * const *domain;
195         HEADER *hp = (HEADER *) answer;
196         u_int dots;
197         int trailing_dot, ret, saved_herrno;
198         int got_nodata = 0, got_servfail = 0, tried_as_is = 0;
199
200         if ((_res.options & RES_INIT) == 0 && res_init() == -1) {
201                 h_errno = NETDB_INTERNAL;
202                 return (-1);
203         }
204         errno = 0;
205         h_errno = HOST_NOT_FOUND;       /* default, if we never query */
206         dots = 0;
207         for (cp = name; *cp; cp++)
208                 dots += (*cp == '.');
209         trailing_dot = 0;
210         if (cp > name && *--cp == '.')
211                 trailing_dot++;
212
213         /* If there aren't any dots, it could be a user-level alias */
214         if (!dots && (cp = hostalias(name)) != NULL)
215                 return (res_query(cp, class, type, answer, anslen));
216
217         /*
218          * If there are dots in the name already, let's just give it a try
219          * 'as is'.  The threshold can be set with the "ndots" option.
220          */
221         saved_herrno = -1;
222         if (dots >= _res.ndots) {
223                 ret = res_querydomain(name, NULL, class, type, answer, anslen);
224                 if (ret > 0)
225                         return (ret);
226                 saved_herrno = h_errno;
227                 tried_as_is++;
228         }
229
230         /*
231          * We do at least one level of search if
232          *      - there is no dot and RES_DEFNAME is set, or
233          *      - there is at least one dot, there is no trailing dot,
234          *        and RES_DNSRCH is set.
235          */
236         if ((!dots && (_res.options & RES_DEFNAMES)) ||
237             (dots && !trailing_dot && (_res.options & RES_DNSRCH))) {
238                 int done = 0;
239
240                 for (domain = (const char * const *)_res.dnsrch;
241                      *domain && !done;
242                      domain++) {
243
244                         ret = res_querydomain(name, *domain, class, type,
245                                               answer, anslen);
246                         if (ret > 0)
247                                 return (ret);
248
249                         /*
250                          * If no server present, give up.
251                          * If name isn't found in this domain,
252                          * keep trying higher domains in the search list
253                          * (if that's enabled).
254                          * On a NO_DATA error, keep trying, otherwise
255                          * a wildcard entry of another type could keep us
256                          * from finding this entry higher in the domain.
257                          * If we get some other error (negative answer or
258                          * server failure), then stop searching up,
259                          * but try the input name below in case it's
260                          * fully-qualified.
261                          */
262                         if (errno == ECONNREFUSED) {
263                                 h_errno = TRY_AGAIN;
264                                 return (-1);
265                         }
266
267                         switch (h_errno) {
268                         case NO_DATA:
269                                 got_nodata++;
270                                 /* FALLTHROUGH */
271                         case HOST_NOT_FOUND:
272                                 /* keep trying */
273                                 break;
274                         case TRY_AGAIN:
275                                 if (hp->rcode == SERVFAIL) {
276                                         /* try next search element, if any */
277                                         got_servfail++;
278                                         break;
279                                 }
280                                 /* FALLTHROUGH */
281                         default:
282                                 /* anything else implies that we're done */
283                                 done++;
284                         }
285
286                         /* if we got here for some reason other than DNSRCH,
287                          * we only wanted one iteration of the loop, so stop.
288                          */
289                         if (!(_res.options & RES_DNSRCH))
290                                 done++;
291                 }
292         }
293
294         /*
295          * If we have not already tried the name "as is", do that now.
296          * note that we do this regardless of how many dots were in the
297          * name or whether it ends with a dot unless NOTLDQUERY is set.
298          */
299         if (!tried_as_is && (dots || !(_res.options & RES_NOTLDQUERY))) {
300                 ret = res_querydomain(name, NULL, class, type, answer, anslen);
301                 if (ret > 0)
302                         return (ret);
303         }
304
305         /* if we got here, we didn't satisfy the search.
306          * if we did an initial full query, return that query's h_errno
307          * (note that we wouldn't be here if that query had succeeded).
308          * else if we ever got a nodata, send that back as the reason.
309          * else send back meaningless h_errno, that being the one from
310          * the last DNSRCH we did.
311          */
312         if (saved_herrno != -1)
313                 h_errno = saved_herrno;
314         else if (got_nodata)
315                 h_errno = NO_DATA;
316         else if (got_servfail)
317                 h_errno = TRY_AGAIN;
318         return (-1);
319 }
320
321 /*
322  * Perform a call on res_query on the concatenation of name and domain,
323  * removing a trailing dot from name if domain is NULL.
324  */
325 int
326 res_querydomain(name, domain, class, type, answer, anslen)
327         const char *name, *domain;
328         int class, type;        /* class and type of query */
329         u_char *answer;         /* buffer to put answer */
330         int anslen;             /* size of answer */
331 {
332         char nbuf[MAXDNAME];
333         const char *longname = nbuf;
334         int n, d;
335
336         if ((_res.options & RES_INIT) == 0 && res_init() == -1) {
337                 h_errno = NETDB_INTERNAL;
338                 return (-1);
339         }
340 #ifdef DEBUG
341         if (_res.options & RES_DEBUG)
342                 printf(";; res_querydomain(%s, %s, %d, %d)\n",
343                        name, domain?domain:"<Nil>", class, type);
344 #endif
345         if (domain == NULL) {
346                 /*
347                  * Check for trailing '.';
348                  * copy without '.' if present.
349                  */
350                 n = strlen(name);
351                 if (n >= MAXDNAME) {
352                         h_errno = NO_RECOVERY;
353                         return (-1);
354                 }
355                 n--;
356                 if (n >= 0 && name[n] == '.') {
357                         strncpy(nbuf, name, n);
358                         nbuf[n] = '\0';
359                 } else
360                         longname = name;
361         } else {
362                 n = strlen(name);
363                 d = strlen(domain);
364                 if (n + d + 1 >= MAXDNAME) {
365                         h_errno = NO_RECOVERY;
366                         return (-1);
367                 }
368                 sprintf(nbuf, "%s.%s", name, domain);
369         }
370         return (res_query(longname, class, type, answer, anslen));
371 }
372
373 const char *
374 hostalias(name)
375         const char *name;
376 {
377         char *cp1, *cp2;
378         FILE *fp;
379         char *file;
380         char buf[BUFSIZ];
381         static char abuf[MAXDNAME];
382
383         if (_res.options & RES_NOALIASES)
384                 return (NULL);
385         if (issetugid())
386                 return (NULL);
387         file = getenv("HOSTALIASES");
388         if (file == NULL || (fp = fopen(file, "r")) == NULL)
389                 return (NULL);
390         setbuf(fp, NULL);
391         buf[sizeof(buf) - 1] = '\0';
392         while (fgets(buf, sizeof(buf), fp)) {
393                 for (cp1 = buf; *cp1 && !isspace((unsigned char)*cp1); ++cp1)
394                         ;
395                 if (!*cp1)
396                         break;
397                 *cp1 = '\0';
398                 if (!strcasecmp(buf, name)) {
399                         while (isspace((unsigned char)*++cp1))
400                                 ;
401                         if (!*cp1)
402                                 break;
403                         for (cp2 = cp1 + 1; *cp2 && !isspace((unsigned char)*cp2); ++cp2)
404                                 ;
405                         abuf[sizeof(abuf) - 1] = *cp2 = '\0';
406                         strncpy(abuf, cp1, sizeof(abuf) - 1);
407                         fclose(fp);
408                         return (abuf);
409                 }
410         }
411         fclose(fp);
412         return (NULL);
413 }
414
415 /*
416  * Weak aliases for applications that use certain private entry points,
417  * and fail to include <resolv.h>.
418  */
419 #undef res_query
420 __weak_reference(__res_query, res_query);
421 #undef res_search
422 __weak_reference(__res_search, res_search);
423 #undef res_querydomain
424 __weak_reference(__res_querydomain, res_querydomain);