xref: /freebsd/contrib/sendmail/src/sm_resolve.c (revision f7c4bd95ba735bd6a5454b4953945a99cefbb80c)
1 /*
2  * Copyright (c) 2000-2004 Sendmail, Inc. and its suppliers.
3  *	All rights reserved.
4  *
5  * By using this file, you agree to the terms and conditions set
6  * forth in the LICENSE file which can be found at the top level of
7  * the sendmail distribution.
8  *
9  */
10 
11 /*
12  * Copyright (c) 1995, 1996, 1997, 1998, 1999 Kungliga Tekniska H�gskolan
13  * (Royal Institute of Technology, Stockholm, Sweden).
14  * All rights reserved.
15  *
16  * Redistribution and use in source and binary forms, with or without
17  * modification, are permitted provided that the following conditions
18  * are met:
19  *
20  * 1. Redistributions of source code must retain the above copyright
21  *    notice, this list of conditions and the following disclaimer.
22  *
23  * 2. Redistributions in binary form must reproduce the above copyright
24  *    notice, this list of conditions and the following disclaimer in the
25  *    documentation and/or other materials provided with the distribution.
26  *
27  * 3. Neither the name of the Institute nor the names of its contributors
28  *    may be used to endorse or promote products derived from this software
29  *    without specific prior written permission.
30  *
31  * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
32  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
33  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
34  * ARE DISCLAIMED.  IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
35  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
36  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
37  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
38  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
39  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
40  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
41  * SUCH DAMAGE.
42  */
43 
44 #include <sendmail.h>
45 #if DNSMAP
46 # if NAMED_BIND
47 #  include "sm_resolve.h"
48 
49 SM_RCSID("$Id: sm_resolve.c,v 8.35 2007/06/25 16:20:14 ca Exp $")
50 
51 static struct stot
52 {
53 	const char	*st_name;
54 	int		st_type;
55 } stot[] =
56 {
57 #  if NETINET
58 	{	"A",		T_A		},
59 #  endif /* NETINET */
60 #  if NETINET6
61 	{	"AAAA",		T_AAAA		},
62 #  endif /* NETINET6 */
63 	{	"NS",		T_NS		},
64 	{	"CNAME",	T_CNAME		},
65 	{	"PTR",		T_PTR		},
66 	{	"MX",		T_MX		},
67 	{	"TXT",		T_TXT		},
68 	{	"AFSDB",	T_AFSDB		},
69 	{	"SRV",		T_SRV		},
70 	{	NULL,		0		}
71 };
72 
73 static DNS_REPLY_T *parse_dns_reply __P((unsigned char *, int));
74 
75 /*
76 **  DNS_STRING_TO_TYPE -- convert resource record name into type
77 **
78 **	Parameters:
79 **		name -- name of resource record type
80 **
81 **	Returns:
82 **		type if succeeded.
83 **		-1 otherwise.
84 */
85 
86 int
87 dns_string_to_type(name)
88 	const char *name;
89 {
90 	struct stot *p = stot;
91 
92 	for (p = stot; p->st_name != NULL; p++)
93 		if (sm_strcasecmp(name, p->st_name) == 0)
94 			return p->st_type;
95 	return -1;
96 }
97 
98 /*
99 **  DNS_TYPE_TO_STRING -- convert resource record type into name
100 **
101 **	Parameters:
102 **		type -- resource record type
103 **
104 **	Returns:
105 **		name if succeeded.
106 **		NULL otherwise.
107 */
108 
109 const char *
110 dns_type_to_string(type)
111 	int type;
112 {
113 	struct stot *p = stot;
114 
115 	for (p = stot; p->st_name != NULL; p++)
116 		if (type == p->st_type)
117 			return p->st_name;
118 	return NULL;
119 }
120 
121 /*
122 **  DNS_FREE_DATA -- free all components of a DNS_REPLY_T
123 **
124 **	Parameters:
125 **		r -- pointer to DNS_REPLY_T
126 **
127 **	Returns:
128 **		none.
129 */
130 
131 void
132 dns_free_data(r)
133 	DNS_REPLY_T *r;
134 {
135 	RESOURCE_RECORD_T *rr;
136 
137 	if (r->dns_r_q.dns_q_domain != NULL)
138 		sm_free(r->dns_r_q.dns_q_domain);
139 	for (rr = r->dns_r_head; rr != NULL; )
140 	{
141 		RESOURCE_RECORD_T *tmp = rr;
142 
143 		if (rr->rr_domain != NULL)
144 			sm_free(rr->rr_domain);
145 		if (rr->rr_u.rr_data != NULL)
146 			sm_free(rr->rr_u.rr_data);
147 		rr = rr->rr_next;
148 		sm_free(tmp);
149 	}
150 	sm_free(r);
151 }
152 
153 /*
154 **  PARSE_DNS_REPLY -- parse DNS reply data.
155 **
156 **	Parameters:
157 **		data -- pointer to dns data
158 **		len -- len of data
159 **
160 **	Returns:
161 **		pointer to DNS_REPLY_T if succeeded.
162 **		NULL otherwise.
163 */
164 
165 static DNS_REPLY_T *
166 parse_dns_reply(data, len)
167 	unsigned char *data;
168 	int len;
169 {
170 	unsigned char *p;
171 	ushort ans_cnt, ui;
172 	int status;
173 	size_t l;
174 	char host[MAXHOSTNAMELEN];
175 	DNS_REPLY_T *r;
176 	RESOURCE_RECORD_T **rr;
177 
178 	r = (DNS_REPLY_T *) sm_malloc(sizeof(*r));
179 	if (r == NULL)
180 		return NULL;
181 	memset(r, 0, sizeof(*r));
182 
183 	p = data;
184 
185 	/* doesn't work on Crays? */
186 	memcpy(&r->dns_r_h, p, sizeof(r->dns_r_h));
187 	p += sizeof(r->dns_r_h);
188 	status = dn_expand(data, data + len, p, host, sizeof(host));
189 	if (status < 0)
190 	{
191 		dns_free_data(r);
192 		return NULL;
193 	}
194 	r->dns_r_q.dns_q_domain = sm_strdup(host);
195 	if (r->dns_r_q.dns_q_domain == NULL)
196 	{
197 		dns_free_data(r);
198 		return NULL;
199 	}
200 
201 	ans_cnt = ntohs((ushort) r->dns_r_h.ancount);
202 
203 	p += status;
204 	GETSHORT(r->dns_r_q.dns_q_type, p);
205 	GETSHORT(r->dns_r_q.dns_q_class, p);
206 	rr = &r->dns_r_head;
207 	ui = 0;
208 	while (p < data + len && ui < ans_cnt)
209 	{
210 		int type, class, ttl, size, txtlen;
211 
212 		status = dn_expand(data, data + len, p, host, sizeof(host));
213 		if (status < 0)
214 		{
215 			dns_free_data(r);
216 			return NULL;
217 		}
218 		++ui;
219 		p += status;
220 		GETSHORT(type, p);
221 		GETSHORT(class, p);
222 		GETLONG(ttl, p);
223 		GETSHORT(size, p);
224 		if (p + size > data + len)
225 		{
226 			/*
227 			**  announced size of data exceeds length of
228 			**  data paket: someone is cheating.
229 			*/
230 
231 			if (LogLevel > 5)
232 				sm_syslog(LOG_WARNING, NOQID,
233 					  "ERROR: DNS RDLENGTH=%d > data len=%d",
234 					  size, len - (p - data));
235 			dns_free_data(r);
236 			return NULL;
237 		}
238 		*rr = (RESOURCE_RECORD_T *) sm_malloc(sizeof(**rr));
239 		if (*rr == NULL)
240 		{
241 			dns_free_data(r);
242 			return NULL;
243 		}
244 		memset(*rr, 0, sizeof(**rr));
245 		(*rr)->rr_domain = sm_strdup(host);
246 		if ((*rr)->rr_domain == NULL)
247 		{
248 			dns_free_data(r);
249 			return NULL;
250 		}
251 		(*rr)->rr_type = type;
252 		(*rr)->rr_class = class;
253 		(*rr)->rr_ttl = ttl;
254 		(*rr)->rr_size = size;
255 		switch (type)
256 		{
257 		  case T_NS:
258 		  case T_CNAME:
259 		  case T_PTR:
260 			status = dn_expand(data, data + len, p, host,
261 					   sizeof(host));
262 			if (status < 0)
263 			{
264 				dns_free_data(r);
265 				return NULL;
266 			}
267 			(*rr)->rr_u.rr_txt = sm_strdup(host);
268 			if ((*rr)->rr_u.rr_txt == NULL)
269 			{
270 				dns_free_data(r);
271 				return NULL;
272 			}
273 			break;
274 
275 		  case T_MX:
276 		  case T_AFSDB:
277 			status = dn_expand(data, data + len, p + 2, host,
278 					   sizeof(host));
279 			if (status < 0)
280 			{
281 				dns_free_data(r);
282 				return NULL;
283 			}
284 			l = strlen(host) + 1;
285 			(*rr)->rr_u.rr_mx = (MX_RECORD_T *)
286 				sm_malloc(sizeof(*((*rr)->rr_u.rr_mx)) + l);
287 			if ((*rr)->rr_u.rr_mx == NULL)
288 			{
289 				dns_free_data(r);
290 				return NULL;
291 			}
292 			(*rr)->rr_u.rr_mx->mx_r_preference = (p[0] << 8) | p[1];
293 			(void) sm_strlcpy((*rr)->rr_u.rr_mx->mx_r_domain,
294 					  host, l);
295 			break;
296 
297 		  case T_SRV:
298 			status = dn_expand(data, data + len, p + 6, host,
299 					   sizeof(host));
300 			if (status < 0)
301 			{
302 				dns_free_data(r);
303 				return NULL;
304 			}
305 			l = strlen(host) + 1;
306 			(*rr)->rr_u.rr_srv = (SRV_RECORDT_T*)
307 				sm_malloc(sizeof(*((*rr)->rr_u.rr_srv)) + l);
308 			if ((*rr)->rr_u.rr_srv == NULL)
309 			{
310 				dns_free_data(r);
311 				return NULL;
312 			}
313 			(*rr)->rr_u.rr_srv->srv_r_priority = (p[0] << 8) | p[1];
314 			(*rr)->rr_u.rr_srv->srv_r_weight = (p[2] << 8) | p[3];
315 			(*rr)->rr_u.rr_srv->srv_r_port = (p[4] << 8) | p[5];
316 			(void) sm_strlcpy((*rr)->rr_u.rr_srv->srv_r_target,
317 					  host, l);
318 			break;
319 
320 		  case T_TXT:
321 
322 			/*
323 			**  The TXT record contains the length as
324 			**  leading byte, hence the value is restricted
325 			**  to 255, which is less than the maximum value
326 			**  of RDLENGTH (size). Nevertheless, txtlen
327 			**  must be less than size because the latter
328 			**  specifies the length of the entire TXT
329 			**  record.
330 			*/
331 
332 			txtlen = *p;
333 			if (txtlen >= size)
334 			{
335 				if (LogLevel > 5)
336 					sm_syslog(LOG_WARNING, NOQID,
337 						  "ERROR: DNS TXT record size=%d <= text len=%d",
338 						  size, txtlen);
339 				dns_free_data(r);
340 				return NULL;
341 			}
342 			(*rr)->rr_u.rr_txt = (char *) sm_malloc(txtlen + 1);
343 			if ((*rr)->rr_u.rr_txt == NULL)
344 			{
345 				dns_free_data(r);
346 				return NULL;
347 			}
348 			(void) sm_strlcpy((*rr)->rr_u.rr_txt, (char*) p + 1,
349 					  txtlen + 1);
350 			break;
351 
352 		  default:
353 			(*rr)->rr_u.rr_data = (unsigned char*) sm_malloc(size);
354 			if ((*rr)->rr_u.rr_data == NULL)
355 			{
356 				dns_free_data(r);
357 				return NULL;
358 			}
359 			(void) memcpy((*rr)->rr_u.rr_data, p, size);
360 			break;
361 		}
362 		p += size;
363 		rr = &(*rr)->rr_next;
364 	}
365 	*rr = NULL;
366 	return r;
367 }
368 
369 /*
370 **  DNS_LOOKUP_INT -- perform dns map lookup (internal helper routine)
371 **
372 **	Parameters:
373 **		domain -- name to lookup
374 **		rr_class -- resource record class
375 **		rr_type -- resource record type
376 **		retrans -- retransmission timeout
377 **		retry -- number of retries
378 **
379 **	Returns:
380 **		result of lookup if succeeded.
381 **		NULL otherwise.
382 */
383 
384 DNS_REPLY_T *
385 dns_lookup_int(domain, rr_class, rr_type, retrans, retry)
386 	const char *domain;
387 	int rr_class;
388 	int rr_type;
389 	time_t retrans;
390 	int retry;
391 {
392 	int len;
393 	unsigned long old_options = 0;
394 	time_t save_retrans = 0;
395 	int save_retry = 0;
396 	DNS_REPLY_T *r = NULL;
397 	unsigned char reply[1024];
398 
399 	if (tTd(8, 16))
400 	{
401 		old_options = _res.options;
402 		_res.options |= RES_DEBUG;
403 		sm_dprintf("dns_lookup(%s, %d, %s)\n", domain,
404 			   rr_class, dns_type_to_string(rr_type));
405 	}
406 	if (retrans > 0)
407 	{
408 		save_retrans = _res.retrans;
409 		_res.retrans = retrans;
410 	}
411 	if (retry > 0)
412 	{
413 		save_retry = _res.retry;
414 		_res.retry = retry;
415 	}
416 	errno = 0;
417 	SM_SET_H_ERRNO(0);
418 	len = res_search(domain, rr_class, rr_type, reply, sizeof(reply));
419 	if (tTd(8, 16))
420 	{
421 		_res.options = old_options;
422 		sm_dprintf("dns_lookup(%s, %d, %s) --> %d\n",
423 			   domain, rr_class, dns_type_to_string(rr_type), len);
424 	}
425 	if (len >= 0)
426 		r = parse_dns_reply(reply, len);
427 	if (retrans > 0)
428 		_res.retrans = save_retrans;
429 	if (retry > 0)
430 		_res.retry = save_retry;
431 	return r;
432 }
433 
434 #  if 0
435 DNS_REPLY_T *
436 dns_lookup(domain, type_name, retrans, retry)
437 	const char *domain;
438 	const char *type_name;
439 	time_t retrans;
440 	int retry;
441 {
442 	int type;
443 
444 	type = dns_string_to_type(type_name);
445 	if (type == -1)
446 	{
447 		if (tTd(8, 16))
448 			sm_dprintf("dns_lookup: unknown resource type: `%s'\n",
449 				type_name);
450 		return NULL;
451 	}
452 	return dns_lookup_int(domain, C_IN, type, retrans, retry);
453 }
454 #  endif /* 0 */
455 # endif /* NAMED_BIND */
456 #endif /* DNSMAP */
457