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