1/** @file
2  Copyright (c) 2012, Intel Corporation. All rights reserved.<BR>
3  This program and the accompanying materials are licensed and made available under
4  the terms and conditions of the BSD License that accompanies this distribution.
5  The full text of the license may be found at
6  http://opensource.org/licenses/bsd-license.php.
7
8  THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
9  WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
10
11 * Portions copyright (c) 1999, 2000
12 * Intel Corporation.
13 * All rights reserved.
14 *
15 * Redistribution and use in source and binary forms, with or without
16 * modification, are permitted provided that the following conditions
17 * are met:
18 *
19 * 1. Redistributions of source code must retain the above copyright
20 *    notice, this list of conditions and the following disclaimer.
21 *
22 * 2. Redistributions in binary form must reproduce the above copyright
23 *    notice, this list of conditions and the following disclaimer in the
24 *    documentation and/or other materials provided with the distribution.
25 *
26 * 3. All advertising materials mentioning features or use of this software
27 *    must display the following acknowledgement:
28 *
29 *    This product includes software developed by Intel Corporation and
30 *    its contributors.
31 *
32 * 4. Neither the name of Intel Corporation or its contributors may be
33 *    used to endorse or promote products derived from this software
34 *    without specific prior written permission.
35 *
36 * THIS SOFTWARE IS PROVIDED BY INTEL CORPORATION AND CONTRIBUTORS ``AS IS''
37 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
38 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
39 * ARE DISCLAIMED.  IN NO EVENT SHALL INTEL CORPORATION OR CONTRIBUTORS BE
40 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
41 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
42 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
43 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
44 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
45 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
46 * THE POSSIBILITY OF SUCH DAMAGE.
47 *
48 * Copyright (c) 1996 by Internet Software Consortium.
49 *
50 * Permission to use, copy, modify, and distribute this software for any
51 * purpose with or without fee is hereby granted, provided that the above
52 * copyright notice and this permission notice appear in all copies.
53 *
54 * THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS
55 * ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
56 * OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE
57 * CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
58 * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
59 * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
60 * ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
61 * SOFTWARE.
62**/
63
64#include <sys/types.h>
65
66#include <netinet/in.h>
67#include <arpa/nameser.h>
68
69#include <errno.h>
70#include <resolv.h>
71#include <string.h>
72
73/* These need to be in the same order as the nres.h:ns_flag enum. */
74struct _ns_flagdata _ns_flagdata[16] = {
75	{ 0x8000, 15 },		/* qr. */
76	{ 0x7800, 11 },		/* opcode. */
77	{ 0x0400, 10 },		/* aa. */
78	{ 0x0200, 9 },		/* tc. */
79	{ 0x0100, 8 },		/* rd. */
80	{ 0x0080, 7 },		/* ra. */
81	{ 0x0040, 6 },		/* z. */
82	{ 0x0020, 5 },		/* ad. */
83	{ 0x0010, 4 },		/* cd. */
84	{ 0x000f, 0 },		/* rcode. */
85	{ 0x0000, 0 },		/* expansion (1/6). */
86	{ 0x0000, 0 },		/* expansion (2/6). */
87	{ 0x0000, 0 },		/* expansion (3/6). */
88	{ 0x0000, 0 },		/* expansion (4/6). */
89	{ 0x0000, 0 },		/* expansion (5/6). */
90	{ 0x0000, 0 },		/* expansion (6/6). */
91};
92
93static int
94skiprr(const u_char *ptr, const u_char *eom, ns_sect section, int count) {
95	const u_char *optr = ptr;
96
97	for ((void)NULL; count > 0; count--) {
98		int b, rdlength;
99
100		b = dn_skipname(ptr, eom);
101		if (b < 0)
102			goto emsgsize;
103		ptr += b/*Name*/ + NS_INT16SZ/*Type*/ + NS_INT16SZ/*Class*/;
104		if (section != ns_s_qd) {
105			if (ptr + NS_INT32SZ > eom)
106				goto emsgsize;
107			ptr += NS_INT32SZ/*TTL*/;
108			if (ptr + NS_INT16SZ > eom)
109				goto emsgsize;
110			NS_GET16(rdlength, ptr);
111			ptr += rdlength/*RData*/;
112		}
113	}
114	if (ptr > eom)
115		goto emsgsize;
116	return ((int)(ptr - optr));
117 emsgsize:
118	errno = EMSGSIZE;
119	return (-1);
120}
121
122int
123ns_initparse(const u_char *msg, int msglen, ns_msg *handle) {
124	const u_char *eom = msg + msglen;
125	int i;
126
127	memset(handle, 0x5e, sizeof *handle);
128	handle->_msg = msg;
129	handle->_eom = eom;
130	if (msg + NS_INT16SZ > eom)
131		goto emsgsize;
132	NS_GET16(handle->_id, msg);
133	if (msg + NS_INT16SZ > eom)
134		goto emsgsize;
135	NS_GET16(handle->_flags, msg);
136	for (i = 0; i < ns_s_max; i++) {
137		if (msg + NS_INT16SZ > eom)
138			goto emsgsize;
139		NS_GET16(handle->_counts[i], msg);
140	}
141	for (i = 0; i < ns_s_max; i++)
142		if (handle->_counts[i] == 0)
143			handle->_sections[i] = NULL;
144		else {
145			int b = skiprr(msg, eom, (ns_sect)i,
146				       handle->_counts[i]);
147
148			if (b < 0)
149				return (-1);
150			handle->_sections[i] = msg;
151			msg += b;
152		}
153	if (msg != eom)
154		goto emsgsize;
155	handle->_sect = ns_s_max;
156	handle->_rrnum = -1;
157	handle->_msg_ptr = NULL;
158	return (0);
159 emsgsize:
160	errno = EMSGSIZE;
161	return (-1);
162}
163
164int
165ns_parserr(ns_msg *handle, ns_sect section, int rrnum, ns_rr *rr) {
166	int b;
167
168	/* Make section right. */
169	if ((unsigned int)section >= ns_s_max)
170		goto enodev;
171	if ((int)section != (int)handle->_sect) {
172		handle->_sect = section;
173		handle->_rrnum = 0;
174		handle->_msg_ptr = handle->_sections[(int)section];
175	}
176
177	/* Make rrnum right. */
178	if (rrnum == -1)
179		rrnum = handle->_rrnum;
180	if (rrnum < 0 || rrnum >= handle->_counts[(int)section])
181		goto enodev;
182	if (rrnum < handle->_rrnum) {
183		handle->_rrnum = 0;
184		handle->_msg_ptr = handle->_sections[(int)section];
185	}
186
187	b = skiprr(handle->_msg, handle->_eom, section,
188		   rrnum - handle->_rrnum);
189	if (b < 0)
190		return (-1);
191	handle->_msg_ptr += b;
192	handle->_rrnum = rrnum;
193
194	/* Do the parse. */
195	b = dn_expand(handle->_msg, handle->_eom,
196		      handle->_msg_ptr, rr->name, NS_MAXDNAME);
197	if (b < 0)
198		return (-1);
199	handle->_msg_ptr += b;
200	if (handle->_msg_ptr + NS_INT16SZ > handle->_eom)
201		goto emsgsize;
202	NS_GET16(rr->type, handle->_msg_ptr);
203	if (handle->_msg_ptr + NS_INT16SZ > handle->_eom)
204		goto emsgsize;
205	NS_GET16(rr->rr_class, handle->_msg_ptr);
206	if (section == ns_s_qd) {
207		rr->ttl = 0;
208		rr->rdlength = 0;
209		rr->rdata = NULL;
210	} else {
211		if (handle->_msg_ptr + NS_INT32SZ > handle->_eom)
212			goto emsgsize;
213		NS_GET32(rr->ttl, handle->_msg_ptr);
214		if (handle->_msg_ptr + NS_INT16SZ > handle->_eom)
215			goto emsgsize;
216		NS_GET16(rr->rdlength, handle->_msg_ptr);
217		if (handle->_msg_ptr + rr->rdlength > handle->_eom)
218			goto emsgsize;
219		rr->rdata = handle->_msg_ptr;
220		handle->_msg_ptr += rr->rdlength;
221	}
222	handle->_rrnum++;
223
224	/* All done. */
225	return (0);
226 enodev:
227	errno = ENODEV;
228	return (-1);
229 emsgsize:
230	errno = EMSGSIZE;
231	return (-1);
232}
233