res_data.c revision 1.8 1 1.8 christos /* $NetBSD: res_data.c,v 1.8 2004/06/09 18:07:03 christos Exp $ */
2 1.1 christos
3 1.1 christos /*
4 1.1 christos * Copyright (c) 2004 by Internet Systems Consortium, Inc. ("ISC")
5 1.1 christos * Copyright (c) 1995-1999 by Internet Software Consortium.
6 1.1 christos *
7 1.1 christos * Permission to use, copy, modify, and distribute this software for any
8 1.1 christos * purpose with or without fee is hereby granted, provided that the above
9 1.1 christos * copyright notice and this permission notice appear in all copies.
10 1.1 christos *
11 1.1 christos * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES
12 1.1 christos * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
13 1.1 christos * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR
14 1.1 christos * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
15 1.1 christos * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
16 1.1 christos * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
17 1.1 christos * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
18 1.1 christos */
19 1.1 christos
20 1.4 christos #include <sys/cdefs.h>
21 1.1 christos #if defined(LIBC_SCCS) && !defined(lint)
22 1.4 christos #ifdef notdef
23 1.1 christos static const char rcsid[] = "Id: res_data.c,v 1.1.206.2 2004/03/16 12:34:18 marka Exp";
24 1.4 christos #else
25 1.8 christos __RCSID("$NetBSD: res_data.c,v 1.8 2004/06/09 18:07:03 christos Exp $");
26 1.4 christos #endif
27 1.1 christos #endif /* LIBC_SCCS and not lint */
28 1.1 christos
29 1.1 christos #include "port_before.h"
30 1.1 christos
31 1.5 christos #include "namespace.h"
32 1.1 christos #include <sys/types.h>
33 1.1 christos #include <sys/param.h>
34 1.1 christos #include <sys/socket.h>
35 1.1 christos #include <sys/time.h>
36 1.1 christos
37 1.1 christos #include <netinet/in.h>
38 1.1 christos #include <arpa/inet.h>
39 1.1 christos #include <arpa/nameser.h>
40 1.1 christos
41 1.1 christos #include <ctype.h>
42 1.1 christos #include <netdb.h>
43 1.1 christos #include <resolv.h>
44 1.1 christos #include <res_update.h>
45 1.1 christos #include <stdio.h>
46 1.1 christos #include <stdlib.h>
47 1.1 christos #include <string.h>
48 1.1 christos #include <unistd.h>
49 1.1 christos
50 1.1 christos #include "port_after.h"
51 1.5 christos
52 1.5 christos #ifdef __weak_alias
53 1.5 christos __weak_alias(res_init,_res_init)
54 1.5 christos __weak_alias(res_mkquery,_res_mkquery)
55 1.7 christos __weak_alias(res_query,_res_query)
56 1.7 christos __weak_alias(res_search,_res_search)
57 1.7 christos __weak_alias(res_send,__res_send)
58 1.7 christos __weak_alias(res_close,__res_close)
59 1.7 christos /* XXX: these leaked in the old bind8 libc */
60 1.7 christos __weak_alias(res_querydomain,__res_querydomain)
61 1.5 christos __weak_alias(res_send_setqhook,__res_send_setqhook)
62 1.5 christos __weak_alias(res_send_setrhook,__res_send_setrhook)
63 1.7 christos #if 0
64 1.7 christos __weak_alias(p_query,__p_query)
65 1.7 christos __weak_alias(fp_query,__fp_query)
66 1.7 christos __weak_alias(fp_nquery,__fp_nquery)
67 1.5 christos __weak_alias(res_isourserver,__res_isourserver)
68 1.5 christos __weak_alias(res_opt,_res_opt)
69 1.5 christos __weak_alias(hostalias,__hostalias)
70 1.5 christos #endif
71 1.6 christos #endif
72 1.5 christos
73 1.1 christos const char *_res_opcodes[] = {
74 1.1 christos "QUERY",
75 1.1 christos "IQUERY",
76 1.1 christos "CQUERYM",
77 1.1 christos "CQUERYU", /* experimental */
78 1.1 christos "NOTIFY", /* experimental */
79 1.1 christos "UPDATE",
80 1.1 christos "6",
81 1.1 christos "7",
82 1.1 christos "8",
83 1.1 christos "9",
84 1.1 christos "10",
85 1.1 christos "11",
86 1.1 christos "12",
87 1.1 christos "13",
88 1.1 christos "ZONEINIT",
89 1.1 christos "ZONEREF",
90 1.1 christos };
91 1.1 christos
92 1.1 christos #ifdef BIND_UPDATE
93 1.1 christos const char *_res_sectioncodes[] = {
94 1.1 christos "ZONE",
95 1.1 christos "PREREQUISITES",
96 1.1 christos "UPDATE",
97 1.1 christos "ADDITIONAL",
98 1.1 christos };
99 1.1 christos #endif
100 1.1 christos
101 1.8 christos #ifndef __BIND_NOSTATIC
102 1.8 christos extern struct __res_state _nres;
103 1.1 christos
104 1.1 christos /* Proto. */
105 1.1 christos
106 1.2 christos int res_ourserver_p(const res_state, const struct sockaddr *);
107 1.1 christos
108 1.1 christos int
109 1.1 christos res_init(void) {
110 1.8 christos int rv;
111 1.1 christos extern int __res_vinit(res_state, int);
112 1.8 christos #ifdef COMPAT__RES
113 1.8 christos /*
114 1.8 christos * Compatibility with program that were accessing _res directly
115 1.8 christos * to set options. We keep another struct res that is the same
116 1.8 christos * size as the original res structure, and then copy fields to
117 1.8 christos * it so that we achieve the same initialization
118 1.8 christos */
119 1.8 christos extern void *__res_get_old_state(void);
120 1.8 christos extern void __res_put_old_state(void *);
121 1.8 christos res_state ores = __res_get_old_state();
122 1.8 christos
123 1.8 christos if (ores->options != 0)
124 1.8 christos _nres.options = ores->options;
125 1.8 christos if (ores->retrans != 0)
126 1.8 christos _nres.retrans = ores->retrans;
127 1.8 christos if (ores->retry != 0)
128 1.8 christos _nres.retry = ores->retry;
129 1.8 christos #endif
130 1.1 christos
131 1.1 christos /*
132 1.1 christos * These three fields used to be statically initialized. This made
133 1.1 christos * it hard to use this code in a shared library. It is necessary,
134 1.1 christos * now that we're doing dynamic initialization here, that we preserve
135 1.1 christos * the old semantics: if an application modifies one of these three
136 1.1 christos * fields of _res before res_init() is called, res_init() will not
137 1.1 christos * alter them. Of course, if an application is setting them to
138 1.1 christos * _zero_ before calling res_init(), hoping to override what used
139 1.1 christos * to be the static default, we can't detect it and unexpected results
140 1.1 christos * will follow. Zero for any of these fields would make no sense,
141 1.1 christos * so one can safely assume that the applications were already getting
142 1.1 christos * unexpected results.
143 1.1 christos *
144 1.8 christos * _nres.options is tricky since some apps were known to diddle the bits
145 1.1 christos * before res_init() was first called. We can't replicate that semantic
146 1.1 christos * with dynamic initialization (they may have turned bits off that are
147 1.1 christos * set in RES_DEFAULT). Our solution is to declare such applications
148 1.1 christos * "broken". They could fool us by setting RES_INIT but none do (yet).
149 1.1 christos */
150 1.8 christos if (!_nres.retrans)
151 1.8 christos _nres.retrans = RES_TIMEOUT;
152 1.8 christos if (!_nres.retry)
153 1.8 christos _nres.retry = 4;
154 1.8 christos if (!(_nres.options & RES_INIT))
155 1.8 christos _nres.options = RES_DEFAULT;
156 1.1 christos
157 1.1 christos /*
158 1.1 christos * This one used to initialize implicitly to zero, so unless the app
159 1.1 christos * has set it to something in particular, we can randomize it now.
160 1.1 christos */
161 1.8 christos if (!_nres.id)
162 1.8 christos _nres.id = res_randomid();
163 1.1 christos
164 1.8 christos rv = __res_vinit(&_nres, 1);
165 1.8 christos #ifdef COMPAT__RES
166 1.8 christos __res_put_old_state(&_nres);
167 1.8 christos #endif
168 1.8 christos return rv;
169 1.1 christos }
170 1.1 christos
171 1.1 christos void
172 1.1 christos p_query(const u_char *msg) {
173 1.1 christos fp_query(msg, stdout);
174 1.1 christos }
175 1.1 christos
176 1.1 christos void
177 1.1 christos fp_query(const u_char *msg, FILE *file) {
178 1.1 christos fp_nquery(msg, PACKETSZ, file);
179 1.1 christos }
180 1.1 christos
181 1.1 christos void
182 1.1 christos fp_nquery(const u_char *msg, int len, FILE *file) {
183 1.8 christos if ((_nres.options & RES_INIT) == 0U && res_init() == -1)
184 1.1 christos return;
185 1.1 christos
186 1.8 christos res_pquery(&_nres, msg, len, file);
187 1.1 christos }
188 1.1 christos
189 1.1 christos int
190 1.1 christos res_mkquery(int op, /* opcode of query */
191 1.1 christos const char *dname, /* domain name */
192 1.1 christos int class, int type, /* class and type of query */
193 1.1 christos const u_char *data, /* resource record data */
194 1.1 christos int datalen, /* length of data */
195 1.1 christos const u_char *newrr_in, /* new rr for modify or append */
196 1.1 christos u_char *buf, /* buffer to put query */
197 1.1 christos int buflen) /* size of buffer */
198 1.1 christos {
199 1.8 christos if ((_nres.options & RES_INIT) == 0U && res_init() == -1) {
200 1.8 christos RES_SET_H_ERRNO(&_nres, NETDB_INTERNAL);
201 1.1 christos return (-1);
202 1.1 christos }
203 1.8 christos return (res_nmkquery(&_nres, op, dname, class, type,
204 1.1 christos data, datalen,
205 1.1 christos newrr_in, buf, buflen));
206 1.1 christos }
207 1.1 christos
208 1.2 christos #ifdef _LIBRESOLV
209 1.1 christos int
210 1.1 christos res_mkupdate(ns_updrec *rrecp_in, u_char *buf, int buflen) {
211 1.8 christos if ((_nres.options & RES_INIT) == 0U && res_init() == -1) {
212 1.8 christos RES_SET_H_ERRNO(&_nres, NETDB_INTERNAL);
213 1.1 christos return (-1);
214 1.1 christos }
215 1.1 christos
216 1.8 christos return (res_nmkupdate(&_nres, rrecp_in, buf, buflen));
217 1.1 christos }
218 1.2 christos #endif
219 1.1 christos
220 1.1 christos int
221 1.1 christos res_query(const char *name, /* domain name */
222 1.1 christos int class, int type, /* class and type of query */
223 1.1 christos u_char *answer, /* buffer to put answer */
224 1.1 christos int anslen) /* size of answer buffer */
225 1.1 christos {
226 1.8 christos if ((_nres.options & RES_INIT) == 0U && res_init() == -1) {
227 1.8 christos RES_SET_H_ERRNO(&_nres, NETDB_INTERNAL);
228 1.1 christos return (-1);
229 1.1 christos }
230 1.8 christos return (res_nquery(&_nres, name, class, type, answer, anslen));
231 1.1 christos }
232 1.1 christos
233 1.1 christos void
234 1.1 christos res_send_setqhook(res_send_qhook hook) {
235 1.8 christos _nres.qhook = hook;
236 1.1 christos }
237 1.1 christos
238 1.1 christos void
239 1.1 christos res_send_setrhook(res_send_rhook hook) {
240 1.8 christos _nres.rhook = hook;
241 1.1 christos }
242 1.1 christos
243 1.1 christos int
244 1.1 christos res_isourserver(const struct sockaddr_in *inp) {
245 1.8 christos return (res_ourserver_p(&_nres, (const struct sockaddr *)(const void *)inp));
246 1.1 christos }
247 1.1 christos
248 1.1 christos int
249 1.1 christos res_send(const u_char *buf, int buflen, u_char *ans, int anssiz) {
250 1.8 christos if ((_nres.options & RES_INIT) == 0U && res_init() == -1) {
251 1.1 christos /* errno should have been set by res_init() in this case. */
252 1.1 christos return (-1);
253 1.1 christos }
254 1.1 christos
255 1.8 christos return (res_nsend(&_nres, buf, buflen, ans, anssiz));
256 1.1 christos }
257 1.1 christos
258 1.2 christos #ifdef _LIBRESOLV
259 1.1 christos int
260 1.1 christos res_sendsigned(const u_char *buf, int buflen, ns_tsig_key *key,
261 1.1 christos u_char *ans, int anssiz)
262 1.1 christos {
263 1.8 christos if ((_nres.options & RES_INIT) == 0U && res_init() == -1) {
264 1.1 christos /* errno should have been set by res_init() in this case. */
265 1.1 christos return (-1);
266 1.1 christos }
267 1.1 christos
268 1.8 christos return (res_nsendsigned(&_nres, buf, buflen, key, ans, anssiz));
269 1.1 christos }
270 1.2 christos #endif
271 1.1 christos
272 1.1 christos void
273 1.1 christos res_close(void) {
274 1.8 christos res_nclose(&_nres);
275 1.1 christos }
276 1.1 christos
277 1.2 christos #ifdef _LIBRESOLV
278 1.1 christos int
279 1.1 christos res_update(ns_updrec *rrecp_in) {
280 1.8 christos if ((_nres.options & RES_INIT) == 0U && res_init() == -1) {
281 1.8 christos RES_SET_H_ERRNO(&_nres, NETDB_INTERNAL);
282 1.1 christos return (-1);
283 1.1 christos }
284 1.1 christos
285 1.8 christos return (res_nupdate(&_nres, rrecp_in, NULL));
286 1.1 christos }
287 1.2 christos #endif
288 1.1 christos
289 1.1 christos int
290 1.1 christos res_search(const char *name, /* domain name */
291 1.1 christos int class, int type, /* class and type of query */
292 1.1 christos u_char *answer, /* buffer to put answer */
293 1.1 christos int anslen) /* size of answer */
294 1.1 christos {
295 1.8 christos if ((_nres.options & RES_INIT) == 0U && res_init() == -1) {
296 1.8 christos RES_SET_H_ERRNO(&_nres, NETDB_INTERNAL);
297 1.1 christos return (-1);
298 1.1 christos }
299 1.1 christos
300 1.8 christos return (res_nsearch(&_nres, name, class, type, answer, anslen));
301 1.1 christos }
302 1.1 christos
303 1.1 christos int
304 1.1 christos res_querydomain(const char *name,
305 1.1 christos const char *domain,
306 1.1 christos int class, int type, /* class and type of query */
307 1.1 christos u_char *answer, /* buffer to put answer */
308 1.1 christos int anslen) /* size of answer */
309 1.1 christos {
310 1.8 christos if ((_nres.options & RES_INIT) == 0U && res_init() == -1) {
311 1.8 christos RES_SET_H_ERRNO(&_nres, NETDB_INTERNAL);
312 1.1 christos return (-1);
313 1.1 christos }
314 1.1 christos
315 1.8 christos return (res_nquerydomain(&_nres, name, domain,
316 1.1 christos class, type,
317 1.1 christos answer, anslen));
318 1.1 christos }
319 1.1 christos
320 1.2 christos int
321 1.2 christos res_opt(int a, u_char *b, int c, int d)
322 1.2 christos {
323 1.8 christos return res_nopt(&_nres, a, b, c, d);
324 1.2 christos }
325 1.2 christos
326 1.1 christos const char *
327 1.1 christos hostalias(const char *name) {
328 1.1 christos static char abuf[MAXDNAME];
329 1.1 christos
330 1.8 christos return (res_hostalias(&_nres, name, abuf, sizeof abuf));
331 1.1 christos }
332 1.1 christos
333 1.1 christos #ifdef ultrix
334 1.1 christos int
335 1.1 christos local_hostname_length(const char *hostname) {
336 1.1 christos int len_host, len_domain;
337 1.1 christos
338 1.8 christos if (!*_nres.defdname)
339 1.1 christos res_init();
340 1.1 christos len_host = strlen(hostname);
341 1.8 christos len_domain = strlen(_nres.defdname);
342 1.1 christos if (len_host > len_domain &&
343 1.8 christos !strcasecmp(hostname + len_host - len_domain, _nres.defdname) &&
344 1.1 christos hostname[len_host - len_domain - 1] == '.')
345 1.1 christos return (len_host - len_domain - 1);
346 1.1 christos return (0);
347 1.1 christos }
348 1.1 christos #endif /*ultrix*/
349 1.1 christos
350 1.1 christos #endif
351