xref: /xnu-11417.140.69/bsd/netinet/dhcp_options.c (revision 43a90889846e00bfb5cf1d255cdc0a701a1e05a4)
1*43a90889SApple OSS Distributions /*
2*43a90889SApple OSS Distributions  * Copyright (c) 2002-2019 Apple Inc. All rights reserved.
3*43a90889SApple OSS Distributions  *
4*43a90889SApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*43a90889SApple OSS Distributions  *
6*43a90889SApple OSS Distributions  * This file contains Original Code and/or Modifications of Original Code
7*43a90889SApple OSS Distributions  * as defined in and that are subject to the Apple Public Source License
8*43a90889SApple OSS Distributions  * Version 2.0 (the 'License'). You may not use this file except in
9*43a90889SApple OSS Distributions  * compliance with the License. The rights granted to you under the License
10*43a90889SApple OSS Distributions  * may not be used to create, or enable the creation or redistribution of,
11*43a90889SApple OSS Distributions  * unlawful or unlicensed copies of an Apple operating system, or to
12*43a90889SApple OSS Distributions  * circumvent, violate, or enable the circumvention or violation of, any
13*43a90889SApple OSS Distributions  * terms of an Apple operating system software license agreement.
14*43a90889SApple OSS Distributions  *
15*43a90889SApple OSS Distributions  * Please obtain a copy of the License at
16*43a90889SApple OSS Distributions  * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*43a90889SApple OSS Distributions  *
18*43a90889SApple OSS Distributions  * The Original Code and all software distributed under the License are
19*43a90889SApple OSS Distributions  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*43a90889SApple OSS Distributions  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*43a90889SApple OSS Distributions  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*43a90889SApple OSS Distributions  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*43a90889SApple OSS Distributions  * Please see the License for the specific language governing rights and
24*43a90889SApple OSS Distributions  * limitations under the License.
25*43a90889SApple OSS Distributions  *
26*43a90889SApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*43a90889SApple OSS Distributions  */
28*43a90889SApple OSS Distributions /*
29*43a90889SApple OSS Distributions  * dhcp_options.c
30*43a90889SApple OSS Distributions  * - routines to parse and access dhcp options
31*43a90889SApple OSS Distributions  *   and create new dhcp option areas
32*43a90889SApple OSS Distributions  * - handles overloaded areas as well as vendor-specific options
33*43a90889SApple OSS Distributions  *   that are encoded using the RFC 2132 encoding
34*43a90889SApple OSS Distributions  */
35*43a90889SApple OSS Distributions 
36*43a90889SApple OSS Distributions /*
37*43a90889SApple OSS Distributions  * Modification History
38*43a90889SApple OSS Distributions  *
39*43a90889SApple OSS Distributions  * March 15, 2002	Dieter Siegmund (dieter@apple)
40*43a90889SApple OSS Distributions  * - imported from bootp project
41*43a90889SApple OSS Distributions  */
42*43a90889SApple OSS Distributions 
43*43a90889SApple OSS Distributions #include <string.h>
44*43a90889SApple OSS Distributions #include <sys/types.h>
45*43a90889SApple OSS Distributions #include <sys/param.h>
46*43a90889SApple OSS Distributions #include <netinet/in.h>
47*43a90889SApple OSS Distributions #include <sys/malloc.h>
48*43a90889SApple OSS Distributions #include <netinet/dhcp.h>
49*43a90889SApple OSS Distributions #include <netinet/dhcp_options.h>
50*43a90889SApple OSS Distributions 
51*43a90889SApple OSS Distributions #ifndef TEST_DHCP_OPTIONS
52*43a90889SApple OSS Distributions #include <libkern/libkern.h>
53*43a90889SApple OSS Distributions 
54*43a90889SApple OSS Distributions #ifdef  DHCP_DEBUG
55*43a90889SApple OSS Distributions #define dprintf(x) printf x;
56*43a90889SApple OSS Distributions #else   /* !DHCP_DEBUG */
57*43a90889SApple OSS Distributions #define dprintf(x)
58*43a90889SApple OSS Distributions #endif  /* DHCP_DEBUG */
59*43a90889SApple OSS Distributions 
60*43a90889SApple OSS Distributions #else
61*43a90889SApple OSS Distributions /*
62*43a90889SApple OSS Distributions  * To build:
63*43a90889SApple OSS Distributions  * xcrun -sdk macosx.internal cc -DTEST_DHCP_OPTIONS -o /tmp/dhcp_options dhcp_options.c -I ..
64*43a90889SApple OSS Distributions  */
65*43a90889SApple OSS Distributions #include <stdlib.h>
66*43a90889SApple OSS Distributions #include <unistd.h>
67*43a90889SApple OSS Distributions #include <stdio.h>
68*43a90889SApple OSS Distributions #define kfree_type(type, n, v)  free(v)
69*43a90889SApple OSS Distributions #define krealloc_type(type, old_n, new_n, ptr, flags) \
70*43a90889SApple OSS Distributions 	realloc(ptr, new_n * sizeof(type)))
71*43a90889SApple OSS Distributions #define dprintf(x) printf x;
72*43a90889SApple OSS Distributions #endif
73*43a90889SApple OSS Distributions 
74*43a90889SApple OSS Distributions /*
75*43a90889SApple OSS Distributions  * Functions: ptrlist_*
76*43a90889SApple OSS Distributions  * Purpose:
77*43a90889SApple OSS Distributions  *   A dynamically growable array of pointers.
78*43a90889SApple OSS Distributions  */
79*43a90889SApple OSS Distributions 
80*43a90889SApple OSS Distributions #define PTRLIST_NUMBER          16
81*43a90889SApple OSS Distributions 
82*43a90889SApple OSS Distributions static void
ptrlist_init(ptrlist_t * list)83*43a90889SApple OSS Distributions ptrlist_init(ptrlist_t * list)
84*43a90889SApple OSS Distributions {
85*43a90889SApple OSS Distributions 	bzero(list, sizeof(*list));
86*43a90889SApple OSS Distributions 	return;
87*43a90889SApple OSS Distributions }
88*43a90889SApple OSS Distributions 
89*43a90889SApple OSS Distributions static void
ptrlist_free(ptrlist_t * list)90*43a90889SApple OSS Distributions ptrlist_free(ptrlist_t * list)
91*43a90889SApple OSS Distributions {
92*43a90889SApple OSS Distributions 	if (list->array) {
93*43a90889SApple OSS Distributions 		kfree_type(const void *, list->size, list->array);
94*43a90889SApple OSS Distributions 	}
95*43a90889SApple OSS Distributions 	ptrlist_init(list);
96*43a90889SApple OSS Distributions 	return;
97*43a90889SApple OSS Distributions }
98*43a90889SApple OSS Distributions 
99*43a90889SApple OSS Distributions static int
ptrlist_count(ptrlist_t * list)100*43a90889SApple OSS Distributions ptrlist_count(ptrlist_t * list)
101*43a90889SApple OSS Distributions {
102*43a90889SApple OSS Distributions 	if (list == NULL || list->array == NULL) {
103*43a90889SApple OSS Distributions 		return 0;
104*43a90889SApple OSS Distributions 	}
105*43a90889SApple OSS Distributions 
106*43a90889SApple OSS Distributions 	return list->count;
107*43a90889SApple OSS Distributions }
108*43a90889SApple OSS Distributions 
109*43a90889SApple OSS Distributions static const void *
ptrlist_element(ptrlist_t * list,int i)110*43a90889SApple OSS Distributions ptrlist_element(ptrlist_t * list, int i)
111*43a90889SApple OSS Distributions {
112*43a90889SApple OSS Distributions 	if (list->array == NULL) {
113*43a90889SApple OSS Distributions 		return NULL;
114*43a90889SApple OSS Distributions 	}
115*43a90889SApple OSS Distributions 	if (i < list->count) {
116*43a90889SApple OSS Distributions 		return list->array[i];
117*43a90889SApple OSS Distributions 	}
118*43a90889SApple OSS Distributions 	return NULL;
119*43a90889SApple OSS Distributions }
120*43a90889SApple OSS Distributions 
121*43a90889SApple OSS Distributions 
122*43a90889SApple OSS Distributions static bool
ptrlist_grow(ptrlist_t * list,uint32_t n)123*43a90889SApple OSS Distributions ptrlist_grow(ptrlist_t * list, uint32_t n)
124*43a90889SApple OSS Distributions {
125*43a90889SApple OSS Distributions 	uint32_t new_size;
126*43a90889SApple OSS Distributions 	const void **arr;
127*43a90889SApple OSS Distributions 
128*43a90889SApple OSS Distributions 	if (os_add_overflow(list->count, n, &n)) {
129*43a90889SApple OSS Distributions 		return false;
130*43a90889SApple OSS Distributions 	}
131*43a90889SApple OSS Distributions 	if (n <= list->size) {
132*43a90889SApple OSS Distributions 		return true;
133*43a90889SApple OSS Distributions 	}
134*43a90889SApple OSS Distributions 
135*43a90889SApple OSS Distributions 	if (list->size == 0) {
136*43a90889SApple OSS Distributions 		new_size = MAX(PTRLIST_NUMBER, n);
137*43a90889SApple OSS Distributions 	} else {
138*43a90889SApple OSS Distributions 		new_size = MAX(list->size * 2, n);
139*43a90889SApple OSS Distributions 	}
140*43a90889SApple OSS Distributions 
141*43a90889SApple OSS Distributions 	arr = krealloc_type(const void *, list->size, new_size, list->array, Z_WAITOK);
142*43a90889SApple OSS Distributions 	if (arr == NULL) {
143*43a90889SApple OSS Distributions 		return false;
144*43a90889SApple OSS Distributions 	}
145*43a90889SApple OSS Distributions 
146*43a90889SApple OSS Distributions 	list->size = new_size;
147*43a90889SApple OSS Distributions 	list->array = arr;
148*43a90889SApple OSS Distributions 	return true;
149*43a90889SApple OSS Distributions }
150*43a90889SApple OSS Distributions 
151*43a90889SApple OSS Distributions static bool
ptrlist_add(ptrlist_t * list,const void * element)152*43a90889SApple OSS Distributions ptrlist_add(ptrlist_t * list, const void * element)
153*43a90889SApple OSS Distributions {
154*43a90889SApple OSS Distributions 	if (!ptrlist_grow(list, 1)) {
155*43a90889SApple OSS Distributions 		return false;
156*43a90889SApple OSS Distributions 	}
157*43a90889SApple OSS Distributions 
158*43a90889SApple OSS Distributions 	list->array[list->count++] = element;
159*43a90889SApple OSS Distributions 	return true;
160*43a90889SApple OSS Distributions }
161*43a90889SApple OSS Distributions 
162*43a90889SApple OSS Distributions /* concatenates extra onto list */
163*43a90889SApple OSS Distributions static bool
ptrlist_concat(ptrlist_t * list,ptrlist_t * extra)164*43a90889SApple OSS Distributions ptrlist_concat(ptrlist_t * list, ptrlist_t * extra)
165*43a90889SApple OSS Distributions {
166*43a90889SApple OSS Distributions 	if (!ptrlist_grow(list, extra->count)) {
167*43a90889SApple OSS Distributions 		return false;
168*43a90889SApple OSS Distributions 	}
169*43a90889SApple OSS Distributions 
170*43a90889SApple OSS Distributions 	bcopy(extra->array, list->array + list->count,
171*43a90889SApple OSS Distributions 	    extra->count * sizeof(*list->array));
172*43a90889SApple OSS Distributions 	list->count += extra->count;
173*43a90889SApple OSS Distributions 	return true;
174*43a90889SApple OSS Distributions }
175*43a90889SApple OSS Distributions 
176*43a90889SApple OSS Distributions 
177*43a90889SApple OSS Distributions /*
178*43a90889SApple OSS Distributions  * Functions: dhcpol_*
179*43a90889SApple OSS Distributions  *
180*43a90889SApple OSS Distributions  * Purpose:
181*43a90889SApple OSS Distributions  *   Routines to parse/access existing options buffers.
182*43a90889SApple OSS Distributions  */
183*43a90889SApple OSS Distributions boolean_t
dhcpol_add(dhcpol_t * list,const void * element)184*43a90889SApple OSS Distributions dhcpol_add(dhcpol_t * list, const void * element)
185*43a90889SApple OSS Distributions {
186*43a90889SApple OSS Distributions 	return ptrlist_add((ptrlist_t *)list, element);
187*43a90889SApple OSS Distributions }
188*43a90889SApple OSS Distributions 
189*43a90889SApple OSS Distributions int
dhcpol_count(dhcpol_t * list)190*43a90889SApple OSS Distributions dhcpol_count(dhcpol_t * list)
191*43a90889SApple OSS Distributions {
192*43a90889SApple OSS Distributions 	return ptrlist_count((ptrlist_t *)list);
193*43a90889SApple OSS Distributions }
194*43a90889SApple OSS Distributions 
195*43a90889SApple OSS Distributions const void *
dhcpol_element(dhcpol_t * list,int i)196*43a90889SApple OSS Distributions dhcpol_element(dhcpol_t * list, int i)
197*43a90889SApple OSS Distributions {
198*43a90889SApple OSS Distributions 	return ptrlist_element((ptrlist_t *)list, i);
199*43a90889SApple OSS Distributions }
200*43a90889SApple OSS Distributions 
201*43a90889SApple OSS Distributions void
dhcpol_init(dhcpol_t * list)202*43a90889SApple OSS Distributions dhcpol_init(dhcpol_t * list)
203*43a90889SApple OSS Distributions {
204*43a90889SApple OSS Distributions 	ptrlist_init((ptrlist_t *)list);
205*43a90889SApple OSS Distributions }
206*43a90889SApple OSS Distributions 
207*43a90889SApple OSS Distributions void
dhcpol_free(dhcpol_t * list)208*43a90889SApple OSS Distributions dhcpol_free(dhcpol_t * list)
209*43a90889SApple OSS Distributions {
210*43a90889SApple OSS Distributions 	ptrlist_free((ptrlist_t *)list);
211*43a90889SApple OSS Distributions }
212*43a90889SApple OSS Distributions 
213*43a90889SApple OSS Distributions boolean_t
dhcpol_concat(dhcpol_t * list,dhcpol_t * extra)214*43a90889SApple OSS Distributions dhcpol_concat(dhcpol_t * list, dhcpol_t * extra)
215*43a90889SApple OSS Distributions {
216*43a90889SApple OSS Distributions 	return ptrlist_concat(list, extra);
217*43a90889SApple OSS Distributions }
218*43a90889SApple OSS Distributions 
219*43a90889SApple OSS Distributions /*
220*43a90889SApple OSS Distributions  * Function: dhcpol_parse_buffer
221*43a90889SApple OSS Distributions  *
222*43a90889SApple OSS Distributions  * Purpose:
223*43a90889SApple OSS Distributions  *   Parse the given buffer into DHCP options, returning the
224*43a90889SApple OSS Distributions  *   list of option pointers in the given dhcpol_t.
225*43a90889SApple OSS Distributions  *   Parsing continues until we hit the end of the buffer or
226*43a90889SApple OSS Distributions  *   the end tag.
227*43a90889SApple OSS Distributions  */
228*43a90889SApple OSS Distributions boolean_t
dhcpol_parse_buffer(dhcpol_t * list,const void * buffer,int length)229*43a90889SApple OSS Distributions dhcpol_parse_buffer(dhcpol_t * list, const void * buffer, int length)
230*43a90889SApple OSS Distributions {
231*43a90889SApple OSS Distributions 	int                 len;
232*43a90889SApple OSS Distributions 	const uint8_t *     scan;
233*43a90889SApple OSS Distributions 	uint8_t             tag;
234*43a90889SApple OSS Distributions 
235*43a90889SApple OSS Distributions 	dhcpol_init(list);
236*43a90889SApple OSS Distributions 
237*43a90889SApple OSS Distributions 	len = length;
238*43a90889SApple OSS Distributions 	tag = dhcptag_pad_e;
239*43a90889SApple OSS Distributions 	for (scan = (const uint8_t *)buffer;
240*43a90889SApple OSS Distributions 	    tag != dhcptag_end_e && len > DHCP_TAG_OFFSET;) {
241*43a90889SApple OSS Distributions 		tag = scan[DHCP_TAG_OFFSET];
242*43a90889SApple OSS Distributions 
243*43a90889SApple OSS Distributions 		switch (tag) {
244*43a90889SApple OSS Distributions 		case dhcptag_end_e:
245*43a90889SApple OSS Distributions 			/* remember that it was terminated */
246*43a90889SApple OSS Distributions 			dhcpol_add(list, scan);
247*43a90889SApple OSS Distributions 			scan++;
248*43a90889SApple OSS Distributions 			len--;
249*43a90889SApple OSS Distributions 			break;
250*43a90889SApple OSS Distributions 		case dhcptag_pad_e: /* ignore pad */
251*43a90889SApple OSS Distributions 			scan++;
252*43a90889SApple OSS Distributions 			len--;
253*43a90889SApple OSS Distributions 			break;
254*43a90889SApple OSS Distributions 		default:
255*43a90889SApple OSS Distributions 			if (len > DHCP_LEN_OFFSET) {
256*43a90889SApple OSS Distributions 				uint8_t       option_len;
257*43a90889SApple OSS Distributions 
258*43a90889SApple OSS Distributions 				option_len = scan[DHCP_LEN_OFFSET];
259*43a90889SApple OSS Distributions 				dhcpol_add(list, scan);
260*43a90889SApple OSS Distributions 				len -= (option_len + DHCP_OPTION_OFFSET);
261*43a90889SApple OSS Distributions 				scan += (option_len + DHCP_OPTION_OFFSET);
262*43a90889SApple OSS Distributions 			} else {
263*43a90889SApple OSS Distributions 				len = -1;
264*43a90889SApple OSS Distributions 			}
265*43a90889SApple OSS Distributions 			break;
266*43a90889SApple OSS Distributions 		}
267*43a90889SApple OSS Distributions 	}
268*43a90889SApple OSS Distributions 	if (len < 0) {
269*43a90889SApple OSS Distributions 		/* ran off the end */
270*43a90889SApple OSS Distributions 		dprintf(("dhcp_options: parse failed near tag %d\n", tag));
271*43a90889SApple OSS Distributions 		dhcpol_free(list);
272*43a90889SApple OSS Distributions 		return FALSE;
273*43a90889SApple OSS Distributions 	}
274*43a90889SApple OSS Distributions 	return TRUE;
275*43a90889SApple OSS Distributions }
276*43a90889SApple OSS Distributions 
277*43a90889SApple OSS Distributions /*
278*43a90889SApple OSS Distributions  * Function: dhcpol_find
279*43a90889SApple OSS Distributions  *
280*43a90889SApple OSS Distributions  * Purpose:
281*43a90889SApple OSS Distributions  *   Finds the first occurence of the given option, and returns its
282*43a90889SApple OSS Distributions  *   length and the option data pointer.
283*43a90889SApple OSS Distributions  *
284*43a90889SApple OSS Distributions  *   The optional start parameter allows this function to
285*43a90889SApple OSS Distributions  *   return the next start point so that successive
286*43a90889SApple OSS Distributions  *   calls will retrieve the next occurence of the option.
287*43a90889SApple OSS Distributions  *   Before the first call, *start should be set to 0.
288*43a90889SApple OSS Distributions  */
289*43a90889SApple OSS Distributions const void *
dhcpol_find(dhcpol_t * list,int tag,int * len_p,int * start)290*43a90889SApple OSS Distributions dhcpol_find(dhcpol_t * list, int tag, int * len_p, int * start)
291*43a90889SApple OSS Distributions {
292*43a90889SApple OSS Distributions 	int         i = 0;
293*43a90889SApple OSS Distributions 
294*43a90889SApple OSS Distributions 	if (tag == dhcptag_end_e || tag == dhcptag_pad_e) {
295*43a90889SApple OSS Distributions 		return NULL;
296*43a90889SApple OSS Distributions 	}
297*43a90889SApple OSS Distributions 
298*43a90889SApple OSS Distributions 	if (start) {
299*43a90889SApple OSS Distributions 		i = *start;
300*43a90889SApple OSS Distributions 	}
301*43a90889SApple OSS Distributions 
302*43a90889SApple OSS Distributions 	for (; i < dhcpol_count(list); i++) {
303*43a90889SApple OSS Distributions 		const uint8_t *         option = dhcpol_element(list, i);
304*43a90889SApple OSS Distributions 
305*43a90889SApple OSS Distributions 		if (option[DHCP_TAG_OFFSET] == tag) {
306*43a90889SApple OSS Distributions 			if (len_p) {
307*43a90889SApple OSS Distributions 				*len_p = option[DHCP_LEN_OFFSET];
308*43a90889SApple OSS Distributions 			}
309*43a90889SApple OSS Distributions 			if (start) {
310*43a90889SApple OSS Distributions 				*start = i + 1;
311*43a90889SApple OSS Distributions 			}
312*43a90889SApple OSS Distributions 			return option + DHCP_OPTION_OFFSET;
313*43a90889SApple OSS Distributions 		}
314*43a90889SApple OSS Distributions 	}
315*43a90889SApple OSS Distributions 	return NULL;
316*43a90889SApple OSS Distributions }
317*43a90889SApple OSS Distributions 
318*43a90889SApple OSS Distributions /*
319*43a90889SApple OSS Distributions  * Function: dhcpol_parse_packet
320*43a90889SApple OSS Distributions  *
321*43a90889SApple OSS Distributions  * Purpose:
322*43a90889SApple OSS Distributions  *    Parse the option areas in the DHCP packet.
323*43a90889SApple OSS Distributions  *    Verifies that the packet has the right magic number,
324*43a90889SApple OSS Distributions  *    then parses and accumulates the option areas.
325*43a90889SApple OSS Distributions  *    First the pkt->dp_options is parsed.  If that contains
326*43a90889SApple OSS Distributions  *    the overload option, it parses pkt->dp_file if specified,
327*43a90889SApple OSS Distributions  *    then parses pkt->dp_sname if specified.
328*43a90889SApple OSS Distributions  */
329*43a90889SApple OSS Distributions boolean_t
dhcpol_parse_packet(dhcpol_t * options,const struct dhcp * pkt,int len)330*43a90889SApple OSS Distributions dhcpol_parse_packet(dhcpol_t * options, const struct dhcp * pkt, int len)
331*43a90889SApple OSS Distributions {
332*43a90889SApple OSS Distributions 	char                rfc_magic[4] = RFC_OPTIONS_MAGIC;
333*43a90889SApple OSS Distributions 
334*43a90889SApple OSS Distributions 	dhcpol_init(options);   /* make sure it's empty */
335*43a90889SApple OSS Distributions 
336*43a90889SApple OSS Distributions 	if (len < (sizeof(*pkt) + RFC_MAGIC_SIZE)) {
337*43a90889SApple OSS Distributions 		dprintf(("dhcp_options: packet is too short: %d < %d\n",
338*43a90889SApple OSS Distributions 		    len, (int)sizeof(*pkt) + RFC_MAGIC_SIZE));
339*43a90889SApple OSS Distributions 		return FALSE;
340*43a90889SApple OSS Distributions 	}
341*43a90889SApple OSS Distributions 	if (bcmp(pkt->dp_options, rfc_magic, RFC_MAGIC_SIZE)) {
342*43a90889SApple OSS Distributions 		dprintf(("dhcp_options: missing magic number\n"));
343*43a90889SApple OSS Distributions 		return FALSE;
344*43a90889SApple OSS Distributions 	}
345*43a90889SApple OSS Distributions 	if (dhcpol_parse_buffer(options, pkt->dp_options + RFC_MAGIC_SIZE,
346*43a90889SApple OSS Distributions 	    len - sizeof(*pkt) - RFC_MAGIC_SIZE) == FALSE) {
347*43a90889SApple OSS Distributions 		return FALSE;
348*43a90889SApple OSS Distributions 	}
349*43a90889SApple OSS Distributions 	{ /* get overloaded options */
350*43a90889SApple OSS Distributions 		const uint8_t * overload;
351*43a90889SApple OSS Distributions 		int             overload_len;
352*43a90889SApple OSS Distributions 
353*43a90889SApple OSS Distributions 		overload = dhcpol_find(options, dhcptag_option_overload_e,
354*43a90889SApple OSS Distributions 		    &overload_len, NULL);
355*43a90889SApple OSS Distributions 		if (overload && overload_len == 1) { /* has overloaded options */
356*43a90889SApple OSS Distributions 			dhcpol_t    extra;
357*43a90889SApple OSS Distributions 
358*43a90889SApple OSS Distributions 			dhcpol_init(&extra);
359*43a90889SApple OSS Distributions 			if (*overload == DHCP_OVERLOAD_FILE
360*43a90889SApple OSS Distributions 			    || *overload == DHCP_OVERLOAD_BOTH) {
361*43a90889SApple OSS Distributions 				if (dhcpol_parse_buffer(&extra, pkt->dp_file,
362*43a90889SApple OSS Distributions 				    sizeof(pkt->dp_file))) {
363*43a90889SApple OSS Distributions 					dhcpol_concat(options, &extra);
364*43a90889SApple OSS Distributions 					dhcpol_free(&extra);
365*43a90889SApple OSS Distributions 				}
366*43a90889SApple OSS Distributions 			}
367*43a90889SApple OSS Distributions 			if (*overload == DHCP_OVERLOAD_SNAME
368*43a90889SApple OSS Distributions 			    || *overload == DHCP_OVERLOAD_BOTH) {
369*43a90889SApple OSS Distributions 				if (dhcpol_parse_buffer(&extra, pkt->dp_sname,
370*43a90889SApple OSS Distributions 				    sizeof(pkt->dp_sname))) {
371*43a90889SApple OSS Distributions 					dhcpol_concat(options, &extra);
372*43a90889SApple OSS Distributions 					dhcpol_free(&extra);
373*43a90889SApple OSS Distributions 				}
374*43a90889SApple OSS Distributions 			}
375*43a90889SApple OSS Distributions 		}
376*43a90889SApple OSS Distributions 	}
377*43a90889SApple OSS Distributions 	return TRUE;
378*43a90889SApple OSS Distributions }
379*43a90889SApple OSS Distributions 
380*43a90889SApple OSS Distributions #ifdef TEST_DHCP_OPTIONS
381*43a90889SApple OSS Distributions char test_empty[] = {
382*43a90889SApple OSS Distributions 	99, 130, 83, 99,
383*43a90889SApple OSS Distributions 	255,
384*43a90889SApple OSS Distributions };
385*43a90889SApple OSS Distributions 
386*43a90889SApple OSS Distributions char test_short[] = {
387*43a90889SApple OSS Distributions 	99, 130, 83, 99,
388*43a90889SApple OSS Distributions 	1,
389*43a90889SApple OSS Distributions };
390*43a90889SApple OSS Distributions 
391*43a90889SApple OSS Distributions char test_simple[] = {
392*43a90889SApple OSS Distributions 	99, 130, 83, 99,
393*43a90889SApple OSS Distributions 	1, 4, 255, 255, 252, 0,
394*43a90889SApple OSS Distributions 	3, 4, 17, 202, 40, 1,
395*43a90889SApple OSS Distributions 	255,
396*43a90889SApple OSS Distributions };
397*43a90889SApple OSS Distributions 
398*43a90889SApple OSS Distributions char test_vendor[] = {
399*43a90889SApple OSS Distributions 	99, 130, 83, 99,
400*43a90889SApple OSS Distributions 	1, 4, 255, 255, 252, 0,
401*43a90889SApple OSS Distributions 	3, 4, 17, 202, 40, 1,
402*43a90889SApple OSS Distributions 	43, 6, 1, 4, 1, 2, 3, 4,
403*43a90889SApple OSS Distributions 	43, 6, 1, 4, 1, 2, 3, 4,
404*43a90889SApple OSS Distributions 	255,
405*43a90889SApple OSS Distributions };
406*43a90889SApple OSS Distributions 
407*43a90889SApple OSS Distributions char test_no_end[] = {
408*43a90889SApple OSS Distributions 	0x63, 0x82, 0x53, 0x63, 0x35, 0x01, 0x05, 0x36,
409*43a90889SApple OSS Distributions 	0x04, 0xc0, 0xa8, 0x01, 0x01, 0x33, 0x04, 0x80,
410*43a90889SApple OSS Distributions 	0x00, 0x80, 0x00, 0x01, 0x04, 0xff, 0xff, 0xff,
411*43a90889SApple OSS Distributions 	0x00, 0x03, 0x04, 0xc0, 0xa8, 0x01, 0x01, 0x06,
412*43a90889SApple OSS Distributions 	0x0c, 0x18, 0x1a, 0xa3, 0x21, 0x18, 0x1a, 0xa3,
413*43a90889SApple OSS Distributions 	0x20, 0x18, 0x5e, 0xa3, 0x21, 0x00, 0x00, 0x00,
414*43a90889SApple OSS Distributions 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
415*43a90889SApple OSS Distributions 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
416*43a90889SApple OSS Distributions };
417*43a90889SApple OSS Distributions 
418*43a90889SApple OSS Distributions char test_no_magic[] = {
419*43a90889SApple OSS Distributions 	0x1
420*43a90889SApple OSS Distributions };
421*43a90889SApple OSS Distributions struct test {
422*43a90889SApple OSS Distributions 	char *              name;
423*43a90889SApple OSS Distributions 	char *              data;
424*43a90889SApple OSS Distributions 	int                 len;
425*43a90889SApple OSS Distributions 	boolean_t           result;
426*43a90889SApple OSS Distributions };
427*43a90889SApple OSS Distributions 
428*43a90889SApple OSS Distributions struct test tests[] = {
429*43a90889SApple OSS Distributions 	{ .name = "empty", .data = test_empty, .len = sizeof(test_empty), .result = TRUE },
430*43a90889SApple OSS Distributions 	{ .name = "simple", .data = test_simple, .len = sizeof(test_simple), .result = TRUE },
431*43a90889SApple OSS Distributions 	{ .name = "vendor", .data = test_vendor, .len = sizeof(test_vendor), .result = TRUE },
432*43a90889SApple OSS Distributions 	{ .name = "no_end", .data = test_no_end, .len = sizeof(test_no_end), .result = TRUE },
433*43a90889SApple OSS Distributions 	{ .name = "no magic", .data = test_no_magic, .len = sizeof(test_no_magic), .result = FALSE },
434*43a90889SApple OSS Distributions 	{ .name = "short", .data = test_short, .len = sizeof(test_short), .result =  FALSE },
435*43a90889SApple OSS Distributions 	{ .name = NULL, .data = NULL, .len = 0, .result = FALSE },
436*43a90889SApple OSS Distributions };
437*43a90889SApple OSS Distributions 
438*43a90889SApple OSS Distributions 
439*43a90889SApple OSS Distributions static char buf[2048];
440*43a90889SApple OSS Distributions 
441*43a90889SApple OSS Distributions int
main(void)442*43a90889SApple OSS Distributions main(void)
443*43a90889SApple OSS Distributions {
444*43a90889SApple OSS Distributions 	int         i;
445*43a90889SApple OSS Distributions 	dhcpol_t    options;
446*43a90889SApple OSS Distributions 	struct dhcp * pkt = (struct dhcp *)buf;
447*43a90889SApple OSS Distributions 
448*43a90889SApple OSS Distributions 	dhcpol_init(&options);
449*43a90889SApple OSS Distributions 
450*43a90889SApple OSS Distributions 	for (i = 0; tests[i].name; i++) {
451*43a90889SApple OSS Distributions 		printf("\nTest %d: ", i);
452*43a90889SApple OSS Distributions 		bcopy(tests[i].data, pkt->dp_options, tests[i].len);
453*43a90889SApple OSS Distributions 		if (dhcpol_parse_packet(&options, pkt,
454*43a90889SApple OSS Distributions 		    sizeof(*pkt) + tests[i].len)
455*43a90889SApple OSS Distributions 		    != tests[i].result) {
456*43a90889SApple OSS Distributions 			printf("test '%s' FAILED\n", tests[i].name);
457*43a90889SApple OSS Distributions 		} else {
458*43a90889SApple OSS Distributions 			printf("test '%s' PASSED\n", tests[i].name);
459*43a90889SApple OSS Distributions 		}
460*43a90889SApple OSS Distributions 		dhcpol_free(&options);
461*43a90889SApple OSS Distributions 	}
462*43a90889SApple OSS Distributions 	exit(0);
463*43a90889SApple OSS Distributions }
464*43a90889SApple OSS Distributions #endif /* TEST_DHCP_OPTIONS */
465