res_data.c revision 1.1 1 1.1 christos /* $NetBSD: res_data.c,v 1.1 2004/05/20 17:18:55 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.1 christos #if defined(LIBC_SCCS) && !defined(lint)
21 1.1 christos static const char rcsid[] = "Id: res_data.c,v 1.1.206.2 2004/03/16 12:34:18 marka Exp";
22 1.1 christos #endif /* LIBC_SCCS and not lint */
23 1.1 christos
24 1.1 christos #include "port_before.h"
25 1.1 christos
26 1.1 christos #include <sys/types.h>
27 1.1 christos #include <sys/param.h>
28 1.1 christos #include <sys/socket.h>
29 1.1 christos #include <sys/time.h>
30 1.1 christos
31 1.1 christos #include <netinet/in.h>
32 1.1 christos #include <arpa/inet.h>
33 1.1 christos #include <arpa/nameser.h>
34 1.1 christos
35 1.1 christos #include <ctype.h>
36 1.1 christos #include <netdb.h>
37 1.1 christos #include <resolv.h>
38 1.1 christos #include <res_update.h>
39 1.1 christos #include <stdio.h>
40 1.1 christos #include <stdlib.h>
41 1.1 christos #include <string.h>
42 1.1 christos #include <unistd.h>
43 1.1 christos
44 1.1 christos #include "port_after.h"
45 1.1 christos #undef _res
46 1.1 christos
47 1.1 christos const char *_res_opcodes[] = {
48 1.1 christos "QUERY",
49 1.1 christos "IQUERY",
50 1.1 christos "CQUERYM",
51 1.1 christos "CQUERYU", /* experimental */
52 1.1 christos "NOTIFY", /* experimental */
53 1.1 christos "UPDATE",
54 1.1 christos "6",
55 1.1 christos "7",
56 1.1 christos "8",
57 1.1 christos "9",
58 1.1 christos "10",
59 1.1 christos "11",
60 1.1 christos "12",
61 1.1 christos "13",
62 1.1 christos "ZONEINIT",
63 1.1 christos "ZONEREF",
64 1.1 christos };
65 1.1 christos
66 1.1 christos #ifdef BIND_UPDATE
67 1.1 christos const char *_res_sectioncodes[] = {
68 1.1 christos "ZONE",
69 1.1 christos "PREREQUISITES",
70 1.1 christos "UPDATE",
71 1.1 christos "ADDITIONAL",
72 1.1 christos };
73 1.1 christos #endif
74 1.1 christos
75 1.1 christos #ifndef __BIND_NOSTATIC
76 1.1 christos struct __res_state _res
77 1.1 christos # if defined(__BIND_RES_TEXT)
78 1.1 christos = { RES_TIMEOUT, } /* Motorola, et al. */
79 1.1 christos # endif
80 1.1 christos ;
81 1.1 christos
82 1.1 christos /* Proto. */
83 1.1 christos
84 1.1 christos int res_ourserver_p(const res_state, const struct sockaddr_in *);
85 1.1 christos
86 1.1 christos int
87 1.1 christos res_init(void) {
88 1.1 christos extern int __res_vinit(res_state, int);
89 1.1 christos
90 1.1 christos /*
91 1.1 christos * These three fields used to be statically initialized. This made
92 1.1 christos * it hard to use this code in a shared library. It is necessary,
93 1.1 christos * now that we're doing dynamic initialization here, that we preserve
94 1.1 christos * the old semantics: if an application modifies one of these three
95 1.1 christos * fields of _res before res_init() is called, res_init() will not
96 1.1 christos * alter them. Of course, if an application is setting them to
97 1.1 christos * _zero_ before calling res_init(), hoping to override what used
98 1.1 christos * to be the static default, we can't detect it and unexpected results
99 1.1 christos * will follow. Zero for any of these fields would make no sense,
100 1.1 christos * so one can safely assume that the applications were already getting
101 1.1 christos * unexpected results.
102 1.1 christos *
103 1.1 christos * _res.options is tricky since some apps were known to diddle the bits
104 1.1 christos * before res_init() was first called. We can't replicate that semantic
105 1.1 christos * with dynamic initialization (they may have turned bits off that are
106 1.1 christos * set in RES_DEFAULT). Our solution is to declare such applications
107 1.1 christos * "broken". They could fool us by setting RES_INIT but none do (yet).
108 1.1 christos */
109 1.1 christos if (!_res.retrans)
110 1.1 christos _res.retrans = RES_TIMEOUT;
111 1.1 christos if (!_res.retry)
112 1.1 christos _res.retry = 4;
113 1.1 christos if (!(_res.options & RES_INIT))
114 1.1 christos _res.options = RES_DEFAULT;
115 1.1 christos
116 1.1 christos /*
117 1.1 christos * This one used to initialize implicitly to zero, so unless the app
118 1.1 christos * has set it to something in particular, we can randomize it now.
119 1.1 christos */
120 1.1 christos if (!_res.id)
121 1.1 christos _res.id = res_randomid();
122 1.1 christos
123 1.1 christos return (__res_vinit(&_res, 1));
124 1.1 christos }
125 1.1 christos
126 1.1 christos void
127 1.1 christos p_query(const u_char *msg) {
128 1.1 christos fp_query(msg, stdout);
129 1.1 christos }
130 1.1 christos
131 1.1 christos void
132 1.1 christos fp_query(const u_char *msg, FILE *file) {
133 1.1 christos fp_nquery(msg, PACKETSZ, file);
134 1.1 christos }
135 1.1 christos
136 1.1 christos void
137 1.1 christos fp_nquery(const u_char *msg, int len, FILE *file) {
138 1.1 christos if ((_res.options & RES_INIT) == 0U && res_init() == -1)
139 1.1 christos return;
140 1.1 christos
141 1.1 christos res_pquery(&_res, msg, len, file);
142 1.1 christos }
143 1.1 christos
144 1.1 christos int
145 1.1 christos res_mkquery(int op, /* opcode of query */
146 1.1 christos const char *dname, /* domain name */
147 1.1 christos int class, int type, /* class and type of query */
148 1.1 christos const u_char *data, /* resource record data */
149 1.1 christos int datalen, /* length of data */
150 1.1 christos const u_char *newrr_in, /* new rr for modify or append */
151 1.1 christos u_char *buf, /* buffer to put query */
152 1.1 christos int buflen) /* size of buffer */
153 1.1 christos {
154 1.1 christos if ((_res.options & RES_INIT) == 0U && res_init() == -1) {
155 1.1 christos RES_SET_H_ERRNO(&_res, NETDB_INTERNAL);
156 1.1 christos return (-1);
157 1.1 christos }
158 1.1 christos return (res_nmkquery(&_res, op, dname, class, type,
159 1.1 christos data, datalen,
160 1.1 christos newrr_in, buf, buflen));
161 1.1 christos }
162 1.1 christos
163 1.1 christos int
164 1.1 christos res_mkupdate(ns_updrec *rrecp_in, u_char *buf, int buflen) {
165 1.1 christos if ((_res.options & RES_INIT) == 0U && res_init() == -1) {
166 1.1 christos RES_SET_H_ERRNO(&_res, NETDB_INTERNAL);
167 1.1 christos return (-1);
168 1.1 christos }
169 1.1 christos
170 1.1 christos return (res_nmkupdate(&_res, rrecp_in, buf, buflen));
171 1.1 christos }
172 1.1 christos
173 1.1 christos int
174 1.1 christos res_query(const char *name, /* domain name */
175 1.1 christos int class, int type, /* class and type of query */
176 1.1 christos u_char *answer, /* buffer to put answer */
177 1.1 christos int anslen) /* size of answer buffer */
178 1.1 christos {
179 1.1 christos if ((_res.options & RES_INIT) == 0U && res_init() == -1) {
180 1.1 christos RES_SET_H_ERRNO(&_res, NETDB_INTERNAL);
181 1.1 christos return (-1);
182 1.1 christos }
183 1.1 christos return (res_nquery(&_res, name, class, type, answer, anslen));
184 1.1 christos }
185 1.1 christos
186 1.1 christos void
187 1.1 christos res_send_setqhook(res_send_qhook hook) {
188 1.1 christos _res.qhook = hook;
189 1.1 christos }
190 1.1 christos
191 1.1 christos void
192 1.1 christos res_send_setrhook(res_send_rhook hook) {
193 1.1 christos _res.rhook = hook;
194 1.1 christos }
195 1.1 christos
196 1.1 christos int
197 1.1 christos res_isourserver(const struct sockaddr_in *inp) {
198 1.1 christos return (res_ourserver_p(&_res, inp));
199 1.1 christos }
200 1.1 christos
201 1.1 christos int
202 1.1 christos res_send(const u_char *buf, int buflen, u_char *ans, int anssiz) {
203 1.1 christos if ((_res.options & RES_INIT) == 0U && res_init() == -1) {
204 1.1 christos /* errno should have been set by res_init() in this case. */
205 1.1 christos return (-1);
206 1.1 christos }
207 1.1 christos
208 1.1 christos return (res_nsend(&_res, buf, buflen, ans, anssiz));
209 1.1 christos }
210 1.1 christos
211 1.1 christos int
212 1.1 christos res_sendsigned(const u_char *buf, int buflen, ns_tsig_key *key,
213 1.1 christos u_char *ans, int anssiz)
214 1.1 christos {
215 1.1 christos if ((_res.options & RES_INIT) == 0U && res_init() == -1) {
216 1.1 christos /* errno should have been set by res_init() in this case. */
217 1.1 christos return (-1);
218 1.1 christos }
219 1.1 christos
220 1.1 christos return (res_nsendsigned(&_res, buf, buflen, key, ans, anssiz));
221 1.1 christos }
222 1.1 christos
223 1.1 christos void
224 1.1 christos res_close(void) {
225 1.1 christos res_nclose(&_res);
226 1.1 christos }
227 1.1 christos
228 1.1 christos int
229 1.1 christos res_update(ns_updrec *rrecp_in) {
230 1.1 christos if ((_res.options & RES_INIT) == 0U && res_init() == -1) {
231 1.1 christos RES_SET_H_ERRNO(&_res, NETDB_INTERNAL);
232 1.1 christos return (-1);
233 1.1 christos }
234 1.1 christos
235 1.1 christos return (res_nupdate(&_res, rrecp_in, NULL));
236 1.1 christos }
237 1.1 christos
238 1.1 christos int
239 1.1 christos res_search(const char *name, /* domain name */
240 1.1 christos int class, int type, /* class and type of query */
241 1.1 christos u_char *answer, /* buffer to put answer */
242 1.1 christos int anslen) /* size of answer */
243 1.1 christos {
244 1.1 christos if ((_res.options & RES_INIT) == 0U && res_init() == -1) {
245 1.1 christos RES_SET_H_ERRNO(&_res, NETDB_INTERNAL);
246 1.1 christos return (-1);
247 1.1 christos }
248 1.1 christos
249 1.1 christos return (res_nsearch(&_res, name, class, type, answer, anslen));
250 1.1 christos }
251 1.1 christos
252 1.1 christos int
253 1.1 christos res_querydomain(const char *name,
254 1.1 christos const char *domain,
255 1.1 christos int class, int type, /* class and type of query */
256 1.1 christos u_char *answer, /* buffer to put answer */
257 1.1 christos int anslen) /* size of answer */
258 1.1 christos {
259 1.1 christos if ((_res.options & RES_INIT) == 0U && res_init() == -1) {
260 1.1 christos RES_SET_H_ERRNO(&_res, NETDB_INTERNAL);
261 1.1 christos return (-1);
262 1.1 christos }
263 1.1 christos
264 1.1 christos return (res_nquerydomain(&_res, name, domain,
265 1.1 christos class, type,
266 1.1 christos answer, anslen));
267 1.1 christos }
268 1.1 christos
269 1.1 christos const char *
270 1.1 christos hostalias(const char *name) {
271 1.1 christos static char abuf[MAXDNAME];
272 1.1 christos
273 1.1 christos return (res_hostalias(&_res, name, abuf, sizeof abuf));
274 1.1 christos }
275 1.1 christos
276 1.1 christos #ifdef ultrix
277 1.1 christos int
278 1.1 christos local_hostname_length(const char *hostname) {
279 1.1 christos int len_host, len_domain;
280 1.1 christos
281 1.1 christos if (!*_res.defdname)
282 1.1 christos res_init();
283 1.1 christos len_host = strlen(hostname);
284 1.1 christos len_domain = strlen(_res.defdname);
285 1.1 christos if (len_host > len_domain &&
286 1.1 christos !strcasecmp(hostname + len_host - len_domain, _res.defdname) &&
287 1.1 christos hostname[len_host - len_domain - 1] == '.')
288 1.1 christos return (len_host - len_domain - 1);
289 1.1 christos return (0);
290 1.1 christos }
291 1.1 christos #endif /*ultrix*/
292 1.1 christos
293 1.1 christos #endif
294