blob: bb86723fb092afa26d6763be79d6e00d34856d3c [file] [log] [blame]
Bernie Innocenti318ed2d2018-08-30 04:05:20 +09001/* $NetBSD: res_init.c,v 1.8 2006/03/19 03:10:08 christos Exp $ */
2
3/*
4 * Copyright (c) 1985, 1989, 1993
5 * The Regents of the University of California. All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3. All advertising materials mentioning features or use of this software
16 * must display the following acknowledgement:
17 * This product includes software developed by the University of
18 * California, Berkeley and its contributors.
19 * 4. Neither the name of the University nor the names of its contributors
20 * may be used to endorse or promote products derived from this software
21 * without specific prior written permission.
22 *
23 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
24 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
25 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
27 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
28 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
29 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
30 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
32 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
33 * SUCH DAMAGE.
34 */
35
36/*
37 * Portions Copyright (c) 1993 by Digital Equipment Corporation.
38 *
39 * Permission to use, copy, modify, and distribute this software for any
40 * purpose with or without fee is hereby granted, provided that the above
41 * copyright notice and this permission notice appear in all copies, and that
42 * the name of Digital Equipment Corporation not be used in advertising or
43 * publicity pertaining to distribution of the document or software without
44 * specific, written prior permission.
45 *
46 * THE SOFTWARE IS PROVIDED "AS IS" AND DIGITAL EQUIPMENT CORP. DISCLAIMS ALL
47 * WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES
48 * OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL DIGITAL EQUIPMENT
49 * CORPORATION BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
50 * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
51 * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
52 * ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
53 * SOFTWARE.
54 */
55
56/*
57 * Copyright (c) 2004 by Internet Systems Consortium, Inc. ("ISC")
58 * Portions Copyright (c) 1996-1999 by Internet Software Consortium.
59 *
60 * Permission to use, copy, modify, and distribute this software for any
61 * purpose with or without fee is hereby granted, provided that the above
62 * copyright notice and this permission notice appear in all copies.
63 *
64 * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES
65 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
66 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR
67 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
68 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
69 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
70 * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
71 */
Bernie Innocenti3952ccc2019-03-03 19:39:53 +090072
Ken Chen5471dca2019-04-15 15:25:35 +080073#define LOG_TAG "resolv"
Bernie Innocenti3952ccc2019-03-03 19:39:53 +090074
Bernie Innocenti318ed2d2018-08-30 04:05:20 +090075#include <sys/param.h>
76#include <sys/socket.h>
77#include <sys/time.h>
Bernie Innocenti8ad893f2018-08-31 14:09:46 +090078#include <sys/types.h>
Bernie Innocenti318ed2d2018-08-30 04:05:20 +090079
Bernie Innocenti318ed2d2018-08-30 04:05:20 +090080#include <arpa/inet.h>
81#include <arpa/nameser.h>
Bernie Innocenti8ad893f2018-08-31 14:09:46 +090082#include <netinet/in.h>
Bernie Innocenti318ed2d2018-08-30 04:05:20 +090083
chenbruceacb832c2019-02-20 19:45:50 +080084#include <android-base/logging.h>
Bernie Innocenti318ed2d2018-08-30 04:05:20 +090085#include <ctype.h>
Bernie Innocentif33a63f2018-08-30 12:04:03 +090086#include <errno.h>
87#include <fcntl.h>
Bernie Innocenti8ad893f2018-08-31 14:09:46 +090088#include <netdb.h>
Bernie Innocenti318ed2d2018-08-30 04:05:20 +090089#include <stdlib.h>
90#include <string.h>
91#include <unistd.h>
Bernie Innocenti318ed2d2018-08-30 04:05:20 +090092
Bernie Innocentiac18b122018-10-01 23:10:18 +090093#include "netd_resolv/resolv.h"
94#include "res_state_ext.h"
Bernie Innocenti318ed2d2018-08-30 04:05:20 +090095#include "resolv_private.h"
Bernie Innocenti318ed2d2018-08-30 04:05:20 +090096
Bernie Innocenti318ed2d2018-08-30 04:05:20 +090097/*
98 * Resolver state default settings.
99 */
100
101/*
102 * Set up default settings. If the configuration file exist, the values
103 * there will have precedence. Otherwise, the server address is set to
104 * INADDR_ANY and the default domain name comes from the gethostname().
105 *
106 * An interrim version of this code (BIND 4.9, pre-4.4BSD) used 127.0.0.1
107 * rather than INADDR_ANY ("0.0.0.0") as the default name server address
108 * since it was noted that INADDR_ANY actually meant ``the first interface
109 * you "ifconfig"'d at boot time'' and if this was a SLIP or PPP interface,
110 * it had to be "up" in order for you to reach your own name server. It
111 * was later decided that since the recommended practice is to always
112 * install local static routes through 127.0.0.1 for all your network
113 * interfaces, that we could solve this problem without a code change.
114 *
115 * The configuration file should always be used, since it is the only way
116 * to specify a default domain. If you are running a server on your local
117 * machine, you should say "nameserver 0.0.0.0" or "nameserver 127.0.0.1"
118 * in the configuration file.
119 *
120 * Return 0 if completes successfully, -1 on error
121 */
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900122int res_ninit(res_state statp) {
Bernie Innocenti2e8540b2018-09-26 11:52:04 +0900123 return res_vinit(statp, 0);
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900124}
125
126/* This function has to be reachable by res_data.c but not publicly. */
Bernie Innocenti2e8540b2018-09-26 11:52:04 +0900127int res_vinit(res_state statp, int preinit) {
Bernie Innocenti9c575932018-09-07 21:10:25 +0900128 char *cp, **pp;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900129 char buf[BUFSIZ];
130 int nserv = 0; /* number of nameserver records read from file */
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900131 int havesearch = 0;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900132 int dots;
nuccachenb980f2f2018-10-23 17:10:58 +0800133 sockaddr_union u[2];
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900134
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900135 if ((statp->options & RES_INIT) != 0U) res_ndestroy(statp);
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900136
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900137 if (!preinit) {
138 statp->netid = NETID_UNSET;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900139 statp->options = RES_DEFAULT;
Luke Huangdda920f2019-02-13 14:05:02 +0800140 statp->id = arc4random_uniform(65536);
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900141 statp->_mark = MARK_UNSET;
142 }
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900143
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900144 memset(u, 0, sizeof(u));
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900145 u[nserv].sin.sin_addr.s_addr = INADDR_ANY;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900146 u[nserv].sin.sin_family = AF_INET;
147 u[nserv].sin.sin_port = htons(NAMESERVER_PORT);
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900148 nserv++;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900149 statp->nscount = 0;
150 statp->ndots = 1;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900151 statp->_vcsock = -1;
152 statp->_flags = 0;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900153 statp->_u._ext.nscount = 0;
Bernie Innocentiac18b122018-10-01 23:10:18 +0900154 statp->_u._ext.ext = (res_state_ext*) malloc(sizeof(*statp->_u._ext.ext));
Mike Yub601ff72018-11-01 20:07:00 +0800155 statp->use_local_nameserver = false;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900156 if (statp->_u._ext.ext != NULL) {
157 memset(statp->_u._ext.ext, 0, sizeof(*statp->_u._ext.ext));
158 statp->_u._ext.ext->nsaddrs[0].sin = statp->nsaddr;
159 strcpy(statp->_u._ext.ext->nsuffix, "ip6.arpa");
160 strcpy(statp->_u._ext.ext->nsuffix2, "ip6.int");
161 }
162 statp->nsort = 0;
163 res_setservers(statp, u, nserv);
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900164
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900165 if (statp->defdname[0] == 0 && gethostname(buf, sizeof(statp->defdname) - 1) == 0 &&
166 (cp = strchr(buf, '.')) != NULL)
167 strcpy(statp->defdname, cp + 1);
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900168
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900169 /* find components of local domain that might be searched */
170 if (havesearch == 0) {
171 pp = statp->dnsrch;
172 *pp++ = statp->defdname;
173 *pp = NULL;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900174
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900175 dots = 0;
176 for (cp = statp->defdname; *cp; cp++) dots += (*cp == '.');
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900177
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900178 cp = statp->defdname;
179 while (pp < statp->dnsrch + MAXDFLSRCH) {
180 if (dots < LOCALDOMAINPARTS) break;
181 cp = strchr(cp, '.') + 1; /* we know there is one */
182 *pp++ = cp;
183 dots--;
184 }
185 *pp = NULL;
Ken Chenffc224a2019-03-19 17:41:28 +0800186 LOG(DEBUG) << __func__ << ": dnsrch list:";
Bernie Innocenti3952ccc2019-03-03 19:39:53 +0900187 for (pp = statp->dnsrch; *pp; pp++) LOG(DEBUG) << "\t" << *pp;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900188 }
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900189
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900190 if (nserv > 0) {
191 statp->nscount = nserv;
192 statp->options |= RES_INIT;
193 }
194 return (0);
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900195}
196
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900197
Bernie Innocentif33a63f2018-08-30 12:04:03 +0900198/*
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900199 * This routine is for closing the socket if a virtual circuit is used and
200 * the program wants to close it. This provides support for endhostent()
201 * which expects to close the socket.
202 *
203 * This routine is not expected to be user visible.
204 */
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900205void res_nclose(res_state statp) {
206 int ns;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900207
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900208 if (statp->_vcsock >= 0) {
209 (void) close(statp->_vcsock);
210 statp->_vcsock = -1;
Bernie Innocenti9c2c06f2019-01-08 18:36:28 +0900211 statp->_flags &= ~RES_F_VC;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900212 }
213 for (ns = 0; ns < statp->_u._ext.nscount; ns++) {
214 if (statp->_u._ext.nssocks[ns] != -1) {
215 (void) close(statp->_u._ext.nssocks[ns]);
216 statp->_u._ext.nssocks[ns] = -1;
217 }
218 }
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900219}
220
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900221void res_ndestroy(res_state statp) {
222 res_nclose(statp);
223 if (statp->_u._ext.ext != NULL) free(statp->_u._ext.ext);
224 statp->options &= ~RES_INIT;
225 statp->_u._ext.ext = NULL;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900226}
227
nuccachenb980f2f2018-10-23 17:10:58 +0800228void res_setservers(res_state statp, const sockaddr_union* set, int cnt) {
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900229 int i, nserv;
230 size_t size;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900231
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900232 /* close open servers */
233 res_nclose(statp);
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900234
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900235 /* cause rtt times to be forgotten */
236 statp->_u._ext.nscount = 0;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900237
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900238 nserv = 0;
239 for (i = 0; i < cnt && nserv < MAXNS; i++) {
240 switch (set->sin.sin_family) {
241 case AF_INET:
242 size = sizeof(set->sin);
243 if (statp->_u._ext.ext)
244 memcpy(&statp->_u._ext.ext->nsaddrs[nserv], &set->sin, size);
245 if (size <= sizeof(statp->nsaddr_list[nserv]))
246 memcpy(&statp->nsaddr_list[nserv], &set->sin, size);
247 else
248 statp->nsaddr_list[nserv].sin_family = 0;
249 nserv++;
250 break;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900251
252#ifdef HAS_INET6_STRUCTS
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900253 case AF_INET6:
254 size = sizeof(set->sin6);
255 if (statp->_u._ext.ext)
256 memcpy(&statp->_u._ext.ext->nsaddrs[nserv], &set->sin6, size);
257 if (size <= sizeof(statp->nsaddr_list[nserv]))
258 memcpy(&statp->nsaddr_list[nserv], &set->sin6, size);
259 else
260 statp->nsaddr_list[nserv].sin_family = 0;
261 nserv++;
262 break;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900263#endif
264
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900265 default:
266 break;
267 }
268 set++;
269 }
270 statp->nscount = nserv;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900271}
272
nuccachenb980f2f2018-10-23 17:10:58 +0800273int res_getservers(res_state statp, sockaddr_union* set, int cnt) {
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900274 int i;
275 size_t size;
276 uint16_t family;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900277
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900278 for (i = 0; i < statp->nscount && i < cnt; i++) {
279 if (statp->_u._ext.ext)
280 family = statp->_u._ext.ext->nsaddrs[i].sin.sin_family;
281 else
282 family = statp->nsaddr_list[i].sin_family;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900283
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900284 switch (family) {
285 case AF_INET:
286 size = sizeof(set->sin);
287 if (statp->_u._ext.ext)
288 memcpy(&set->sin, &statp->_u._ext.ext->nsaddrs[i], size);
289 else
290 memcpy(&set->sin, &statp->nsaddr_list[i], size);
291 break;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900292
293#ifdef HAS_INET6_STRUCTS
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900294 case AF_INET6:
295 size = sizeof(set->sin6);
296 if (statp->_u._ext.ext)
297 memcpy(&set->sin6, &statp->_u._ext.ext->nsaddrs[i], size);
298 else
299 memcpy(&set->sin6, &statp->nsaddr_list[i], size);
300 break;
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900301#endif
302
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900303 default:
304 set->sin.sin_family = 0;
305 break;
306 }
307 set++;
308 }
309 return (statp->nscount);
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900310}
311
lifr94981782019-05-17 21:15:19 +0800312void res_setnetcontext(res_state statp, const struct android_net_context* netcontext,
313 android::net::NetworkDnsEventReported* _Nonnull event) {
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900314 if (statp != NULL) {
315 statp->netid = netcontext->dns_netid;
316 statp->_mark = netcontext->dns_mark;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900317 if (netcontext->flags & NET_CONTEXT_FLAG_USE_EDNS) {
318 statp->options |= RES_USE_EDNS0 | RES_USE_DNSSEC;
319 }
Mike Yub601ff72018-11-01 20:07:00 +0800320 if (netcontext->flags & NET_CONTEXT_FLAG_USE_LOCAL_NAMESERVERS) {
321 statp->use_local_nameserver = true;
322 }
lifr94981782019-05-17 21:15:19 +0800323 statp->event = event;
Bernie Innocenti8ad893f2018-08-31 14:09:46 +0900324 }
Bernie Innocenti318ed2d2018-08-30 04:05:20 +0900325}